diff --git a/.clang-format b/.clang-format deleted file mode 100644 index 5af12d3a..00000000 --- a/.clang-format +++ /dev/null @@ -1,37 +0,0 @@ ---- -BraceWrapping: - AfterCaseLabel: false - AfterClass: false - AfterControlStatement: false - AfterEnum: false - AfterFunction: false - AfterNamespace: false - AfterObjCDeclaration: false - AfterStruct: false - AfterUnion: false - AfterExternBlock: false - BeforeCatch: false - BeforeElse: false - IndentBraces: false - SplitEmptyFunction: false - SplitEmptyRecord: false - SplitEmptyNamespace: false - BeforeLambdaBody: false - BeforeWhile: false -BreakBeforeBraces: Attach -ColumnLimit: 0 -IndentWidth: 4 -IndentCaseLabels: true -IncludeBlocks: Regroup -IncludeCategories: - - Regex: '<[[:alnum:].]+\.h>' - Priority: 1 - - Regex: '<[[:alnum:].]+>' - Priority: 2 - - Regex: '.*/.*' - Priority: 3 - - Regex: '.*' - Priority: 4 -DerivePointerAlignment: false -PointerAlignment: Left -... diff --git a/.clang-tidy b/.clang-tidy deleted file mode 100644 index ea50801d..00000000 --- a/.clang-tidy +++ /dev/null @@ -1,17 +0,0 @@ -Checks: '-*,readability-*,performance-*,modernize-*,-modernize-use-trailing-return-type,bugprone-*' -WarningsAsErrors: true -HeaderFilterRegex: '' -FormatStyle: none -CheckOptions: - - key: readability-identifier-naming.ClassCase - value: CamelCase - - key: readability-identifier-naming.ClassMethodCase - value: CamelCase - - key: readability-identifier-naming.ClassMemberPrefix - value: m_ - - key: readability-identifier-naming.ClassMemberCase - value: CamelCase - - key: readability-identifier-naming.ClassConstantCase - value: UPPER_CASE - - key: readability-identifier-naming.FunctionCase - value: CamelCase \ No newline at end of file diff --git a/.env.example b/.env.example index 15385906..462fb17f 100644 --- a/.env.example +++ b/.env.example @@ -14,4 +14,4 @@ EXTERNAL_IP=localhost MARIADB_USER=darkflame MARIADB_PASSWORD=SECRET_VALUE_CHANGE_ME MARIADB_ROOT_PASSWORD=SECRET_VALUE_CHANGE_ME -MARIADB_DATABASE=darkflame \ No newline at end of file +MARIADB_DATABASE=darkflame diff --git a/.git-blame-ignore-revs b/.git-blame-ignore-revs index 37567a02..14e48fbe 100644 --- a/.git-blame-ignore-revs +++ b/.git-blame-ignore-revs @@ -6,3 +6,6 @@ # convert to unix line endings 72477e01e2711e0f61cdb192ee266e5e21b8846f + +# enum cleanup +faf42d2f8cf432df2993b031f079b0b8c6d7dbe7 diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index 5a36df4b..8a81def7 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -10,12 +10,13 @@ jobs: build-and-test: name: Build & Test (${{ matrix.os }}) runs-on: ${{ matrix.os }} + continue-on-error: true strategy: matrix: os: [ windows-2022, ubuntu-20.04, macos-11 ] steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v3 with: submodules: true - name: Add msbuild to PATH (Windows only) @@ -34,23 +35,17 @@ jobs: buildPreset: "ci-${{matrix.os}}" testPreset: "ci-${{matrix.os}}" - name: artifacts - uses: actions/upload-artifact@v2 - if: ${{ github.ref == 'ref/head/main' }} + uses: actions/upload-artifact@v3 with: name: build-${{matrix.os}} path: | - build - !build/tests - !build/Testing - !build/CMakeFiles - !build/DartConfiguration.tcl - !build/CTestTestfile.cmake - !build/CMakeCache.txt - !build/build.ninja - !build/_deps - !build/cmake_install.cmake - !build/*.a - !build/*.lib - !build/*.dir - !build/*.vcxproj - !build/*.vcxproj.filters + build/*Server* + build/*.ini + build/*.so + build/*.dll + build/vanity/ + build/navmeshes/ + build/migrations/ + build/*.dcf + !build/*.pdb + !build/d*/ diff --git a/.gitignore b/.gitignore index 3777608d..e093ba4b 100644 --- a/.gitignore +++ b/.gitignore @@ -121,4 +121,4 @@ docker/__pycache__ docker-compose.override.yml -!/tests/TestBitStreams/*.bin +!*Test.bin diff --git a/.gitmodules b/.gitmodules index 6fb56bde..33193447 100644 --- a/.gitmodules +++ b/.gitmodules @@ -14,12 +14,6 @@ path = thirdparty/mariadb-connector-cpp url = https://github.com/mariadb-corporation/mariadb-connector-cpp.git ignore = dirty -[submodule "thirdparty/docker-utils"] - path = thirdparty/docker-utils - url = https://github.com/lcdr/utils.git -[submodule "thirdparty/LUnpack"] - path = thirdparty/LUnpack - url = https://github.com/Xiphoseer/LUnpack.git [submodule "thirdparty/AccountManager"] path = thirdparty/AccountManager url = https://github.com/DarkflameUniverse/AccountManager diff --git a/CMakeLists.txt b/CMakeLists.txt index 1a977a64..8f8981ee 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -1,4 +1,4 @@ -cmake_minimum_required(VERSION 3.14) +cmake_minimum_required(VERSION 3.18) project(Darkflame) include(CTest) @@ -58,7 +58,7 @@ if(UNIX) if(APPLE) set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++17 -O2 -Wuninitialized -D_GLIBCXX_USE_CXX11_ABI=0 -D_GLIBCXX_USE_CXX17_ABI=0 -fPIC") else() - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++17 -O2 -Wuninitialized -D_GLIBCXX_USE_CXX11_ABI=0 -D_GLIBCXX_USE_CXX17_ABI=0 -static-libgcc -fPIC") + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++17 -O2 -Wuninitialized -D_GLIBCXX_USE_CXX11_ABI=0 -D_GLIBCXX_USE_CXX17_ABI=0 -static-libgcc -fPIC -lstdc++fs") endif() if (__dynamic AND CMAKE_CXX_COMPILER_ID STREQUAL "GNU") set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -rdynamic") @@ -76,30 +76,88 @@ endif() # Our output dir set(CMAKE_BINARY_DIR ${PROJECT_BINARY_DIR}) + +# TODO make this not have to override the build type directories +set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELWITHDEBINFO ${CMAKE_BINARY_DIR}) +set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY_RELWITHDEBINFO ${CMAKE_BINARY_DIR}) +set(CMAKE_LIBRARY_OUTPUT_DIRECTORY_RELWITHDEBINFO ${CMAKE_BINARY_DIR}) +set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE ${CMAKE_BINARY_DIR}) +set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY_RELEASE ${CMAKE_BINARY_DIR}) +set(CMAKE_LIBRARY_OUTPUT_DIRECTORY_RELEASE ${CMAKE_BINARY_DIR}) + set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}) set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}) set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}) -# Create a /res directory -make_directory(${CMAKE_BINARY_DIR}/res) - -# Create a /locale directory -make_directory(${CMAKE_BINARY_DIR}/locale) +# Create a /resServer directory +make_directory(${CMAKE_BINARY_DIR}/resServer) # Create a /logs directory make_directory(${CMAKE_BINARY_DIR}/logs) # Copy resource files on first build -set(RESOURCE_FILES "authconfig.ini" "chatconfig.ini" "worldconfig.ini" "masterconfig.ini" "blacklist.dcf") -foreach(resource_file ${RESOURCE_FILES}) - if (NOT EXISTS ${PROJECT_BINARY_DIR}/${resource_file}) +set(RESOURCE_FILES "sharedconfig.ini" "authconfig.ini" "chatconfig.ini" "worldconfig.ini" "masterconfig.ini" "blacklist.dcf") +message(STATUS "Checking resource file integrity") +foreach (resource_file ${RESOURCE_FILES}) + set(file_size 0) + if (EXISTS ${PROJECT_BINARY_DIR}/${resource_file}) + file(SIZE ${PROJECT_BINARY_DIR}/${resource_file} file_size) + endif() + if (${file_size} EQUAL 0) configure_file( ${CMAKE_SOURCE_DIR}/resources/${resource_file} ${PROJECT_BINARY_DIR}/${resource_file} COPYONLY ) - message("Moved ${resource_file} to project binary directory") + message(STATUS "Moved " ${resource_file} " to project binary directory") + elseif (resource_file MATCHES ".ini") + message(STATUS "Checking " ${resource_file} " for missing config options") + file(READ ${PROJECT_BINARY_DIR}/${resource_file} current_file_contents) + string(REPLACE "\\\n" "" current_file_contents ${current_file_contents}) + string(REPLACE "\n" ";" current_file_contents ${current_file_contents}) + set(parsed_current_file_contents "") + # Remove comment lines so they do not interfere with the variable parsing + foreach (line ${current_file_contents}) + string(FIND ${line} "#" is_comment) + if (NOT ${is_comment} EQUAL 0) + string(APPEND parsed_current_file_contents ${line}) + endif() + endforeach() + file(READ ${CMAKE_SOURCE_DIR}/resources/${resource_file} depot_file_contents) + string(REPLACE "\\\n" "" depot_file_contents ${depot_file_contents}) + string(REPLACE "\n" ";" depot_file_contents ${depot_file_contents}) + set(line_to_add "") + foreach (line ${depot_file_contents}) + string(FIND ${line} "#" is_comment) + if (NOT ${is_comment} EQUAL 0) + string(REPLACE "=" ";" line_split ${line}) + list(GET line_split 0 variable_name) + if (NOT ${parsed_current_file_contents} MATCHES ${variable_name}) + message(STATUS "Adding missing config option " ${variable_name} " to " ${resource_file}) + set(line_to_add ${line_to_add} ${line}) + foreach (line_to_append ${line_to_add}) + file(APPEND ${PROJECT_BINARY_DIR}/${resource_file} "\n" ${line_to_append}) + endforeach() + file(APPEND ${PROJECT_BINARY_DIR}/${resource_file} "\n") + endif() + set(line_to_add "") + else() + set(line_to_add ${line_to_add} ${line}) + endif() + endforeach() endif() endforeach() +message(STATUS "Resource file integrity check complete") + +# Copy navmesh data on first build and extract it +if (NOT EXISTS ${PROJECT_BINARY_DIR}/navmeshes/) + configure_file( + ${CMAKE_SOURCE_DIR}/resources/navmeshes.zip ${PROJECT_BINARY_DIR}/navmeshes.zip + COPYONLY + ) + + file(ARCHIVE_EXTRACT INPUT ${PROJECT_BINARY_DIR}/navmeshes.zip) + file(REMOVE ${PROJECT_BINARY_DIR}/navmeshes.zip) +endif() # Copy vanity files on first build set(VANITY_FILES "CREDITS.md" "INFO.md" "TESTAMENT.md" "NPC.xml") @@ -108,13 +166,25 @@ foreach(file ${VANITY_FILES}) endforeach() # Move our migrations for MasterServer to run -file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/migrations/) +file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/migrations/dlu/) file(GLOB SQL_FILES ${CMAKE_SOURCE_DIR}/migrations/dlu/*.sql) foreach(file ${SQL_FILES}) get_filename_component(file ${file} NAME) - if (NOT EXISTS ${PROJECT_BINARY_DIR}/migrations/${file}) + if (NOT EXISTS ${PROJECT_BINARY_DIR}/migrations/dlu/${file}) configure_file( - ${CMAKE_SOURCE_DIR}/migrations/dlu/${file} ${PROJECT_BINARY_DIR}/migrations/${file} + ${CMAKE_SOURCE_DIR}/migrations/dlu/${file} ${PROJECT_BINARY_DIR}/migrations/dlu/${file} + COPYONLY + ) + endif() +endforeach() + +file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/migrations/cdserver/) +file(GLOB SQL_FILES ${CMAKE_SOURCE_DIR}/migrations/cdserver/*.sql) +foreach(file ${SQL_FILES}) + get_filename_component(file ${file} NAME) + if (NOT EXISTS ${PROJECT_BINARY_DIR}/migrations/cdserver/${file}) + configure_file( + ${CMAKE_SOURCE_DIR}/migrations/cdserver/${file} ${PROJECT_BINARY_DIR}/migrations/cdserver/${file} COPYONLY ) endif() @@ -123,6 +193,8 @@ endforeach() # Create our list of include directories set(INCLUDED_DIRECTORIES "dCommon" + "dCommon/dClient" + "dCommon/dEnums" "dChatFilter" "dGame" "dGame/dBehaviors" @@ -131,6 +203,8 @@ set(INCLUDED_DIRECTORIES "dGame/dInventory" "dGame/dMission" "dGame/dEntity" + "dGame/dPropertyBehaviors" + "dGame/dPropertyBehaviors/ControlBehaviorMessages" "dGame/dUtilities" "dPhysics" "dNavigation" @@ -140,12 +214,86 @@ set(INCLUDED_DIRECTORIES "dDatabase/Tables" "dNet" "dScripts" + "dScripts/02_server" + "dScripts/ai" + "dScripts/client" + "dScripts/EquipmentScripts" + "dScripts/EquipmentTriggers" + "dScripts/zone" + "dScripts/02_server/DLU" + "dScripts/02_server/Enemy" + "dScripts/02_server/Equipment" + "dScripts/02_server/Map" + "dScripts/02_server/Minigame" + "dScripts/02_server/Objects" + "dScripts/02_server/Pets" + "dScripts/02_server/Enemy/AG" + "dScripts/02_server/Enemy/AM" + "dScripts/02_server/Enemy/FV" + "dScripts/02_server/Enemy/General" + "dScripts/02_server/Enemy/Survival" + "dScripts/02_server/Enemy/VE" + "dScripts/02_server/Enemy/Waves" + "dScripts/02_server/Map/AG" + "dScripts/02_server/Map/AG_Spider_Queen" + "dScripts/02_server/Map/AM" + "dScripts/02_server/Map/FV" + "dScripts/02_server/Map/General" + "dScripts/02_server/Map/GF" + "dScripts/02_server/Map/njhub" + "dScripts/02_server/Map/NS" + "dScripts/02_server/Map/NT" + "dScripts/02_server/Map/PR" + "dScripts/02_server/Map/Property" + "dScripts/02_server/Map/SS" + "dScripts/02_server/Map/VE" + "dScripts/02_server/Map/FV/Racing" + "dScripts/02_server/Map/General/Ninjago" + "dScripts/02_server/Map/njhub/boss_instance" + "dScripts/02_server/Map/NS/Waves" + "dScripts/02_server/Map/Property/AG_Med" + "dScripts/02_server/Map/Property/AG_Small" + "dScripts/02_server/Map/Property/NS_Med" + "dScripts/02_server/Minigame/General" + "dScripts/ai/ACT" + "dScripts/ai/AG" + "dScripts/ai/FV" + "dScripts/ai/GENERAL" + "dScripts/ai/GF" + "dScripts/ai/MINIGAME" + "dScripts/ai/NP" + "dScripts/ai/NS" + "dScripts/ai/PETS" + "dScripts/ai/PROPERTY" + "dScripts/ai/RACING" + "dScripts/ai/SPEC" + "dScripts/ai/WILD" + "dScripts/ai/ACT/FootRace" + "dScripts/ai/MINIGAME/SG_GF" + "dScripts/ai/MINIGAME/SG_GF/SERVER" + "dScripts/ai/NS/NS_PP_01" + "dScripts/ai/NS/WH" + "dScripts/ai/PROPERTY/AG" + "dScripts/ai/RACING/OBJECTS" + "dScripts/client/ai" + "dScripts/client/ai/PR" + "dScripts/zone/AG" + "dScripts/zone/LUPs" + "dScripts/zone/PROPERTY" + "dScripts/zone/PROPERTY/FV" + "dScripts/zone/PROPERTY/GF" + "dScripts/zone/PROPERTY/NS" "thirdparty/raknet/Source" "thirdparty/tinyxml2" "thirdparty/recastnavigation" "thirdparty/SQLite" "thirdparty/cpplinq" + + "tests" + "tests/dCommonTests" + "tests/dGameTests" + "tests/dGameTests/dComponentsTests" ) # Add system specfic includes for Apple, Windows and Other Unix OS' (including Linux) @@ -160,7 +308,6 @@ elseif (UNIX) set(INCLUDED_DIRECTORIES ${INCLUDED_DIRECTORIES} "thirdparty/libbcrypt/include/bcrypt") endif() -include_directories(${ZLIB_INCLUDE_DIRS}) # Add binary directory as an include directory include_directories(${PROJECT_BINARY_DIR}) @@ -228,8 +375,6 @@ if (UNIX) endif() endif() -add_subdirectory(tests) - # Include all of our binary directories add_subdirectory(dWorldServer) add_subdirectory(dAuthServer) @@ -262,3 +407,7 @@ target_precompile_headers( tinyxml2 PRIVATE "$<$:${PROJECT_SOURCE_DIR}/thirdparty/tinyxml2/tinyxml2.h>" ) + +if (${__enable_testing__} MATCHES "1") + add_subdirectory(tests) +endif() diff --git a/CMakePresets.json b/CMakePresets.json index 133d6a3c..f8170755 100644 --- a/CMakePresets.json +++ b/CMakePresets.json @@ -1,126 +1,128 @@ { - "version": 3, - "cmakeMinimumRequired": { - "major": 3, - "minor": 14, - "patch": 0 - }, - "configurePresets": [ - { - "name": "default", - "displayName": "Default configure step", - "description": "Use 'build' dir and Unix makefiles", - "binaryDir": "${sourceDir}/build", - "generator": "Unix Makefiles" + "version": 3, + "cmakeMinimumRequired": { + "major": 3, + "minor": 14, + "patch": 0 }, - { - "name": "ci-ubuntu-20.04", - "displayName": "CI configure step for Ubuntu", - "description": "Same as default, Used in GitHub actions workflow", - "inherits": "default" - }, - { - "name": "ci-macos-11", - "displayName": "CI configure step for MacOS", - "description": "Same as default, Used in GitHub actions workflow", - "inherits": "default", - "cacheVariables": { - "OPENSSL_ROOT_DIR": "/usr/local/Cellar/openssl@3/3.0.5/" - } - }, - { - "name": "ci-windows-2022", - "displayName": "CI configure step for Windows", - "description": "Set architecture to 64-bit (b/c RakNet)", - "inherits": "default", - "generator": "Visual Studio 17 2022", - "architecture": { - "value": "x64" + "configurePresets": [ + { + "name": "default", + "displayName": "Default configure step", + "description": "Use 'build' dir and Unix makefiles", + "binaryDir": "${sourceDir}/build", + "generator": "Unix Makefiles" }, - "cacheVariables": { - "CMAKE_BUILD_TYPE": "RelWithDebInfo" + { + "name": "ci-ubuntu-20.04", + "displayName": "CI configure step for Ubuntu", + "description": "Same as default, Used in GitHub actions workflow", + "inherits": "default" + }, + { + "name": "ci-macos-11", + "displayName": "CI configure step for MacOS", + "description": "Same as default, Used in GitHub actions workflow", + "inherits": "default" + }, + { + "name": "ci-windows-2022", + "displayName": "CI configure step for Windows", + "description": "Set architecture to 64-bit (b/c RakNet)", + "inherits": "default", + "generator": "Visual Studio 17 2022", + "architecture": { + "value": "x64" + }, + "cacheVariables": { + "CMAKE_BUILD_TYPE": "RelWithDebInfo" + } + }, + { + "name": "windows-default", + "inherits": "ci-windows-2022", + "displayName": "Windows only Configure Settings", + "description": "Sets build and install directories", + "generator": "Ninja", + "architecture": { + "value": "x64", + "strategy": "external" + } } - }, - { - "name": "windows-default", - "inherits": "ci-windows-2022", - "displayName": "Windows only Configure Settings", - "description": "Sets build and install directories", - "generator": "Ninja", - "architecture": { - "value": "x64", - "strategy": "external" - } - } - ], - "buildPresets": [ - { - "name": "default", - "configurePreset": "default", - "displayName": "Default Build", - "description": "Default Build", - "jobs": 2 - }, - { - "name": "ci-windows-2022", - "configurePreset": "ci-windows-2022", - "displayName": "Windows CI Build", - "description": "This preset is used by the CI build on windows", - "configuration": "RelWithDebInfo", - "jobs": 2 - }, - { - "name": "ci-ubuntu-20.04", - "configurePreset": "ci-ubuntu-20.04", - "displayName": "Linux CI Build", - "description": "This preset is used by the CI build on linux", - "jobs": 2 - }, - { - "name": "ci-macos-11", - "configurePreset": "ci-macos-11", - "displayName": "MacOS CI Build", - "description": "This preset is used by the CI build on MacOS", - "jobs": 2 - } - ], - "testPresets": [ - { - "name": "ci-ubuntu-20.04", - "configurePreset": "ci-ubuntu-20.04", - "displayName": "CI Tests on Linux", - "description": "Runs all tests on a linux configuration", - "execution": { + ], + "buildPresets": [ + { + "name": "default", + "configurePreset": "default", + "displayName": "Default Build", + "description": "Default Build", "jobs": 2 }, - "output": { - "outputOnFailure": true - } - }, - { - "name": "ci-macos-11", - "configurePreset": "ci-macos-11", - "displayName": "CI Tests on MacOS", - "description": "Runs all tests on a Mac configuration", - "execution": { + { + "name": "ci-windows-2022", + "configurePreset": "ci-windows-2022", + "displayName": "Windows CI Build", + "description": "This preset is used by the CI build on windows", + "configuration": "RelWithDebInfo", "jobs": 2 }, - "output": { - "outputOnFailure": true - } - }, - { - "name": "ci-windows-2022", - "configurePreset": "ci-windows-2022", - "displayName": "CI Tests on windows", - "description": "Runs all tests on a windows configuration", - "configuration": "RelWithDebInfo", - "execution": { + { + "name": "ci-ubuntu-20.04", + "configurePreset": "ci-ubuntu-20.04", + "displayName": "Linux CI Build", + "description": "This preset is used by the CI build on linux", "jobs": 2 }, - "output": { - "outputOnFailure": true + { + "name": "ci-macos-11", + "configurePreset": "ci-macos-11", + "displayName": "MacOS CI Build", + "description": "This preset is used by the CI build on MacOS", + "jobs": 2 } - } - ] -} + ], + "testPresets": [ + { + "name": "ci-ubuntu-20.04", + "configurePreset": "ci-ubuntu-20.04", + "displayName": "CI Tests on Linux", + "description": "Runs all tests on a linux configuration", + "execution": { + "jobs": 2 + }, + "output": { + "outputOnFailure": true + } + }, + { + "name": "ci-macos-11", + "configurePreset": "ci-macos-11", + "displayName": "CI Tests on MacOS", + "description": "Runs all tests on a Mac configuration", + "execution": { + "jobs": 2 + }, + "output": { + "outputOnFailure": true + } + }, + { + "name": "ci-windows-2022", + "configurePreset": "ci-windows-2022", + "displayName": "CI Tests on windows", + "description": "Runs all tests on a windows configuration", + "configuration": "RelWithDebInfo", + "execution": { + "jobs": 2 + }, + "output": { + "outputOnFailure": true + }, + "filter": { + "exclude": { + "name": "((example)|(minigzip))+" + } + } + } + ] + } diff --git a/CMakeVariables.txt b/CMakeVariables.txt index 7a8d6b71..67b78aac 100644 --- a/CMakeVariables.txt +++ b/CMakeVariables.txt @@ -1,20 +1,20 @@ PROJECT_VERSION_MAJOR=1 PROJECT_VERSION_MINOR=0 -PROJECT_VERSION_PATCH=4 +PROJECT_VERSION_PATCH=1 # LICENSE LICENSE=AGPL-3.0 -# The network version. -# 171023 - Darkflame Universe client -# 171022 - Unmodded client -NET_VERSION=171022 # Debugging -__dynamic=1 # Set __dynamic to 1 to enable the -rdynamic flag for the linker, yielding some symbols in crashlogs. -# __ggdb=1 +__dynamic=1 # Set __ggdb to 1 to enable the -ggdb flag for the linker, including more debug info. -# __include_backtrace__=1 +# __ggdb=1 # Set __include_backtrace__ to 1 to includes the backtrace library for better crashlogs. +# __include_backtrace__=1 +# Set __compile_backtrace__ to 1 to compile the backtrace library instead of using system libraries. # __compile_backtrace__=1 -# Set __compile_backtrace__ to 1 to compile the backtrace library instead of using system libraries. -__maria_db_connector_compile_jobs__=1 # Set to the number of jobs (make -j equivalent) to compile the mariadbconn files with. +__maria_db_connector_compile_jobs__=1 +# When set to 1 and uncommented, compiling and linking testing folders and libraries will be done. +__enable_testing__=1 +# The path to OpenSSL. Change this if your OpenSSL install path is different than the default. +OPENSSL_ROOT_DIR=/usr/local/opt/openssl@3/ diff --git a/Docker.md b/Docker.md index 54d0ce19..b06bd0fe 100644 --- a/Docker.md +++ b/Docker.md @@ -4,7 +4,7 @@ - [Docker](https://docs.docker.com/get-docker/) (Docker Desktop or on Linux normal Docker) - [Docker Compose](https://docs.docker.com/compose/install/) (Included in Docker Desktop) -- LEGO® Universe packed Client. Check the main [README](./README.md) for details on this. +- LEGO® Universe Client. Check the main [README](./README.md) for details on this. ## Run server inside Docker diff --git a/Docker_Windows.md b/Docker_Windows.md index 1cc633cc..984bbe57 100644 --- a/Docker_Windows.md +++ b/Docker_Windows.md @@ -25,7 +25,7 @@ 11. Once the command has completed (you can see you path again and can enter commands), close the window. 12. Inside the downloaded folder, copy `.env.example` and name the copy `.env` 13. Open `.env` with Notepad by right-clicking it and selecting _Open With_ -> _More apps_ -> _Notepad_. -14. Change the text after `CLIENT_PATH=` to the location of your client. The folder you are pointing to must contain a folder called `client` which should contain the client files. +14. Change the text after `CLIENT_PATH=` to the location of your client. This folder must contain either a folder `client` or `legouniverse.exe`. > If you need the extra performance, place the client files in `\\wsl$\\...` to avoid working across file systems, see [Docker Best Practices](https://docs.docker.com/desktop/windows/wsl/#best-practices) and [WSL documentation](https://docs.microsoft.com/en-us/windows/wsl/filesystems#file-storage-and-performance-across-file-systems). 15. Optionally, you can change the number after `BUILD_THREADS=` to the number of cores / threads your processor has. If your computer crashes while building, you can try to reduce this value. diff --git a/README.md b/README.md index de20e7ad..b51fa4b0 100644 --- a/README.md +++ b/README.md @@ -18,204 +18,192 @@ Darkflame Universe is licensed under AGPLv3, please read [LICENSE](LICENSE). Som Throughout the entire build and setup process a level of familiarity with the command line and preferably a Unix-like development environment is greatly advantageous. ### Hosting a server -We do not recommend hosting public servers. DLU is intended for small scale deployment, for example within a group of friends. It has not been tested for large scale deployment which comes with additional security risks. +We do not recommend hosting public servers. Darkflame Universe is intended for small scale deployment, for example within a group of friends. It has not been tested for large scale deployment which comes with additional security risks. ### Supply of resource files -Darkflame Universe is a server emulator and does not distribute any LEGO® Universe files. A separate game client is required to setup this server emulator and play the game, which we cannot supply. Users are strongly suggested to refer to the safe checksums listed in the resources tab below when checking if a client will work. +Darkflame Universe is a server emulator and does not distribute any LEGO® Universe files. A separate game client is required to setup this server emulator and play the game, which we cannot supply. Users are strongly suggested to refer to the safe checksums listed [here](#verifying-your-client-files) to see if a client will work. -## Build -Development of the latest iteration of Darkflame Universe has been done primarily in a Unix-like environment and is where it has been tested and designed for deployment. It is therefore highly recommended that Darkflame Universe be built and deployed using a Unix-like environment for the most streamlined experience. +## Steps to setup server +* [Clone this repository](#clone-the-repository) +* [Install dependencies](#install-dependencies) +* [Database setup](#database-setup) +* [Build the server](#build-the-server) +* [Configuring your server](#configuring-your-server) + * [Required Configuration](#required-configuration) + * [Optional Configuration](#optional-configuration) +* [Verify your setup](#verify-your-setup) +* [Running the server](#running-the-server) +* [User Guide](#user-guide) -### Prerequisites -#### Clone the repository +## Clone the repository +If you are on Windows, you will need to download and install git from [here](https://git-scm.com/download/win) + +Then run the following command ```bash git clone --recursive https://github.com/DarkflameUniverse/DarkflameServer ``` -#### Python -Some tools utilized to streamline the setup process require Python 3, make sure you have it installed. +## Install dependencies +### Windows packages +Ensure that you have either the [MSVC C++ compiler](https://visualstudio.microsoft.com/vs/features/cplusplus/) (recommended) or the [Clang compiler](https://github.com/llvm/llvm-project/releases/) installed. +You'll also need to download and install [CMake](https://cmake.org/download/) (version **CMake version 3.18** or later!). -### Choosing the right version for your client -DLU clients identify themselves using a higher version number than the regular live clients out there. -This was done make sure that older and incomplete clients wouldn't produce false positive bug reports for us, and because we made bug fixes and new content for the client. +### MacOS packages +Ensure you have [brew](https://brew.sh) installed. +You will need to install the following packages +```bash +brew install cmake gcc mariadb openssl zlib +``` -If you're using a DLU client you'll have to go into the "CMakeVariables.txt" file and change the NET_VERSION variable to 171023 to match the modified client's version number. +### Linux packages +Make sure packages like `gcc`, and `zlib` are installed. Depending on the distribution, these packages might already be installed. Note that on systems like Ubuntu, you will need the `zlib1g-dev` package so that the header files are available. `libssl-dev` will also be required as well as `openssl`. You will also need a MySQL database solution to use. We recommend using `mariadb-server`. -### Using Docker -Refer to [Docker.md](/Docker.md). +For Ubuntu, you would run the following commands. On other systems, the package install command will differ. -For Windows, refer to [Docker_Windows.md](/Docker_Windows.md). +```bash +sudo apt update && sudo apt upgrade -### Linux builds -Make sure packages like `gcc`, `cmake`, and `zlib` are installed. Depending on the distribution, these packages might already be installed. Note that on systems like Ubuntu, you will need the `zlib1g-dev` package so that the header files are available. `libssl-dev` will also be required as well as `openssl`. +# Install packages +sudo apt install build-essential gcc zlib1g-dev libssl-dev openssl mariadb-server cmake +``` -CMake must be version 3.14 or higher! +#### Required CMake version +This project uses **CMake version 3.18** or higher and as such you will need to ensure you have this version installed. +You can check your CMake version by using the following command in a terminal. +```bash +cmake --version +``` -#### Build the repository +If you are going to be using an Ubuntu environment to run the server, you may need to get a more recent version of `cmake` than the packages available may provide. +The general approach to do so would be to obtain a copy of the signing key and then add the CMake repository to your apt. +You can do so with the following commands. + +[Source of the below commands](https://askubuntu.com/questions/355565/how-do-i-install-the-latest-version-of-cmake-from-the-command-line) + +```bash +# Remove the old version of CMake +sudo apt purge --auto-remove cmake + +# Prepare for installation +sudo apt update && sudo apt install -y software-properties-common lsb-release && sudo apt clean all + +# Obtain a copy of the signing key +wget -O - https://apt.kitware.com/keys/kitware-archive-latest.asc 2>/dev/null | gpg --dearmor - | sudo tee /etc/apt/trusted.gpg.d/kitware.gpg >/dev/null + +# Add the repository to your sources list. +sudo apt-add-repository "deb https://apt.kitware.com/ubuntu/ $(lsb_release -cs) main" + +# Next you'll want to ensure that Kitware's keyring stays up to date +sudo apt update +sudo apt install kitware-archive-keyring +sudo rm /etc/apt/trusted.gpg.d/kitware.gpg + +# If sudo apt update above returned an error, copy the public key at the end of the error message and run the following command +# if the error message was "The following signatures couldn't be verified because the public key is not available: NO_PUBKEY 6AF7F09730B3F0A4" +# then the below command would be "sudo apt-key adv --keyserver keyserver.ubuntu.com --recv-keys 6AF7F09730B3F0A4" +sudo apt-key adv --keyserver keyserver.ubuntu.com --recv-keys + +# Finally update and install +sudo apt update +sudo apt install cmake +``` + +## Database setup +First you'll need to start MariaDB. + +For Windows the service is always running by default. + +For MacOS, run the following command +```bash +brew services start mariadb +``` + +For Linux, run the following command +```bash +sudo systemctl start mysql +# If systemctl is not a known command on your distribution, try the following instead +sudo service mysql start +``` + +**You will need to run this command every time you restart your environment** + +If you are using Linux and `systemctl` and want the MariaDB instance to start on startup, run the following command +```bash +sudo systemctl enable --now mysql +``` + +Once MariaDB is started, you'll need to create a user and an empty database for Darkflame Universe to use. + +First, login to the MariaDB instance. + +To do this on Ubuntu/Linux, MacOS, or another Unix like operating system, run the following command in a terminal +```bash +# Logs you into the MariaDB instance as root +sudo mysql +``` + +For Windows, run the following command in the `Command Prompt (MariaDB xx.xx)` terminal +```bash +# Logs you into the mysql instance +mysql -u root -p +# You will then be prompted for the password you set for root during installation of MariaDB +``` + +Now that you are logged in, run the following commands. + +```bash +# Creates a user for this computer which uses a password and grant said user all privileges. +# Change mydarkflameuser to a custom username and password to a custom password. +GRANT ALL ON *.* TO 'mydarkflameuser'@'localhost' IDENTIFIED BY 'password' WITH GRANT OPTION; +FLUSH PRIVILEGES; + +# Then create a database for Darkflame Universe to use. +CREATE DATABASE darkflame; +``` + +## Build the server You can either run `build.sh` when in the root folder of the repository: ```bash ./build.sh ``` -Or manually run the commands used in `build.sh`: +Or manually run the commands used in [build.sh](build.sh). +If you would like to build the server faster, append `-j` where number is the number of simultaneous compile jobs to run at once. It is recommended that you have this number always be 1 less than your core count to prevent slowdowns. The command would look like this if you would build with 4 jobs at once: ```bash -# Create the build directory, preserving it if it already exists -mkdir -p build -cd build - -# Run CMake to generate make files -cmake .. - -# To build utilizing multiple cores, append `-j` and the amount of cores to utilize, for example `cmake --build . --config Release -j8' -cmake --build . --config Release +./build.sh -j4 ``` +### Notes +Depending on your operating system, you may need to adjust some pre-processor defines in [CMakeVariables.txt](./CMakeVariables.txt) before building: +* If you are on MacOS, ensure OPENSSL_ROOT_DIR is pointing to the openssl root directory. +* If you are using a Darkflame Universe client, ensure `client_net_version` in `build/sharedconfig.ini` is changed to 171023. -### MacOS builds -Ensure `cmake`, `zlib` and `open ssl` are installed as well as a compiler (e.g `clang` or `gcc`). +## Configuring your server +This server has a few steps that need to be taken to configure the server for your use case. -In the repository root folder run the following. Ensure -DOPENSSL_ROOT_DIR=/path/to/openssl points to your openssl install location -```bash -# Create the build directory, preserving it if it already exists -mkdir -p build -cd build +### Required Configuration +Darkflame Universe can run with either a packed or an unpacked client. +Navigate to `build/sharedconfig.ini` and fill in the following fields: +* `mysql_host` (This is the IP address or hostname of your MariaDB server. This is highly likely `localhost`) + * If you setup your MariaDB instance on a port other than 3306, which can be done on a Windows install, you will need to make this value `tcp://localhost:portNum` where portNum is replaced with the port you chose to run MariaDB on. +* `mysql_database` (This is the database you created for the server) +* `mysql_username` (This is the user you created for the server) +* `mysql_password` (This is the password for the user you created for the server) +* `client_location` (This is the location of the client files. This should be the folder path of a packed or unpacked client) + * Ideally the path to the client should not contain any spaces. -# Run CMake to generate build files -cmake .. -DOPENSSL_ROOT_DIR=/path/to/openssl - -# Get cmake to build the project. If make files are being used then using make and appending `-j` and the amount of cores to utilize may be preferable, for example `make -j8` -cmake --build . --config Release -``` - -### Windows builds (native) -Ensure that you have either the [MSVC](https://visualstudio.microsoft.com/vs/) or the [Clang](https://github.com/llvm/llvm-project/releases/) (recommended) compiler installed. You will also need to install [CMake](https://cmake.org/download/). Currently on native Windows the server will only work in Release mode. - -#### Build the repository -```batch -:: Create the build directory -mkdir build -cd build - -:: Run CMake to generate make files -cmake .. - -:: Run CMake with build flag to build -cmake --build . --config Release -``` -#### Windows for ARM has not been tested but should build by doing the following -```batch -:: Create the build directory -mkdir build -cd build - -:: Run CMake to generate make files -cmake .. -DMARIADB_BUILD_SOURCE=ON - -:: Run CMake with build flag to build -cmake --build . --config Release -``` - -### Windows builds (WSL) -This section will go through how to install [WSL](https://docs.microsoft.com/en-us/windows/wsl/install) and building in a Linux environment under Windows. WSL requires Windows 10 version 2004 and higher (Build 19041 and higher) or Windows 11. - -#### Open the Command Prompt application with Administrator permissions and run the following: -```bash -# Installing Windows Subsystem for Linux -wsl --install -``` - -#### Open the Ubuntu application and run the following: -```bash -# Make sure the install is up to date -apt update && apt upgrade - -# Make sure the gcc, cmake, and build-essentials are installed -sudo apt install gcc -sudo apt install cmake -sudo apt install build-essential -``` - -[**Follow the Linux instructions**](#linux-builds) - -### ARM builds -AArch64 builds should work on linux and MacOS using their respective build steps. Windows ARM should build but it has not been tested - -### Updating your build -To update your server to the latest version navigate to your cloned directory -```bash -cd /path/to/DarkflameServer -``` -run the following commands to update to the latest changes -```bash -git pull -git submodule update --init --recursive -``` -now follow the build section for your system - -## Setting up the environment - -### Resources - -#### LEGO® Universe 1.10.64 - -This repository does not distribute any LEGO® Universe files. A full install of LEGO® Universe version 1.10.64 (latest) is required to finish setting up Darkflame Universe. - -Known good SHA256 checksums of the client: -- `8f6c7e84eca3bab93232132a88c4ae6f8367227d7eafeaa0ef9c40e86c14edf5` (packed client, rar compressed) -- `c1531bf9401426042e8bab2de04ba1b723042dc01d9907c2635033d417de9e05` (packed client, includes extra locales, rar compressed) -- `0d862f71eedcadc4494c4358261669721b40b2131101cbd6ef476c5a6ec6775b` (unpacked client, includes extra locales, rar compressed) - -Known good *SHA1* checksum of the DLU client: -- `91498e09b83ce69f46baf9e521d48f23fe502985` (packed client, zip compressed) - -How to generate a SHA256 checksum: -```bash -# Replace with the file path to the client - -# If on Linux or MacOS -shasum -a 256 - -# If on Windows -certutil -hashfile SHA256 -``` - -#### Unpacking the client -* Clone lcdr's utilities repository [here](https://github.com/lcdr/utils) -* Use `pkextractor.pyw` to unpack the client files if they are not already unpacked - -#### Setup resource directory -* In the `build` directory create a `res` directory if it does not already exist. -* Copy over or create symlinks from `macros`, `BrickModels`, `chatplus_en_us.txt`, `names`, and `maps` in your client `res` directory to the server `build/res` directory -* Unzip the navmeshes [here](./resources/navmeshes.zip) and place them in `build/res/maps/navmeshes` - -#### Setup locale -* In the `build` directory create a `locale` directory if it does not already exist -* Copy over or create symlinks from `locale.xml` in your client `locale` directory to the `build/locale` directory - -#### Client database -* Use `fdb_to_sqlite.py` in lcdr's utilities on `res/cdclient.fdb` in the unpacked client to convert the client database to `cdclient.sqlite` -* Move and rename `cdclient.sqlite` into `build/res/CDServer.sqlite` -* Run each SQL file in the order at which they appear [here](migrations/cdserver/) on the SQLite database - -### Database -Darkflame Universe utilizes a MySQL/MariaDB database for account and character information. - -Initial setup can vary drastically based on which operating system or distribution you are running; there are instructions out there for most setups, follow those and come back here when you have a database up and running. -* Create a database for Darkflame Universe to use - -#### Configuration - -After the server has been built there should be four `ini` files in the build director: `authconfig.ini`, `chatconfig.ini`, `masterconfig.ini`, and `worldconfig.ini`. Go through them and fill in the database credentials and configure other settings if necessary. - -#### Setup and Migrations - -Use the command `./MasterServer -m` to setup the tables in the database. The first time this command is run on a database, the tables will be up to date with the most recent version. To update your database tables, run this command again. Multiple invocations will not affect any functionality. - -#### Verify +### Optional Configuration +* After the server has been built there should be five `ini` files in the build directory: `sharedconfig.ini`, `authconfig.ini`, `chatconfig.ini`, `masterconfig.ini`, and `worldconfig.ini`. +* `authconfig.ini` contains an option to enable or disable play keys on your server. Do not change the default port for auth. +* `chatconfig.ini` contains a port option. +* `masterconfig.ini` contains options related to permissions you want to run your servers with. +* `sharedconfig.ini` contains several options that are shared across all servers +* `worldconfig.ini` contains several options to turn on QOL improvements should you want them. If you would like the most vanilla experience possible, you will need to turn some of these settings off. +## Verify your setup Your build directory should now look like this: * AuthServer * ChatServer @@ -224,212 +212,105 @@ Your build directory should now look like this: * authconfig.ini * chatconfig.ini * masterconfig.ini +* sharedconfig.ini * worldconfig.ini -* **locale/** - * locale.xml -* **res/** - * CDServer.sqlite - * chatplus_en_us.txt - * **macros/** - * ... - * **BrickModels/** - * ... - * **maps/** - * **navmeshes/** - * ... - * ... * ... ## Running the server -If everything has been configured correctly you should now be able to run the `MasterServer` binary. Darkflame Universe utilizes port numbers under 1024, so under Linux you either have to give the binary network permissions or run it under sudo. +If everything has been configured correctly you should now be able to run the `MasterServer` binary which is located in the `build` directory. Darkflame Universe utilizes port numbers under 1024, so under Linux you either have to give the `AuthServer` binary network permissions or run it under sudo. +To give `AuthServer` network permissions and not require sudo, run the following command +```bash +sudo setcap 'cap_net_bind_service=+ep' AuthServer +``` +and then go to `build/masterconfig.ini` and change `use_sudo_auth` to 0. ### First admin user Run `MasterServer -a` to get prompted to create an admin account. This method is only intended for the system administrator as a means to get started, do NOT use this method to create accounts for other users! -### Account Manager +### Account management tool (Nexus Dashboard) +**If you are just using this server for yourself, you can skip setting up Nexus Dashboard** -Follow the instructions [here](https://github.com/DarkflameUniverse/AccountManager) to setup the DLU account management Python web application. This is the intended way for users to create accounts. +Follow the instructions [here](https://github.com/DarkflameUniverse/NexusDashboard) to setup the DLU Nexus Dashboard web application. This is the intended way for users to create accounts and the intended way for moderators to approve names/pets/properties and do other moderation actions. ### Admin levels +The admin level, or Game Master level (hereafter referred to as gmlevel), is specified in the `accounts.gm_level` column in the MySQL database. Normal players should have this set to `0`, which comes with no special privileges. The system administrator will have this set to `9`, which comes will all privileges. gmlevel `8` should be used to give a player a majority of privileges without the safety critical once. -The admin level, or game master level, is specified in the `accounts.gm_level` column in the MySQL database. Normal players should have this set to `0`, which comes with no special privileges. The system administrator will have this set to `9`, which comes will all privileges. Admin level `8` should be used to give a player a majority of privileges without the safety critical once. +While a character has a gmlevel of anything but `0`, some gameplay behavior will change. When testing gameplay, you should always use a character with a gmlevel of `0`. -While a character has a gmlevel of anything but 0, some gameplay behavior will change. When testing gameplay, you should always use a character with a gmlevel of 0. +# User guide +Some changes to the client `boot.cfg` file are needed to play on your server. -## User guide -A few modifications have to be made to the client. - -### Client configuration +## Allowing a user to connect to your server To connect to a server follow these steps: * In the client directory, locate `boot.cfg` * Open it in a text editor and locate where it says `AUTHSERVERIP=0:` * Replace the contents after to `:` and the following `,` with what you configured as the server's public facing IP. For example `AUTHSERVERIP=0:localhost` for locally hosted servers +* Next locate the line `UGCUSE3DSERVICES=7:` +* Ensure the number after the 7 is a `0` * Launch `legouniverse.exe`, through `wine` if on a Unix-like operating system * Note that if you are on WSL2, you will need to configure the public IP in the server and client to be the IP of the WSL2 instance and not localhost, which can be found by running `ifconfig` in the terminal. Windows defaults to WSL1, so this will not apply to most users. -### Survival +## Updating your server +To update your server to the latest version navigate to your cloned directory +```bash +cd path/to/DarkflameServer +``` +Run the following commands to update to the latest changes +```bash +git pull +git submodule update --init --recursive +``` +Now follow the [build](#build-the-server) section for your system and your server is up to date. -The client script for the survival minigame has a bug in it which can cause the minigame to not load. To fix this, follow these instructions: -* Open `res/scripts/ai/minigame/survival/l_zone_survival_client.lua` -* Navigate to line `617` -* Change `PlayerReady(self)` to `onPlayerReady(self)` -* Save the file, overriding readonly mode if required +## In-game commands +* A list of all in-game commands can be found [here](./docs/Commands.md). -If you still experience the bug, try deleting/renaming `res/pack/scripts.pk`. +## Verifying your client files -### Brick-By-Brick building +### LEGO® Universe 1.10.64 +To verify that you are indeed using a LEGO® Universe 1.10.64 client, make sure you have the full client compressed **in a rar file** and run the following command. +```bash +# Replace with the file path to the zipped client -Brick-By-Brick building requires `PATCHSERVERIP=0:` in the `boot.cfg` to point to a HTTP server which always returns `HTTP 404 - Not Found` for all requests. This can be achieved by pointing it to `localhost` while having `sudo python -m http.server 80` running in the background. +# If on Linux or MacOS +shasum -a 256 -### In-game commands -Here is a summary of the commands available in-game. All commands are prefixed by `/` and typed in the in-game chat window. Some commands requires admin privileges. Operands within `<>` are required, operands within `()` are not. For the full list of in-game commands, please checkout [the source file](./dGame/dUtilities/SlashCommandHandler.cpp). +# If on Windows using the Command Prompt +certutil -hashfile SHA256 +``` - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- Command - - Usage - - Description - - Admin Level Requirement -
- info - - /info - - Displays server info to the user, including where to find the server's source code. - -
- credits - - /credits - - Displays the names of the people behind Darkflame Universe. - -
- instanceinfo - - /instanceinfo - - Displays in the chat the current zone, clone, and instance id. - -
- gmlevel - - /gmlevel <level> - - Within the authorized range of levels for the current account, changes the character's game master level to the specified value. This is required to use certain commands. - -
- testmap - - /testmap <zone> (clone-id) - - Transfers you to the given zone by id and clone id. - - 1 -
- ban - - /ban <username> - - Bans a user from the server. - - 4 -
- gmadditem - - /gmadditem <id> (count) - - Adds the given item to your inventory by id. - - 8 -
- spawn - - /spawn <id> - - Spawns an object at your location by id. - - 8 -
- metrics - - /metrics - - Prints some information about the server's performance. - - 8 -
+Below are known good SHA256 checksums of the client: +* `8f6c7e84eca3bab93232132a88c4ae6f8367227d7eafeaa0ef9c40e86c14edf5` (packed client, rar compressed) +* `c1531bf9401426042e8bab2de04ba1b723042dc01d9907c2635033d417de9e05` (packed client, includes extra locales, rar compressed) +* `0d862f71eedcadc4494c4358261669721b40b2131101cbd6ef476c5a6ec6775b` (unpacked client, includes extra locales, rar compressed) + +If the returned hash matches one of the lines above then you can continue with setting up the server. If you are using a fully downloaded and complete client from live, then it will work, but the hash above may not match. Otherwise you must obtain a full install of LEGO® Universe 1.10.64. + +### Darkflame Universe Client +Darkflame Universe clients identify themselves using a higher version number than the regular live clients out there. +This was done make sure that older and incomplete clients wouldn't produce false positive bug reports for us, and because we made bug fixes and new content for the client. + +To verify that you are indeed using a Darkflame Universe client, make sure you have the full client compressed **in a zip file** and run the following command. + +```bash +# Replace with the file path to the zipped client + +# If on Linux or MacOS +shasum -a 1 + +# If on Windows using the Command Prompt +certutil -hashfile SHA1 +``` + +Known good *SHA1* checksum of the Darkflame Universe client: +- `91498e09b83ce69f46baf9e521d48f23fe502985` (packed client, zip compressed) + +# Development Documentation +This is a Work in Progress, but below are some quick links to documentaion for systems and structs in the server +[Networked message structs](https://lcdruniverse.org/lu_packets/lu_packets/index.html) +[General system documentation](https://docs.lu-dev.net/en/latest/index.html) # Credits -## Active Contributors -* [EmosewaMC](https://github.com/EmosewaMC) -* [Jettford](https://github.com/Jettford) -* [Aaron K.](https://github.com/aronwk-aaron) ## DLU Team * [DarwinAnim8or](https://github.com/DarwinAnim8or) @@ -438,25 +319,30 @@ Here is a summary of the commands available in-game. All commands are prefixed b * [averysumner](https://github.com/codeshaunted) * [Jon002](https://github.com/jaller200) * [Jonny](https://github.com/cuzitsjonny) +* [Aaron K.](https://github.com/aronwk-aaron) -### Research and tools +### Research and Tools * [lcdr](https://github.com/lcdr) * [Xiphoseer](https://github.com/Xiphoseer) -### Community management +### Community Management * [Neal](https://github.com/NealSpellman) -### Former contributors -* TheMachine -* Matthew -* [Raine](https://github.com/Rainebannister) -* Bricknave - ### Logo * Cole Peterson (BlasterBuilder) -## Special thanks +## Active Contributors +* [EmosewaMC](https://github.com/EmosewaMC) +* [Jettford](https://github.com/Jettford) + +## Former Contributors +* TheMachine +* Matthew +* [Raine](https://github.com/uwainium) +* Bricknave + +## Special Thanks * humanoid24 * pwjones1969 * [Simon](https://github.com/SimonNitzsche) -* ALL OF THE NETDEVIL AND LEGO TEAMS! +* [ALL OF THE NETDEVIL AND LEGO TEAMS!](https://www.mobygames.com/game/macintosh/lego-universe/credits) diff --git a/build.sh b/build.sh index e61c1cb4..b8d33492 100755 --- a/build.sh +++ b/build.sh @@ -1,3 +1,6 @@ +# Error if any command fails +set -e + # Create the build directory, preserving it if it already exists mkdir -p build cd build @@ -6,7 +9,5 @@ cd build cmake .. # To build utilizing multiple cores, append `-j` and the amount of cores to utilize, for example `cmake --build . --config Release -j8' -cmake --build . --config Release +cmake --build . --config Release $1 -# Run migrations -./MasterServer -m diff --git a/dAuthServer/AuthServer.cpp b/dAuthServer/AuthServer.cpp index 73cb4212..38910823 100644 --- a/dAuthServer/AuthServer.cpp +++ b/dAuthServer/AuthServer.cpp @@ -11,47 +11,55 @@ #include "Database.h" #include "dConfig.h" #include "Diagnostics.h" +#include "BinaryPathFinder.h" //RakNet includes: #include "RakNetDefines.h" +#include //Auth includes: #include "AuthPackets.h" -#include "dMessageIdentifiers.h" +#include "eConnectionType.h" +#include "eServerMessageType.h" +#include "eAuthMessageType.h" #include "Game.h" namespace Game { - dLogger* logger; - dServer* server; - dConfig* config; + dLogger* logger = nullptr; + dServer* server = nullptr; + dConfig* config = nullptr; + bool shouldShutdown = false; + std::mt19937 randomEngine; } dLogger* SetupLogger(); void HandlePacket(Packet* packet); int main(int argc, char** argv) { + constexpr uint32_t authFramerate = mediumFramerate; + constexpr uint32_t authFrameDelta = mediumFrameDelta; Diagnostics::SetProcessName("Auth"); Diagnostics::SetProcessFileName(argv[0]); Diagnostics::Initialize(); //Create all the objects we need to run our service: Game::logger = SetupLogger(); - if (!Game::logger) return 0; + if (!Game::logger) return EXIT_FAILURE; + + //Read our config: + Game::config = new dConfig((BinaryPathFinder::GetBinaryDir() / "authconfig.ini").string()); + Game::logger->SetLogToConsole(Game::config->GetValue("log_to_console") != "0"); + Game::logger->SetLogDebugStatements(Game::config->GetValue("log_debug_statements") == "1"); + Game::logger->Log("AuthServer", "Starting Auth server..."); Game::logger->Log("AuthServer", "Version: %i.%i", PROJECT_VERSION_MAJOR, PROJECT_VERSION_MINOR); Game::logger->Log("AuthServer", "Compiled on: %s", __TIMESTAMP__); - //Read our config: - dConfig config("authconfig.ini"); - Game::config = &config; - Game::logger->SetLogToConsole(bool(std::stoi(config.GetValue("log_to_console")))); - Game::logger->SetLogDebugStatements(config.GetValue("log_debug_statements") == "1"); - //Connect to the MySQL Database - std::string mysql_host = config.GetValue("mysql_host"); - std::string mysql_database = config.GetValue("mysql_database"); - std::string mysql_username = config.GetValue("mysql_username"); - std::string mysql_password = config.GetValue("mysql_password"); + std::string mysql_host = Game::config->GetValue("mysql_host"); + std::string mysql_database = Game::config->GetValue("mysql_database"); + std::string mysql_username = Game::config->GetValue("mysql_username"); + std::string mysql_password = Game::config->GetValue("mysql_password"); try { Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); @@ -60,12 +68,12 @@ int main(int argc, char** argv) { Database::Destroy("AuthServer"); delete Game::server; delete Game::logger; - return 0; + return EXIT_FAILURE; } //Find out the master's IP: std::string masterIP; - int masterPort = 1500; + uint32_t masterPort = 1500; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -76,27 +84,31 @@ int main(int argc, char** argv) { delete res; delete stmt; - //It's safe to pass 'localhost' here, as the IP is only used as the external IP. - int maxClients = 50; - int ourPort = 1001; //LU client is hardcoded to use this for auth port, so I'm making it the default. - if (config.GetValue("max_clients") != "") maxClients = std::stoi(config.GetValue("max_clients")); - if (config.GetValue("port") != "") ourPort = std::atoi(config.GetValue("port").c_str()); + Game::randomEngine = std::mt19937(time(0)); - Game::server = new dServer(config.GetValue("external_ip"), ourPort, 0, maxClients, false, true, Game::logger, masterIP, masterPort, ServerType::Auth); + //It's safe to pass 'localhost' here, as the IP is only used as the external IP. + uint32_t maxClients = 50; + uint32_t ourPort = 1001; //LU client is hardcoded to use this for auth port, so I'm making it the default. + if (Game::config->GetValue("max_clients") != "") maxClients = std::stoi(Game::config->GetValue("max_clients")); + if (Game::config->GetValue("port") != "") ourPort = std::atoi(Game::config->GetValue("port").c_str()); + + Game::server = new dServer(Game::config->GetValue("external_ip"), ourPort, 0, maxClients, false, true, Game::logger, masterIP, masterPort, ServerType::Auth, Game::config, &Game::shouldShutdown); //Run it until server gets a kill message from Master: auto t = std::chrono::high_resolution_clock::now(); Packet* packet = nullptr; - int framesSinceLastFlush = 0; - int framesSinceMasterDisconnect = 0; - int framesSinceLastSQLPing = 0; + constexpr uint32_t logFlushTime = 30 * authFramerate; // 30 seconds in frames + constexpr uint32_t sqlPingTime = 10 * 60 * authFramerate; // 10 minutes in frames + uint32_t framesSinceLastFlush = 0; + uint32_t framesSinceMasterDisconnect = 0; + uint32_t framesSinceLastSQLPing = 0; - while (true) { + while (!Game::shouldShutdown) { //Check if we're still connected to master: if (!Game::server->GetIsConnectedToMaster()) { framesSinceMasterDisconnect++; - if (framesSinceMasterDisconnect >= 30) + if (framesSinceMasterDisconnect >= authFramerate) break; //Exit our loop, shut down. } else framesSinceMasterDisconnect = 0; @@ -112,16 +124,16 @@ int main(int argc, char** argv) { } //Push our log every 30s: - if (framesSinceLastFlush >= 900) { + if (framesSinceLastFlush >= logFlushTime) { Game::logger->Flush(); framesSinceLastFlush = 0; } else framesSinceLastFlush++; //Every 10 min we ping our sql server to keep it alive hopefully: - if (framesSinceLastSQLPing >= 40000) { + if (framesSinceLastSQLPing >= sqlPingTime) { //Find out the master's IP for absolutely no reason: std::string masterIP; - int masterPort; + uint32_t masterPort; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -136,7 +148,7 @@ int main(int argc, char** argv) { } else framesSinceLastSQLPing++; //Sleep our thread since auth can afford to. - t += std::chrono::milliseconds(mediumFramerate); //Auth can run at a lower "fps" + t += std::chrono::milliseconds(authFrameDelta); //Auth can run at a lower "fps" std::this_thread::sleep_until(t); } @@ -144,13 +156,13 @@ int main(int argc, char** argv) { Database::Destroy("AuthServer"); delete Game::server; delete Game::logger; + delete Game::config; - exit(EXIT_SUCCESS); return EXIT_SUCCESS; } dLogger* SetupLogger() { - std::string logPath = "./logs/AuthServer_" + std::to_string(time(nullptr)) + ".log"; + std::string logPath = (BinaryPathFinder::GetBinaryDir() / ("logs/AuthServer_" + std::to_string(time(nullptr)) + ".log")).string(); bool logToConsole = false; bool logDebugStatements = false; #ifdef _DEBUG @@ -162,13 +174,15 @@ dLogger* SetupLogger() { } void HandlePacket(Packet* packet) { + if (packet->length < 4) return; + if (packet->data[0] == ID_USER_PACKET_ENUM) { - if (packet->data[1] == SERVER) { - if (packet->data[3] == MSG_SERVER_VERSION_CONFIRM) { + if (static_cast(packet->data[1]) == eConnectionType::SERVER) { + if (static_cast(packet->data[3]) == eServerMessageType::VERSION_CONFIRM) { AuthPackets::HandleHandshake(Game::server, packet); } - } else if (packet->data[1] == AUTH) { - if (packet->data[3] == MSG_AUTH_LOGIN_REQUEST) { + } else if (static_cast(packet->data[1]) == eConnectionType::AUTH) { + if (static_cast(packet->data[3]) == eAuthMessageType::LOGIN_REQUEST) { AuthPackets::HandleLoginRequest(Game::server, packet); } } diff --git a/dAuthServer/CMakeLists.txt b/dAuthServer/CMakeLists.txt index 353f2a54..00fa6e7a 100644 --- a/dAuthServer/CMakeLists.txt +++ b/dAuthServer/CMakeLists.txt @@ -1,4 +1,2 @@ -set(DAUTHSERVER_SOURCES "AuthServer.cpp") - -add_executable(AuthServer ${DAUTHSERVER_SOURCES}) +add_executable(AuthServer "AuthServer.cpp") target_link_libraries(AuthServer ${COMMON_LIBRARIES}) diff --git a/dChatFilter/dChatFilter.cpp b/dChatFilter/dChatFilter.cpp index ea38f8cd..92da9556 100644 --- a/dChatFilter/dChatFilter.cpp +++ b/dChatFilter/dChatFilter.cpp @@ -12,6 +12,7 @@ #include "dConfig.h" #include "Database.h" #include "Game.h" +#include "eGameMasterLevel.h" using namespace dChatFilterDCF; @@ -108,8 +109,8 @@ void dChatFilter::ExportWordlistToDCF(const std::string& filepath, bool whiteLis } } -std::vector> dChatFilter::IsSentenceOkay(const std::string& message, int gmLevel, bool whiteList) { - if (gmLevel > GAME_MASTER_LEVEL_FORUM_MODERATOR) return { }; //If anything but a forum mod, return true. +std::vector> dChatFilter::IsSentenceOkay(const std::string& message, eGameMasterLevel gmLevel, bool whiteList) { + if (gmLevel > eGameMasterLevel::FORUM_MODERATOR) return { }; //If anything but a forum mod, return true. if (message.empty()) return { }; if (!whiteList && m_DeniedWords.empty()) return { { 0, message.length() } }; diff --git a/dChatFilter/dChatFilter.h b/dChatFilter/dChatFilter.h index 7e7dd859..d00525ce 100644 --- a/dChatFilter/dChatFilter.h +++ b/dChatFilter/dChatFilter.h @@ -4,6 +4,7 @@ #include "dCommonVars.h" +enum class eGameMasterLevel : uint8_t; namespace dChatFilterDCF { static const uint32_t header = ('D' + ('C' << 8) + ('F' << 16) + ('B' << 24)); static const uint32_t formatVersion = 2; @@ -23,7 +24,7 @@ public: void ReadWordlistPlaintext(const std::string& filepath, bool whiteList); bool ReadWordlistDCF(const std::string& filepath, bool whiteList); void ExportWordlistToDCF(const std::string& filepath, bool whiteList); - std::vector> IsSentenceOkay(const std::string& message, int gmLevel, bool whiteList = true); + std::vector> IsSentenceOkay(const std::string& message, eGameMasterLevel gmLevel, bool whiteList = true); private: bool m_DontGenerateDCF; diff --git a/dChatServer/CMakeLists.txt b/dChatServer/CMakeLists.txt index 948593fb..9a47803d 100644 --- a/dChatServer/CMakeLists.txt +++ b/dChatServer/CMakeLists.txt @@ -1,6 +1,10 @@ -set(DCHATSERVER_SOURCES "ChatPacketHandler.cpp" - "ChatServer.cpp" - "PlayerContainer.cpp") - -add_executable(ChatServer ${DCHATSERVER_SOURCES}) -target_link_libraries(ChatServer ${COMMON_LIBRARIES} dChatFilter) +set(DCHATSERVER_SOURCES + "ChatPacketHandler.cpp" + "PlayerContainer.cpp" +) + +add_executable(ChatServer "ChatServer.cpp") +add_library(dChatServer ${DCHATSERVER_SOURCES}) + +target_link_libraries(dChatServer ${COMMON_LIBRARIES} dChatFilter) +target_link_libraries(ChatServer ${COMMON_LIBRARIES} dChatFilter dChatServer) diff --git a/dChatServer/ChatPacketHandler.cpp b/dChatServer/ChatPacketHandler.cpp index a51716ca..878cc71c 100644 --- a/dChatServer/ChatPacketHandler.cpp +++ b/dChatServer/ChatPacketHandler.cpp @@ -3,24 +3,28 @@ #include "Database.h" #include #include "PacketUtils.h" -#include "dMessageIdentifiers.h" #include "Game.h" #include "dServer.h" #include "GeneralUtils.h" #include "dLogger.h" -#include "AddFriendResponseCode.h" -#include "AddFriendResponseType.h" +#include "eAddFriendResponseCode.h" +#include "eAddFriendResponseType.h" #include "RakString.h" #include "dConfig.h" +#include "eObjectBits.h" +#include "eConnectionType.h" +#include "eChatMessageType.h" +#include "eChatInternalMessageType.h" +#include "eClientMessageType.h" +#include "eGameMessageType.h" extern PlayerContainer playerContainer; void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { //Get from the packet which player we want to do something with: - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = 0; inStream.Read(playerID); - inStream.Read(playerID); auto player = playerContainer.GetPlayerData(playerID); if (!player) return; @@ -33,9 +37,10 @@ void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { "WHEN friend_id = ? THEN player_id " "END AS requested_player, best_friend FROM friends) AS fr " "JOIN charinfo AS ci ON ci.id = fr.requested_player " - "WHERE fr.requested_player IS NOT NULL;")); + "WHERE fr.requested_player IS NOT NULL AND fr.requested_player != ?;")); stmt->setUInt(1, static_cast(playerID)); stmt->setUInt(2, static_cast(playerID)); + stmt->setUInt(3, static_cast(playerID)); std::vector friends; @@ -44,8 +49,8 @@ void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { FriendData fd; fd.isFTP = false; // not a thing in DLU fd.friendID = res->getUInt(1); - GeneralUtils::SetBit(fd.friendID, static_cast(eObjectBits::OBJECT_BIT_PERSISTENT)); - GeneralUtils::SetBit(fd.friendID, static_cast(eObjectBits::OBJECT_BIT_CHARACTER)); + GeneralUtils::SetBit(fd.friendID, eObjectBits::PERSISTENT); + GeneralUtils::SetBit(fd.friendID, eObjectBits::CHARACTER); fd.isBestFriend = res->getInt(2) == 3; //0 = friends, 1 = left_requested, 2 = right_requested, 3 = both_accepted - are now bffs if (fd.isBestFriend) player->countOfBestFriends += 1; @@ -70,11 +75,11 @@ void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { //Now, we need to send the friendlist to the server they came from: CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(playerID); //portion that will get routed: - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_GET_FRIENDS_LIST_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::GET_FRIENDS_LIST_RESPONSE); bitStream.Write(0); bitStream.Write(1); //Length of packet -- just writing one as it doesn't matter, client skips it. bitStream.Write((uint16_t)friends.size()); @@ -93,10 +98,9 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { auto maxNumberOfBestFriendsAsString = Game::config->GetValue("max_number_of_best_friends"); // If this config option doesn't exist, default to 5 which is what live used. auto maxNumberOfBestFriends = maxNumberOfBestFriendsAsString != "" ? std::stoi(maxNumberOfBestFriendsAsString) : 5U; - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID requestorPlayerID; inStream.Read(requestorPlayerID); - inStream.Read(requestorPlayerID); uint32_t spacing{}; inStream.Read(spacing); std::string playerName = ""; @@ -113,6 +117,10 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { inStream.Read(isBestFriendRequest); auto requestor = playerContainer.GetPlayerData(requestorPlayerID); + if (requestor->playerName == playerName) { + SendFriendResponse(requestor, requestor, eAddFriendResponseType::MYTHRAN); + return; + }; std::unique_ptr requestee(playerContainer.GetPlayerData(playerName)); // Check if player is online first @@ -148,7 +156,7 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { requestee.reset(new PlayerData()); requestee->playerName = playerName; - SendFriendResponse(requestor, requestee.get(), result->next() ? AddFriendResponseType::NOTONLINE : AddFriendResponseType::INVALIDCHARACTER); + SendFriendResponse(requestor, requestee.get(), result->next() ? eAddFriendResponseType::NOTONLINE : eAddFriendResponseType::INVALIDCHARACTER); return; } @@ -173,10 +181,10 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { bestFriendStatus = oldBestFriendStatus; // Set the bits - GeneralUtils::SetBit(queryPlayerID, static_cast(eObjectBits::OBJECT_BIT_CHARACTER)); - GeneralUtils::SetBit(queryPlayerID, static_cast(eObjectBits::OBJECT_BIT_PERSISTENT)); - GeneralUtils::SetBit(queryFriendID, static_cast(eObjectBits::OBJECT_BIT_CHARACTER)); - GeneralUtils::SetBit(queryFriendID, static_cast(eObjectBits::OBJECT_BIT_PERSISTENT)); + GeneralUtils::SetBit(queryPlayerID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(queryPlayerID, eObjectBits::PERSISTENT); + GeneralUtils::SetBit(queryFriendID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(queryFriendID, eObjectBits::PERSISTENT); // Since this player can either be the friend of someone else or be friends with someone else // their column in the database determines what bit gets set. When the value hits 3, they @@ -192,10 +200,10 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { if (oldBestFriendStatus != bestFriendStatus) { if (requestee->countOfBestFriends >= maxNumberOfBestFriends || requestor->countOfBestFriends >= maxNumberOfBestFriends) { if (requestee->countOfBestFriends >= maxNumberOfBestFriends) { - SendFriendResponse(requestor, requestee.get(), AddFriendResponseType::THEIRFRIENDLISTFULL, false); + SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::THEIRFRIENDLISTFULL, false); } if (requestor->countOfBestFriends >= maxNumberOfBestFriends) { - SendFriendResponse(requestor, requestee.get(), AddFriendResponseType::YOURFRIENDSLISTFULL, false); + SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::YOURFRIENDSLISTFULL, false); } } else { // Then update the database with this new info. @@ -210,8 +218,8 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { if (bestFriendStatus == 3U) { requestee->countOfBestFriends += 1; requestor->countOfBestFriends += 1; - if (requestee->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestee.get(), requestor, AddFriendResponseType::ACCEPTED, false, true); - if (requestor->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestor, requestee.get(), AddFriendResponseType::ACCEPTED, false, true); + if (requestee->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestee.get(), requestor, eAddFriendResponseType::ACCEPTED, false, true); + if (requestor->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::ACCEPTED, false, true); for (auto& friendData : requestor->friends) { if (friendData.friendID == requestee->playerID) { friendData.isBestFriend = true; @@ -225,7 +233,7 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { } } } else { - if (requestor->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestor, requestee.get(), AddFriendResponseType::WAITINGAPPROVAL, true, true); + if (requestor->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::WAITINGAPPROVAL, true, true); } } else { // Do not send this if we are requesting to be a best friend. @@ -237,12 +245,11 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { } void ChatPacketHandler::HandleFriendResponse(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; inStream.Read(playerID); - inStream.Read(playerID); - AddFriendResponseCode clientResponseCode = static_cast(packet->data[0x14]); + eAddFriendResponseCode clientResponseCode = static_cast(packet->data[0x14]); std::string friendName = PacketUtils::ReadString(0x15, packet, true); //Now to try and find both of these: @@ -250,29 +257,29 @@ void ChatPacketHandler::HandleFriendResponse(Packet* packet) { auto requestee = playerContainer.GetPlayerData(friendName); if (!requestor || !requestee) return; - AddFriendResponseType serverResponseCode{}; + eAddFriendResponseType serverResponseCode{}; uint8_t isAlreadyBestFriends = 0U; // We need to convert this response code to one we can actually send back to the client. switch (clientResponseCode) { - case AddFriendResponseCode::ACCEPTED: - serverResponseCode = AddFriendResponseType::ACCEPTED; + case eAddFriendResponseCode::ACCEPTED: + serverResponseCode = eAddFriendResponseType::ACCEPTED; break; - case AddFriendResponseCode::BUSY: - serverResponseCode = AddFriendResponseType::BUSY; + case eAddFriendResponseCode::BUSY: + serverResponseCode = eAddFriendResponseType::BUSY; break; - case AddFriendResponseCode::CANCELLED: - serverResponseCode = AddFriendResponseType::CANCELLED; + case eAddFriendResponseCode::CANCELLED: + serverResponseCode = eAddFriendResponseType::CANCELLED; break; - case AddFriendResponseCode::REJECTED: - serverResponseCode = AddFriendResponseType::DECLINED; + case eAddFriendResponseCode::REJECTED: + serverResponseCode = eAddFriendResponseType::DECLINED; break; } // Now that we have handled the base cases, we need to check the other cases. - if (serverResponseCode == AddFriendResponseType::ACCEPTED) { + if (serverResponseCode == eAddFriendResponseType::ACCEPTED) { for (auto friendData : requestor->friends) { if (friendData.friendID == requestee->playerID) { - serverResponseCode = AddFriendResponseType::ALREADYFRIEND; + serverResponseCode = eAddFriendResponseType::ALREADYFRIEND; if (friendData.isBestFriend) { isAlreadyBestFriends = 1U; } @@ -281,7 +288,7 @@ void ChatPacketHandler::HandleFriendResponse(Packet* packet) { } // This message is NOT sent for best friends and is handled differently for those requests. - if (serverResponseCode == AddFriendResponseType::ACCEPTED) { + if (serverResponseCode == eAddFriendResponseType::ACCEPTED) { // Add the each player to the others friend list. FriendData requestorData; requestorData.zoneID = requestor->zoneID; @@ -308,15 +315,14 @@ void ChatPacketHandler::HandleFriendResponse(Packet* packet) { statement->execute(); } - if (serverResponseCode != AddFriendResponseType::DECLINED) SendFriendResponse(requestor, requestee, serverResponseCode, isAlreadyBestFriends); - if (serverResponseCode != AddFriendResponseType::ALREADYFRIEND) SendFriendResponse(requestee, requestor, serverResponseCode, isAlreadyBestFriends); + if (serverResponseCode != eAddFriendResponseType::DECLINED) SendFriendResponse(requestor, requestee, serverResponseCode, isAlreadyBestFriends); + if (serverResponseCode != eAddFriendResponseType::ALREADYFRIEND) SendFriendResponse(requestee, requestor, serverResponseCode, isAlreadyBestFriends); } void ChatPacketHandler::HandleRemoveFriend(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; inStream.Read(playerID); - inStream.Read(playerID); std::string friendName = PacketUtils::ReadString(0x14, packet, true); //we'll have to query the db here to find the user, since you can delete them while they're offline. @@ -331,8 +337,8 @@ void ChatPacketHandler::HandleRemoveFriend(Packet* packet) { } // Convert friendID to LWOOBJID - GeneralUtils::SetBit(friendID, static_cast(eObjectBits::OBJECT_BIT_PERSISTENT)); - GeneralUtils::SetBit(friendID, static_cast(eObjectBits::OBJECT_BIT_CHARACTER)); + GeneralUtils::SetBit(friendID, eObjectBits::PERSISTENT); + GeneralUtils::SetBit(friendID, eObjectBits::CHARACTER); std::unique_ptr deletestmt(Database::CreatePreppedStmt("DELETE FROM friends WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1;")); deletestmt->setUInt(1, static_cast(playerID)); @@ -371,10 +377,9 @@ void ChatPacketHandler::HandleRemoveFriend(Packet* packet) { } void ChatPacketHandler::HandleChatMessage(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); auto* sender = playerContainer.GetPlayerData(playerID); @@ -389,7 +394,7 @@ void ChatPacketHandler::HandleChatMessage(Packet* packet) { uint8_t channel = 0; inStream.Read(channel); - std::string message = PacketUtils::ReadString(0x66, packet, true); + std::string message = PacketUtils::ReadString(0x66, packet, true, 512); Game::logger->Log("ChatPacketHandler", "Got a message from (%s) [%d]: %s", senderName.c_str(), channel, message.c_str()); @@ -407,10 +412,10 @@ void ChatPacketHandler::HandleChatMessage(Packet* packet) { const auto otherName = std::string(otherMember->playerName.c_str()); CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(otherMember->playerID); - PacketUtils::WriteHeader(bitStream, CHAT, MSG_CHAT_PRIVATE_CHAT_MESSAGE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT, eChatMessageType::PRIVATE_CHAT_MESSAGE); bitStream.Write(otherMember->playerID); bitStream.Write(8); bitStream.Write(69); @@ -431,7 +436,7 @@ void ChatPacketHandler::HandleChatMessage(Packet* packet) { void ChatPacketHandler::HandlePrivateChatMessage(Packet* packet) { LWOOBJID senderID = PacketUtils::ReadPacketS64(0x08, packet); std::string receiverName = PacketUtils::ReadString(0x66, packet, true); - std::string message = PacketUtils::ReadString(0xAA, packet, true); + std::string message = PacketUtils::ReadString(0xAA, packet, true, 512); //Get the bois: auto goonA = playerContainer.GetPlayerData(senderID); @@ -446,10 +451,10 @@ void ChatPacketHandler::HandlePrivateChatMessage(Packet* packet) { //To the sender: { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(goonA->playerID); - PacketUtils::WriteHeader(bitStream, CHAT, MSG_CHAT_PRIVATE_CHAT_MESSAGE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT, eChatMessageType::PRIVATE_CHAT_MESSAGE); bitStream.Write(goonA->playerID); bitStream.Write(7); bitStream.Write(69); @@ -469,10 +474,10 @@ void ChatPacketHandler::HandlePrivateChatMessage(Packet* packet) { //To the receiver: { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(goonB->playerID); - PacketUtils::WriteHeader(bitStream, CHAT, MSG_CHAT_PRIVATE_CHAT_MESSAGE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT, eChatMessageType::PRIVATE_CHAT_MESSAGE); bitStream.Write(goonA->playerID); bitStream.Write(7); bitStream.Write(69); @@ -491,10 +496,9 @@ void ChatPacketHandler::HandlePrivateChatMessage(Packet* packet) { } void ChatPacketHandler::HandleTeamInvite(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; inStream.Read(playerID); - inStream.Read(playerID); std::string invitedPlayer = PacketUtils::ReadString(0x14, packet, true); auto* player = playerContainer.GetPlayerData(playerID); @@ -532,10 +536,9 @@ void ChatPacketHandler::HandleTeamInvite(Packet* packet) { } void ChatPacketHandler::HandleTeamInviteResponse(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); uint32_t size = 0; inStream.Read(size); char declined = 0; @@ -566,10 +569,9 @@ void ChatPacketHandler::HandleTeamInviteResponse(Packet* packet) { } void ChatPacketHandler::HandleTeamLeave(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); uint32_t size = 0; inStream.Read(size); @@ -583,10 +585,9 @@ void ChatPacketHandler::HandleTeamLeave(Packet* packet) { } void ChatPacketHandler::HandleTeamKick(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); std::string kickedPlayer = PacketUtils::ReadString(0x14, packet, true); @@ -614,10 +615,9 @@ void ChatPacketHandler::HandleTeamKick(Packet* packet) { } void ChatPacketHandler::HandleTeamPromote(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); std::string promotedPlayer = PacketUtils::ReadString(0x14, packet, true); @@ -637,10 +637,9 @@ void ChatPacketHandler::HandleTeamPromote(Packet* packet) { } void ChatPacketHandler::HandleTeamLootOption(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); uint32_t size = 0; inStream.Read(size); @@ -661,10 +660,9 @@ void ChatPacketHandler::HandleTeamLootOption(Packet* packet) { } void ChatPacketHandler::HandleTeamStatusRequest(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID = LWOOBJID_EMPTY; inStream.Read(playerID); - inStream.Read(playerID); auto* team = playerContainer.GetTeam(playerID); auto* data = playerContainer.GetPlayerData(playerID); @@ -711,11 +709,11 @@ void ChatPacketHandler::HandleTeamStatusRequest(Packet* packet) { void ChatPacketHandler::SendTeamInvite(PlayerData* receiver, PlayerData* sender) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_TEAM_INVITE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::TEAM_INVITE); PacketUtils::WritePacketWString(sender->playerName.c_str(), 33, &bitStream); bitStream.Write(sender->playerID); @@ -726,14 +724,14 @@ void ChatPacketHandler::SendTeamInvite(PlayerData* receiver, PlayerData* sender) void ChatPacketHandler::SendTeamInviteConfirm(PlayerData* receiver, bool bLeaderIsFreeTrial, LWOOBJID i64LeaderID, LWOZONEID i64LeaderZoneID, uint8_t ucLootFlag, uint8_t ucNumOfOtherPlayers, uint8_t ucResponseCode, std::u16string wsLeaderName) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: CMSGHEADER; bitStream.Write(receiver->playerID); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_INVITE_CONFIRM); + bitStream.Write(eGameMessageType::TEAM_INVITE_CONFIRM); bitStream.Write(bLeaderIsFreeTrial); bitStream.Write(i64LeaderID); @@ -753,14 +751,14 @@ void ChatPacketHandler::SendTeamInviteConfirm(PlayerData* receiver, bool bLeader void ChatPacketHandler::SendTeamStatus(PlayerData* receiver, LWOOBJID i64LeaderID, LWOZONEID i64LeaderZoneID, uint8_t ucLootFlag, uint8_t ucNumOfOtherPlayers, std::u16string wsLeaderName) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: CMSGHEADER; bitStream.Write(receiver->playerID); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_GET_STATUS_RESPONSE); + bitStream.Write(eGameMessageType::TEAM_GET_STATUS_RESPONSE); bitStream.Write(i64LeaderID); bitStream.Write(i64LeaderZoneID); @@ -778,14 +776,14 @@ void ChatPacketHandler::SendTeamStatus(PlayerData* receiver, LWOOBJID i64LeaderI void ChatPacketHandler::SendTeamSetLeader(PlayerData* receiver, LWOOBJID i64PlayerID) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: CMSGHEADER; bitStream.Write(receiver->playerID); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_SET_LEADER); + bitStream.Write(eGameMessageType::TEAM_SET_LEADER); bitStream.Write(i64PlayerID); @@ -795,14 +793,14 @@ void ChatPacketHandler::SendTeamSetLeader(PlayerData* receiver, LWOOBJID i64Play void ChatPacketHandler::SendTeamAddPlayer(PlayerData* receiver, bool bIsFreeTrial, bool bLocal, bool bNoLootOnDeath, LWOOBJID i64PlayerID, std::u16string wsPlayerName, LWOZONEID zoneID) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: CMSGHEADER; bitStream.Write(receiver->playerID); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_ADD_PLAYER); + bitStream.Write(eGameMessageType::TEAM_ADD_PLAYER); bitStream.Write(bIsFreeTrial); bitStream.Write(bLocal); @@ -824,14 +822,14 @@ void ChatPacketHandler::SendTeamAddPlayer(PlayerData* receiver, bool bIsFreeTria void ChatPacketHandler::SendTeamRemovePlayer(PlayerData* receiver, bool bDisband, bool bIsKicked, bool bIsLeaving, bool bLocal, LWOOBJID i64LeaderID, LWOOBJID i64PlayerID, std::u16string wsPlayerName) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: CMSGHEADER; bitStream.Write(receiver->playerID); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_REMOVE_PLAYER); + bitStream.Write(eGameMessageType::TEAM_REMOVE_PLAYER); bitStream.Write(bDisband); bitStream.Write(bIsKicked); @@ -850,14 +848,14 @@ void ChatPacketHandler::SendTeamRemovePlayer(PlayerData* receiver, bool bDisband void ChatPacketHandler::SendTeamSetOffWorldFlag(PlayerData* receiver, LWOOBJID i64PlayerID, LWOZONEID zoneID) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: CMSGHEADER; bitStream.Write(receiver->playerID); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_SET_OFF_WORLD_FLAG); + bitStream.Write(eGameMessageType::TEAM_SET_OFF_WORLD_FLAG); bitStream.Write(i64PlayerID); if (receiver->zoneID.GetCloneID() == zoneID.GetCloneID()) { @@ -884,11 +882,11 @@ void ChatPacketHandler::SendFriendUpdate(PlayerData* friendData, PlayerData* pla [bool] - is FTP*/ CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(friendData->playerID); //portion that will get routed: - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_UPDATE_FRIEND_NOTIFY); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::UPDATE_FRIEND_NOTIFY); bitStream.Write(notifyType); std::string playerName = playerData->playerName.c_str(); @@ -917,17 +915,17 @@ void ChatPacketHandler::SendFriendRequest(PlayerData* receiver, PlayerData* send //Make sure people aren't requesting people that they're already friends with: for (auto fr : receiver->friends) { if (fr.friendID == sender->playerID) { - SendFriendResponse(sender, receiver, AddFriendResponseType::ALREADYFRIEND, fr.isBestFriend); + SendFriendResponse(sender, receiver, eAddFriendResponseType::ALREADYFRIEND, fr.isBestFriend); return; //we have this player as a friend, yeet this function so it doesn't send another request. } } CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_ADD_FRIEND_REQUEST); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::ADD_FRIEND_REQUEST); PacketUtils::WritePacketWString(sender->playerName.c_str(), 33, &bitStream); bitStream.Write(0); // This is a BFF flag however this is unused in live and does not have an implementation client side. @@ -935,22 +933,22 @@ void ChatPacketHandler::SendFriendRequest(PlayerData* receiver, PlayerData* send SEND_PACKET; } -void ChatPacketHandler::SendFriendResponse(PlayerData* receiver, PlayerData* sender, AddFriendResponseType responseCode, uint8_t isBestFriendsAlready, uint8_t isBestFriendRequest) { +void ChatPacketHandler::SendFriendResponse(PlayerData* receiver, PlayerData* sender, eAddFriendResponseType responseCode, uint8_t isBestFriendsAlready, uint8_t isBestFriendRequest) { if (!receiver || !sender) return; CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); // Portion that will get routed: - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_ADD_FRIEND_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::ADD_FRIEND_RESPONSE); bitStream.Write(responseCode); // For all requests besides accepted, write a flag that says whether or not we are already best friends with the receiver. - bitStream.Write(responseCode != AddFriendResponseType::ACCEPTED ? isBestFriendsAlready : sender->sysAddr != UNASSIGNED_SYSTEM_ADDRESS); + bitStream.Write(responseCode != eAddFriendResponseType::ACCEPTED ? isBestFriendsAlready : sender->sysAddr != UNASSIGNED_SYSTEM_ADDRESS); // Then write the player name PacketUtils::WritePacketWString(sender->playerName.c_str(), 33, &bitStream); // Then if this is an acceptance code, write the following extra info. - if (responseCode == AddFriendResponseType::ACCEPTED) { + if (responseCode == eAddFriendResponseType::ACCEPTED) { bitStream.Write(sender->playerID); bitStream.Write(sender->zoneID); bitStream.Write(isBestFriendRequest); //isBFF @@ -964,11 +962,11 @@ void ChatPacketHandler::SendRemoveFriend(PlayerData* receiver, std::string& pers if (!receiver) return; CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ROUTE_TO_PLAYER); bitStream.Write(receiver->playerID); //portion that will get routed: - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_REMOVE_FRIEND_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::REMOVE_FRIEND_RESPONSE); bitStream.Write(isSuccessful); //isOnline PacketUtils::WritePacketWString(personToRemove, 33, &bitStream); diff --git a/dChatServer/ChatPacketHandler.h b/dChatServer/ChatPacketHandler.h index fffd1ca4..f2d83502 100644 --- a/dChatServer/ChatPacketHandler.h +++ b/dChatServer/ChatPacketHandler.h @@ -4,7 +4,7 @@ #include "BitStream.h" struct PlayerData; -enum class AddFriendResponseType : uint8_t; +enum class eAddFriendResponseType : uint8_t; namespace ChatPacketHandler { void HandleFriendlistRequest(Packet* packet); @@ -35,6 +35,6 @@ namespace ChatPacketHandler { void SendFriendUpdate(PlayerData* friendData, PlayerData* playerData, uint8_t notifyType, uint8_t isBestFriend); void SendFriendRequest(PlayerData* receiver, PlayerData* sender); - void SendFriendResponse(PlayerData* receiver, PlayerData* sender, AddFriendResponseType responseCode, uint8_t isBestFriendsAlready = 0U, uint8_t isBestFriendRequest = 0U); + void SendFriendResponse(PlayerData* receiver, PlayerData* sender, eAddFriendResponseType responseCode, uint8_t isBestFriendsAlready = 0U, uint8_t isBestFriendRequest = 0U); void SendRemoveFriend(PlayerData* receiver, std::string& personToRemove, bool isSuccessful); }; diff --git a/dChatServer/ChatServer.cpp b/dChatServer/ChatServer.cpp index 273921eb..185bccbf 100644 --- a/dChatServer/ChatServer.cpp +++ b/dChatServer/ChatServer.cpp @@ -9,23 +9,33 @@ #include "dLogger.h" #include "Database.h" #include "dConfig.h" -#include "dMessageIdentifiers.h" #include "dChatFilter.h" #include "Diagnostics.h" - +#include "AssetManager.h" +#include "BinaryPathFinder.h" +#include "eConnectionType.h" #include "PlayerContainer.h" #include "ChatPacketHandler.h" +#include "eChatMessageType.h" +#include "eChatInternalMessageType.h" +#include "eWorldMessageType.h" #include "Game.h" -namespace Game { - dLogger* logger; - dServer* server; - dConfig* config; - dChatFilter* chatFilter; -} //RakNet includes: #include "RakNetDefines.h" +#include + +namespace Game { + dLogger* logger = nullptr; + dServer* server = nullptr; + dConfig* config = nullptr; + dChatFilter* chatFilter = nullptr; + AssetManager* assetManager = nullptr; + bool shouldShutdown = false; + std::mt19937 randomEngine; +} + dLogger* SetupLogger(); void HandlePacket(Packet* packet); @@ -33,28 +43,45 @@ void HandlePacket(Packet* packet); PlayerContainer playerContainer; int main(int argc, char** argv) { + constexpr uint32_t chatFramerate = mediumFramerate; + constexpr uint32_t chatFrameDelta = mediumFrameDelta; Diagnostics::SetProcessName("Chat"); Diagnostics::SetProcessFileName(argv[0]); Diagnostics::Initialize(); //Create all the objects we need to run our service: Game::logger = SetupLogger(); - if (!Game::logger) return 0; + if (!Game::logger) return EXIT_FAILURE; + + //Read our config: + Game::config = new dConfig((BinaryPathFinder::GetBinaryDir() / "chatconfig.ini").string()); + Game::logger->SetLogToConsole(Game::config->GetValue("log_to_console") != "0"); + Game::logger->SetLogDebugStatements(Game::config->GetValue("log_debug_statements") == "1"); + Game::logger->Log("ChatServer", "Starting Chat server..."); Game::logger->Log("ChatServer", "Version: %i.%i", PROJECT_VERSION_MAJOR, PROJECT_VERSION_MINOR); Game::logger->Log("ChatServer", "Compiled on: %s", __TIMESTAMP__); - //Read our config: - dConfig config("chatconfig.ini"); - Game::config = &config; - Game::logger->SetLogToConsole(bool(std::stoi(config.GetValue("log_to_console")))); - Game::logger->SetLogDebugStatements(config.GetValue("log_debug_statements") == "1"); + try { + std::string clientPathStr = Game::config->GetValue("client_location"); + if (clientPathStr.empty()) clientPathStr = "./res"; + std::filesystem::path clientPath = std::filesystem::path(clientPathStr); + if (clientPath.is_relative()) { + clientPath = BinaryPathFinder::GetBinaryDir() / clientPath; + } + + Game::assetManager = new AssetManager(clientPath); + } catch (std::runtime_error& ex) { + Game::logger->Log("ChatServer", "Got an error while setting up assets: %s", ex.what()); + + return EXIT_FAILURE; + } //Connect to the MySQL Database - std::string mysql_host = config.GetValue("mysql_host"); - std::string mysql_database = config.GetValue("mysql_database"); - std::string mysql_username = config.GetValue("mysql_username"); - std::string mysql_password = config.GetValue("mysql_password"); + std::string mysql_host = Game::config->GetValue("mysql_host"); + std::string mysql_database = Game::config->GetValue("mysql_database"); + std::string mysql_username = Game::config->GetValue("mysql_username"); + std::string mysql_password = Game::config->GetValue("mysql_password"); try { Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); @@ -63,12 +90,12 @@ int main(int argc, char** argv) { Database::Destroy("ChatServer"); delete Game::server; delete Game::logger; - return 0; + return EXIT_FAILURE; } //Find out the master's IP: std::string masterIP; - int masterPort = 1000; + uint32_t masterPort = 1000; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -80,28 +107,32 @@ int main(int argc, char** argv) { delete stmt; //It's safe to pass 'localhost' here, as the IP is only used as the external IP. - int maxClients = 50; - int ourPort = 1501; - if (config.GetValue("max_clients") != "") maxClients = std::stoi(config.GetValue("max_clients")); - if (config.GetValue("port") != "") ourPort = std::atoi(config.GetValue("port").c_str()); + uint32_t maxClients = 50; + uint32_t ourPort = 1501; + if (Game::config->GetValue("max_clients") != "") maxClients = std::stoi(Game::config->GetValue("max_clients")); + if (Game::config->GetValue("port") != "") ourPort = std::atoi(Game::config->GetValue("port").c_str()); - Game::server = new dServer(config.GetValue("external_ip"), ourPort, 0, maxClients, false, true, Game::logger, masterIP, masterPort, ServerType::Chat); + Game::server = new dServer(Game::config->GetValue("external_ip"), ourPort, 0, maxClients, false, true, Game::logger, masterIP, masterPort, ServerType::Chat, Game::config, &Game::shouldShutdown); - Game::chatFilter = new dChatFilter("./res/chatplus_en_us", bool(std::stoi(config.GetValue("dont_generate_dcf")))); + Game::chatFilter = new dChatFilter(Game::assetManager->GetResPath().string() + "/chatplus_en_us", bool(std::stoi(Game::config->GetValue("dont_generate_dcf")))); + + Game::randomEngine = std::mt19937(time(0)); //Run it until server gets a kill message from Master: auto t = std::chrono::high_resolution_clock::now(); Packet* packet = nullptr; - int framesSinceLastFlush = 0; - int framesSinceMasterDisconnect = 0; - int framesSinceLastSQLPing = 0; + constexpr uint32_t logFlushTime = 30 * chatFramerate; // 30 seconds in frames + constexpr uint32_t sqlPingTime = 10 * 60 * chatFramerate; // 10 minutes in frames + uint32_t framesSinceLastFlush = 0; + uint32_t framesSinceMasterDisconnect = 0; + uint32_t framesSinceLastSQLPing = 0; - while (true) { + while (!Game::shouldShutdown) { //Check if we're still connected to master: if (!Game::server->GetIsConnectedToMaster()) { framesSinceMasterDisconnect++; - if (framesSinceMasterDisconnect >= 30) + if (framesSinceMasterDisconnect >= chatFramerate) break; //Exit our loop, shut down. } else framesSinceMasterDisconnect = 0; @@ -117,16 +148,16 @@ int main(int argc, char** argv) { } //Push our log every 30s: - if (framesSinceLastFlush >= 900) { + if (framesSinceLastFlush >= logFlushTime) { Game::logger->Flush(); framesSinceLastFlush = 0; } else framesSinceLastFlush++; //Every 10 min we ping our sql server to keep it alive hopefully: - if (framesSinceLastSQLPing >= 40000) { + if (framesSinceLastSQLPing >= sqlPingTime) { //Find out the master's IP for absolutely no reason: std::string masterIP; - int masterPort; + uint32_t masterPort; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -141,7 +172,7 @@ int main(int argc, char** argv) { } else framesSinceLastSQLPing++; //Sleep our thread since auth can afford to. - t += std::chrono::milliseconds(mediumFramerate); //Chat can run at a lower "fps" + t += std::chrono::milliseconds(chatFrameDelta); //Chat can run at a lower "fps" std::this_thread::sleep_until(t); } @@ -149,13 +180,13 @@ int main(int argc, char** argv) { Database::Destroy("ChatServer"); delete Game::server; delete Game::logger; + delete Game::config; - exit(EXIT_SUCCESS); return EXIT_SUCCESS; } dLogger* SetupLogger() { - std::string logPath = "./logs/ChatServer_" + std::to_string(time(nullptr)) + ".log"; + std::string logPath = (BinaryPathFinder::GetBinaryDir() / ("logs/ChatServer_" + std::to_string(time(nullptr)) + ".log")).string(); bool logToConsole = false; bool logDebugStatements = false; #ifdef _DEBUG @@ -175,25 +206,27 @@ void HandlePacket(Packet* packet) { Game::logger->Log("ChatServer", "A server is connecting, awaiting user list."); } - if (packet->data[1] == CHAT_INTERNAL) { - switch (packet->data[3]) { - case MSG_CHAT_INTERNAL_PLAYER_ADDED_NOTIFICATION: + if (packet->length < 4) return; // Nothing left to process. Need 4 bytes to continue. + + if (static_cast(packet->data[1]) == eConnectionType::CHAT_INTERNAL) { + switch (static_cast(packet->data[3])) { + case eChatInternalMessageType::PLAYER_ADDED_NOTIFICATION: playerContainer.InsertPlayer(packet); break; - case MSG_CHAT_INTERNAL_PLAYER_REMOVED_NOTIFICATION: + case eChatInternalMessageType::PLAYER_REMOVED_NOTIFICATION: playerContainer.RemovePlayer(packet); break; - case MSG_CHAT_INTERNAL_MUTE_UPDATE: + case eChatInternalMessageType::MUTE_UPDATE: playerContainer.MuteUpdate(packet); break; - case MSG_CHAT_INTERNAL_CREATE_TEAM: + case eChatInternalMessageType::CREATE_TEAM: playerContainer.CreateTeamServer(packet); break; - case MSG_CHAT_INTERNAL_ANNOUNCEMENT: { + case eChatInternalMessageType::ANNOUNCEMENT: { //we just forward this packet to every connected server CINSTREAM; Game::server->Send(&inStream, packet->systemAddress, true); //send to everyone except origin @@ -205,67 +238,67 @@ void HandlePacket(Packet* packet) { } } - if (packet->data[1] == CHAT) { - switch (packet->data[3]) { - case MSG_CHAT_GET_FRIENDS_LIST: + if (static_cast(packet->data[1]) == eConnectionType::CHAT) { + switch (static_cast(packet->data[3])) { + case eChatMessageType::GET_FRIENDS_LIST: ChatPacketHandler::HandleFriendlistRequest(packet); break; - case MSG_CHAT_GET_IGNORE_LIST: + case eChatMessageType::GET_IGNORE_LIST: Game::logger->Log("ChatServer", "Asked for ignore list, but is unimplemented right now."); break; - case MSG_CHAT_TEAM_GET_STATUS: + case eChatMessageType::TEAM_GET_STATUS: ChatPacketHandler::HandleTeamStatusRequest(packet); break; - case MSG_CHAT_ADD_FRIEND_REQUEST: + case eChatMessageType::ADD_FRIEND_REQUEST: //this involves someone sending the initial request, the response is below, response as in from the other player. //We basically just check to see if this player is online or not and route the packet. ChatPacketHandler::HandleFriendRequest(packet); break; - case MSG_CHAT_ADD_FRIEND_RESPONSE: + case eChatMessageType::ADD_FRIEND_RESPONSE: //This isn't the response a server sent, rather it is a player's response to a received request. //Here, we'll actually have to add them to eachother's friend lists depending on the response code. ChatPacketHandler::HandleFriendResponse(packet); break; - case MSG_CHAT_REMOVE_FRIEND: + case eChatMessageType::REMOVE_FRIEND: ChatPacketHandler::HandleRemoveFriend(packet); break; - case MSG_CHAT_GENERAL_CHAT_MESSAGE: + case eChatMessageType::GENERAL_CHAT_MESSAGE: ChatPacketHandler::HandleChatMessage(packet); break; - case MSG_CHAT_PRIVATE_CHAT_MESSAGE: + case eChatMessageType::PRIVATE_CHAT_MESSAGE: //This message is supposed to be echo'd to both the sender and the receiver //BUT: they have to have different responseCodes, so we'll do some of the ol hacky wacky to fix that right up. ChatPacketHandler::HandlePrivateChatMessage(packet); break; - case MSG_CHAT_TEAM_INVITE: + case eChatMessageType::TEAM_INVITE: ChatPacketHandler::HandleTeamInvite(packet); break; - case MSG_CHAT_TEAM_INVITE_RESPONSE: + case eChatMessageType::TEAM_INVITE_RESPONSE: ChatPacketHandler::HandleTeamInviteResponse(packet); break; - case MSG_CHAT_TEAM_LEAVE: + case eChatMessageType::TEAM_LEAVE: ChatPacketHandler::HandleTeamLeave(packet); break; - case MSG_CHAT_TEAM_SET_LEADER: + case eChatMessageType::TEAM_SET_LEADER: ChatPacketHandler::HandleTeamPromote(packet); break; - case MSG_CHAT_TEAM_KICK: + case eChatMessageType::TEAM_KICK: ChatPacketHandler::HandleTeamKick(packet); break; - case MSG_CHAT_TEAM_SET_LOOT: + case eChatMessageType::TEAM_SET_LOOT: ChatPacketHandler::HandleTeamLootOption(packet); break; @@ -274,9 +307,9 @@ void HandlePacket(Packet* packet) { } } - if (packet->data[1] == WORLD) { - switch (packet->data[3]) { - case MSG_WORLD_CLIENT_ROUTE_PACKET: { + if (static_cast(packet->data[1]) == eConnectionType::WORLD) { + switch (static_cast(packet->data[3])) { + case eWorldMessageType::ROUTE_PACKET: { Game::logger->Log("ChatServer", "Routing packet from world"); break; } diff --git a/dChatServer/PlayerContainer.cpp b/dChatServer/PlayerContainer.cpp index 6bf3ccd1..689ffd77 100644 --- a/dChatServer/PlayerContainer.cpp +++ b/dChatServer/PlayerContainer.cpp @@ -6,9 +6,10 @@ #include "dLogger.h" #include "ChatPacketHandler.h" #include "GeneralUtils.h" -#include "dMessageIdentifiers.h" #include "PacketUtils.h" #include "Database.h" +#include "eConnectionType.h" +#include "eChatInternalMessageType.h" PlayerContainer::PlayerContainer() { } @@ -18,9 +19,8 @@ PlayerContainer::~PlayerContainer() { } void PlayerContainer::InsertPlayer(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; PlayerData* data = new PlayerData(); - inStream.SetReadOffset(inStream.GetReadOffset() + 64); inStream.Read(data->playerID); uint32_t len; @@ -51,9 +51,8 @@ void PlayerContainer::InsertPlayer(Packet* packet) { } void PlayerContainer::RemovePlayer(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; - inStream.Read(playerID); //skip header inStream.Read(playerID); //Before they get kicked, we need to also send a message to their friends saying that they disconnected. @@ -96,9 +95,8 @@ void PlayerContainer::RemovePlayer(Packet* packet) { } void PlayerContainer::MuteUpdate(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; - inStream.Read(playerID); //skip header inStream.Read(playerID); time_t expire = 0; inStream.Read(expire); @@ -117,9 +115,8 @@ void PlayerContainer::MuteUpdate(Packet* packet) { } void PlayerContainer::CreateTeamServer(Packet* packet) { - CINSTREAM; + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; - inStream.Read(playerID); //skip header inStream.Read(playerID); size_t membersSize = 0; inStream.Read(membersSize); @@ -149,7 +146,7 @@ void PlayerContainer::CreateTeamServer(Packet* packet) { void PlayerContainer::BroadcastMuteUpdate(LWOOBJID player, time_t time) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_MUTE_UPDATE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::MUTE_UPDATE); bitStream.Write(player); bitStream.Write(time); @@ -348,7 +345,7 @@ void PlayerContainer::TeamStatusUpdate(TeamData* team) { void PlayerContainer::UpdateTeamsOnWorld(TeamData* team, bool deleteTeam) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_TEAM_UPDATE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::TEAM_UPDATE); bitStream.Write(team->teamID); bitStream.Write(deleteTeam); diff --git a/dCommon/AMFDeserialize.cpp b/dCommon/AMFDeserialize.cpp index df4a7cb6..648d1ed1 100644 --- a/dCommon/AMFDeserialize.cpp +++ b/dCommon/AMFDeserialize.cpp @@ -1,77 +1,79 @@ #include "AMFDeserialize.h" -#include "AMFFormat.h" +#include + +#include "Amf3.h" /** * AMF3 Reference document https://rtmp.veriskope.com/pdf/amf3-file-format-spec.pdf * AMF3 Deserializer written by EmosewaMC */ -AMFValue* AMFDeserialize::Read(RakNet::BitStream* inStream) { +AMFBaseValue* AMFDeserialize::Read(RakNet::BitStream* inStream) { if (!inStream) return nullptr; - AMFValue* returnValue = nullptr; + AMFBaseValue* returnValue = nullptr; // Read in the value type from the bitStream - int8_t marker; + eAmf marker; inStream->Read(marker); // Based on the typing, create the value associated with that and return the base value class switch (marker) { - case AMFValueType::AMFUndefined: { - returnValue = new AMFUndefinedValue(); + case eAmf::Undefined: { + returnValue = new AMFBaseValue(); break; } - case AMFValueType::AMFNull: { + case eAmf::Null: { returnValue = new AMFNullValue(); break; } - case AMFValueType::AMFFalse: { - returnValue = new AMFFalseValue(); + case eAmf::False: { + returnValue = new AMFBoolValue(false); break; } - case AMFValueType::AMFTrue: { - returnValue = new AMFTrueValue(); + case eAmf::True: { + returnValue = new AMFBoolValue(true); break; } - case AMFValueType::AMFInteger: { + case eAmf::Integer: { returnValue = ReadAmfInteger(inStream); break; } - case AMFValueType::AMFDouble: { + case eAmf::Double: { returnValue = ReadAmfDouble(inStream); break; } - case AMFValueType::AMFString: { + case eAmf::String: { returnValue = ReadAmfString(inStream); break; } - case AMFValueType::AMFArray: { + case eAmf::Array: { returnValue = ReadAmfArray(inStream); break; } - // TODO We do not need these values, but if someone wants to implement them - // then please do so and add the corresponding unit tests. - case AMFValueType::AMFXMLDoc: - case AMFValueType::AMFDate: - case AMFValueType::AMFObject: - case AMFValueType::AMFXML: - case AMFValueType::AMFByteArray: - case AMFValueType::AMFVectorInt: - case AMFValueType::AMFVectorUInt: - case AMFValueType::AMFVectorDouble: - case AMFValueType::AMFVectorObject: - case AMFValueType::AMFDictionary: { - throw static_cast(marker); + // These values are unimplemented in the live client and will remain unimplemented + // unless someone modifies the client to allow serializing of these values. + case eAmf::XMLDoc: + case eAmf::Date: + case eAmf::Object: + case eAmf::XML: + case eAmf::ByteArray: + case eAmf::VectorInt: + case eAmf::VectorUInt: + case eAmf::VectorDouble: + case eAmf::VectorObject: + case eAmf::Dictionary: { + throw marker; break; } default: - throw static_cast(marker); + throw std::invalid_argument("Invalid AMF3 marker" + std::to_string(static_cast(marker))); break; } return returnValue; @@ -99,7 +101,7 @@ uint32_t AMFDeserialize::ReadU29(RakNet::BitStream* inStream) { return actualNumber; } -std::string AMFDeserialize::ReadString(RakNet::BitStream* inStream) { +const std::string AMFDeserialize::ReadString(RakNet::BitStream* inStream) { auto length = ReadU29(inStream); // Check if this is a reference bool isReference = length % 2 == 1; @@ -113,48 +115,39 @@ std::string AMFDeserialize::ReadString(RakNet::BitStream* inStream) { return value; } else { // Length is a reference to a previous index - use that as the read in value - return accessedElements[length]; + return accessedElements.at(length); } } -AMFValue* AMFDeserialize::ReadAmfDouble(RakNet::BitStream* inStream) { - auto doubleValue = new AMFDoubleValue(); +AMFBaseValue* AMFDeserialize::ReadAmfDouble(RakNet::BitStream* inStream) { double value; inStream->Read(value); - doubleValue->SetDoubleValue(value); - return doubleValue; + return new AMFDoubleValue(value); } -AMFValue* AMFDeserialize::ReadAmfArray(RakNet::BitStream* inStream) { +AMFBaseValue* AMFDeserialize::ReadAmfArray(RakNet::BitStream* inStream) { auto arrayValue = new AMFArrayValue(); // Read size of dense array auto sizeOfDenseArray = (ReadU29(inStream) >> 1); - - // Then read Key'd portion + // Then read associative portion while (true) { auto key = ReadString(inStream); - // No more values when we encounter an empty string + // No more associative values when we encounter an empty string key if (key.size() == 0) break; - arrayValue->InsertValue(key, Read(inStream)); + arrayValue->Insert(key, Read(inStream)); } - // Finally read dense portion for (uint32_t i = 0; i < sizeOfDenseArray; i++) { - arrayValue->PushBackValue(Read(inStream)); + arrayValue->Insert(i, Read(inStream)); } - return arrayValue; } -AMFValue* AMFDeserialize::ReadAmfString(RakNet::BitStream* inStream) { - auto stringValue = new AMFStringValue(); - stringValue->SetStringValue(ReadString(inStream)); - return stringValue; +AMFBaseValue* AMFDeserialize::ReadAmfString(RakNet::BitStream* inStream) { + return new AMFStringValue(ReadString(inStream)); } -AMFValue* AMFDeserialize::ReadAmfInteger(RakNet::BitStream* inStream) { - auto integerValue = new AMFIntegerValue(); - integerValue->SetIntegerValue(ReadU29(inStream)); - return integerValue; +AMFBaseValue* AMFDeserialize::ReadAmfInteger(RakNet::BitStream* inStream) { + return new AMFIntValue(ReadU29(inStream)); } diff --git a/dCommon/AMFDeserialize.h b/dCommon/AMFDeserialize.h index a49cdcd2..5e2729eb 100644 --- a/dCommon/AMFDeserialize.h +++ b/dCommon/AMFDeserialize.h @@ -5,7 +5,8 @@ #include #include -class AMFValue; +class AMFBaseValue; + class AMFDeserialize { public: /** @@ -14,7 +15,7 @@ public: * @param inStream inStream to read value from. * @return Returns an AMFValue with all the information from the bitStream in it. */ - AMFValue* Read(RakNet::BitStream* inStream); + AMFBaseValue* Read(RakNet::BitStream* inStream); private: /** * @brief Private method to read a U29 integer from a bitstream @@ -30,7 +31,7 @@ private: * @param inStream bitStream to read data from * @return The read string */ - std::string ReadString(RakNet::BitStream* inStream); + const std::string ReadString(RakNet::BitStream* inStream); /** * @brief Read an AMFDouble value from a bitStream @@ -38,7 +39,7 @@ private: * @param inStream bitStream to read data from * @return Double value represented as an AMFValue */ - AMFValue* ReadAmfDouble(RakNet::BitStream* inStream); + AMFBaseValue* ReadAmfDouble(RakNet::BitStream* inStream); /** * @brief Read an AMFArray from a bitStream @@ -46,7 +47,7 @@ private: * @param inStream bitStream to read data from * @return Array value represented as an AMFValue */ - AMFValue* ReadAmfArray(RakNet::BitStream* inStream); + AMFBaseValue* ReadAmfArray(RakNet::BitStream* inStream); /** * @brief Read an AMFString from a bitStream @@ -54,7 +55,7 @@ private: * @param inStream bitStream to read data from * @return String value represented as an AMFValue */ - AMFValue* ReadAmfString(RakNet::BitStream* inStream); + AMFBaseValue* ReadAmfString(RakNet::BitStream* inStream); /** * @brief Read an AMFInteger from a bitStream @@ -62,7 +63,7 @@ private: * @param inStream bitStream to read data from * @return Integer value represented as an AMFValue */ - AMFValue* ReadAmfInteger(RakNet::BitStream* inStream); + AMFBaseValue* ReadAmfInteger(RakNet::BitStream* inStream); /** * List of strings read so far saved to be read by reference. diff --git a/dCommon/AMFFormat.cpp b/dCommon/AMFFormat.cpp deleted file mode 100644 index 822f994d..00000000 --- a/dCommon/AMFFormat.cpp +++ /dev/null @@ -1,171 +0,0 @@ -#include "AMFFormat.h" - -// AMFInteger -void AMFIntegerValue::SetIntegerValue(uint32_t value) { - this->value = value; -} - -uint32_t AMFIntegerValue::GetIntegerValue() { - return this->value; -} - -// AMFDouble -void AMFDoubleValue::SetDoubleValue(double value) { - this->value = value; -} - -double AMFDoubleValue::GetDoubleValue() { - return this->value; -} - -// AMFString -void AMFStringValue::SetStringValue(const std::string& value) { - this->value = value; -} - -std::string AMFStringValue::GetStringValue() { - return this->value; -} - -// AMFXMLDoc -void AMFXMLDocValue::SetXMLDocValue(const std::string& value) { - this->xmlData = value; -} - -std::string AMFXMLDocValue::GetXMLDocValue() { - return this->xmlData; -} - -// AMFDate -void AMFDateValue::SetDateValue(uint64_t value) { - this->millisecondTimestamp = value; -} - -uint64_t AMFDateValue::GetDateValue() { - return this->millisecondTimestamp; -} - -// AMFArray Insert Value -void AMFArrayValue::InsertValue(const std::string& key, AMFValue* value) { - this->associative.insert(std::make_pair(key, value)); -} - -// AMFArray Remove Value -void AMFArrayValue::RemoveValue(const std::string& key) { - _AMFArrayMap_::iterator it = this->associative.find(key); - if (it != this->associative.end()) { - this->associative.erase(it); - } -} - -// AMFArray Find Value -AMFValue* AMFArrayValue::FindValue(const std::string& key) { - _AMFArrayMap_::iterator it = this->associative.find(key); - if (it != this->associative.end()) { - return it->second; - } - - return nullptr; -} - -// AMFArray Get Associative Iterator Begin -_AMFArrayMap_::iterator AMFArrayValue::GetAssociativeIteratorValueBegin() { - return this->associative.begin(); -} - -// AMFArray Get Associative Iterator End -_AMFArrayMap_::iterator AMFArrayValue::GetAssociativeIteratorValueEnd() { - return this->associative.end(); -} - -// AMFArray Push Back Value -void AMFArrayValue::PushBackValue(AMFValue* value) { - this->dense.push_back(value); -} - -// AMFArray Pop Back Value -void AMFArrayValue::PopBackValue() { - this->dense.pop_back(); -} - -// AMFArray Get Dense List Size -uint32_t AMFArrayValue::GetDenseValueSize() { - return (uint32_t)this->dense.size(); -} - -// AMFArray Get value at index in Dense List -AMFValue* AMFArrayValue::GetValueAt(uint32_t index) { - return this->dense.at(index); -} - -// AMFArray Get Dense Iterator Begin -_AMFArrayList_::iterator AMFArrayValue::GetDenseIteratorBegin() { - return this->dense.begin(); -} - -// AMFArray Get Dense Iterator End -_AMFArrayList_::iterator AMFArrayValue::GetDenseIteratorEnd() { - return this->dense.end(); -} - -AMFArrayValue::~AMFArrayValue() { - for (auto valueToDelete : GetDenseArray()) { - if (valueToDelete) delete valueToDelete; - } - for (auto valueToDelete : GetAssociativeMap()) { - if (valueToDelete.second) delete valueToDelete.second; - } -} - -// AMFObject Constructor -AMFObjectValue::AMFObjectValue(std::vector> traits) { - this->traits.reserve(traits.size()); - std::vector>::iterator it = traits.begin(); - while (it != traits.end()) { - this->traits.insert(std::make_pair(it->first, std::make_pair(it->second, new AMFNullValue()))); - it++; - } -} - -// AMFObject Set Value -void AMFObjectValue::SetTraitValue(const std::string& trait, AMFValue* value) { - if (value) { - _AMFObjectTraits_::iterator it = this->traits.find(trait); - if (it != this->traits.end()) { - if (it->second.first == value->GetValueType()) { - it->second.second = value; - } - } - } -} - -// AMFObject Get Value -AMFValue* AMFObjectValue::GetTraitValue(const std::string& trait) { - _AMFObjectTraits_::iterator it = this->traits.find(trait); - if (it != this->traits.end()) { - return it->second.second; - } - - return nullptr; -} - -// AMFObject Get Trait Iterator Begin -_AMFObjectTraits_::iterator AMFObjectValue::GetTraitsIteratorBegin() { - return this->traits.begin(); -} - -// AMFObject Get Trait Iterator End -_AMFObjectTraits_::iterator AMFObjectValue::GetTraitsIteratorEnd() { - return this->traits.end(); -} - -// AMFObject Get Trait Size -uint32_t AMFObjectValue::GetTraitArrayCount() { - return (uint32_t)this->traits.size(); -} - -AMFObjectValue::~AMFObjectValue() { - for (auto valueToDelete = GetTraitsIteratorBegin(); valueToDelete != GetTraitsIteratorEnd(); valueToDelete++) { - if (valueToDelete->second.second) delete valueToDelete->second.second; - } -} diff --git a/dCommon/AMFFormat.h b/dCommon/AMFFormat.h deleted file mode 100644 index eaa342cd..00000000 --- a/dCommon/AMFFormat.h +++ /dev/null @@ -1,384 +0,0 @@ -#pragma once - -// Custom Classes -#include "dCommonVars.h" - -// C++ -#include -#include - -/*! - \file AMFFormat.hpp - \brief A class for managing AMF values - */ - -class AMFValue; // Forward declaration - -// Definitions -#define _AMFArrayMap_ std::unordered_map -#define _AMFArrayList_ std::vector - -#define _AMFObjectTraits_ std::unordered_map> -#define _AMFObjectDynamicTraits_ std::unordered_map - -//! An enum for each AMF value type -enum AMFValueType : unsigned char { - AMFUndefined = 0x00, //!< An undefined AMF Value - AMFNull = 0x01, //!< A null AMF value - AMFFalse = 0x02, //!< A false AMF value - AMFTrue = 0x03, //!< A true AMF value - AMFInteger = 0x04, //!< An integer AMF value - AMFDouble = 0x05, //!< A double AMF value - AMFString = 0x06, //!< A string AMF value - AMFXMLDoc = 0x07, //!< An XML Doc AMF value - AMFDate = 0x08, //!< A date AMF value - AMFArray = 0x09, //!< An array AMF value - AMFObject = 0x0A, //!< An object AMF value - AMFXML = 0x0B, //!< An XML AMF value - AMFByteArray = 0x0C, //!< A byte array AMF value - AMFVectorInt = 0x0D, //!< An integer vector AMF value - AMFVectorUInt = 0x0E, //!< An unsigned integer AMF value - AMFVectorDouble = 0x0F, //!< A double vector AMF value - AMFVectorObject = 0x10, //!< An object vector AMF value - AMFDictionary = 0x11 //!< A dictionary AMF value -}; - -//! An enum for the object value types -enum AMFObjectValueType : unsigned char { - AMFObjectAnonymous = 0x01, - AMFObjectTyped = 0x02, - AMFObjectDynamic = 0x03, - AMFObjectExternalizable = 0x04 -}; - -//! The base AMF value class -class AMFValue { -public: - //! Returns the AMF value type - /*! - \return The AMF value type - */ - virtual AMFValueType GetValueType() = 0; - virtual ~AMFValue() {}; -}; - -//! A typedef for a pointer to an AMF value -typedef AMFValue* NDGFxValue; - - -// The various AMF value types - -//! The undefined value AMF type -class AMFUndefinedValue : public AMFValue { -private: - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFUndefined; } -}; - -//! The null value AMF type -class AMFNullValue : public AMFValue { -private: - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFNull; } -}; - -//! The false value AMF type -class AMFFalseValue : public AMFValue { -private: - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFFalse; } -}; - -//! The true value AMF type -class AMFTrueValue : public AMFValue { -private: - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFTrue; } -}; - -//! The integer value AMF type -class AMFIntegerValue : public AMFValue { -private: - uint32_t value; //!< The value of the AMF type - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFInteger; } - -public: - //! Sets the integer value - /*! - \param value The value to set - */ - void SetIntegerValue(uint32_t value); - - //! Gets the integer value - /*! - \return The integer value - */ - uint32_t GetIntegerValue(); -}; - -//! The double value AMF type -class AMFDoubleValue : public AMFValue { -private: - double value; //!< The value of the AMF type - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFDouble; } - -public: - //! Sets the double value - /*! - \param value The value to set to - */ - void SetDoubleValue(double value); - - //! Gets the double value - /*! - \return The double value - */ - double GetDoubleValue(); -}; - -//! The string value AMF type -class AMFStringValue : public AMFValue { -private: - std::string value; //!< The value of the AMF type - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFString; } - -public: - //! Sets the string value - /*! - \param value The string value to set to - */ - void SetStringValue(const std::string& value); - - //! Gets the string value - /*! - \return The string value - */ - std::string GetStringValue(); -}; - -//! The XML doc value AMF type -class AMFXMLDocValue : public AMFValue { -private: - std::string xmlData; //!< The value of the AMF type - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFXMLDoc; } - -public: - //! Sets the XML Doc value - /*! - \param value The value to set to - */ - void SetXMLDocValue(const std::string& value); - - //! Gets the XML Doc value - /*! - \return The XML Doc value - */ - std::string GetXMLDocValue(); -}; - -//! The date value AMF type -class AMFDateValue : public AMFValue { -private: - uint64_t millisecondTimestamp; //!< The time in milliseconds since the ephoch - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFDate; } - -public: - //! Sets the date time - /*! - \param value The value to set to - */ - void SetDateValue(uint64_t value); - - //! Gets the date value - /*! - \return The date value in milliseconds since the epoch - */ - uint64_t GetDateValue(); -}; - -//! The array value AMF type -// This object will manage it's own memory map and list. Do not delete its values. -class AMFArrayValue : public AMFValue { -private: - _AMFArrayMap_ associative; //!< The array map (associative part) - _AMFArrayList_ dense; //!< The array list (dense part) - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFArray; } - -public: - ~AMFArrayValue() override; - //! Inserts an item into the array map for a specific key - /*! - \param key The key to set - \param value The value to add - */ - void InsertValue(const std::string& key, AMFValue* value); - - //! Removes an item for a specific key - /*! - \param key The key to remove - */ - void RemoveValue(const std::string& key); - - //! Finds an AMF value - /*! - \return The AMF value if found, nullptr otherwise - */ - AMFValue* FindValue(const std::string& key); - - //! Returns where the associative iterator begins - /*! - \return Where the array map iterator begins - */ - _AMFArrayMap_::iterator GetAssociativeIteratorValueBegin(); - - //! Returns where the associative iterator ends - /*! - \return Where the array map iterator ends - */ - _AMFArrayMap_::iterator GetAssociativeIteratorValueEnd(); - - //! Pushes back a value into the array list - /*! - \param value The value to push back - */ - void PushBackValue(AMFValue* value); - - //! Pops back the last value in the array list - void PopBackValue(); - - //! Gets the count of the dense list - /*! - \return The dense list size - */ - uint32_t GetDenseValueSize(); - - //! Gets a specific value from the list for the specified index - /*! - \param index The index to get - */ - AMFValue* GetValueAt(uint32_t index); - - //! Returns where the dense iterator begins - /*! - \return Where the iterator begins - */ - _AMFArrayList_::iterator GetDenseIteratorBegin(); - - //! Returns where the dense iterator ends - /*! - \return Where the iterator ends - */ - _AMFArrayList_::iterator GetDenseIteratorEnd(); - - //! Returns the associative map - /*! - \return The associative map - */ - _AMFArrayMap_ GetAssociativeMap() { return this->associative; }; - - //! Returns the dense array - /*! - \return The dense array - */ - _AMFArrayList_ GetDenseArray() { return this->dense; }; -}; - -//! The anonymous object value AMF type -class AMFObjectValue : public AMFValue { -private: - _AMFObjectTraits_ traits; //!< The object traits - - //! Returns the AMF value type - /*! - \return The AMF value type - */ - AMFValueType GetValueType() { return AMFObject; } - ~AMFObjectValue() override; - -public: - //! Constructor - /*! - \param traits The traits to set - */ - AMFObjectValue(std::vector> traits); - - //! Gets the object value type - /*! - \return The object value type - */ - virtual AMFObjectValueType GetObjectValueType() { return AMFObjectAnonymous; } - - //! Sets the value of a trait - /*! - \param trait The trait to set the value for - \param value The AMF value to set - */ - void SetTraitValue(const std::string& trait, AMFValue* value); - - //! Gets a trait value - /*! - \param trait The trait to get the value for - \return The trait value - */ - AMFValue* GetTraitValue(const std::string& trait); - - //! Gets the beginning of the object traits iterator - /*! - \return The AMF trait array iterator begin - */ - _AMFObjectTraits_::iterator GetTraitsIteratorBegin(); - - //! Gets the end of the object traits iterator - /*! - \return The AMF trait array iterator begin - */ - _AMFObjectTraits_::iterator GetTraitsIteratorEnd(); - - //! Gets the amount of traits - /*! - \return The amount of traits - */ - uint32_t GetTraitArrayCount(); -}; diff --git a/dCommon/AMFFormat_BitStream.cpp b/dCommon/AMFFormat_BitStream.cpp deleted file mode 100644 index b603ba31..00000000 --- a/dCommon/AMFFormat_BitStream.cpp +++ /dev/null @@ -1,250 +0,0 @@ -#include "AMFFormat_BitStream.h" - -// Writes an AMFValue pointer to a RakNet::BitStream -template<> -void RakNet::BitStream::Write(AMFValue* value) { - if (value != nullptr) { - AMFValueType type = value->GetValueType(); - - switch (type) { - case AMFUndefined: { - AMFUndefinedValue* v = (AMFUndefinedValue*)value; - this->Write(*v); - break; - } - - case AMFNull: { - AMFNullValue* v = (AMFNullValue*)value; - this->Write(*v); - break; - } - - case AMFFalse: { - AMFFalseValue* v = (AMFFalseValue*)value; - this->Write(*v); - break; - } - - case AMFTrue: { - AMFTrueValue* v = (AMFTrueValue*)value; - this->Write(*v); - break; - } - - case AMFInteger: { - AMFIntegerValue* v = (AMFIntegerValue*)value; - this->Write(*v); - break; - } - - case AMFDouble: { - AMFDoubleValue* v = (AMFDoubleValue*)value; - this->Write(*v); - break; - } - - case AMFString: { - AMFStringValue* v = (AMFStringValue*)value; - this->Write(*v); - break; - } - - case AMFXMLDoc: { - AMFXMLDocValue* v = (AMFXMLDocValue*)value; - this->Write(*v); - break; - } - - case AMFDate: { - AMFDateValue* v = (AMFDateValue*)value; - this->Write(*v); - break; - } - - case AMFArray: { - this->Write((AMFArrayValue*)value); - break; - } - } - } -} - -/** - * A private function to write an value to a RakNet::BitStream - * RakNet writes in the correct byte order - do not reverse this. - */ -void WriteUInt29(RakNet::BitStream* bs, uint32_t v) { - unsigned char b4 = (unsigned char)v; - if (v < 0x00200000) { - b4 = b4 & 0x7F; - if (v > 0x7F) { - unsigned char b3; - v = v >> 7; - b3 = ((unsigned char)(v)) | 0x80; - if (v > 0x7F) { - unsigned char b2; - v = v >> 7; - b2 = ((unsigned char)(v)) | 0x80; - bs->Write(b2); - } - - bs->Write(b3); - } - } else { - unsigned char b1; - unsigned char b2; - unsigned char b3; - - v = v >> 8; - b3 = ((unsigned char)(v)) | 0x80; - v = v >> 7; - b2 = ((unsigned char)(v)) | 0x80; - v = v >> 7; - b1 = ((unsigned char)(v)) | 0x80; - - bs->Write(b1); - bs->Write(b2); - bs->Write(b3); - } - - bs->Write(b4); -} - -/** - * Writes a flag number to a RakNet::BitStream - * RakNet writes in the correct byte order - do not reverse this. - */ -void WriteFlagNumber(RakNet::BitStream* bs, uint32_t v) { - v = (v << 1) | 0x01; - WriteUInt29(bs, v); -} - -/** - * Writes an AMFString to a RakNet::BitStream - * - * RakNet writes in the correct byte order - do not reverse this. - */ -void WriteAMFString(RakNet::BitStream* bs, const std::string& str) { - WriteFlagNumber(bs, (uint32_t)str.size()); - bs->Write(str.c_str(), (uint32_t)str.size()); -} - -/** - * Writes an U16 to a bitstream - * - * RakNet writes in the correct byte order - do not reverse this. - */ -void WriteAMFU16(RakNet::BitStream* bs, uint16_t value) { - bs->Write(value); -} - -/** - * Writes an U32 to a bitstream - * - * RakNet writes in the correct byte order - do not reverse this. - */ -void WriteAMFU32(RakNet::BitStream* bs, uint32_t value) { - bs->Write(value); -} - -/** - * Writes an U64 to a bitstream - * - * RakNet writes in the correct byte order - do not reverse this. - */ -void WriteAMFU64(RakNet::BitStream* bs, uint64_t value) { - bs->Write(value); -} - - -// Writes an AMFUndefinedValue to BitStream -template<> -void RakNet::BitStream::Write(AMFUndefinedValue value) { - this->Write(AMFUndefined); -} - -// Writes an AMFNullValue to BitStream -template<> -void RakNet::BitStream::Write(AMFNullValue value) { - this->Write(AMFNull); -} - -// Writes an AMFFalseValue to BitStream -template<> -void RakNet::BitStream::Write(AMFFalseValue value) { - this->Write(AMFFalse); -} - -// Writes an AMFTrueValue to BitStream -template<> -void RakNet::BitStream::Write(AMFTrueValue value) { - this->Write(AMFTrue); -} - -// Writes an AMFIntegerValue to BitStream -template<> -void RakNet::BitStream::Write(AMFIntegerValue value) { - this->Write(AMFInteger); - WriteUInt29(this, value.GetIntegerValue()); -} - -// Writes an AMFDoubleValue to BitStream -template<> -void RakNet::BitStream::Write(AMFDoubleValue value) { - this->Write(AMFDouble); - double d = value.GetDoubleValue(); - WriteAMFU64(this, *((unsigned long long*) & d)); -} - -// Writes an AMFStringValue to BitStream -template<> -void RakNet::BitStream::Write(AMFStringValue value) { - this->Write(AMFString); - std::string v = value.GetStringValue(); - WriteAMFString(this, v); -} - -// Writes an AMFXMLDocValue to BitStream -template<> -void RakNet::BitStream::Write(AMFXMLDocValue value) { - this->Write(AMFXMLDoc); - std::string v = value.GetXMLDocValue(); - WriteAMFString(this, v); -} - -// Writes an AMFDateValue to BitStream -template<> -void RakNet::BitStream::Write(AMFDateValue value) { - this->Write(AMFDate); - uint64_t date = value.GetDateValue(); - WriteAMFU64(this, date); -} - -// Writes an AMFArrayValue to BitStream -template<> -void RakNet::BitStream::Write(AMFArrayValue* value) { - this->Write(AMFArray); - uint32_t denseSize = value->GetDenseValueSize(); - WriteFlagNumber(this, denseSize); - - _AMFArrayMap_::iterator it = value->GetAssociativeIteratorValueBegin(); - _AMFArrayMap_::iterator end = value->GetAssociativeIteratorValueEnd(); - - while (it != end) { - WriteAMFString(this, it->first); - this->Write(it->second); - it++; - } - - this->Write(AMFNull); - - if (denseSize > 0) { - _AMFArrayList_::iterator it2 = value->GetDenseIteratorBegin(); - _AMFArrayList_::iterator end2 = value->GetDenseIteratorEnd(); - - while (it2 != end2) { - this->Write(*it2); - it2++; - } - } -} diff --git a/dCommon/AMFFormat_BitStream.h b/dCommon/AMFFormat_BitStream.h deleted file mode 100644 index caa49337..00000000 --- a/dCommon/AMFFormat_BitStream.h +++ /dev/null @@ -1,92 +0,0 @@ -#pragma once - -// Custom Classes -#include "AMFFormat.h" - -// RakNet -#include - -/*! - \file AMFFormat_BitStream.h - \brief A class that implements native writing of AMF values to RakNet::BitStream - */ - - // We are using the RakNet namespace -namespace RakNet { - //! Writes an AMFValue pointer to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFValue* value); - - //! Writes an AMFUndefinedValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFUndefinedValue value); - - //! Writes an AMFNullValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFNullValue value); - - //! Writes an AMFFalseValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFFalseValue value); - - //! Writes an AMFTrueValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFTrueValue value); - - //! Writes an AMFIntegerValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFIntegerValue value); - - //! Writes an AMFDoubleValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFDoubleValue value); - - //! Writes an AMFStringValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFStringValue value); - - //! Writes an AMFXMLDocValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFXMLDocValue value); - - //! Writes an AMFDateValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFDateValue value); - - //! Writes an AMFArrayValue to a RakNet::BitStream - /*! - \param value The value to write - */ - template <> - void RakNet::BitStream::Write(AMFArrayValue* value); -} // namespace RakNet diff --git a/dCommon/AddFriendResponseCode.h b/dCommon/AddFriendResponseCode.h deleted file mode 100644 index bb2faff7..00000000 --- a/dCommon/AddFriendResponseCode.h +++ /dev/null @@ -1,15 +0,0 @@ -#pragma once - -#ifndef __ADDFRIENDRESPONSECODE__H__ -#define __ADDFRIENDRESPONSECODE__H__ - -#include - -enum class AddFriendResponseCode : uint8_t { - ACCEPTED = 0, - REJECTED, - BUSY, - CANCELLED -}; - -#endif //!__ADDFRIENDRESPONSECODE__H__ diff --git a/dCommon/AddFriendResponseType.h b/dCommon/AddFriendResponseType.h deleted file mode 100644 index 305796e8..00000000 --- a/dCommon/AddFriendResponseType.h +++ /dev/null @@ -1,24 +0,0 @@ -#pragma once - -#ifndef __ADDFRIENDRESPONSETYPE__H__ -#define __ADDFRIENDRESPONSETYPE__H__ - -#include - -enum class AddFriendResponseType : uint8_t { - ACCEPTED = 0, - ALREADYFRIEND, - INVALIDCHARACTER, - GENERALERROR, - YOURFRIENDSLISTFULL, - THEIRFRIENDLISTFULL, - DECLINED, - BUSY, - NOTONLINE, - WAITINGAPPROVAL, - MYTHRAN, - CANCELLED, - FRIENDISFREETRIAL -}; - -#endif //!__ADDFRIENDRESPONSETYPE__H__ diff --git a/dCommon/Amf3.h b/dCommon/Amf3.h new file mode 100644 index 00000000..4c649524 --- /dev/null +++ b/dCommon/Amf3.h @@ -0,0 +1,367 @@ +#ifndef __AMF3__H__ +#define __AMF3__H__ + +#include "dCommonVars.h" +#include "dLogger.h" +#include "Game.h" + +#include +#include + +enum class eAmf : uint8_t { + Undefined = 0x00, // An undefined AMF Value + Null = 0x01, // A null AMF value + False = 0x02, // A false AMF value + True = 0x03, // A true AMF value + Integer = 0x04, // An integer AMF value + Double = 0x05, // A double AMF value + String = 0x06, // A string AMF value + XMLDoc = 0x07, // Unused in the live client and cannot be serialized without modification. An XML Doc AMF value + Date = 0x08, // Unused in the live client and cannot be serialized without modification. A date AMF value + Array = 0x09, // An array AMF value + Object = 0x0A, // Unused in the live client and cannot be serialized without modification. An object AMF value + XML = 0x0B, // Unused in the live client and cannot be serialized without modification. An XML AMF value + ByteArray = 0x0C, // Unused in the live client and cannot be serialized without modification. A byte array AMF value + VectorInt = 0x0D, // Unused in the live client and cannot be serialized without modification. An integer vector AMF value + VectorUInt = 0x0E, // Unused in the live client and cannot be serialized without modification. An unsigned integer AMF value + VectorDouble = 0x0F, // Unused in the live client and cannot be serialized without modification. A double vector AMF value + VectorObject = 0x10, // Unused in the live client and cannot be serialized without modification. An object vector AMF value + Dictionary = 0x11 // Unused in the live client and cannot be serialized without modification. A dictionary AMF value +}; + +class AMFBaseValue { +public: + virtual eAmf GetValueType() { return eAmf::Undefined; }; + AMFBaseValue() {}; + virtual ~AMFBaseValue() {}; +}; + +template +class AMFValue : public AMFBaseValue { +public: + AMFValue() {}; + AMFValue(ValueType value) { SetValue(value); }; + virtual ~AMFValue() override {}; + + eAmf GetValueType() override { return eAmf::Undefined; }; + + const ValueType& GetValue() { return data; }; + void SetValue(ValueType value) { data = value; }; +protected: + ValueType data; +}; + +// As a string this is much easier to write and read from a BitStream. +template<> +class AMFValue : public AMFBaseValue { +public: + AMFValue() {}; + AMFValue(const char* value) { SetValue(std::string(value)); }; + virtual ~AMFValue() override {}; + + eAmf GetValueType() override { return eAmf::String; }; + + const std::string& GetValue() { return data; }; + void SetValue(std::string value) { data = value; }; +protected: + std::string data; +}; + +typedef AMFValue AMFNullValue; +typedef AMFValue AMFBoolValue; +typedef AMFValue AMFIntValue; +typedef AMFValue AMFStringValue; +typedef AMFValue AMFDoubleValue; + +template<> inline eAmf AMFValue::GetValueType() { return eAmf::Null; }; +template<> inline eAmf AMFValue::GetValueType() { return this->data ? eAmf::True : eAmf::False; }; +template<> inline eAmf AMFValue::GetValueType() { return eAmf::Integer; }; +template<> inline eAmf AMFValue::GetValueType() { return eAmf::Integer; }; +template<> inline eAmf AMFValue::GetValueType() { return eAmf::String; }; +template<> inline eAmf AMFValue::GetValueType() { return eAmf::Double; }; + +/** + * The AMFArrayValue object holds 2 types of lists: + * An associative list where a key maps to a value + * A Dense list where elements are stored back to back + * + * Objects that are Registered are owned by this object + * and are not to be deleted by a caller. + */ +class AMFArrayValue : public AMFBaseValue { + + typedef std::unordered_map AMFAssociative; + typedef std::vector AMFDense; + +public: + eAmf GetValueType() override { return eAmf::Array; }; + + ~AMFArrayValue() override { + for (auto valueToDelete : GetDense()) { + if (valueToDelete) { + delete valueToDelete; + valueToDelete = nullptr; + } + } + for (auto valueToDelete : GetAssociative()) { + if (valueToDelete.second) { + delete valueToDelete.second; + valueToDelete.second = nullptr; + } + } + }; + + /** + * Returns the Associative portion of the object + */ + inline AMFAssociative& GetAssociative() { return this->associative; }; + + /** + * Returns the dense portion of the object + */ + inline AMFDense& GetDense() { return this->dense; }; + + /** + * Inserts an AMFValue into the associative portion with the given key. + * If a duplicate is attempted to be inserted, it is ignored and the + * first value with that key is kept in the map. + * + * These objects are not to be deleted by the caller as they are owned by + * the AMFArray object which manages its own memory. + * + * @param key The key to associate with the value + * @param value The value to insert + * + * @return The inserted element if the type matched, + * or nullptr if a key existed and was not the same type + */ + template + std::pair*, bool> Insert(const std::string& key, ValueType value) { + auto element = associative.find(key); + AMFValue* val = nullptr; + bool found = true; + if (element == associative.end()) { + val = new AMFValue(value); + associative.insert(std::make_pair(key, val)); + } else { + val = dynamic_cast*>(element->second); + found = false; + } + return std::make_pair(val, found); + }; + + // Associates an array with a string key + std::pair Insert(const std::string& key) { + auto element = associative.find(key); + AMFArrayValue* val = nullptr; + bool found = true; + if (element == associative.end()) { + val = new AMFArrayValue(); + associative.insert(std::make_pair(key, val)); + } else { + val = dynamic_cast(element->second); + found = false; + } + return std::make_pair(val, found); + }; + + // Associates an array with an integer key + std::pair Insert(const uint32_t& index) { + AMFArrayValue* val = nullptr; + bool inserted = false; + if (index >= dense.size()) { + dense.resize(index + 1); + val = new AMFArrayValue(); + dense.at(index) = val; + inserted = true; + } + return std::make_pair(dynamic_cast(dense.at(index)), inserted); + }; + + /** + * @brief Inserts an AMFValue into the AMFArray key'd by index. + * Attempting to insert the same key to the same value twice overwrites + * the previous value with the new one. + * + * @param index The index to associate with the value + * @param value The value to insert + * @return The inserted element, or nullptr if the type did not match + * what was at the index. + */ + template + std::pair*, bool> Insert(const uint32_t& index, ValueType value) { + AMFValue* val = nullptr; + bool inserted = false; + if (index >= this->dense.size()) { + this->dense.resize(index + 1); + val = new AMFValue(value); + this->dense.at(index) = val; + inserted = true; + } + return std::make_pair(dynamic_cast*>(this->dense.at(index)), inserted); + }; + + /** + * Inserts an AMFValue into the associative portion with the given key. + * If a duplicate is attempted to be inserted, it replaces the original + * + * The inserted element is now owned by this object and is not to be deleted + * + * @param key The key to associate with the value + * @param value The value to insert + */ + void Insert(const std::string& key, AMFBaseValue* value) { + auto element = associative.find(key); + if (element != associative.end() && element->second) { + delete element->second; + element->second = value; + } else { + associative.insert(std::make_pair(key, value)); + } + }; + + /** + * Inserts an AMFValue into the associative portion with the given index. + * If a duplicate is attempted to be inserted, it replaces the original + * + * The inserted element is now owned by this object and is not to be deleted + * + * @param key The key to associate with the value + * @param value The value to insert + */ + void Insert(const uint32_t index, AMFBaseValue* value) { + if (index < dense.size()) { + AMFDense::iterator itr = dense.begin() + index; + if (*itr) delete dense.at(index); + } else { + dense.resize(index + 1); + } + dense.at(index) = value; + }; + + /** + * Pushes an AMFValue into the back of the dense portion. + * + * These objects are not to be deleted by the caller as they are owned by + * the AMFArray object which manages its own memory. + * + * @param value The value to insert + * + * @return The inserted pointer, or nullptr should the key already be in use. + */ + template + inline AMFValue* Push(ValueType value) { + return Insert(this->dense.size(), value).first; + }; + + /** + * Removes the key from the associative portion + * + * The pointer removed is now no longer managed by this container + * + * @param key The key to remove from the associative portion + */ + void Remove(const std::string& key, bool deleteValue = true) { + AMFAssociative::iterator it = this->associative.find(key); + if (it != this->associative.end()) { + if (deleteValue) delete it->second; + this->associative.erase(it); + } + } + + /** + * Pops the last element in the dense portion, deleting it in the process. + */ + void Remove(const uint32_t index) { + if (!this->dense.empty() && index < this->dense.size()) { + auto itr = this->dense.begin() + index; + if (*itr) delete (*itr); + this->dense.erase(itr); + } + } + + void Pop() { + if (!this->dense.empty()) Remove(this->dense.size() - 1); + } + + AMFArrayValue* GetArray(const std::string& key) { + AMFAssociative::const_iterator it = this->associative.find(key); + if (it != this->associative.end()) { + return dynamic_cast(it->second); + } + return nullptr; + }; + + AMFArrayValue* GetArray(const uint32_t index) { + return index >= this->dense.size() ? nullptr : dynamic_cast(this->dense.at(index)); + }; + + inline AMFArrayValue* InsertArray(const std::string& key) { + return static_cast(Insert(key).first); + }; + + inline AMFArrayValue* InsertArray(const uint32_t index) { + return static_cast(Insert(index).first); + }; + + inline AMFArrayValue* PushArray() { + return static_cast(Insert(this->dense.size()).first); + }; + + /** + * Gets an AMFValue by the key from the associative portion and converts it + * to the AmfValue template type. If the key did not exist, it is inserted. + * + * @tparam The target object type + * @param key The key to lookup + * + * @return The AMFValue + */ + template + AMFValue* Get(const std::string& key) const { + AMFAssociative::const_iterator it = this->associative.find(key); + return it != this->associative.end() ? + dynamic_cast*>(it->second) : + nullptr; + }; + + // Get from the array but dont cast it + AMFBaseValue* Get(const std::string& key) const { + AMFAssociative::const_iterator it = this->associative.find(key); + return it != this->associative.end() ? it->second : nullptr; + }; + + /** + * @brief Get an AMFValue object at a position in the dense portion. + * Gets an AMFValue by the index from the dense portion and converts it + * to the AmfValue template type. If the index did not exist, it is inserted. + * + * @tparam The target object type + * @param index The index to get + * @return The casted object, or nullptr. + */ + template + AMFValue* Get(uint32_t index) const { + return index < this->dense.size() ? + dynamic_cast*>(this->dense.at(index)) : + nullptr; + }; + + // Get from the dense but dont cast it + AMFBaseValue* Get(const uint32_t index) const { + return index < this->dense.size() ? this->dense.at(index) : nullptr; + }; +private: + /** + * The associative portion. These values are key'd with strings to an AMFValue. + */ + AMFAssociative associative; + + /** + * The dense portion. These AMFValue's are stored one after + * another with the most recent addition being at the back. + */ + AMFDense dense; +}; + +#endif //!__AMF3__H__ diff --git a/dCommon/AmfSerialize.cpp b/dCommon/AmfSerialize.cpp new file mode 100644 index 00000000..79ba5e2d --- /dev/null +++ b/dCommon/AmfSerialize.cpp @@ -0,0 +1,184 @@ +#include "AmfSerialize.h" + +#include "Game.h" +#include "dLogger.h" + +// Writes an AMFValue pointer to a RakNet::BitStream +template<> +void RakNet::BitStream::Write(AMFBaseValue& value) { + eAmf type = value.GetValueType(); + this->Write(type); + switch (type) { + case eAmf::Integer: { + this->Write(*static_cast(&value)); + break; + } + + case eAmf::Double: { + this->Write(*static_cast(&value)); + break; + } + + case eAmf::String: { + this->Write(*static_cast(&value)); + break; + } + + case eAmf::Array: { + this->Write(*static_cast(&value)); + break; + } + default: { + Game::logger->Log("AmfSerialize", "Encountered unwritable AMFType %i!", type); + } + case eAmf::Undefined: + case eAmf::Null: + case eAmf::False: + case eAmf::True: + case eAmf::Date: + case eAmf::Object: + case eAmf::XML: + case eAmf::XMLDoc: + case eAmf::ByteArray: + case eAmf::VectorInt: + case eAmf::VectorUInt: + case eAmf::VectorDouble: + case eAmf::VectorObject: + case eAmf::Dictionary: + break; + } +} + +/** + * A private function to write an value to a RakNet::BitStream + * RakNet writes in the correct byte order - do not reverse this. + */ +void WriteUInt29(RakNet::BitStream* bs, uint32_t v) { + unsigned char b4 = (unsigned char)v; + if (v < 0x00200000) { + b4 = b4 & 0x7F; + if (v > 0x7F) { + unsigned char b3; + v = v >> 7; + b3 = ((unsigned char)(v)) | 0x80; + if (v > 0x7F) { + unsigned char b2; + v = v >> 7; + b2 = ((unsigned char)(v)) | 0x80; + bs->Write(b2); + } + + bs->Write(b3); + } + } else { + unsigned char b1; + unsigned char b2; + unsigned char b3; + + v = v >> 8; + b3 = ((unsigned char)(v)) | 0x80; + v = v >> 7; + b2 = ((unsigned char)(v)) | 0x80; + v = v >> 7; + b1 = ((unsigned char)(v)) | 0x80; + + bs->Write(b1); + bs->Write(b2); + bs->Write(b3); + } + + bs->Write(b4); +} + +/** + * Writes a flag number to a RakNet::BitStream + * RakNet writes in the correct byte order - do not reverse this. + */ +void WriteFlagNumber(RakNet::BitStream* bs, uint32_t v) { + v = (v << 1) | 0x01; + WriteUInt29(bs, v); +} + +/** + * Writes an AMFString to a RakNet::BitStream + * + * RakNet writes in the correct byte order - do not reverse this. + */ +void WriteAMFString(RakNet::BitStream* bs, const std::string& str) { + WriteFlagNumber(bs, (uint32_t)str.size()); + bs->Write(str.c_str(), (uint32_t)str.size()); +} + +/** + * Writes an U16 to a bitstream + * + * RakNet writes in the correct byte order - do not reverse this. + */ +void WriteAMFU16(RakNet::BitStream* bs, uint16_t value) { + bs->Write(value); +} + +/** + * Writes an U32 to a bitstream + * + * RakNet writes in the correct byte order - do not reverse this. + */ +void WriteAMFU32(RakNet::BitStream* bs, uint32_t value) { + bs->Write(value); +} + +/** + * Writes an U64 to a bitstream + * + * RakNet writes in the correct byte order - do not reverse this. + */ +void WriteAMFU64(RakNet::BitStream* bs, uint64_t value) { + bs->Write(value); +} + +// Writes an AMFIntegerValue to BitStream +template<> +void RakNet::BitStream::Write(AMFIntValue& value) { + WriteUInt29(this, value.GetValue()); +} + +// Writes an AMFDoubleValue to BitStream +template<> +void RakNet::BitStream::Write(AMFDoubleValue& value) { + double d = value.GetValue(); + WriteAMFU64(this, *reinterpret_cast(&d)); +} + +// Writes an AMFStringValue to BitStream +template<> +void RakNet::BitStream::Write(AMFStringValue& value) { + WriteAMFString(this, value.GetValue()); +} + +// Writes an AMFArrayValue to BitStream +template<> +void RakNet::BitStream::Write(AMFArrayValue& value) { + uint32_t denseSize = value.GetDense().size(); + WriteFlagNumber(this, denseSize); + + auto it = value.GetAssociative().begin(); + auto end = value.GetAssociative().end(); + + while (it != end) { + WriteAMFString(this, it->first); + this->Write(*it->second); + it++; + } + + this->Write(eAmf::Null); + + if (denseSize > 0) { + auto it2 = value.GetDense().begin(); + auto end2 = value.GetDense().end(); + + while (it2 != end2) { + this->Write(**it2); + it2++; + } + } +} diff --git a/dCommon/AmfSerialize.h b/dCommon/AmfSerialize.h new file mode 100644 index 00000000..9a6f56f2 --- /dev/null +++ b/dCommon/AmfSerialize.h @@ -0,0 +1,50 @@ +#pragma once + +// Custom Classes +#include "Amf3.h" + +// RakNet +#include + +/*! + \file AmfSerialize.h + \brief A class that implements native writing of AMF values to RakNet::BitStream + */ + + // We are using the RakNet namespace +namespace RakNet { + //! Writes an AMFValue pointer to a RakNet::BitStream + /*! + \param value The value to write + */ + template <> + void RakNet::BitStream::Write(AMFBaseValue& value); + + //! Writes an AMFIntegerValue to a RakNet::BitStream + /*! + \param value The value to write + */ + template <> + void RakNet::BitStream::Write(AMFIntValue& value); + + //! Writes an AMFDoubleValue to a RakNet::BitStream + /*! + \param value The value to write + */ + template <> + void RakNet::BitStream::Write(AMFDoubleValue& value); + + //! Writes an AMFStringValue to a RakNet::BitStream + /*! + \param value The value to write + */ + template <> + void RakNet::BitStream::Write(AMFStringValue& value); + + //! Writes an AMFArrayValue to a RakNet::BitStream + /*! + \param value The value to write + */ + template <> + void RakNet::BitStream::Write(AMFArrayValue& value); +} // namespace RakNet diff --git a/dCommon/BinaryIO.cpp b/dCommon/BinaryIO.cpp index 7cb18331..22e4de60 100644 --- a/dCommon/BinaryIO.cpp +++ b/dCommon/BinaryIO.cpp @@ -10,7 +10,7 @@ void BinaryIO::WriteString(const std::string& stringToWrite, std::ofstream& outs } //For reading null-terminated strings -std::string BinaryIO::ReadString(std::ifstream& instream) { +std::string BinaryIO::ReadString(std::istream& instream) { std::string toReturn; char buffer; @@ -25,7 +25,7 @@ std::string BinaryIO::ReadString(std::ifstream& instream) { } //For reading strings of a specific size -std::string BinaryIO::ReadString(std::ifstream& instream, size_t size) { +std::string BinaryIO::ReadString(std::istream& instream, size_t size) { std::string toReturn; char buffer; @@ -37,7 +37,7 @@ std::string BinaryIO::ReadString(std::ifstream& instream, size_t size) { return toReturn; } -std::string BinaryIO::ReadWString(std::ifstream& instream) { +std::string BinaryIO::ReadWString(std::istream& instream) { size_t size; BinaryRead(instream, size); //toReturn.resize(size); diff --git a/dCommon/BinaryIO.h b/dCommon/BinaryIO.h index 1f9aaefd..a117ad0d 100644 --- a/dCommon/BinaryIO.h +++ b/dCommon/BinaryIO.h @@ -10,16 +10,15 @@ namespace BinaryIO { template std::istream& BinaryRead(std::istream& stream, T& value) { - if (!stream.good()) - printf("bla"); + if (!stream.good()) throw std::runtime_error("Failed to read from istream."); return stream.read(reinterpret_cast(&value), sizeof(T)); } void WriteString(const std::string& stringToWrite, std::ofstream& outstream); - std::string ReadString(std::ifstream& instream); - std::string ReadString(std::ifstream& instream, size_t size); - std::string ReadWString(std::ifstream& instream); + std::string ReadString(std::istream& instream); + std::string ReadString(std::istream& instream, size_t size); + std::string ReadWString(std::istream& instream); inline bool DoesFileExist(const std::string& name) { std::ifstream f(name.c_str()); diff --git a/dCommon/BinaryPathFinder.cpp b/dCommon/BinaryPathFinder.cpp new file mode 100644 index 00000000..b1c2afef --- /dev/null +++ b/dCommon/BinaryPathFinder.cpp @@ -0,0 +1,71 @@ +#include +#include +#include "BinaryPathFinder.h" +#include "dPlatforms.h" + +#if defined(DARKFLAME_PLATFORM_WIN32) +#include +#elif defined(DARKFLAME_PLATFORM_MACOS) || defined(DARKFLAME_PLATFORM_IOS) +#include +#elif defined(DARKFLAME_PLATFORM_FREEBSD) +#include +#include +#include +#endif + +std::filesystem::path BinaryPathFinder::binaryDir; + +std::filesystem::path BinaryPathFinder::GetBinaryDir() { + if (!binaryDir.empty()) { + return binaryDir; + } + + std::string pathStr; + + // Derived from boost::dll::program_location, licensed under the Boost Software License: http://www.boost.org/LICENSE_1_0.txt +#if defined(DARKFLAME_PLATFORM_WIN32) + char path[MAX_PATH]; + GetModuleFileName(NULL, path, MAX_PATH); + pathStr = std::string(path); +#elif defined(DARKFLAME_PLATFORM_MACOS) || defined(DARKFLAME_PLATFORM_IOS) + char path[1024]; + uint32_t size = sizeof(path); + if (_NSGetExecutablePath(path, &size) == 0) { + pathStr = std::string(path); + } else { + // The filepath size is greater than our initial buffer size, so try again with the size + // that _NSGetExecutablePath told us it actually is + char *p = new char[size]; + if (_NSGetExecutablePath(p, &size) != 0) { + throw std::runtime_error("Failed to get binary path from _NSGetExecutablePath"); + } + + pathStr = std::string(p); + delete[] p; + } +#elif defined(DARKFLAME_PLATFORM_FREEBSD) + int mib[4]; + mib[0] = CTL_KERN; + mib[1] = KERN_PROC; + mib[2] = KERN_PROC_PATHNAME; + mib[3] = -1; + char buf[10240]; + size_t cb = sizeof(buf); + sysctl(mib, 4, buf, &cb, NULL, 0); + pathStr = std::string(buf); +#else // DARKFLAME_PLATFORM_LINUX || DARKFLAME_PLATFORM_UNIX || DARKFLAME_PLATFORM_ANDROID + pathStr = std::filesystem::read_symlink("/proc/self/exe"); +#endif + + // Some methods like _NSGetExecutablePath could return a symlink + // Either way, we need to get the parent path because we want the directory, not the binary itself + // We also ensure that it is an absolute path so that it is valid if we need to construct a path + // to exucute on unix systems (eg sudo BinaryPathFinder::GetBinaryDir() / WorldServer) + if (std::filesystem::is_symlink(pathStr)) { + binaryDir = std::filesystem::absolute(std::filesystem::read_symlink(pathStr).parent_path()); + } else { + binaryDir = std::filesystem::absolute(std::filesystem::path(pathStr).parent_path()); + } + + return binaryDir; +} diff --git a/dCommon/BinaryPathFinder.h b/dCommon/BinaryPathFinder.h new file mode 100644 index 00000000..c4ca6da2 --- /dev/null +++ b/dCommon/BinaryPathFinder.h @@ -0,0 +1,15 @@ +#pragma once + +#ifndef __BINARYPATHFINDER__H__ +#define __BINARYPATHFINDER__H__ + +#include + +class BinaryPathFinder { +private: + static std::filesystem::path binaryDir; +public: + static std::filesystem::path GetBinaryDir(); +}; + +#endif //!__BINARYPATHFINDER__H__ diff --git a/dCommon/Brick.h b/dCommon/Brick.h new file mode 100644 index 00000000..e8bd747e --- /dev/null +++ b/dCommon/Brick.h @@ -0,0 +1,11 @@ +#ifndef __BRICK__H__ +#define __BRICK__H__ + +#include + +struct Brick { + uint32_t designerID; + uint32_t materialID; +}; + +#endif //!__BRICK__H__ diff --git a/dCommon/BrickByBrickFix.cpp b/dCommon/BrickByBrickFix.cpp new file mode 100644 index 00000000..15194bf9 --- /dev/null +++ b/dCommon/BrickByBrickFix.cpp @@ -0,0 +1,180 @@ +#include "BrickByBrickFix.h" + +#include +#include +#include + +#include "tinyxml2.h" + +#include "Database.h" +#include "Game.h" +#include "ZCompression.h" +#include "dLogger.h" + +//! Forward declarations + +std::unique_ptr GetModelsFromDatabase(); +void WriteSd0Magic(char* input, uint32_t chunkSize); +bool CheckSd0Magic(sql::Blob* streamToCheck); + +/** + * @brief Truncates all models with broken data from the database. + * + * @return The number of models deleted + */ +uint32_t BrickByBrickFix::TruncateBrokenBrickByBrickXml() { + uint32_t modelsTruncated{}; + auto modelsToTruncate = GetModelsFromDatabase(); + bool previousCommitValue = Database::GetAutoCommit(); + Database::SetAutoCommit(false); + while (modelsToTruncate->next()) { + std::unique_ptr ugcModelToDelete(Database::CreatePreppedStmt("DELETE FROM ugc WHERE ugc.id = ?;")); + std::unique_ptr pcModelToDelete(Database::CreatePreppedStmt("DELETE FROM properties_contents WHERE ugc_id = ?;")); + std::string completeUncompressedModel{}; + uint32_t chunkCount{}; + uint64_t modelId = modelsToTruncate->getInt(1); + std::unique_ptr modelAsSd0(modelsToTruncate->getBlob(2)); + // Check that header is sd0 by checking for the sd0 magic. + if (CheckSd0Magic(modelAsSd0.get())) { + while (true) { + uint32_t chunkSize{}; + modelAsSd0->read(reinterpret_cast(&chunkSize), sizeof(uint32_t)); // Extract chunk size from istream + + // Check if good here since if at the end of an sd0 file, this will have eof flagged. + if (!modelAsSd0->good()) break; + + std::unique_ptr compressedChunk(new uint8_t[chunkSize]); + for (uint32_t i = 0; i < chunkSize; i++) { + compressedChunk[i] = modelAsSd0->get(); + } + + // Ignore the valgrind warning about uninitialized values. These are discarded later when we know the actual uncompressed size. + std::unique_ptr uncompressedChunk(new uint8_t[ZCompression::MAX_SD0_CHUNK_SIZE]); + int32_t err{}; + int32_t actualUncompressedSize = ZCompression::Decompress( + compressedChunk.get(), chunkSize, uncompressedChunk.get(), ZCompression::MAX_SD0_CHUNK_SIZE, err); + + if (actualUncompressedSize != -1) { + uint32_t previousSize = completeUncompressedModel.size(); + completeUncompressedModel.append((char*)uncompressedChunk.get()); + completeUncompressedModel.resize(previousSize + actualUncompressedSize); + } else { + Game::logger->Log("BrickByBrickFix", "Failed to inflate chunk %i for model %llu. Error: %i", chunkCount, modelId, err); + break; + } + chunkCount++; + } + std::unique_ptr document = std::make_unique(); + if (!document) { + Game::logger->Log("BrickByBrickFix", "Failed to initialize tinyxml document. Aborting."); + return 0; + } + + if (!(document->Parse(completeUncompressedModel.c_str(), completeUncompressedModel.size()) == tinyxml2::XML_SUCCESS)) { + if (completeUncompressedModel.find( + "", + completeUncompressedModel.length() >= 15 ? completeUncompressedModel.length() - 15 : 0) == std::string::npos + ) { + Game::logger->Log("BrickByBrickFix", + "Brick-by-brick model %llu will be deleted!", modelId); + ugcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); + pcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); + ugcModelToDelete->execute(); + pcModelToDelete->execute(); + modelsTruncated++; + } + } + } else { + Game::logger->Log("BrickByBrickFix", + "Brick-by-brick model %llu will be deleted!", modelId); + ugcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); + pcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); + ugcModelToDelete->execute(); + pcModelToDelete->execute(); + modelsTruncated++; + } + } + + Database::Commit(); + Database::SetAutoCommit(previousCommitValue); + return modelsTruncated; +} + +/** + * @brief Updates all current models in the database to have the Segmented Data 0 (SD0) format. + * Any models that do not start with zlib and best compression magic will not be updated. + * + * @return The number of models updated to SD0 + */ +uint32_t BrickByBrickFix::UpdateBrickByBrickModelsToSd0() { + uint32_t updatedModels = 0; + auto modelsToUpdate = GetModelsFromDatabase(); + auto previousAutoCommitState = Database::GetAutoCommit(); + Database::SetAutoCommit(false); + std::unique_ptr insertionStatement(Database::CreatePreppedStmt("UPDATE ugc SET lxfml = ? WHERE id = ?;")); + while (modelsToUpdate->next()) { + int64_t modelId = modelsToUpdate->getInt64(1); + std::unique_ptr oldLxfml(modelsToUpdate->getBlob(2)); + // Check if the stored blob starts with zlib magic (0x78 0xDA - best compression of zlib) + // If it does, convert it to sd0. + if (oldLxfml->get() == 0x78 && oldLxfml->get() == 0xDA) { + + // Get and save size of zlib compressed chunk. + oldLxfml->seekg(0, std::ios::end); + uint32_t oldLxfmlSize = static_cast(oldLxfml->tellg()); + oldLxfml->seekg(0); + + // Allocate 9 extra bytes. 5 for sd0 magic, 4 for the only zlib compressed size. + uint32_t oldLxfmlSizeWithHeader = oldLxfmlSize + 9; + std::unique_ptr sd0ConvertedModel(new char[oldLxfmlSizeWithHeader]); + + WriteSd0Magic(sd0ConvertedModel.get(), oldLxfmlSize); + for (uint32_t i = 9; i < oldLxfmlSizeWithHeader; i++) { + sd0ConvertedModel.get()[i] = oldLxfml->get(); + } + + std::string outputString(sd0ConvertedModel.get(), oldLxfmlSizeWithHeader); + std::istringstream outputStringStream(outputString); + + insertionStatement->setBlob(1, static_cast(&outputStringStream)); + insertionStatement->setInt64(2, modelId); + try { + insertionStatement->executeUpdate(); + Game::logger->Log("BrickByBrickFix", "Updated model %i to sd0", modelId); + updatedModels++; + } catch (sql::SQLException exception) { + Game::logger->Log( + "BrickByBrickFix", + "Failed to update model %i. This model should be inspected manually to see why." + "The database error is %s", modelId, exception.what()); + } + } + } + Database::Commit(); + Database::SetAutoCommit(previousAutoCommitState); + return updatedModels; +} + +std::unique_ptr GetModelsFromDatabase() { + std::unique_ptr modelsRawDataQuery(Database::CreatePreppedStmt("SELECT id, lxfml FROM ugc;")); + return std::unique_ptr(modelsRawDataQuery->executeQuery()); +} + +/** + * @brief Writes sd0 magic at the front of a char* + * + * @param input the char* to write at the front of + * @param chunkSize The size of the first chunk to write the size of + */ +void WriteSd0Magic(char* input, uint32_t chunkSize) { + input[0] = 's'; + input[1] = 'd'; + input[2] = '0'; + input[3] = 0x01; + input[4] = 0xFF; + *reinterpret_cast(input + 5) = chunkSize; // Write the integer to the character array +} + +bool CheckSd0Magic(sql::Blob* streamToCheck) { + return streamToCheck->get() == 's' && streamToCheck->get() == 'd' && streamToCheck->get() == '0' && streamToCheck->get() == 0x01 && streamToCheck->get() == 0xFF; +} diff --git a/dCommon/BrickByBrickFix.h b/dCommon/BrickByBrickFix.h new file mode 100644 index 00000000..7450fb71 --- /dev/null +++ b/dCommon/BrickByBrickFix.h @@ -0,0 +1,20 @@ +#pragma once + +#include + +namespace BrickByBrickFix { + /** + * @brief Deletes all broken BrickByBrick models that have invalid XML + * + * @return The number of BrickByBrick models that were truncated + */ + uint32_t TruncateBrokenBrickByBrickXml(); + + /** + * @brief Updates all BrickByBrick models in the database to be + * in the sd0 format as opposed to a zlib compressed format. + * + * @return The number of BrickByBrick models that were updated + */ + uint32_t UpdateBrickByBrickModelsToSd0(); +}; diff --git a/dCommon/CMakeLists.txt b/dCommon/CMakeLists.txt index 028eb29f..2517c048 100644 --- a/dCommon/CMakeLists.txt +++ b/dCommon/CMakeLists.txt @@ -1,6 +1,6 @@ -set(DCOMMON_SOURCES "AMFFormat.cpp" +set(DCOMMON_SOURCES "AMFDeserialize.cpp" - "AMFFormat_BitStream.cpp" + "AmfSerialize.cpp" "BinaryIO.cpp" "dConfig.cpp" "Diagnostics.cpp" @@ -13,15 +13,51 @@ set(DCOMMON_SOURCES "AMFFormat.cpp" "NiQuaternion.cpp" "SHA512.cpp" "Type.cpp" - "ZCompression.cpp") + "ZCompression.cpp" + "BrickByBrickFix.cpp" + "BinaryPathFinder.cpp" + "FdbToSqlite.cpp" +) + +add_subdirectory(dClient) + +foreach(file ${DCOMMON_DCLIENT_SOURCES}) + set(DCOMMON_SOURCES ${DCOMMON_SOURCES} "dClient/${file}") +endforeach() include_directories(${PROJECT_SOURCE_DIR}/dCommon/) add_library(dCommon STATIC ${DCOMMON_SOURCES}) -target_link_libraries(dCommon bcrypt) +target_link_libraries(dCommon bcrypt dDatabase tinyxml2) if (UNIX) find_package(ZLIB REQUIRED) - target_link_libraries(dCommon ZLIB::ZLIB) -endif() +elseif (WIN32) + include(FetchContent) + + # TODO Keep an eye on the zlib repository for an update to disable testing. Don't forget to update CMakePresets + FetchContent_Declare( + zlib + URL https://github.com/madler/zlib/archive/refs/tags/v1.2.11.zip + URL_HASH MD5=9d6a627693163bbbf3f26403a3a0b0b1 + ) + + # Disable warning about no project version. + set(CMAKE_POLICY_DEFAULT_CMP0048 NEW) + # Disable warning about the minimum version of cmake used for bcrypt being deprecated in the future + set(CMAKE_WARN_DEPRECATED OFF CACHE BOOL "" FORCE) + + FetchContent_MakeAvailable(zlib) + + set(ZLIB_INCLUDE_DIRS ${zlib_SOURCE_DIR} ${zlib_BINARY_DIR}) + set_target_properties(zlib PROPERTIES INTERFACE_INCLUDE_DIRECTORIES "${ZLIB_INCLUDE_DIRS}") + add_library(ZLIB::ZLIB ALIAS zlib) +else () + message( + FATAL_ERROR + "This platform does not have a way to use zlib.\nCreate an issue on GitHub with your build system so it can be configured." + ) +endif () + +target_link_libraries(dCommon ZLIB::ZLIB) diff --git a/dCommon/Diagnostics.cpp b/dCommon/Diagnostics.cpp index 4c2c8beb..58d558de 100644 --- a/dCommon/Diagnostics.cpp +++ b/dCommon/Diagnostics.cpp @@ -1,4 +1,6 @@ #include "Diagnostics.h" +#include "Game.h" +#include "dLogger.h" // If we're on Win32, we'll include our minidump writer #ifdef _WIN32 @@ -26,7 +28,7 @@ void make_minidump(EXCEPTION_POINTERS* e) { "_%4d%02d%02d_%02d%02d%02d.dmp", t.wYear, t.wMonth, t.wDay, t.wHour, t.wMinute, t.wSecond); } - + Game::logger->Log("Diagnostics", "Creating crash dump %s", name); auto hFile = CreateFileA(name, GENERIC_WRITE, FILE_SHARE_READ, 0, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0); if (hFile == INVALID_HANDLE_VALUE) return; @@ -81,6 +83,7 @@ struct bt_ctx { static inline void Bt(struct backtrace_state* state) { std::string fileName = Diagnostics::GetOutDirectory() + "crash_" + Diagnostics::GetProcessName() + "_" + std::to_string(getpid()) + ".log"; + Game::logger->Log("Diagnostics", "backtrace is enabled, crash dump located at %s", fileName.c_str()); FILE* file = fopen(fileName.c_str(), "w+"); if (file != nullptr) { backtrace_print(state, 2, file); @@ -108,12 +111,14 @@ static void ErrorCallback(void* data, const char* msg, int errnum) { void GenerateDump() { std::string cmd = "sudo gcore " + std::to_string(getpid()); - system(cmd.c_str()); + int ret = system(cmd.c_str()); // Saving a return just to prevent warning } void CatchUnhandled(int sig) { #ifndef __include_backtrace__ + std::string fileName = Diagnostics::GetOutDirectory() + "crash_" + Diagnostics::GetProcessName() + "_" + std::to_string(getpid()) + ".log"; + Game::logger->Log("Diagnostics", "Encountered signal %i, creating crash dump %s", sig, fileName.c_str()); if (Diagnostics::GetProduceMemoryDump()) { GenerateDump(); } @@ -124,7 +129,6 @@ void CatchUnhandled(int sig) { // get void*'s for all entries on the stack size = backtrace(array, 10); - printf("Fatal error %i\nStacktrace:\n", sig); #if defined(__GNUG__) and defined(__dynamic) // Loop through the returned addresses, and get the symbols to be demangled @@ -142,19 +146,18 @@ void CatchUnhandled(int sig) { demangled = demangle(functionName.c_str()); if (demangled.empty()) { - printf("[%02zu] %s\n", i, demangled.c_str()); + Game::logger->Log("Diagnostics", "[%02zu] %s", i, demangled.c_str()); } else { - printf("[%02zu] %s\n", i, functionName.c_str()); + Game::logger->Log("Diagnostics", "[%02zu] %s", i, functionName.c_str()); } } else { - printf("[%02zu] %s\n", i, functionName.c_str()); + Game::logger->Log("Diagnostics", "[%02zu] %s", i, functionName.c_str()); } } #else backtrace_symbols_fd(array, size, STDOUT_FILENO); #endif - std::string fileName = Diagnostics::GetOutDirectory() + "crash_" + Diagnostics::GetProcessName() + "_" + std::to_string(getpid()) + ".log"; FILE* file = fopen(fileName.c_str(), "w+"); if (file != NULL) { // print out all the frames to stderr diff --git a/dCommon/DluAssert.h b/dCommon/DluAssert.h new file mode 100644 index 00000000..c54dd54e --- /dev/null +++ b/dCommon/DluAssert.h @@ -0,0 +1,12 @@ +#ifndef __DLUASSERT__H__ +#define __DLUASSERT__H__ + +#include + +#ifdef _DEBUG +# define DluAssert(expression) assert(expression) +#else +# define DluAssert(expression) +#endif + +#endif //!__DLUASSERT__H__ diff --git a/dCommon/FdbToSqlite.cpp b/dCommon/FdbToSqlite.cpp new file mode 100644 index 00000000..e05286a9 --- /dev/null +++ b/dCommon/FdbToSqlite.cpp @@ -0,0 +1,247 @@ +#include "FdbToSqlite.h" + +#include +#include +#include +#include + +#include "BinaryIO.h" +#include "CDClientDatabase.h" +#include "GeneralUtils.h" +#include "Game.h" +#include "dLogger.h" +#include "AssetManager.h" + +#include "eSqliteDataType.h" + +std::map FdbToSqlite::Convert::m_SqliteType = { + { eSqliteDataType::NONE, "none"}, + { eSqliteDataType::INT32, "int32"}, + { eSqliteDataType::REAL, "real"}, + { eSqliteDataType::TEXT_4, "text_4"}, + { eSqliteDataType::INT_BOOL, "int_bool"}, + { eSqliteDataType::INT64, "int64"}, + { eSqliteDataType::TEXT_8, "text_8"} +}; + +FdbToSqlite::Convert::Convert(std::string binaryOutPath) { + this->m_BinaryOutPath = binaryOutPath; +} + +bool FdbToSqlite::Convert::ConvertDatabase(AssetMemoryBuffer& buffer) { + if (m_ConversionStarted) return false; + + std::istream cdClientBuffer(&buffer); + + this->m_ConversionStarted = true; + try { + CDClientDatabase::Connect(m_BinaryOutPath + "/CDServer.sqlite"); + + CDClientDatabase::ExecuteQuery("BEGIN TRANSACTION;"); + + int32_t numberOfTables = ReadInt32(cdClientBuffer); + ReadTables(numberOfTables, cdClientBuffer); + + CDClientDatabase::ExecuteQuery("COMMIT;"); + } catch (CppSQLite3Exception& e) { + Game::logger->Log("FdbToSqlite", "Encountered error %s converting FDB to SQLite", e.errorMessage()); + return false; + } + + return true; +} + +int32_t FdbToSqlite::Convert::ReadInt32(std::istream& cdClientBuffer) { + int32_t nextInt{}; + BinaryIO::BinaryRead(cdClientBuffer, nextInt); + return nextInt; +} + +int64_t FdbToSqlite::Convert::ReadInt64(std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + int64_t value{}; + BinaryIO::BinaryRead(cdClientBuffer, value); + + cdClientBuffer.seekg(prevPosition); + return value; +} + +std::string FdbToSqlite::Convert::ReadString(std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + auto readString = BinaryIO::ReadString(cdClientBuffer); + + cdClientBuffer.seekg(prevPosition); + return readString; +} + +int32_t FdbToSqlite::Convert::SeekPointer(std::istream& cdClientBuffer) { + int32_t position{}; + BinaryIO::BinaryRead(cdClientBuffer, position); + int32_t prevPosition = cdClientBuffer.tellg(); + cdClientBuffer.seekg(position); + return prevPosition; +} + +std::string FdbToSqlite::Convert::ReadColumnHeader(std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + int32_t numberOfColumns = ReadInt32(cdClientBuffer); + std::string tableName = ReadString(cdClientBuffer); + + auto columns = ReadColumns(numberOfColumns, cdClientBuffer); + std::string newTable = "CREATE TABLE IF NOT EXISTS '" + tableName + "' (" + columns + ");"; + CDClientDatabase::ExecuteDML(newTable); + + cdClientBuffer.seekg(prevPosition); + + return tableName; +} + +void FdbToSqlite::Convert::ReadTables(int32_t& numberOfTables, std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + for (int32_t i = 0; i < numberOfTables; i++) { + auto columnHeader = ReadColumnHeader(cdClientBuffer); + ReadRowHeader(columnHeader, cdClientBuffer); + } + + cdClientBuffer.seekg(prevPosition); +} + +std::string FdbToSqlite::Convert::ReadColumns(int32_t& numberOfColumns, std::istream& cdClientBuffer) { + std::stringstream columnsToCreate; + int32_t prevPosition = SeekPointer(cdClientBuffer); + + std::string name{}; + eSqliteDataType dataType{}; + for (int32_t i = 0; i < numberOfColumns; i++) { + if (i != 0) columnsToCreate << ", "; + dataType = static_cast(ReadInt32(cdClientBuffer)); + name = ReadString(cdClientBuffer); + columnsToCreate << "'" << name << "' " << FdbToSqlite::Convert::m_SqliteType[dataType]; + } + + cdClientBuffer.seekg(prevPosition); + return columnsToCreate.str(); +} + +void FdbToSqlite::Convert::ReadRowHeader(std::string& tableName, std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + int32_t numberOfAllocatedRows = ReadInt32(cdClientBuffer); + if (numberOfAllocatedRows != 0) assert((numberOfAllocatedRows & (numberOfAllocatedRows - 1)) == 0); // assert power of 2 allocation size + ReadRows(numberOfAllocatedRows, tableName, cdClientBuffer); + + cdClientBuffer.seekg(prevPosition); +} + +void FdbToSqlite::Convert::ReadRows(int32_t& numberOfAllocatedRows, std::string& tableName, std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + int32_t rowid = 0; + for (int32_t row = 0; row < numberOfAllocatedRows; row++) { + int32_t rowPointer = ReadInt32(cdClientBuffer); + if (rowPointer == -1) rowid++; + else ReadRow(rowPointer, tableName, cdClientBuffer); + } + + cdClientBuffer.seekg(prevPosition); +} + +void FdbToSqlite::Convert::ReadRow(int32_t& position, std::string& tableName, std::istream& cdClientBuffer) { + int32_t prevPosition = cdClientBuffer.tellg(); + cdClientBuffer.seekg(position); + + while (true) { + ReadRowInfo(tableName, cdClientBuffer); + int32_t linked = ReadInt32(cdClientBuffer); + if (linked == -1) break; + cdClientBuffer.seekg(linked); + } + + cdClientBuffer.seekg(prevPosition); +} + +void FdbToSqlite::Convert::ReadRowInfo(std::string& tableName, std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + int32_t numberOfColumns = ReadInt32(cdClientBuffer); + ReadRowValues(numberOfColumns, tableName, cdClientBuffer); + + cdClientBuffer.seekg(prevPosition); +} + +void FdbToSqlite::Convert::ReadRowValues(int32_t& numberOfColumns, std::string& tableName, std::istream& cdClientBuffer) { + int32_t prevPosition = SeekPointer(cdClientBuffer); + + int32_t emptyValue{}; + int32_t intValue{}; + float_t floatValue{}; + std::string stringValue{}; + int32_t boolValue{}; + int64_t int64Value{}; + bool insertedFirstEntry = false; + std::stringstream insertedRow; + insertedRow << "INSERT INTO " << tableName << " values ("; + + for (int32_t i = 0; i < numberOfColumns; i++) { + if (i != 0) insertedRow << ", "; // Only append comma and space after first entry in row. + switch (static_cast(ReadInt32(cdClientBuffer))) { + case eSqliteDataType::NONE: + BinaryIO::BinaryRead(cdClientBuffer, emptyValue); + assert(emptyValue == 0); + insertedRow << "NULL"; + break; + + case eSqliteDataType::INT32: + intValue = ReadInt32(cdClientBuffer); + insertedRow << intValue; + break; + + case eSqliteDataType::REAL: + BinaryIO::BinaryRead(cdClientBuffer, floatValue); + insertedRow << std::fixed << std::setprecision(34) << floatValue; // maximum precision of floating point number + break; + + case eSqliteDataType::TEXT_4: + case eSqliteDataType::TEXT_8: { + stringValue = ReadString(cdClientBuffer); + size_t position = 0; + + // Need to escape quote with a double of ". + while (position < stringValue.size()) { + if (stringValue.at(position) == '\"') { + stringValue.insert(position, "\""); + position++; + } + position++; + } + insertedRow << "\"" << stringValue << "\""; + break; + } + + case eSqliteDataType::INT_BOOL: + BinaryIO::BinaryRead(cdClientBuffer, boolValue); + insertedRow << static_cast(boolValue); + break; + + case eSqliteDataType::INT64: + int64Value = ReadInt64(cdClientBuffer); + insertedRow << std::to_string(int64Value); + break; + + default: + throw std::invalid_argument("Unsupported SQLite type encountered."); + break; + + } + } + + insertedRow << ");"; + + auto copiedString = insertedRow.str(); + CDClientDatabase::ExecuteDML(copiedString); + cdClientBuffer.seekg(prevPosition); +} diff --git a/dCommon/FdbToSqlite.h b/dCommon/FdbToSqlite.h new file mode 100644 index 00000000..7aad2703 --- /dev/null +++ b/dCommon/FdbToSqlite.h @@ -0,0 +1,145 @@ +#ifndef __FDBTOSQLITE__H__ +#define __FDBTOSQLITE__H__ + +#pragma once + +#include +#include +#include + +class AssetMemoryBuffer; + +enum class eSqliteDataType : int32_t; + +namespace FdbToSqlite { + class Convert { + public: + /** + * Create a new convert object with an input .fdb file and an output binary path. + * + * @param inputFile The file which ends in .fdb to be converted + * @param binaryPath The base path where the file will be saved + */ + Convert(std::string binaryOutPath); + + /** + * Converts the input file to sqlite. Calling multiple times is safe. + * + * @return true if the database was converted properly, false otherwise. + */ + bool ConvertDatabase(AssetMemoryBuffer& buffer); + + /** + * @brief Reads a 32 bit int from the fdb file. + * + * @return The read value + */ + int32_t ReadInt32(std::istream& cdClientBuffer); + + /** + * @brief Reads a 64 bit integer from the fdb file. + * + * @return The read value + */ + int64_t ReadInt64(std::istream& cdClientBuffer); + + /** + * @brief Reads a string from the fdb file. + * + * @return The read string + * + * TODO This needs to be translated to latin-1! + */ + std::string ReadString(std::istream& cdClientBuffer); + + /** + * @brief Seeks to a pointer position. + * + * @return The previous position before the seek + */ + int32_t SeekPointer(std::istream& cdClientBuffer); + + /** + * @brief Reads a column header from the fdb file and creates the table in the database + * + * @return The table name + */ + std::string ReadColumnHeader(std::istream& cdClientBuffer); + + /** + * @brief Read the tables from the fdb file. + * + * @param numberOfTables The number of tables to read + */ + void ReadTables(int32_t& numberOfTables, std::istream& cdClientBuffer); + + /** + * @brief Reads the columns from the fdb file. + * + * @param numberOfColumns The number of columns to read + * @return All columns of the table formatted for a sql query + */ + std::string ReadColumns(int32_t& numberOfColumns, std::istream& cdClientBuffer); + + /** + * @brief Reads the row header from the fdb file. + * + * @param tableName The tables name + */ + void ReadRowHeader(std::string& tableName, std::istream& cdClientBuffer); + + /** + * @brief Read the rows from the fdb file., + * + * @param numberOfAllocatedRows The number of rows that were allocated. Always a power of 2! + * @param tableName The tables name. + */ + void ReadRows(int32_t& numberOfAllocatedRows, std::string& tableName, std::istream& cdClientBuffer); + + /** + * @brief Reads a row from the fdb file. + * + * @param position The position to seek in the fdb to + * @param tableName The tables name + */ + void ReadRow(int32_t& position, std::string& tableName, std::istream& cdClientBuffer); + + /** + * @brief Reads the row info from the fdb file. + * + * @param tableName The tables name + */ + void ReadRowInfo(std::string& tableName, std::istream& cdClientBuffer); + + /** + * @brief Reads each row and its values from the fdb file and inserts them into the database + * + * @param numberOfColumns The number of columns to read in + * @param tableName The tables name + */ + void ReadRowValues(int32_t& numberOfColumns, std::string& tableName, std::istream& cdClientBuffer); + private: + + /** + * Maps each sqlite data type to its string equivalent. + */ + static std::map m_SqliteType; + + /** + * Base path of the folder containing the fdb file + */ + std::string m_BasePath{}; + + /** + * Whether or not a conversion was started. If one was started, do not attempt to convert the file again. + */ + bool m_ConversionStarted{}; + + /** + * The path where the CDServer will be stored + */ + std::string m_BinaryOutPath{}; + }; //! class FdbToSqlite +}; //! namespace FdbToSqlite + +#endif //!__FDBTOSQLITE__H__ diff --git a/dCommon/Game.h b/dCommon/Game.h index f4862602..09ac6f6e 100644 --- a/dCommon/Game.h +++ b/dCommon/Game.h @@ -5,22 +5,25 @@ class dServer; class dLogger; class InstanceManager; -class dpWorld; class dChatFilter; class dConfig; -class dLocale; class RakPeerInterface; +class AssetManager; struct SystemAddress; +class EntityManager; +class dZoneManager; namespace Game { extern dLogger* logger; extern dServer* server; extern InstanceManager* im; - extern dpWorld* physicsWorld; extern dChatFilter* chatFilter; extern dConfig* config; - extern dLocale* locale; extern std::mt19937 randomEngine; extern RakPeerInterface* chatServer; + extern AssetManager* assetManager; extern SystemAddress chatSysAddr; + extern bool shouldShutdown; + extern EntityManager* entityManager; + extern dZoneManager* zoneManager; } diff --git a/dCommon/GeneralUtils.cpp b/dCommon/GeneralUtils.cpp index 4a6c4739..99ca687a 100644 --- a/dCommon/GeneralUtils.cpp +++ b/dCommon/GeneralUtils.cpp @@ -4,6 +4,8 @@ #include #include #include +#include +#include template inline size_t MinSize(size_t size, const std::basic_string_view& string) { @@ -50,6 +52,7 @@ bool _IsSuffixChar(uint8_t c) { bool GeneralUtils::_NextUTF8Char(std::string_view& slice, uint32_t& out) { size_t rem = slice.length(); + if (slice.empty()) return false; const uint8_t* bytes = (const uint8_t*)&slice.front(); if (rem > 0) { uint8_t first = bytes[0]; @@ -238,7 +241,7 @@ std::vector GeneralUtils::SplitString(std::wstring& str, wchar_t d return vector; } -std::vector GeneralUtils::SplitString(std::u16string& str, char16_t delimiter) { +std::vector GeneralUtils::SplitString(const std::u16string& str, char16_t delimiter) { std::vector vector = std::vector(); std::u16string current; @@ -289,51 +292,34 @@ std::u16string GeneralUtils::ReadWString(RakNet::BitStream* inStream) { return string; } -#ifdef _WIN32 -#define WIN32_LEAN_AND_MEAN -#include - -std::vector GeneralUtils::GetFileNamesFromFolder(const std::string& folder) { - std::vector names; - std::string search_path = folder + "/*.*"; - WIN32_FIND_DATA fd; - HANDLE hFind = ::FindFirstFile(search_path.c_str(), &fd); - if (hFind != INVALID_HANDLE_VALUE) { - do { - if (!(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { - names.push_back(fd.cFileName); - } - } while (::FindNextFile(hFind, &fd)); - ::FindClose(hFind); - } - return names; -} -#else -#include -#include -#include -#include -#include -#include - -std::vector GeneralUtils::GetFileNamesFromFolder(const std::string& folder) { - std::vector names; - struct dirent* entry; - DIR* dir = opendir(folder.c_str()); - if (dir == NULL) { - return names; +std::vector GeneralUtils::GetSqlFileNamesFromFolder(const std::string& folder) { + // Because we dont know how large the initial number before the first _ is we need to make it a map like so. + std::map filenames{}; + for (auto& t : std::filesystem::directory_iterator(folder)) { + auto filename = t.path().filename().string(); + auto index = std::stoi(GeneralUtils::SplitString(filename, '_').at(0)); + filenames.insert(std::make_pair(index, filename)); } - while ((entry = readdir(dir)) != NULL) { - std::string value(entry->d_name, strlen(entry->d_name)); - if (value == "." || value == "..") { - continue; + // Now sort the map by the oldest migration. + std::vector sortedFiles{}; + auto fileIterator = filenames.begin(); + std::map::iterator oldest = filenames.begin(); + while (!filenames.empty()) { + if (fileIterator == filenames.end()) { + sortedFiles.push_back(oldest->second); + filenames.erase(oldest); + fileIterator = filenames.begin(); + oldest = filenames.begin(); + continue; } - names.push_back(value); + if (oldest->first > fileIterator->first) oldest = fileIterator; + fileIterator++; } - closedir(dir); - - return names; + return sortedFiles; +} + +bool GeneralUtils::TryParse(const std::string& x, const std::string& y, const std::string& z, NiPoint3& dst) { + return TryParse(x.c_str(), dst.x) && TryParse(y.c_str(), dst.y) && TryParse(z.c_str(), dst.z); } -#endif diff --git a/dCommon/GeneralUtils.h b/dCommon/GeneralUtils.h index 898616d2..e9e20ba0 100644 --- a/dCommon/GeneralUtils.h +++ b/dCommon/GeneralUtils.h @@ -10,8 +10,14 @@ #include #include #include +#include "NiPoint3.h" #include "Game.h" +#include "dLogger.h" + +enum eInventoryType : uint32_t; +enum class eObjectBits : size_t; +enum class eReplicaComponentType : uint32_t; /*! \file GeneralUtils.hpp @@ -61,9 +67,9 @@ namespace GeneralUtils { //! Sets a bit on a numerical value template - void SetBit(T& value, size_t index) { + inline void SetBit(T& value, eObjectBits bits) { static_assert(std::is_arithmetic::value, "Not an arithmetic type"); - + auto index = static_cast(bits); if (index > (sizeof(T) * 8) - 1) { return; } @@ -73,9 +79,9 @@ namespace GeneralUtils { //! Clears a bit on a numerical value template - void ClearBit(T& value, size_t index) { + inline void ClearBit(T& value, eObjectBits bits) { static_assert(std::is_arithmetic::value, "Not an arithmetic type"); - + auto index = static_cast(bits); if (index > (sizeof(T) * 8 - 1)) { return; } @@ -105,40 +111,17 @@ namespace GeneralUtils { */ bool CheckBit(int64_t value, uint32_t index); - // MARK: Random Number Generation - - //! Generates a random number - /*! - \param min The minimum the generate from - \param max The maximum to generate to - */ - template - inline T GenerateRandomNumber(std::size_t min, std::size_t max) { - // Make sure it is a numeric type - static_assert(std::is_arithmetic::value, "Not an arithmetic type"); - - if constexpr (std::is_integral_v) { // constexpr only necessary on first statement - std::uniform_int_distribution distribution(min, max); - return distribution(Game::randomEngine); - } else if (std::is_floating_point_v) { - std::uniform_real_distribution distribution(min, max); - return distribution(Game::randomEngine); - } - - return T(); - } - bool ReplaceInString(std::string& str, const std::string& from, const std::string& to); std::u16string ReadWString(RakNet::BitStream* inStream); std::vector SplitString(std::wstring& str, wchar_t delimiter); - std::vector SplitString(std::u16string& str, char16_t delimiter); + std::vector SplitString(const std::u16string& str, char16_t delimiter); std::vector SplitString(const std::string& str, char delimiter); - std::vector GetFileNamesFromFolder(const std::string& folder); + std::vector GetSqlFileNamesFromFolder(const std::string& folder); template T Parse(const char* value); @@ -173,6 +156,16 @@ namespace GeneralUtils { return std::stoull(value); } + template <> + inline eInventoryType Parse(const char* value) { + return static_cast(std::stoul(value)); + } + + template <> + inline eReplicaComponentType Parse(const char* value) { + return static_cast(std::stoul(value)); + } + template bool TryParse(const char* value, T& dst) { try { @@ -194,6 +187,8 @@ namespace GeneralUtils { return TryParse(value.c_str(), dst); } + bool TryParse(const std::string& x, const std::string& y, const std::string& z, NiPoint3& dst); + template std::u16string to_u16string(T value) { return GeneralUtils::ASCIIToUTF16(std::to_string(value)); @@ -205,4 +200,42 @@ namespace GeneralUtils { std::hash h; s ^= h(v) + 0x9e3779b9 + (s << 6) + (s >> 2); } + + // MARK: Random Number Generation + + //! Generates a random number + /*! + \param min The minimum the generate from + \param max The maximum to generate to + */ + template + inline T GenerateRandomNumber(std::size_t min, std::size_t max) { + // Make sure it is a numeric type + static_assert(std::is_arithmetic::value, "Not an arithmetic type"); + + if constexpr (std::is_integral_v) { // constexpr only necessary on first statement + std::uniform_int_distribution distribution(min, max); + return distribution(Game::randomEngine); + } else if (std::is_floating_point_v) { + std::uniform_real_distribution distribution(min, max); + return distribution(Game::randomEngine); + } + + return T(); + } + +// on Windows we need to undef these or else they conflict with our numeric limits calls +// DEVELOPERS DEVELOPERS DEVELOPERS DEVELOPERS DEVELOPERS DEVELOPERS DEVELOPERS DEVELOPERS +#ifdef _WIN32 +#undef min +#undef max +#endif + + template + inline T GenerateRandomNumber() { + // Make sure it is a numeric type + static_assert(std::is_arithmetic::value, "Not an arithmetic type"); + + return GenerateRandomNumber(std::numeric_limits::min(), std::numeric_limits::max()); + } } diff --git a/dCommon/LDFFormat.cpp b/dCommon/LDFFormat.cpp index 767ec81a..cb921842 100644 --- a/dCommon/LDFFormat.cpp +++ b/dCommon/LDFFormat.cpp @@ -3,122 +3,174 @@ // Custom Classes #include "GeneralUtils.h" +#include "Game.h" +#include "dLogger.h" + // C++ -#include +#include #include +using LDFKey = std::string_view; +using LDFTypeAndValue = std::string_view; + +using LDFType = std::string_view; +using LDFValue = std::string_view; + //! Returns a pointer to a LDFData value based on string format -LDFBaseData* LDFBaseData::DataFromString(const std::string& format) { +LDFBaseData* LDFBaseData::DataFromString(const std::string_view& format) { + // A valid LDF must be at least 3 characters long (=0:) is the shortest valid LDF (empty UTF-16 key with no initial value) + if (format.empty() || format.length() <= 2) return nullptr; + auto equalsPosition = format.find('='); + // You can have an empty key, just make sure the type and value might exist + if (equalsPosition == std::string::npos || equalsPosition == (format.size() - 1)) return nullptr; - // First, check the format - std::istringstream ssFormat(format); - std::string token; + std::pair keyValue; + keyValue.first = format.substr(0, equalsPosition); + keyValue.second = format.substr(equalsPosition + 1, format.size()); - std::vector keyValueArray; - while (std::getline(ssFormat, token, '=')) { - keyValueArray.push_back(token); + std::u16string key = GeneralUtils::ASCIIToUTF16(keyValue.first); + + auto colonPosition = keyValue.second.find(':'); + + // If : is the first thing after an =, then this is an invalid LDF since + // we dont have a type to use. + if (colonPosition == std::string::npos || colonPosition == 0) return nullptr; + + std::pair ldfTypeAndValue; + ldfTypeAndValue.first = keyValue.second.substr(0, colonPosition); + ldfTypeAndValue.second = keyValue.second.substr(colonPosition + 1, keyValue.second.size()); + + // Only allow empty values for string values. + if (ldfTypeAndValue.second.size() == 0 && !(ldfTypeAndValue.first == "0" || ldfTypeAndValue.first == "13")) return nullptr; + + eLDFType type; + char* storage; + try { + type = static_cast(strtol(ldfTypeAndValue.first.data(), &storage, 10)); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Attempted to process invalid ldf type (%s) from string (%s)", ldfTypeAndValue.first.data(), format.data()); + return nullptr; } - if (keyValueArray.size() == 2) { - std::u16string key = GeneralUtils::ASCIIToUTF16(keyValueArray[0]); + LDFBaseData* returnValue = nullptr; + switch (type) { + case LDF_TYPE_UTF_16: { + std::u16string data = GeneralUtils::UTF8ToUTF16(ldfTypeAndValue.second); + returnValue = new LDFData(key, data); + break; + } - std::vector dataArray; - std::istringstream ssData(keyValueArray[1]); - while (std::getline(ssData, token, ':')) { - dataArray.push_back(token); + case LDF_TYPE_S32: { + try { + int32_t data = static_cast(strtoul(ldfTypeAndValue.second.data(), &storage, 10)); + returnValue = new LDFData(key, data); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid int32 value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + return nullptr; } + break; + } - if (dataArray.size() > 2) { // hacky fix for strings with colons in them - std::vector newDataArray; - newDataArray.push_back(dataArray[0]); - std::string value = ""; - for (size_t i = 1; i < dataArray.size(); ++i) { - value += dataArray[i] + ':'; - } - value.pop_back(); // remove last colon - newDataArray.push_back(value); - dataArray = newDataArray; + case LDF_TYPE_FLOAT: { + try { + float data = strtof(ldfTypeAndValue.second.data(), &storage); + returnValue = new LDFData(key, data); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid float value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + return nullptr; } + break; + } - if ((dataArray[0] == "0" || dataArray[0] == "13") && dataArray.size() == 1) { - dataArray.push_back(""); + case LDF_TYPE_DOUBLE: { + try { + double data = strtod(ldfTypeAndValue.second.data(), &storage); + returnValue = new LDFData(key, data); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid double value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + return nullptr; } + break; + } - if (dataArray.size() == 2) { - eLDFType type = static_cast(stoi(dataArray[0])); + case LDF_TYPE_U32: + { + uint32_t data; - switch (type) { - case LDF_TYPE_UTF_16: { - std::u16string data = GeneralUtils::UTF8ToUTF16(dataArray[1]); - return new LDFData(key, data); - } - - case LDF_TYPE_S32: { - int32_t data = static_cast(stoull(dataArray[1])); - return new LDFData(key, data); - } - - case LDF_TYPE_FLOAT: { - float data = static_cast(stof(dataArray[1])); - return new LDFData(key, data); - } - - case LDF_TYPE_DOUBLE: { - double data = static_cast(stod(dataArray[1])); - return new LDFData(key, data); - } - - case LDF_TYPE_U32: - { - uint32_t data; - - if (dataArray[1] == "true") { - data = 1; - } else if (dataArray[1] == "false") { - data = 0; - } else { - data = static_cast(stoul(dataArray[1])); - } - - return new LDFData(key, data); - } - - case LDF_TYPE_BOOLEAN: { - bool data; - - if (dataArray[1] == "true") { - data = true; - } else if (dataArray[1] == "false") { - data = false; - } else { - data = static_cast(stoi(dataArray[1])); - } - - return new LDFData(key, data); - } - - case LDF_TYPE_U64: { - uint64_t data = static_cast(stoull(dataArray[1])); - return new LDFData(key, data); - } - - case LDF_TYPE_OBJID: { - LWOOBJID data = static_cast(stoll(dataArray[1])); - return new LDFData(key, data); - } - - case LDF_TYPE_UTF_8: { - std::string data = dataArray[1]; - return new LDFData(key, data); - } - - case LDF_TYPE_UNKNOWN: { + if (ldfTypeAndValue.second == "true") { + data = 1; + } else if (ldfTypeAndValue.second == "false") { + data = 0; + } else { + try { + data = static_cast(strtoul(ldfTypeAndValue.second.data(), &storage, 10)); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid uint32 value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); return nullptr; } - } } + + returnValue = new LDFData(key, data); + break; } - return nullptr; + case LDF_TYPE_BOOLEAN: { + bool data; + if (ldfTypeAndValue.second == "true") { + data = true; + } else if (ldfTypeAndValue.second == "false") { + data = false; + } else { + try { + data = static_cast(strtol(ldfTypeAndValue.second.data(), &storage, 10)); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid bool value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + return nullptr; + } + } + + returnValue = new LDFData(key, data); + break; + } + + case LDF_TYPE_U64: { + try { + uint64_t data = static_cast(strtoull(ldfTypeAndValue.second.data(), &storage, 10)); + returnValue = new LDFData(key, data); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid uint64 value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + return nullptr; + } + break; + } + + case LDF_TYPE_OBJID: { + try { + LWOOBJID data = static_cast(strtoll(ldfTypeAndValue.second.data(), &storage, 10)); + returnValue = new LDFData(key, data); + } catch (std::exception) { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid LWOOBJID value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + return nullptr; + } + break; + } + + case LDF_TYPE_UTF_8: { + std::string data = ldfTypeAndValue.second.data(); + returnValue = new LDFData(key, data); + break; + } + + case LDF_TYPE_UNKNOWN: { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid unknown value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data()); + break; + } + + default: { + Game::logger->Log("LDFFormat", "Warning: Attempted to process invalid LDF type (%d) from string (%s)", type, format.data()); + break; + } + } + return returnValue; } diff --git a/dCommon/LDFFormat.h b/dCommon/LDFFormat.h index 9b62efa7..0921d04c 100644 --- a/dCommon/LDFFormat.h +++ b/dCommon/LDFFormat.h @@ -1,4 +1,5 @@ -#pragma once +#ifndef __LDFFORMAT__H__ +#define __LDFFORMAT__H__ // Custom Classes #include "dCommonVars.h" @@ -6,18 +7,12 @@ // C++ #include +#include #include // RakNet +#include "BitStream.h" -#include "../thirdparty/raknet/Source/BitStream.h" - -/*! - \file LDFFormat.hpp - \brief A collection of LDF format classes - */ - - //! An enum for LDF Data Types enum eLDFType { LDF_TYPE_UNKNOWN = -1, //!< Unknown data type LDF_TYPE_UTF_16 = 0, //!< UTF-16 wstring data type @@ -31,36 +26,21 @@ enum eLDFType { LDF_TYPE_UTF_8 = 13, //!< UTF-8 string data type }; -//! A base class for the LDF data class LDFBaseData { public: - //! Destructor - virtual ~LDFBaseData(void) {} + virtual ~LDFBaseData() {} - //! Writes the data to a packet - /*! - \param packet The packet - */ virtual void WriteToPacket(RakNet::BitStream* packet) = 0; - //! Gets the key - /*! - \return The key - */ - virtual const std::u16string& GetKey(void) = 0; + virtual const std::u16string& GetKey() = 0; - //! Gets the value type - /*! - \return The value type - */ - virtual eLDFType GetValueType(void) = 0; + virtual eLDFType GetValueType() = 0; - //! Gets a string from the key/value pair - /*! - \param includeKey Whether or not to include the key in the data - \param includeTypeId Whether or not to include the type id in the data - \return The string representation of the data + /** Gets a string from the key/value pair + * @param includeKey Whether or not to include the key in the data + * @param includeTypeId Whether or not to include the type id in the data + * @return The string representation of the data */ virtual std::string GetString(bool includeKey = true, bool includeTypeId = true) = 0; @@ -68,19 +48,15 @@ public: virtual LDFBaseData* Copy() = 0; - // MARK: Functions - - //! Returns a pointer to a LDFData value based on string format - /*! - \param format The format + /** + * Given an input string, return the data as a LDF key. */ - static LDFBaseData* DataFromString(const std::string& format); + static LDFBaseData* DataFromString(const std::string_view& format); }; -//! A structure for an LDF key-value pair template -class LDFData : public LDFBaseData { +class LDFData: public LDFBaseData { private: std::u16string key; T value; @@ -164,15 +140,11 @@ public: if (includeKey) { const std::string& sKey = GeneralUtils::UTF16ToWTF8(this->key, this->key.size()); - - stream << sKey << "="; + stream << sKey << '='; } if (includeTypeId) { - const std::string& sType = std::to_string(this->GetValueType()); - - - stream << sType << ":"; + stream << this->GetValueType() << ':'; } const std::string& sData = this->GetValueString(); @@ -234,20 +206,18 @@ inline void LDFData::WriteValue(RakNet::BitStream* packet) { } } -// MARK: String Data -template<> inline std::string LDFData::GetValueString(void) { - //std::string toReturn(this->value.begin(), this->value.end()); - //return toReturn; - +template<> inline std::string LDFData::GetValueString() { return GeneralUtils::UTF16ToWTF8(this->value, this->value.size()); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } +template<> inline std::string LDFData::GetValueString() { return std::to_string(this->value); } -template<> inline std::string LDFData::GetValueString(void) { return this->value; } +template<> inline std::string LDFData::GetValueString() { return this->value; } + +#endif //!__LDFFORMAT__H__ diff --git a/dCommon/NiPoint3.cpp b/dCommon/NiPoint3.cpp index a539c4df..20780815 100644 --- a/dCommon/NiPoint3.cpp +++ b/dCommon/NiPoint3.cpp @@ -128,6 +128,14 @@ NiPoint3 NiPoint3::operator+(const NiPoint3& point) const { return NiPoint3(this->x + point.x, this->y + point.y, this->z + point.z); } +//! Operator for addition of vectors +NiPoint3& NiPoint3::operator+=(const NiPoint3& point) { + this->x += point.x; + this->y += point.y; + this->z += point.z; + return *this; +} + //! Operator for subtraction of vectors NiPoint3 NiPoint3::operator-(const NiPoint3& point) const { return NiPoint3(this->x - point.x, this->y - point.y, this->z - point.z); diff --git a/dCommon/NiPoint3.h b/dCommon/NiPoint3.h index 6c0b9d3d..c956b654 100644 --- a/dCommon/NiPoint3.h +++ b/dCommon/NiPoint3.h @@ -135,6 +135,9 @@ public: //! Operator for addition of vectors NiPoint3 operator+(const NiPoint3& point) const; + //! Operator for addition of vectors + NiPoint3& operator+=(const NiPoint3& point); + //! Operator for subtraction of vectors NiPoint3 operator-(const NiPoint3& point) const; diff --git a/dCommon/PermissionMap.h b/dCommon/PermissionMap.h deleted file mode 100644 index 8c271f17..00000000 --- a/dCommon/PermissionMap.h +++ /dev/null @@ -1,42 +0,0 @@ -#pragma once - -#include - -/** - * Bitmap of permissions and restrictions for characters. - */ -enum class PermissionMap : uint64_t -{ - /** - * Reserved for future use, bit 0-3. - */ - - /** - * The character has restricted trade acccess, bit 4. - */ - RestrictedTradeAccess = 0x1 << 4, - - /** - * The character has restricted mail access, bit 5. - */ - RestrictedMailAccess = 0x1 << 5, - - /** - * The character has restricted chat access, bit 6. - */ - RestrictedChatAccess = 0x1 << 6, - - // - // Combined permissions - // - - /** - * The character is marked as 'old', restricted from trade and mail. - */ - Old = RestrictedTradeAccess | RestrictedMailAccess, - - /** - * The character is soft banned, restricted from trade, mail, and chat. - */ - SoftBanned = RestrictedTradeAccess | RestrictedMailAccess | RestrictedChatAccess, -}; diff --git a/dPhysics/Singleton.h b/dCommon/Singleton.h similarity index 100% rename from dPhysics/Singleton.h rename to dCommon/Singleton.h diff --git a/dCommon/ZCompression.cpp b/dCommon/ZCompression.cpp index 70f23500..d5d4c126 100644 --- a/dCommon/ZCompression.cpp +++ b/dCommon/ZCompression.cpp @@ -1,7 +1,5 @@ #include "ZCompression.h" -#ifndef _WIN32 - #include namespace ZCompression { @@ -27,7 +25,6 @@ namespace ZCompression { } deflateEnd(&zInfo); // zlib function return(nRet); - } int32_t Decompress(const uint8_t* abSrc, int32_t nLenSrc, uint8_t* abDst, int32_t nLenDst, int32_t& nErr) { @@ -48,26 +45,6 @@ namespace ZCompression { } inflateEnd(&zInfo); // zlib function return(nRet); - - /* - z_stream zInfo = { 0 }; - zInfo.total_in = zInfo.avail_in = nLenSrc; - zInfo.total_out = zInfo.avail_out = nLenDst; - zInfo.next_in = const_cast(abSrc); - zInfo.next_out = const_cast(abDst); - - int nRet = -1; - nErr = inflateInit(&zInfo); // zlib function - if (nErr == Z_OK) { - nErr = inflate(&zInfo, Z_FINISH); // zlib function - if (nErr == Z_STREAM_END) { - nRet = zInfo.total_out; - } - } - inflateEnd(&zInfo); // zlib function - return(nRet); // -1 or len of output - */ } } -#endif diff --git a/dCommon/ZCompression.h b/dCommon/ZCompression.h index 7b987cfa..84e8a9b4 100644 --- a/dCommon/ZCompression.h +++ b/dCommon/ZCompression.h @@ -2,16 +2,17 @@ #include -#include "dPlatforms.h" - -#ifndef DARKFLAME_PLATFORM_WIN32 - namespace ZCompression { int32_t GetMaxCompressedLength(int32_t nLenSrc); int32_t Compress(const uint8_t* abSrc, int32_t nLenSrc, uint8_t* abDst, int32_t nLenDst); int32_t Decompress(const uint8_t* abSrc, int32_t nLenSrc, uint8_t* abDst, int32_t nLenDst, int32_t& nErr); + + /** + * @brief Max size of an inflated sd0 zlib chunk + * + */ + constexpr uint32_t MAX_SD0_CHUNK_SIZE = 1024 * 256; } -#endif diff --git a/dCommon/dClient/AssetManager.cpp b/dCommon/dClient/AssetManager.cpp new file mode 100644 index 00000000..00cedba9 --- /dev/null +++ b/dCommon/dClient/AssetManager.cpp @@ -0,0 +1,183 @@ +#include + +#include "AssetManager.h" +#include "Game.h" +#include "dLogger.h" + +#include + +AssetManager::AssetManager(const std::filesystem::path& path) { + if (!std::filesystem::is_directory(path)) { + throw std::runtime_error("Attempted to load asset bundle (" + path.string() + ") however it is not a valid directory."); + } + + m_Path = path; + + if (std::filesystem::exists(m_Path / "client") && std::filesystem::exists(m_Path / "versions")) { + m_AssetBundleType = eAssetBundleType::Packed; + + m_RootPath = m_Path; + m_ResPath = (m_Path / "client" / "res"); + } else if (std::filesystem::exists(m_Path / ".." / "versions") && std::filesystem::exists(m_Path / "res")) { + m_AssetBundleType = eAssetBundleType::Packed; + + m_RootPath = (m_Path / ".."); + m_ResPath = (m_Path / "res"); + } else if (std::filesystem::exists(m_Path / "pack") && std::filesystem::exists(m_Path / ".." / ".." / "versions")) { + m_AssetBundleType = eAssetBundleType::Packed; + + m_RootPath = (m_Path / ".." / ".."); + m_ResPath = m_Path; + } else if ((std::filesystem::exists(m_Path / "res" / "cdclient.fdb") || std::filesystem::exists(m_Path / "res" / "CDServer.sqlite")) && !std::filesystem::exists(m_Path / "res" / "pack")) { + m_AssetBundleType = eAssetBundleType::Unpacked; + + m_ResPath = (m_Path / "res"); + } else if ((std::filesystem::exists(m_Path / "cdclient.fdb") || std::filesystem::exists(m_Path / "CDServer.sqlite")) && !std::filesystem::exists(m_Path / "pack")) { + m_AssetBundleType = eAssetBundleType::Unpacked; + + m_ResPath = m_Path; + } + + if (m_AssetBundleType == eAssetBundleType::None) { + throw std::runtime_error("Failed to identify client type, cannot read client data."); + } + + switch (m_AssetBundleType) { + case eAssetBundleType::Packed: { + this->LoadPackIndex(); + break; + } + case eAssetBundleType::None: + case eAssetBundleType::Unpacked: { + break; + } + } +} + +void AssetManager::LoadPackIndex() { + m_PackIndex = new PackIndex(m_RootPath); +} + +std::filesystem::path AssetManager::GetResPath() { + return m_ResPath; +} + +eAssetBundleType AssetManager::GetAssetBundleType() { + return m_AssetBundleType; +} + +bool AssetManager::HasFile(const char* name) { + auto fixedName = std::string(name); + std::transform(fixedName.begin(), fixedName.end(), fixedName.begin(), [](uint8_t c) { return std::tolower(c); }); + + // Special case for unpacked client have BrickModels in upper case + if (this->m_AssetBundleType == eAssetBundleType::Unpacked) GeneralUtils::ReplaceInString(fixedName, "brickmodels", "BrickModels"); + + std::replace(fixedName.begin(), fixedName.end(), '\\', '/'); + if (std::filesystem::exists(m_ResPath / fixedName)) return true; + + if (this->m_AssetBundleType == eAssetBundleType::Unpacked) return false; + + std::replace(fixedName.begin(), fixedName.end(), '/', '\\'); + if (fixedName.rfind("client\\res\\", 0) != 0) fixedName = "client\\res\\" + fixedName; + + uint32_t crc = crc32b(0xFFFFFFFF, (uint8_t*)fixedName.c_str(), fixedName.size()); + crc = crc32b(crc, (Bytef*)"\0\0\0\0", 4); + + for (const auto& item : this->m_PackIndex->GetPackFileIndices()) { + if (item.m_Crc == crc) { + return true; + } + } + + return false; +} + +bool AssetManager::GetFile(const char* name, char** data, uint32_t* len) { + auto fixedName = std::string(name); + std::transform(fixedName.begin(), fixedName.end(), fixedName.begin(), [](uint8_t c) { return std::tolower(c); }); + std::replace(fixedName.begin(), fixedName.end(), '\\', '/'); // On the off chance someone has the wrong slashes, force forward slashes + + // Special case for unpacked client have BrickModels in upper case + if (this->m_AssetBundleType == eAssetBundleType::Unpacked) GeneralUtils::ReplaceInString(fixedName, "brickmodels", "BrickModels"); + + if (std::filesystem::exists(m_ResPath / fixedName)) { + FILE* file; +#ifdef _WIN32 + fopen_s(&file, (m_ResPath / fixedName).string().c_str(), "rb"); +#elif __APPLE__ + // macOS has 64bit file IO by default + file = fopen((m_ResPath / fixedName).string().c_str(), "rb"); +#else + file = fopen64((m_ResPath / fixedName).string().c_str(), "rb"); +#endif + fseek(file, 0, SEEK_END); + *len = ftell(file); + *data = (char*)malloc(*len); + fseek(file, 0, SEEK_SET); + int32_t readInData = fread(*data, sizeof(uint8_t), *len, file); + fclose(file); + + return true; + } + + if (this->m_AssetBundleType == eAssetBundleType::Unpacked) return false; + + // The crc in side of the pack always uses backslashes, so we need to convert them again... + std::replace(fixedName.begin(), fixedName.end(), '/', '\\'); + if (fixedName.rfind("client\\res\\", 0) != 0) { + fixedName = "client\\res\\" + fixedName; + } + int32_t packIndex = -1; + uint32_t crc = crc32b(0xFFFFFFFF, (uint8_t*)fixedName.c_str(), fixedName.size()); + crc = crc32b(crc, (Bytef*)"\0\0\0\0", 4); + + for (const auto& item : this->m_PackIndex->GetPackFileIndices()) { + if (item.m_Crc == crc) { + packIndex = item.m_PackFileIndex; + crc = item.m_Crc; + break; + } + } + + if (packIndex == -1 || !crc) { + return false; + } + + auto packs = this->m_PackIndex->GetPacks(); + auto* pack = packs.at(packIndex); + + bool success = pack->ReadFileFromPack(crc, data, len); + + return success; +} + +AssetMemoryBuffer AssetManager::GetFileAsBuffer(const char* name) { + char* buf; + uint32_t len; + + bool success = this->GetFile(name, &buf, &len); + + return AssetMemoryBuffer(buf, len, success); +} + +uint32_t AssetManager::crc32b(uint32_t base, uint8_t* message, size_t l) { + size_t i, j; + uint32_t crc, msb; + + crc = base; + for (i = 0; i < l; i++) { + // xor next byte to upper bits of crc + crc ^= (((unsigned int)message[i]) << 24); + for (j = 0; j < 8; j++) { // Do eight times. + msb = crc >> 31; + crc <<= 1; + crc ^= (0 - msb) & 0x04C11DB7; + } + } + return crc; // don't complement crc on output +} + +AssetManager::~AssetManager() { + delete m_PackIndex; +} diff --git a/dCommon/dClient/AssetManager.h b/dCommon/dClient/AssetManager.h new file mode 100644 index 00000000..bc7e5ff7 --- /dev/null +++ b/dCommon/dClient/AssetManager.h @@ -0,0 +1,77 @@ +#pragma once + +#include +#include +#include +#include + +#include "Pack.h" +#include "PackIndex.h" + +enum class eAssetBundleType { + None, + Unpacked, + Packed +}; + +struct AssetMemoryBuffer : std::streambuf { + char* m_Base; + bool m_Success; + + AssetMemoryBuffer(char* base, std::ptrdiff_t n, bool success) { + m_Base = base; + m_Success = success; + if (!m_Success) return; + this->setg(base, base, base + n); + } + + pos_type seekpos(pos_type sp, std::ios_base::openmode which) override { + return seekoff(sp - pos_type(off_type(0)), std::ios_base::beg, which); + } + + pos_type seekoff(off_type off, + std::ios_base::seekdir dir, + std::ios_base::openmode which = std::ios_base::in) override { + if (dir == std::ios_base::cur) + gbump(off); + else if (dir == std::ios_base::end) + setg(eback(), egptr() + off, egptr()); + else if (dir == std::ios_base::beg) + setg(eback(), eback() + off, egptr()); + return gptr() - eback(); + } + + void close() { + delete m_Base; + } +}; + +class AssetManager { +public: + AssetManager(const std::filesystem::path& path); + ~AssetManager(); + + std::filesystem::path GetResPath(); + eAssetBundleType GetAssetBundleType(); + + bool HasFile(const char* name); + bool GetFile(const char* name, char** data, uint32_t* len); + AssetMemoryBuffer GetFileAsBuffer(const char* name); + +private: + void LoadPackIndex(); + + // Modified crc algorithm (mpeg2) + // Reference: https://stackoverflow.com/questions/54339800/how-to-modify-crc-32-to-crc-32-mpeg-2 + inline uint32_t crc32b(uint32_t base, uint8_t* message, size_t l); + + bool m_SuccessfullyLoaded; + + std::filesystem::path m_Path; + std::filesystem::path m_RootPath; + std::filesystem::path m_ResPath; + + eAssetBundleType m_AssetBundleType = eAssetBundleType::None; + + PackIndex* m_PackIndex; +}; diff --git a/dCommon/dClient/CMakeLists.txt b/dCommon/dClient/CMakeLists.txt new file mode 100644 index 00000000..69bb1712 --- /dev/null +++ b/dCommon/dClient/CMakeLists.txt @@ -0,0 +1,6 @@ +set(DCOMMON_DCLIENT_SOURCES + "PackIndex.cpp" + "Pack.cpp" + "AssetManager.cpp" + PARENT_SCOPE +) diff --git a/dCommon/dClient/Pack.cpp b/dCommon/dClient/Pack.cpp new file mode 100644 index 00000000..11345fc9 --- /dev/null +++ b/dCommon/dClient/Pack.cpp @@ -0,0 +1,119 @@ +#include "Pack.h" + +#include "BinaryIO.h" +#include "ZCompression.h" + +Pack::Pack(const std::filesystem::path& filePath) { + m_FilePath = filePath; + + if (!std::filesystem::exists(filePath)) { + return; + } + + m_FileStream = std::ifstream(filePath, std::ios::in | std::ios::binary); + + m_FileStream.read(m_Version, 7); + + m_FileStream.seekg(-8, std::ios::end); // move file pointer to 8 bytes before the end (location of the address of the record count) + + uint32_t recordCountPos = 0; + BinaryIO::BinaryRead(m_FileStream, recordCountPos); + + m_FileStream.seekg(recordCountPos, std::ios::beg); + + BinaryIO::BinaryRead(m_FileStream, m_RecordCount); + + for (int i = 0; i < m_RecordCount; i++) { + PackRecord record; + BinaryIO::BinaryRead(m_FileStream, record); + + m_Records.push_back(record); + } + + m_FileStream.close(); +} + +bool Pack::HasFile(uint32_t crc) { + for (const auto& record : m_Records) { + if (record.m_Crc == crc) { + return true; + } + } + + return false; +} + +bool Pack::ReadFileFromPack(uint32_t crc, char** data, uint32_t* len) { + // Time for some wacky C file reading for speed reasons + + PackRecord pkRecord{}; + + for (const auto& record : m_Records) { + if (record.m_Crc == crc) { + pkRecord = record; + break; + } + } + + if (pkRecord.m_Crc == 0) return false; + + size_t pos = 0; + pos += pkRecord.m_FilePointer; + + bool isCompressed = (pkRecord.m_IsCompressed & 0xff) > 0; + auto inPackSize = isCompressed ? pkRecord.m_CompressedSize : pkRecord.m_UncompressedSize; + + FILE* file; +#ifdef _WIN32 + fopen_s(&file, m_FilePath.string().c_str(), "rb"); +#elif __APPLE__ + // macOS has 64bit file IO by default + file = fopen(m_FilePath.string().c_str(), "rb"); +#else + file = fopen64(m_FilePath.string().c_str(), "rb"); +#endif + + fseek(file, pos, SEEK_SET); + + if (!isCompressed) { + char* tempData = (char*)malloc(pkRecord.m_UncompressedSize); + int32_t readInData = fread(tempData, sizeof(uint8_t), pkRecord.m_UncompressedSize, file); + + *data = tempData; + *len = pkRecord.m_UncompressedSize; + fclose(file); + + return true; + } + + pos += 5; // skip header + + fseek(file, pos, SEEK_SET); + + char* decompressedData = (char*)malloc(pkRecord.m_UncompressedSize); + uint32_t currentReadPos = 0; + + while (true) { + if (currentReadPos >= pkRecord.m_UncompressedSize) break; + + uint32_t size; + int32_t readInData = fread(&size, sizeof(uint32_t), 1, file); + pos += 4; // Move pointer position 4 to the right + + char* chunk = (char*)malloc(size); + int32_t readInData2 = fread(chunk, sizeof(int8_t), size, file); + pos += size; // Move pointer position the amount of bytes read to the right + + int32_t err; + currentReadPos += ZCompression::Decompress((uint8_t*)chunk, size, reinterpret_cast(decompressedData + currentReadPos), ZCompression::MAX_SD0_CHUNK_SIZE, err); + + free(chunk); + } + + *data = decompressedData; + *len = pkRecord.m_UncompressedSize; + + fclose(file); + + return true; +} diff --git a/dCommon/dClient/Pack.h b/dCommon/dClient/Pack.h new file mode 100644 index 00000000..3e95b00a --- /dev/null +++ b/dCommon/dClient/Pack.h @@ -0,0 +1,38 @@ +#pragma once + +#include +#include +#include + +#pragma pack(push, 1) +struct PackRecord { + uint32_t m_Crc; + int32_t m_LowerCrc; + int32_t m_UpperCrc; + uint32_t m_UncompressedSize; + char m_UncompressedHash[32]; + uint32_t m_Padding1; + uint32_t m_CompressedSize; + char m_CompressedHash[32]; + uint32_t m_Padding2; + uint32_t m_FilePointer; + uint32_t m_IsCompressed; // u32 bool +}; +#pragma pack(pop) + +class Pack { +public: + Pack(const std::filesystem::path& filePath); + ~Pack() = default; + + bool HasFile(uint32_t crc); + bool ReadFileFromPack(uint32_t crc, char** data, uint32_t* len); +private: + std::ifstream m_FileStream; + std::filesystem::path m_FilePath; + + char m_Version[7]; + + uint32_t m_RecordCount; + std::vector m_Records; +}; diff --git a/dCommon/dClient/PackIndex.cpp b/dCommon/dClient/PackIndex.cpp new file mode 100644 index 00000000..5d96abeb --- /dev/null +++ b/dCommon/dClient/PackIndex.cpp @@ -0,0 +1,54 @@ +#include "PackIndex.h" +#include "BinaryIO.h" +#include "Game.h" +#include "dLogger.h" + +PackIndex::PackIndex(const std::filesystem::path& filePath) { + m_FileStream = std::ifstream(filePath / "versions" / "primary.pki", std::ios::in | std::ios::binary); + + BinaryIO::BinaryRead(m_FileStream, m_Version); + BinaryIO::BinaryRead(m_FileStream, m_PackPathCount); + + for (int i = 0; i < m_PackPathCount; i++) { + uint32_t stringLen = 0; + BinaryIO::BinaryRead(m_FileStream, stringLen); + + std::string path; + + for (int j = 0; j < stringLen; j++) { + char inChar; + BinaryIO::BinaryRead(m_FileStream, inChar); + + path += inChar; + } + + m_PackPaths.push_back(path); + } + + BinaryIO::BinaryRead(m_FileStream, m_PackFileIndexCount); + + for (int i = 0; i < m_PackFileIndexCount; i++) { + PackFileIndex packFileIndex; + BinaryIO::BinaryRead(m_FileStream, packFileIndex); + + m_PackFileIndices.push_back(packFileIndex); + } + + Game::logger->Log("PackIndex", "Loaded pack catalog with %i pack files and %i files", m_PackPaths.size(), m_PackFileIndices.size()); + + for (auto& item : m_PackPaths) { + std::replace(item.begin(), item.end(), '\\', '/'); + + auto* pack = new Pack(filePath / item); + + m_Packs.push_back(pack); + } + + m_FileStream.close(); +} + +PackIndex::~PackIndex() { + for (const auto* item : m_Packs) { + delete item; + } +} diff --git a/dCommon/dClient/PackIndex.h b/dCommon/dClient/PackIndex.h new file mode 100644 index 00000000..bf10b809 --- /dev/null +++ b/dCommon/dClient/PackIndex.h @@ -0,0 +1,40 @@ +#pragma once + +#include + +#include +#include +#include + +#include "Pack.h" + +#pragma pack(push, 1) +struct PackFileIndex { + uint32_t m_Crc; + int32_t m_LowerCrc; + int32_t m_UpperCrc; + uint32_t m_PackFileIndex; + uint32_t m_IsCompressed; // u32 bool? +}; +#pragma pack(pop) + +class PackIndex { +public: + PackIndex(const std::filesystem::path& filePath); + ~PackIndex(); + + const std::vector& GetPackPaths() { return m_PackPaths; } + const std::vector& GetPackFileIndices() { return m_PackFileIndices; } + const std::vector& GetPacks() { return m_Packs; } +private: + std::ifstream m_FileStream; + + uint32_t m_Version; + + uint32_t m_PackPathCount; + std::vector m_PackPaths; + uint32_t m_PackFileIndexCount; + std::vector m_PackFileIndices; + + std::vector m_Packs; +}; diff --git a/dCommon/dCommonVars.h b/dCommon/dCommonVars.h deleted file mode 100644 index 4c0e15fa..00000000 --- a/dCommon/dCommonVars.h +++ /dev/null @@ -1,650 +0,0 @@ -#pragma once - -#include -#include -#include -#include "../thirdparty/raknet/Source/BitStream.h" - -#pragma warning (disable:4251) //Disables SQL warnings - -typedef int RESTICKET; - -const int highFrameRate = 16; //60fps -const int mediumFramerate = 33; //30fps -const int lowFramerate = 66; //15fps - -//========== MACROS =========== - -#define CBITSTREAM RakNet::BitStream bitStream; -#define CINSTREAM RakNet::BitStream inStream(packet->data, packet->length, false); -#define CMSGHEADER PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_GAME_MSG); -#define SEND_PACKET Game::server->Send(&bitStream, sysAddr, false); -#define SEND_PACKET_BROADCAST Game::server->Send(&bitStream, UNASSIGNED_SYSTEM_ADDRESS, true); - -//=========== TYPEDEFS ========== - -typedef int32_t LOT; //!< A LOT -typedef int64_t LWOOBJID; //!< An object ID (should be unsigned actually but ok) -typedef int32_t TSkillID; //!< A skill ID -typedef uint32_t LWOCLONEID; //!< Used for Clone IDs -typedef uint16_t LWOMAPID; //!< Used for Map IDs -typedef uint16_t LWOINSTANCEID; //!< Used for Instance IDs -typedef uint32_t PROPERTYCLONELIST; //!< Used for Property Clone IDs - -typedef int32_t PetTamingPiece; //!< Pet Taming Pieces - -const LWOOBJID LWOOBJID_EMPTY = 0; //!< An empty object ID -const LOT LOT_NULL = -1; //!< A null LOT -const int32_t LOOTTYPE_NONE = 0; //!< No loot type available -const float SECONDARY_PRIORITY = 1.0f; //!< Secondary Priority -const uint32_t INVENTORY_INVALID = -1; //!< Invalid Inventory -const uint32_t INVENTORY_DEFAULT = -1; //!< Default Inventory -const uint32_t StatusChangeInfo = 0; //!< Status Change Info (???) -const uint32_t INVENTORY_MAX = 9999999; //!< The Maximum Inventory Size -const uint32_t LWOCLONEID_INVALID = -1; //!< Invalid LWOCLONEID -const uint16_t LWOINSTANCEID_INVALID = -1; //!< Invalid LWOINSTANCEID -const uint16_t LWOMAPID_INVALID = -1; //!< Invalid LWOMAPID -const uint64_t LWOZONEID_INVALID = 0; //!< Invalid LWOZONEID - -typedef std::set TSetObjID; - -const float PI = 3.14159f; - -#if defined(__unix) || defined(__APPLE__) -//For Linux: -typedef __int64_t __int64; -#endif - -//============ STRUCTS ============== - -struct LWOSCENEID { -public: - LWOSCENEID() { m_sceneID = -1; m_layerID = 0; } - LWOSCENEID(int sceneID) { m_sceneID = sceneID; m_layerID = 0; } - LWOSCENEID(int sceneID, unsigned int layerID) { m_sceneID = sceneID; m_layerID = layerID; } - - LWOSCENEID& operator=(const LWOSCENEID& rhs) { m_sceneID = rhs.m_sceneID; m_layerID = rhs.m_layerID; return *this; } - LWOSCENEID& operator=(const int rhs) { m_sceneID = rhs; m_layerID = 0; return *this; } - - bool operator<(const LWOSCENEID& rhs) const { return (m_sceneID < rhs.m_sceneID || (m_sceneID == rhs.m_sceneID && m_layerID < rhs.m_layerID)); } - bool operator<(const int rhs) const { return m_sceneID < rhs; } - - bool operator==(const LWOSCENEID& rhs) const { return (m_sceneID == rhs.m_sceneID && m_layerID == rhs.m_layerID); } - bool operator==(const int rhs) const { return m_sceneID == rhs; } - - const int GetSceneID() const { return m_sceneID; } - const unsigned int GetLayerID() const { return m_layerID; } - - void SetSceneID(const int sceneID) { m_sceneID = sceneID; } - void SetLayerID(const unsigned int layerID) { m_layerID = layerID; } - -private: - int m_sceneID; - unsigned int m_layerID; -}; - -struct LWOZONEID { -public: - const LWOMAPID& GetMapID() const { return m_MapID; } - const LWOINSTANCEID& GetInstanceID() const { return m_InstanceID; } - const LWOCLONEID& GetCloneID() const { return m_CloneID; } - - //In order: def constr, constr, assign op - LWOZONEID() { m_MapID = LWOMAPID_INVALID; m_InstanceID = LWOINSTANCEID_INVALID; m_CloneID = LWOCLONEID_INVALID; } - LWOZONEID(const LWOMAPID& mapID, const LWOINSTANCEID& instanceID, const LWOCLONEID& cloneID) { m_MapID = mapID; m_InstanceID = instanceID; m_CloneID = cloneID; } - LWOZONEID(const LWOZONEID& replacement) { *this = replacement; } - -private: - LWOMAPID m_MapID; //1000 for VE, 1100 for AG, etc... - LWOINSTANCEID m_InstanceID; //Instances host the same world, but on a different dWorld process. - LWOCLONEID m_CloneID; //To differentiate between "your property" and "my property". Always 0 for non-prop worlds. -}; - -const LWOSCENEID LWOSCENEID_INVALID = -1; - -struct LWONameValue { - uint32_t length = 0; //!< The length of the name - std::u16string name; //!< The name - - LWONameValue(void) {} - - LWONameValue(const std::u16string& name) { - this->name = name; - this->length = static_cast(name.length()); - } - - ~LWONameValue(void) {} -}; - -struct FriendData { -public: - bool isOnline = false; - bool isBestFriend = false; - bool isFTP = false; - LWOZONEID zoneID; - LWOOBJID friendID; - std::string friendName; - - void Serialize(RakNet::BitStream& bitStream) { - bitStream.Write(isOnline); - bitStream.Write(isBestFriend); - bitStream.Write(isFTP); - bitStream.Write(0); //??? - bitStream.Write(0); //??? - bitStream.Write(zoneID.GetMapID()); - bitStream.Write(zoneID.GetInstanceID()); - bitStream.Write(zoneID.GetCloneID()); - bitStream.Write(friendID); - - uint32_t maxSize = 33; - uint32_t size = static_cast(friendName.length()); - uint32_t remSize = static_cast(maxSize - size); - - if (size > maxSize) size = maxSize; - - for (uint32_t i = 0; i < size; ++i) { - bitStream.Write(static_cast(friendName[i])); - } - - for (uint32_t j = 0; j < remSize; ++j) { - bitStream.Write(static_cast(0)); - } - - bitStream.Write(0); //??? - bitStream.Write(0); //??? - } -}; - -struct Brick { - uint32_t designerID; - uint32_t materialID; -}; - -//This union is used by the behavior system -union suchar { - unsigned char usigned; - char svalue; -}; - -//=========== DLU ENUMS ============ - -enum eGameMasterLevel : int32_t { - GAME_MASTER_LEVEL_CIVILIAN = 0, // Normal player. - GAME_MASTER_LEVEL_FORUM_MODERATOR = 1, // No permissions on live servers. - GAME_MASTER_LEVEL_JUNIOR_MODERATOR = 2, // Can kick/mute and pull chat logs. - GAME_MASTER_LEVEL_MODERATOR = 3, // Can return lost items. - GAME_MASTER_LEVEL_SENIOR_MODERATOR = 4, // Can ban. - GAME_MASTER_LEVEL_LEAD_MODERATOR = 5, // Can approve properties. - GAME_MASTER_LEVEL_JUNIOR_DEVELOPER = 6, // Junior developer & future content team. Civilan on live. - GAME_MASTER_LEVEL_INACTIVE_DEVELOPER = 7, // Inactive developer, limited permissions. - GAME_MASTER_LEVEL_DEVELOPER = 8, // Active developer, full permissions on live. - GAME_MASTER_LEVEL_OPERATOR = 9 // Can shutdown server for restarts & updates. -}; - -//=========== LU ENUMS ============ - -//! An enum for object ID bits -enum eObjectBits : int32_t { - OBJECT_BIT_PERSISTENT = 32, //!< The 32 bit index - OBJECT_BIT_CLIENT = 46, //!< The 46 bit index - OBJECT_BIT_SPAWNED = 58, //!< The 58 bit index - OBJECT_BIT_CHARACTER = 60 //!< The 60 bit index -}; - -//! An enum for MatchUpdate types -enum eMatchUpdate : int { - MATCH_UPDATE_PLAYER_JOINED = 0, - MATCH_UPDATE_PLAYER_LEFT = 1, - MATCH_UPDATE_TIME = 3, - MATCH_UPDATE_TIME_START_DELAY = 4, - MATCH_UPDATE_PLAYER_READY = 5, - MATCH_UPDATE_PLAYER_UNREADY = 6 -}; - -//! An enum for camera cycling modes -enum eCyclingMode : uint32_t { - ALLOW_CYCLE_TEAMMATES, - DISALLOW_CYCLING -}; - -enum eCinematicEvent : uint32_t { - STARTED, - WAYPOINT, - ENDED, -}; - -//! An enum for character creation responses -enum eCreationResponse : uint8_t { - CREATION_RESPONSE_SUCCESS = 0, //!< The creation was successful - CREATION_RESPONSE_OBJECT_ID_UNAVAILABLE, //!< The Object ID can't be used - CREATION_RESPONSE_NAME_NOT_ALLOWED, //!< The name is not allowed - CREATION_RESPONSE_PREDEFINED_NAME_IN_USE, //!< The predefined name is already in use - CREATION_RESPONSE_CUSTOM_NAME_IN_USE //!< The custom name is already in use -}; - -//! An enum for login responses -enum eLoginResponse : uint8_t { - LOGIN_RESPONSE_GENERAL_FAILED = 0, - LOGIN_RESPONSE_SUCCESS = 1, - LOGIN_RESPONSE_BANNED = 2, - LOGIN_RESPONSE_PERMISSIONS_NOT_HIGH_ENOUGH = 5, - LOGIN_RESPONSE_WRONG_PASS_OR_USER = 6, - LOGIN_RESPONSE_ACCOUNT_LOCKED = 7 -}; - -//! An enum for character rename responses -enum eRenameResponse : uint8_t { - RENAME_RESPONSE_SUCCESS = 0, //!< The renaming was successful - RENAME_RESPONSE_UNKNOWN_ERROR, //!< There was an unknown error - RENAME_RESPONSE_NAME_UNAVAILABLE, //!< The name is unavailable - RENAME_RESPONSE_NAME_IN_USE //!< The name is already in use -}; - -//! A replica packet type -enum eReplicaPacketType { - PACKET_TYPE_CONSTRUCTION, //!< A construction packet - PACKET_TYPE_SERIALIZATION, //!< A serialization packet - PACKET_TYPE_DESTRUCTION //!< A destruction packet -}; - -enum ServerDisconnectIdentifiers { - SERVER_DISCON_UNKNOWN_SERVER_ERROR = 0, //!< Unknown server error - SERVER_DISCON_DUPLICATE_LOGIN = 4, //!< Used when another user with the same username is logged in (duplicate login) - SERVER_DISCON_SERVER_SHUTDOWN = 5, //!< Used when the server is shutdown - SERVER_DISCON_SERVER_MAP_LOAD_FAILURE = 6, //!< Used when the server cannot load a map - SERVER_DISCON_INVALID_SESSION_KEY = 7, //!< Used if the session is invalid - SERVER_DISCON_ACCOUNT_NOT_IN_PENDING_LIST = 8, //!< ??? - SERVER_DISCON_CHARACTER_NOT_FOUND = 9, //!< Used if a character that the server has is not found (i.e, corruption with user-player data) - SERVER_DISCON_CHARACTER_CORRUPTED = 10, //!< Similar to abovce - SERVER_DISCON_KICK = 11, //!< Used if the user is kicked from the server - SERVER_DISCON_FREE_TRIAL_EXPIRED = 12, //!< Used if the user's free trial expired - SERVER_DISCON_PLAY_SCHEDULE_TIME_DONE = 13 //!< Used if the user's play time is used up -}; - -//! The Behavior Types for use with the AI system -enum eCombatBehaviorTypes : uint32_t { - PASSIVE = 0, //!< The object is passive - AGGRESSIVE = 1, //!< The object is aggressive - PASSIVE_TURRET = 2, //!< The object is a passive turret - AGGRESSIVE_TURRET = 3 //!< The object is an aggressive turret -}; - -//! The Combat Role Type for use with the AI system -enum eCombatRoleType : uint32_t { - MELEE = 0, //!< Used for melee attacks - RANGED = 1, //!< Used for range attacks - SUPPORT = 2 //!< Used for support -}; - -//! The kill types for the Die packet -enum eKillType : uint32_t { - VIOLENT, - SILENT -}; - -//! The various world states used throughout the server -enum eObjectWorldState { - WORLDSTATE_INWORLD, //!< Probably used when the object is in the world - WORLDSTATE_ATTACHED, //!< Probably used when the object is attached to another object - WORLDSTATE_INVENTORY //!< Probably used when the object is in an inventory -}; - -//! The trigger stats (???) -enum eTriggerStat { - INVALID_STAT, //!< ??? - HEALTH, //!< Probably used for health - ARMOR, //!< Probably used for armor - IMAGINATION //!< Probably used for imagination -}; - -//! The trigger operations (???) -enum eTriggerOperator { - INVALID_OPER, //!< ??? - EQUAL, //!< ??? - NOT_EQUAL, //!< ??? - GREATER, //!< ??? - GREATER_EQUAL, //!< ??? - LESS, //!< ??? - LESS_EQUAL //!< ??? -}; - -//! The various build types -enum eBuildType { - BUILD_NOWHERE, //!< Used if something can't be built anywhere - BUILD_IN_WORLD, //!< Used if something can be built in the world - BUILD_ON_PROPERTY //!< Used if something can be build on a property -}; - -//! Quickbuild fail reasons -enum eFailReason : uint32_t { - REASON_NOT_GIVEN, - REASON_OUT_OF_IMAGINATION, - REASON_CANCELED_EARLY, - REASON_BUILD_ENDED -}; - -//! Terminate interaction type -enum eTerminateType : uint32_t { - RANGE, - USER, - FROM_INTERACTION -}; - -//! The combat state -enum eCombatState { - IDLE, //!< The AI is in an idle state - AGGRO, //!< The AI is in an aggressive state - TETHER, //!< The AI is being redrawn back to tether point - SPAWN, //!< The AI is spawning - DEAD //!< The AI is dead -}; - -enum eControlSceme { - SCHEME_A, - SCHEME_D, - SCHEME_GAMEPAD, - SCHEME_E, - SCHEME_FPS, - SCHEME_DRIVING, - SCHEME_TAMING, - SCHEME_MODULAR_BUILD, - SCHEME_WEAR_A_ROBOT //== freecam? -}; - -enum eStunState { - PUSH, - POP -}; - -enum eNotifyType { - NOTIFY_TYPE_SUCCESS, - NOTIFY_TYPE_QUIT, - NOTIFY_TYPE_FAILED, - NOTIFY_TYPE_BEGIN, - NOTIFY_TYPE_READY, - NOTIFY_TYPE_NAMINGPET -}; - -enum eReplicaComponentType : int32_t { - COMPONENT_TYPE_CONTROLLABLE_PHYSICS = 1, //!< The ControllablePhysics Component - COMPONENT_TYPE_RENDER = 2, //!< The Render Component - COMPONENT_TYPE_SIMPLE_PHYSICS = 3, //!< The SimplePhysics Component - COMPONENT_TYPE_CHARACTER = 4, //!< The Character Component - COMPONENT_TYPE_SCRIPT = 5, //!< The Script Component - COMPONENT_TYPE_BOUNCER = 6, //!< The Bouncer Component - COMPONENT_TYPE_BUFF = 7, //!< The Buff Component - COMPONENT_TYPE_SKILL = 9, //!< The Skill Component - COMPONENT_TYPE_ITEM = 11, //!< The Item Component - COMPONENT_TYPE_VENDOR = 16, //!< The Vendor Component - COMPONENT_TYPE_INVENTORY = 17, //!< The Inventory Component - COMPONENT_TYPE_SHOOTING_GALLERY = 19, //!< The Shooting Gallery Component - COMPONENT_TYPE_RIGID_BODY_PHANTOM_PHYSICS = 20, //!< The RigidBodyPhantomPhysics Component - COMPONENT_TYPE_COLLECTIBLE = 23, //!< The Collectible Component - COMPONENT_TYPE_MOVING_PLATFORM = 25, //!< The MovingPlatform Component - COMPONENT_TYPE_PET = 26, //!< The Pet Component - COMPONENT_TYPE_VEHICLE_PHYSICS = 30, //!< The VehiclePhysics Component - COMPONENT_TYPE_MOVEMENT_AI = 31, //!< The MovementAI Component - COMPONENT_TYPE_PROPERTY = 36, //!< The Property Component - COMPONENT_TYPE_SCRIPTED_ACTIVITY = 39, //!< The ScriptedActivity Component - COMPONENT_TYPE_PHANTOM_PHYSICS = 40, //!< The PhantomPhysics Component - COMPONENT_TYPE_MODEL = 42, //!< The Model Component - COMPONENT_TYPE_PROPERTY_ENTRANCE = 43, //!< The PhantomPhysics Component - COMPONENT_TYPE_PROPERTY_MANAGEMENT = 45, //!< The PropertyManagement Component - COMPONENT_TYPE_REBUILD = 48, //!< The Rebuild Component - COMPONENT_TYPE_SWITCH = 49, //!< The Switch Component - COMPONENT_TYPE_ZONE_CONTROL = 50, //!< The ZoneControl Component - COMPONENT_TYPE_PACKAGE = 53, //!< The Package Component - COMPONENT_TYPE_PLAYER_FLAG = 58, //!< The PlayerFlag Component - COMPONENT_TYPE_BASE_COMBAT_AI = 60, //!< The BaseCombatAI Component - COMPONENT_TYPE_MODULE_ASSEMBLY = 61, //!< The ModuleAssembly Component - COMPONENT_TYPE_PROPERTY_VENDOR = 65, //!< The PropertyVendor Component - COMPONENT_TYPE_ROCKET_LAUNCH = 67, //!< The RocketLaunch Component - COMPONENT_TYPE_RACING_CONTROL = 71, //!< The RacingControl Component - COMPONENT_TYPE_MISSION_OFFER = 73, //!< The MissionOffer Component - COMPONENT_TYPE_EXHIBIT = 75, //!< The Exhibit Component - COMPONENT_TYPE_RACING_STATS = 74, //!< The Racing Stats Component - COMPONENT_TYPE_SOUND_TRIGGER = 77, //!< The Sound Trigger Component - COMPONENT_TYPE_PROXIMITY_MONITOR = 78, //!< The Proximity Monitor Component - COMPONENT_TYPE_MISSION = 84, //!< The Mission Component - COMPONENT_TYPE_ROCKET_LAUNCH_LUP = 97, //!< The LUP Launchpad Componen - COMPONENT_TYPE_RAIL_ACTIVATOR = 104, //!< The Rail Activator Component - COMPONENT_TYPE_PLAYER_FORCED_MOVEMENT = 106, //!< The Player Forced Movement Component - COMPONENT_TYPE_POSSESSABLE = 108, //!< The Possessable Component - COMPONENT_TYPE_LEVEL_PROGRESSION = 109, //!< The Level Progression Component - COMPONENT_TYPE_POSSESSOR = 110, //!< The Possessor Component - COMPONENT_TYPE_BUILD_BORDER = 114, //!< The Build Border Component - COMPONENT_TYPE_DESTROYABLE = 1000, //!< The Destroyable Component -}; - -enum class UseItemResponse : uint32_t { - NoImaginationForPet = 1, - FailedPrecondition, - MountsNotAllowed -}; - -/** - * Represents the different types of inventories an entity may have - */ -enum eInventoryType : uint32_t { - ITEMS = 0, - VAULT_ITEMS, - BRICKS, - TEMP_ITEMS = 4, - MODELS, - TEMP_MODELS, - BEHAVIORS, - PROPERTY_DEEDS, - VENDOR_BUYBACK = 11, - HIDDEN = 12, //Used for missional items - VAULT_MODELS = 14, - ITEM_SETS, //internal - INVALID // made up, for internal use!!! -}; - -enum eRebuildState : uint32_t { - REBUILD_OPEN, - REBUILD_COMPLETED = 2, - REBUILD_RESETTING = 4, - REBUILD_BUILDING, - REBUILD_INCOMPLETE -}; - -/** - * The loot source's type. - */ -enum eLootSourceType : int32_t { - LOOT_SOURCE_NONE = 0, - LOOT_SOURCE_CHEST, - LOOT_SOURCE_MISSION, - LOOT_SOURCE_MAIL, - LOOT_SOURCE_CURRENCY, - LOOT_SOURCE_ACHIEVEMENT, - LOOT_SOURCE_TRADE, - LOOT_SOURCE_QUICKBUILD, - LOOT_SOURCE_DELETION, - LOOT_SOURCE_VENDOR, - LOOT_SOURCE_ACTIVITY, - LOOT_SOURCE_PICKUP, - LOOT_SOURCE_BRICK, - LOOT_SOURCE_PROPERTY, - LOOT_SOURCE_MODERATION, - LOOT_SOURCE_EXHIBIT, - LOOT_SOURCE_INVENTORY, - LOOT_SOURCE_CLAIMCODE, - LOOT_SOURCE_CONSUMPTION, - LOOT_SOURCE_CRAFTING, - LOOT_SOURCE_LEVEL_REWARD, - LOOT_SOURCE_RELOCATE -}; - -enum eGameActivities : uint32_t { - ACTIVITY_NONE, - ACTIVITY_QUICKBUILDING, - ACTIVITY_SHOOTING_GALLERY, - ACTIVITY_RACING, - ACTIVITY_PINBALL, - ACTIVITY_PET_TAMING -}; - -enum ePlayerFlags { - BTARR_TESTING = 0, - PLAYER_HAS_ENTERED_PET_RANCH = 1, - MINIMAP_UNLOCKED = 2, - ACTIVITY_REBUILDING_FAIL_TIME = 3, - ACTIVITY_REBUILDING_FAIL_RANGE = 4, - ACTIVITY_SHOOTING_GALLERY_HELP = 5, - HELP_WALKING_CONTROLS = 6, - FIRST_SMASHABLE = 7, - FIRST_IMAGINATION_PICKUP = 8, - FIRST_DAMAGE = 9, - FIRST_ITEM = 10, - FIRST_BRICK = 11, - FIRST_CONSUMABLE = 12, - FIRST_EQUIPPABLE = 13, - CHAT_HELP = 14, - FIRST_PET_TAMING_MINIGAME = 15, - FIRST_PET_ON_SWITCH = 16, - FIRST_PET_JUMPED_ON_SWITCH = 17, - FIRST_PET_FOUND_TREASURE = 18, - FIRST_PET_DUG_TREASURE = 19, - FIRST_PET_OWNER_ON_PET_BOUNCER = 20, - FIRST_PET_DESPAWN_NO_IMAGINATION = 21, - FIRST_PET_SELECTED_ENOUGH_BRICKS = 22, - FIRST_EMOTE_UNLOCKED = 23, - GF_PIRATE_REP = 24, - AG_BOB_CINEMATIC_EVENT = 25, - HELP_JUMPING_CONTROLS = 26, - HELP_DOUBLE_JUMP_CONTROLS = 27, - HELP_CAMERA_CONTROLS = 28, - HELP_ROTATE_CONTROLS = 29, - HELP_SMASH = 30, - MONUMENT_INTRO_MUSIC_PLAYED = 31, - BEGINNING_ZONE_SUMMARY_DISPLAYED = 32, - AG_FINISH_LINE_BUILT = 33, - AG_BOSS_AREA_FOUND = 34, - AG_LANDED_IN_BATTLEFIELD = 35, - GF_PLAYER_HAS_BEEN_TO_THE_RAVINE = 36, - MODULAR_BUILD_STARTED = 37, - MODULAR_BUILD_FINISHED_CLICK_BUTTON = 38, - THINKING_HAT_RECEIVED_GO_TO_MODULAR_BUILD_AREA = 39, - BUILD_AREA_ENTERED_MOD_NOT_ACTIVATED_PUT_ON_HAT = 40, - HAT_ON_INSIDE_OF_MOD_BUILD_EQUIP_A_MODULE_FROM_LEG = 41, - MODULE_EQUIPPED_PLACE_ON_GLOWING_BLUE_SPOT = 42, - FIRST_MODULE_PLACED_CORRECTLY_NOW_DO_THE_REST = 43, - ROCKET_COMPLETE_NOW_LAUNCH_FROM_PAD = 44, - JOINED_A_FACTION = 45, - VENTURE_FACTION = 46, - ASSEMBLY_FACTION = 47, - PARADOX_FACTION = 48, - SENTINEL_FACTION = 49, - LUP_WORLD_ACCESS = 50, - AG_FIRST_FLAG_COLLECTED = 51, - TOOLTIP_TALK_TO_SKYLAND_TO_GET_HAT = 52, - MODULAR_BUILD_PLAYER_PLACES_FIRST_MODEL_IN_SCRATCH = 53, - MODULAR_BUILD_FIRST_ARROW_DISPLAY_FOR_MODULE = 54, - AG_BEACON_QB_SO_THE_PLAYER_CAN_ALWAYS_BUILD_THEM = 55, - GF_PET_DIG_FLAG_1 = 56, - GF_PET_DIG_FLAG_2 = 57, - GF_PET_DIG_FLAG_3 = 58, - SUPPRESS_SPACESHIP_CINEMATIC_FLYTHROUGH = 59, - GF_PLAYER_FALL_DEATH = 60, - GF_PLAYER_CAN_GET_FLAG_1 = 61, - GF_PLAYER_CAN_GET_FLAG_2 = 62, - GF_PLAYER_CAN_GET_FLAG_3 = 63, - ENTER_BBB_FROM_PROPERTY_EDIT_CONFIRMATION_DIALOG = 64, - AG_FIRST_COMBAT_COMPLETE = 65, - AG_COMPLETE_BOB_MISSION = 66, - NJ_GARMADON_CINEMATIC_SEEN = 125, - ELEPHANT_PET_3050 = 801, - CAT_PET_3054 = 802, - TRICERATOPS_PET_3195 = 803, - TERRIER_PET_3254 = 804, - SKUNK_PET_3261 = 805, - LION_PET_3520 = 806, - BUNNY_PET_3672 = 807, - CROCODILE_PET_3994 = 808, - DOBERMAN_PET_5635 = 809, - BUFFALO_PET_5636 = 810, - ROBOT_DOG_PET_5637 = 811, - EUROPEAN_DRAGON_PET_5639 = 812, - TORTOISE_PET_5640 = 813, - ASIAN_DRAGON_PET_5641 = 814, - MANTIS_PET_5642 = 815, - PANDA_PET_5643 = 816, - WARTHOG_PET_6720 = 817, - GOAT_PET_7638 = 818, - CRAB_PET_7694 = 819, - AG_SPACE_SHIP_BINOC_AT_LAUNCH = 1001, - AG_SPACE_SHIP_BINOC_AT_LAUNCH_PLATFORM = 1002, - AG_SPACE_SHIP_BINOC_ON_PLATFORM_TO_LEFT_OF_START = 1003, - AG_SPACE_SHIP_BINOC_ON_PLATFORM_TO_RIGHT_OF_START = 1004, - AG_SPACE_SHIP_BINOC_AT_BOB = 1005, - AG_BATTLE_BINOC_FOR_TRICERETOPS = 1101, - AG_BATTLE_BINOC_AT_PARADOX = 1102, - AG_BATTLE_BINOC_AT_MISSION_GIVER = 1103, - AG_BATTLE_BINOC_AT_BECK = 1104, - AG_MONUMENT_BINOC_INTRO = 1105, - AG_MONUMENT_BINOC_OUTRO = 1106, - AG_LAUNCH_BINOC_INTRO = 1107, - AG_LAUNCH_BINOC_BISON = 1108, - AG_LAUNCH_BINOC_SHARK = 1109, - NS_BINOC_CONCERT_TRANSITION = 1201, - NS_BINOC_RACE_PLACE_TRANSITION = 1202, - NS_BINOC_BRICK_ANNEX_TRANSITION = 1203, - NS_BINOC_GF_LAUNCH = 1204, - NS_BINOC_FV_LAUNCH = 1205, - NS_BINOC_BRICK_ANNEX_WATER = 1206, - NS_BINOC_AG_LAUNCH_AT_RACE_PLACE = 1207, - NS_BINOC_AG_LAUNCH_AT_BRICK_ANNEX = 1208, - NS_BINOC_AG_LAUNCH_AT_PLAZA = 1209, - NS_BINOC_TBA = 1210, - NS_FLAG_COLLECTABLE_1_BY_JONNY_THUNDER = 1211, - NS_FLAG_COLLECTABLE_2_UNDER_CONCERT_BRIDGE = 1212, - NS_FLAG_COLLECTABLE_3_BY_FV_LAUNCH = 1213, - NS_FLAG_COLLECTABLE_4_IN_PLAZA_BEHIND_BUILDING = 1214, - NS_FLAG_COLLECTABLE_5_BY_GF_LAUNCH = 1215, - NS_FLAG_COLLECTABLE_6_BY_DUCK_SG = 1216, - NS_FLAG_COLLECTABLE_7_BY_LUP_LAUNCH = 1217, - NS_FLAG_COLLECTABLE_8_BY_NT_LUANCH = 1218, - NS_FLAG_COLLECTABLE_9_BY_RACE_BUILD = 1219, - NS_FLAG_COLLECTABLE_10_ON_AG_LAUNCH_PATH = 1220, - PR_BINOC_AT_LAUNCH_PAD = 1251, - PR_BINOC_AT_BEGINNING_OF_ISLAND_B = 1252, - PR_BINOC_AT_FIRST_PET_BOUNCER = 1253, - PR_BINOC_ON_BY_CROWS_NEST = 1254, - PR_PET_DIG_AT_BEGINNING_OF_ISLAND_B = 1261, - PR_PET_DIG_AT_THE_LOCATION_OF_OLD_BOUNCE_BACK = 1262, - PR_PET_DIG_UNDER_QB_BRIDGE = 1263, - PR_PET_DIG_BACK_SIDE_BY_PARTNER_BOUNCE = 1264, - PR_PET_DIG_BY_LAUNCH_PAD = 1265, - PR_PET_DIG_BY_FIRST_PET_BOUNCER = 1266, - GF_BINOC_ON_LANDING_PAD = 1301, - GF_BINOC_AT_RAVINE_START = 1302, - GF_BINOC_ON_TOP_OF_RAVINE_HEAD = 1303, - GF_BINOC_AT_TURTLE_AREA = 1304, - GF_BINOC_IN_TUNNEL_TO_ELEPHANTS = 1305, - GF_BINOC_IN_ELEPHANTS_AREA = 1306, - GF_BINOC_IN_RACING_AREA = 1307, - GF_BINOC_IN_CROC_AREA = 1308, - GF_BINOC_IN_JAIL_AREA = 1309, - GF_BINOC_TELESCOPE_NEXT_TO_CAPTAIN_JACK = 1310, - NT_PLINTH_REBUILD = 1919, - NT_FACTION_SPY_DUKE = 1974, - NT_FACTION_SPY_OVERBUILD = 1976, - NT_FACTION_SPY_HAEL = 1977, - NJ_EARTH_SPINJITZU = 2030, - NJ_LIGHTNING_SPINJITZU = 2031, - NJ_ICE_SPINJITZU = 2032, - NJ_FIRE_SPINJITZU = 2033, - NJ_WU_SHOW_DAILY_CHEST = 2099 -}; - -//======== FUNC =========== - -template -inline T const& clamp(const T& val, const T& low, const T& high) { - if (val < low) return low; - else if (val > high) return high; - - return val; -} diff --git a/dCommon/dConfig.cpp b/dCommon/dConfig.cpp index 9750238a..f09a44c1 100644 --- a/dCommon/dConfig.cpp +++ b/dCommon/dConfig.cpp @@ -1,45 +1,53 @@ #include "dConfig.h" + #include +#include "BinaryPathFinder.h" +#include "GeneralUtils.h" + dConfig::dConfig(const std::string& filepath) { - m_EmptyString = ""; - std::ifstream in(filepath); + m_ConfigFilePath = filepath; + LoadConfig(); +} + +void dConfig::LoadConfig() { + std::ifstream in(BinaryPathFinder::GetBinaryDir() / m_ConfigFilePath); if (!in.good()) return; - std::string line; + std::string line{}; while (std::getline(in, line)) { - if (line.length() > 0) { - if (line[0] != '#') ProcessLine(line); - } + if (!line.empty() && line.front() != '#') ProcessLine(line); + } + + std::ifstream sharedConfig(BinaryPathFinder::GetBinaryDir() / "sharedconfig.ini", std::ios::in); + if (!sharedConfig.good()) return; + + line.clear(); + while (std::getline(sharedConfig, line)) { + if (!line.empty() && line.front() != '#') ProcessLine(line); } } -dConfig::~dConfig(void) { +void dConfig::ReloadConfig() { + this->m_ConfigValues.clear(); + LoadConfig(); } const std::string& dConfig::GetValue(std::string key) { - for (size_t i = 0; i < m_Keys.size(); ++i) { - if (m_Keys[i] == key) return m_Values[i]; - } - - return m_EmptyString; + return this->m_ConfigValues[key]; } void dConfig::ProcessLine(const std::string& line) { - std::stringstream ss(line); - std::string segment; - std::vector seglist; + auto splitLine = GeneralUtils::SplitString(line, '='); - while (std::getline(ss, segment, '=')) { - seglist.push_back(segment); - } - - if (seglist.size() != 2) return; + if (splitLine.size() != 2) return; //Make sure that on Linux, we remove special characters: - if (!seglist[1].empty() && seglist[1][seglist[1].size() - 1] == '\r') - seglist[1].erase(seglist[1].size() - 1); + auto& key = splitLine.at(0); + auto& value = splitLine.at(1); + if (!value.empty() && value.at(value.size() - 1) == '\r') value.erase(value.size() - 1); - m_Keys.push_back(seglist[0]); - m_Values.push_back(seglist[1]); + if (this->m_ConfigValues.find(key) != this->m_ConfigValues.end()) return; + + this->m_ConfigValues.insert(std::make_pair(key, value)); } diff --git a/dCommon/dConfig.h b/dCommon/dConfig.h index 7764fa54..562c1ce9 100644 --- a/dCommon/dConfig.h +++ b/dCommon/dConfig.h @@ -1,20 +1,33 @@ #pragma once #include +#include #include -#include class dConfig { public: dConfig(const std::string& filepath); - ~dConfig(void); + /** + * Gets the specified key from the config. Returns an empty string if the value is not found. + * + * @param key Key to find + * @return The keys value in the config + */ const std::string& GetValue(std::string key); + /** + * Loads the config from a file + */ + void LoadConfig(); + + /** + * Reloads the config file to reset values + */ + void ReloadConfig(); private: void ProcessLine(const std::string& line); private: - std::vector m_Keys; - std::vector m_Values; - std::string m_EmptyString; + std::map m_ConfigValues; + std::string m_ConfigFilePath; }; diff --git a/dCommon/dEnums/dCommonVars.h b/dCommon/dEnums/dCommonVars.h new file mode 100644 index 00000000..f67145da --- /dev/null +++ b/dCommon/dEnums/dCommonVars.h @@ -0,0 +1,162 @@ +#pragma once + +#ifndef __DCOMMONVARS__H__ +#define __DCOMMONVARS__H__ + +#include +#include +#include +#include "BitStream.h" +#include "eConnectionType.h" +#include "eClientMessageType.h" + +#pragma warning (disable:4251) //Disables SQL warnings + +// These are the same define, but they mean two different things in different contexts +// so a different define to distinguish what calculation is happening will help clarity. +#define FRAMES_TO_MS(x) 1000 / x +#define MS_TO_FRAMES(x) 1000 / x + +//=========== FRAME TIMINGS =========== +constexpr uint32_t highFramerate = 60; +constexpr uint32_t mediumFramerate = 30; +constexpr uint32_t lowFramerate = 15; + +constexpr uint32_t highFrameDelta = FRAMES_TO_MS(highFramerate); +constexpr uint32_t mediumFrameDelta = FRAMES_TO_MS(mediumFramerate); +constexpr uint32_t lowFrameDelta = FRAMES_TO_MS(lowFramerate); + +//========== MACROS =========== + +#define HEADER_SIZE 8 +#define CBITSTREAM RakNet::BitStream bitStream; +#define CINSTREAM RakNet::BitStream inStream(packet->data, packet->length, false); +#define CINSTREAM_SKIP_HEADER CINSTREAM if (inStream.GetNumberOfUnreadBits() >= BYTES_TO_BITS(HEADER_SIZE)) inStream.IgnoreBytes(HEADER_SIZE); else inStream.IgnoreBits(inStream.GetNumberOfUnreadBits()); +#define CMSGHEADER PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::GAME_MSG); +#define SEND_PACKET Game::server->Send(&bitStream, sysAddr, false); +#define SEND_PACKET_BROADCAST Game::server->Send(&bitStream, UNASSIGNED_SYSTEM_ADDRESS, true); + +//=========== TYPEDEFS ========== + +typedef int32_t LOT; //!< A LOT +typedef int64_t LWOOBJID; //!< An object ID (should be unsigned actually but ok) +typedef int32_t TSkillID; //!< A skill ID +typedef uint32_t LWOCLONEID; //!< Used for Clone IDs +typedef uint16_t LWOMAPID; //!< Used for Map IDs +typedef uint16_t LWOINSTANCEID; //!< Used for Instance IDs +typedef uint32_t PROPERTYCLONELIST; //!< Used for Property Clone IDs +typedef uint32_t StripId; + +const LWOOBJID LWOOBJID_EMPTY = 0; //!< An empty object ID +const LOT LOT_NULL = -1; //!< A null LOT +const int32_t LOOTTYPE_NONE = 0; //!< No loot type available +const float SECONDARY_PRIORITY = 1.0f; //!< Secondary Priority +const uint32_t INVENTORY_MAX = 9999999; //!< The Maximum Inventory Size +const uint32_t LWOCLONEID_INVALID = -1; //!< Invalid LWOCLONEID +const uint16_t LWOINSTANCEID_INVALID = -1; //!< Invalid LWOINSTANCEID +const uint16_t LWOMAPID_INVALID = -1; //!< Invalid LWOMAPID +const uint64_t LWOZONEID_INVALID = 0; //!< Invalid LWOZONEID + +const float PI = 3.14159f; + +//============ STRUCTS ============== + +struct LWOSCENEID { +public: + LWOSCENEID() { m_sceneID = -1; m_layerID = 0; } + LWOSCENEID(int sceneID) { m_sceneID = sceneID; m_layerID = 0; } + LWOSCENEID(int sceneID, unsigned int layerID) { m_sceneID = sceneID; m_layerID = layerID; } + + LWOSCENEID& operator=(const LWOSCENEID& rhs) { m_sceneID = rhs.m_sceneID; m_layerID = rhs.m_layerID; return *this; } + LWOSCENEID& operator=(const int rhs) { m_sceneID = rhs; m_layerID = 0; return *this; } + + bool operator<(const LWOSCENEID& rhs) const { return (m_sceneID < rhs.m_sceneID || (m_sceneID == rhs.m_sceneID && m_layerID < rhs.m_layerID)); } + bool operator<(const int rhs) const { return m_sceneID < rhs; } + + bool operator==(const LWOSCENEID& rhs) const { return (m_sceneID == rhs.m_sceneID && m_layerID == rhs.m_layerID); } + bool operator==(const int rhs) const { return m_sceneID == rhs; } + + const int GetSceneID() const { return m_sceneID; } + const unsigned int GetLayerID() const { return m_layerID; } + + void SetSceneID(const int sceneID) { m_sceneID = sceneID; } + void SetLayerID(const unsigned int layerID) { m_layerID = layerID; } + +private: + int m_sceneID; + unsigned int m_layerID; +}; + +struct LWOZONEID { +public: + const LWOMAPID& GetMapID() const { return m_MapID; } + const LWOINSTANCEID& GetInstanceID() const { return m_InstanceID; } + const LWOCLONEID& GetCloneID() const { return m_CloneID; } + + //In order: def constr, constr, assign op + LWOZONEID() { m_MapID = LWOMAPID_INVALID; m_InstanceID = LWOINSTANCEID_INVALID; m_CloneID = LWOCLONEID_INVALID; } + LWOZONEID(const LWOMAPID& mapID, const LWOINSTANCEID& instanceID, const LWOCLONEID& cloneID) { m_MapID = mapID; m_InstanceID = instanceID; m_CloneID = cloneID; } + LWOZONEID(const LWOZONEID& replacement) { *this = replacement; } + +private: + LWOMAPID m_MapID; //1000 for VE, 1100 for AG, etc... + LWOINSTANCEID m_InstanceID; //Instances host the same world, but on a different dWorld process. + LWOCLONEID m_CloneID; //To differentiate between "your property" and "my property". Always 0 for non-prop worlds. +}; + +const LWOSCENEID LWOSCENEID_INVALID = -1; + +struct LWONameValue { + uint32_t length = 0; //!< The length of the name + std::u16string name; //!< The name + + LWONameValue(void) {} + + LWONameValue(const std::u16string& name) { + this->name = name; + this->length = static_cast(name.length()); + } + + ~LWONameValue(void) {} +}; + +struct FriendData { +public: + bool isOnline = false; + bool isBestFriend = false; + bool isFTP = false; + LWOZONEID zoneID; + LWOOBJID friendID; + std::string friendName; + + void Serialize(RakNet::BitStream& bitStream) { + bitStream.Write(isOnline); + bitStream.Write(isBestFriend); + bitStream.Write(isFTP); + bitStream.Write(0); //??? + bitStream.Write(0); //??? + bitStream.Write(zoneID.GetMapID()); + bitStream.Write(zoneID.GetInstanceID()); + bitStream.Write(zoneID.GetCloneID()); + bitStream.Write(friendID); + + uint32_t maxSize = 33; + uint32_t size = static_cast(friendName.length()); + uint32_t remSize = static_cast(maxSize - size); + + if (size > maxSize) size = maxSize; + + for (uint32_t i = 0; i < size; ++i) { + bitStream.Write(static_cast(friendName[i])); + } + + for (uint32_t j = 0; j < remSize; ++j) { + bitStream.Write(static_cast(0)); + } + + bitStream.Write(0); //??? + bitStream.Write(0); //??? + } +}; + +#endif //!__DCOMMONVARS__H__ diff --git a/dCommon/dPlatforms.h b/dCommon/dEnums/dPlatforms.h similarity index 100% rename from dCommon/dPlatforms.h rename to dCommon/dEnums/dPlatforms.h diff --git a/dPhysics/dpCollisionGroups.h b/dCommon/dEnums/dpCollisionGroups.h similarity index 100% rename from dPhysics/dpCollisionGroups.h rename to dCommon/dEnums/dpCollisionGroups.h diff --git a/dPhysics/dpCommon.h b/dCommon/dEnums/dpCommon.h similarity index 100% rename from dPhysics/dpCommon.h rename to dCommon/dEnums/dpCommon.h diff --git a/dCommon/dEnums/eAddFriendResponseCode.h b/dCommon/dEnums/eAddFriendResponseCode.h new file mode 100644 index 00000000..56304c82 --- /dev/null +++ b/dCommon/dEnums/eAddFriendResponseCode.h @@ -0,0 +1,15 @@ +#pragma once + +#ifndef __EADDFRIENDRESPONSECODE__H__ +#define __EADDFRIENDRESPONSECODE__H__ + +#include + +enum class eAddFriendResponseCode : uint8_t { + ACCEPTED = 0, + REJECTED, + BUSY, + CANCELLED +}; + +#endif //!__ADDFRIENDRESPONSECODE__H__ diff --git a/dCommon/dEnums/eAddFriendResponseType.h b/dCommon/dEnums/eAddFriendResponseType.h new file mode 100644 index 00000000..5568aafb --- /dev/null +++ b/dCommon/dEnums/eAddFriendResponseType.h @@ -0,0 +1,24 @@ +#pragma once + +#ifndef __EADDFRIENDRESPONSETYPE__H__ +#define __EADDFRIENDRESPONSETYPE__H__ + +#include + +enum class eAddFriendResponseType : uint8_t { + ACCEPTED = 0, + ALREADYFRIEND, + INVALIDCHARACTER, + GENERALERROR, + YOURFRIENDSLISTFULL, + THEIRFRIENDLISTFULL, + DECLINED, + BUSY, + NOTONLINE, + WAITINGAPPROVAL, + MYTHRAN, + CANCELLED, + FRIENDISFREETRIAL +}; + +#endif //!__EADDFRIENDRESPONSETYPE__H__ diff --git a/dCommon/dEnums/eAninmationFlags.h b/dCommon/dEnums/eAninmationFlags.h new file mode 100644 index 00000000..ce235ae9 --- /dev/null +++ b/dCommon/dEnums/eAninmationFlags.h @@ -0,0 +1,44 @@ +#pragma once + +#ifndef __EANINMATIONFLAGS__H__ +#define __EANINMATIONFLAGS__H__ + +#include + +enum class eAnimationFlags : uint32_t { + IDLE_NONE = 0, + IDLE_BASIC, + IDLE_SWIM, + IDLE_CARRY, + IDLE_SWORD, + IDLE_HAMMER, + IDLE_SPEAR, + IDLE_PISTOL, + IDLE_BOW, + IDLE_COMBAT, + IDLE_JETPACK, + IDLE_HORSE, + IDLE_SG, + IDLE_ORGAN, + IDLE_SKATEBOARD, + IDLE_DAREDEVIL, + IDLE_SAMURAI, + IDLE_SUMMONER, + IDLE_BUCCANEER, + IDLE_MISC, + IDLE_NINJA, + IDLE_MISC1, + IDLE_MISC2, + IDLE_MISC3, + IDLE_MISC4, + IDLE_MISC5, + IDLE_MISC6, + IDLE_MISC7, + IDLE_MISC8, + IDLE_MISC9, + IDLE_MISC10, + IDLE_MISC11, + IDLE_MISC12 +}; + +#endif //!__EANINMATIONFLAGS__H__ diff --git a/dCommon/dEnums/eAuthMessageType.h b/dCommon/dEnums/eAuthMessageType.h new file mode 100644 index 00000000..ecc17a37 --- /dev/null +++ b/dCommon/dEnums/eAuthMessageType.h @@ -0,0 +1,15 @@ +#ifndef __EAUTHMESSAGETYPE__H__ +#define __EAUTHMESSAGETYPE__H__ + +#include + +enum class eAuthMessageType : uint32_t { + LOGIN_REQUEST = 0, + LOGOUT_REQUEST, + CREATE_NEW_ACCOUNT_REQUEST, + LEGOINTERFACE_AUTH_RESPONSE, + SESSIONKEY_RECEIVED_CONFIRM, + RUNTIME_CONFIG +}; + +#endif //!__EAUTHMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eBasicAttackSuccessTypes.h b/dCommon/dEnums/eBasicAttackSuccessTypes.h new file mode 100644 index 00000000..8c06da8a --- /dev/null +++ b/dCommon/dEnums/eBasicAttackSuccessTypes.h @@ -0,0 +1,12 @@ +#ifndef __EBASICATTACKSUCCESSTYPES__H__ +#define __EBASICATTACKSUCCESSTYPES__H__ + +#include + +enum class eBasicAttackSuccessTypes : uint8_t { + SUCCESS = 1, + FAILARMOR, + FAILIMMUNE +}; + +#endif //!__EBASICATTACKSUCCESSTYPES__H__ diff --git a/dCommon/dEnums/eBlueprintSaveResponseType.h b/dCommon/dEnums/eBlueprintSaveResponseType.h new file mode 100644 index 00000000..29d15695 --- /dev/null +++ b/dCommon/dEnums/eBlueprintSaveResponseType.h @@ -0,0 +1,26 @@ +#pragma once + +#ifndef __EBLUEPRINTSAVERESPONSETYPE__H__ +#define __EBLUEPRINTSAVERESPONSETYPE__H__ + +#include + +enum class eBlueprintSaveResponseType : uint32_t { + EverythingWorked = 0, + SaveCancelled, + CantBeginTransaction, + SaveBlueprintFailed, + SaveUgobjectFailed, + CantEndTransaction, + SaveFilesFailed, + BadInput, + NotEnoughBricks, + InventoryFull, + ModelGenerationFailed, + PlacementFailed, + GmLevelInsufficient, + WaitForPreviousSave, + FindMatchesFailed +}; + +#endif //!__EBLUEPRINTSAVERESPONSETYPE__H__ diff --git a/dCommon/dEnums/eBubbleType.h b/dCommon/dEnums/eBubbleType.h new file mode 100644 index 00000000..9ceef4b5 --- /dev/null +++ b/dCommon/dEnums/eBubbleType.h @@ -0,0 +1,14 @@ +#pragma once + +#ifndef __EBUBBLETYPE__H__ +#define __EBUBBLETYPE__H__ + +#include + +enum class eBubbleType : uint32_t { + DEFAULT = 0, + ENERGY, + SKUNK +}; + +#endif //!__EBUBBLETYPE__H__ diff --git a/dCommon/dEnums/eBuildType.h b/dCommon/dEnums/eBuildType.h new file mode 100644 index 00000000..f28f43cc --- /dev/null +++ b/dCommon/dEnums/eBuildType.h @@ -0,0 +1,12 @@ +#ifndef __EBUILDTYPE__H__ +#define __EBUILDTYPE__H__ + +#include + +enum class eBuildType :uint32_t { + NOWHERE, + IN_WORLD, + ON_PROPERTY +}; + +#endif //!__EBUILDTYPE__H__ diff --git a/dCommon/dEnums/eCharacterCreationResponse.h b/dCommon/dEnums/eCharacterCreationResponse.h new file mode 100644 index 00000000..da1ec0f2 --- /dev/null +++ b/dCommon/dEnums/eCharacterCreationResponse.h @@ -0,0 +1,14 @@ +#ifndef __ECHARACTERCREATIONRESPONSE__H__ +#define __ECHARACTERCREATIONRESPONSE__H__ + +#include + +enum class eCharacterCreationResponse : uint8_t { + SUCCESS = 0, + OBJECT_ID_UNAVAILABLE, + NAME_NOT_ALLOWED, + PREDEFINED_NAME_IN_USE, + CUSTOM_NAME_IN_USE +}; + +#endif //!__ECHARACTERCREATIONRESPONSE__H__ diff --git a/dCommon/dEnums/eCharacterVersion.h b/dCommon/dEnums/eCharacterVersion.h new file mode 100644 index 00000000..0fab4498 --- /dev/null +++ b/dCommon/dEnums/eCharacterVersion.h @@ -0,0 +1,21 @@ +#pragma once + +#ifndef __ECHARACTERVERSION__H__ +#define __ECHARACTERVERSION__H__ + +#include + +enum class eCharacterVersion : uint32_t { +// Versions from the live game + RELEASE = 0, // Initial release of the game + LIVE, // Fixes for the 1.9 release bug fixes for missions leading up to joining a faction +// New versions for DLU fixes + // Fixes the "Joined a faction" player flag not being set properly + PLAYER_FACTION_FLAGS, + // Fixes vault size value + VAULT_SIZE, + // Fixes speed base value in level component + UP_TO_DATE, // will become SPEED_BASE +}; + +#endif //!__ECHARACTERVERSION__H__ diff --git a/dCommon/dEnums/eChatInternalMessageType.h b/dCommon/dEnums/eChatInternalMessageType.h new file mode 100644 index 00000000..d3b7020b --- /dev/null +++ b/dCommon/dEnums/eChatInternalMessageType.h @@ -0,0 +1,31 @@ +#ifndef __ECHATINTERNALMESSAGETYPE__H__ +#define __ECHATINTERNALMESSAGETYPE__H__ + +#include + +enum eChatInternalMessageType : uint32_t { + PLAYER_ADDED_NOTIFICATION = 0, + PLAYER_REMOVED_NOTIFICATION, + ADD_FRIEND, + ADD_BEST_FRIEND, + ADD_TO_TEAM, + ADD_BLOCK, + REMOVE_FRIEND, + REMOVE_BLOCK, + REMOVE_FROM_TEAM, + DELETE_TEAM, + REPORT, + PRIVATE_CHAT, + PRIVATE_CHAT_RESPONSE, + ANNOUNCEMENT, + MAIL_COUNT_UPDATE, + MAIL_SEND_NOTIFY, + REQUEST_USER_LIST, + FRIEND_LIST, + ROUTE_TO_PLAYER, + TEAM_UPDATE, + MUTE_UPDATE, + CREATE_TEAM, +}; + +#endif //!__ECHATINTERNALMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eChatMessageType.h b/dCommon/dEnums/eChatMessageType.h new file mode 100644 index 00000000..52895ba3 --- /dev/null +++ b/dCommon/dEnums/eChatMessageType.h @@ -0,0 +1,78 @@ +#ifndef __ECHATMESSAGETYPE__H__ +#define __ECHATMESSAGETYPE__H__ + +#include + +//! The Internal Chat Packet Identifiers +enum class eChatMessageType :uint32_t { + LOGIN_SESSION_NOTIFY = 0, + GENERAL_CHAT_MESSAGE, + PRIVATE_CHAT_MESSAGE, + USER_CHANNEL_CHAT_MESSAGE, + WORLD_DISCONNECT_REQUEST, + WORLD_PROXIMITY_RESPONSE, + WORLD_PARCEL_RESPONSE, + ADD_FRIEND_REQUEST, + ADD_FRIEND_RESPONSE, + REMOVE_FRIEND, + GET_FRIENDS_LIST, + ADD_IGNORE, + REMOVE_IGNORE, + GET_IGNORE_LIST, + TEAM_MISSED_INVITE_CHECK, + TEAM_INVITE, + TEAM_INVITE_RESPONSE, + TEAM_KICK, + TEAM_LEAVE, + TEAM_SET_LOOT, + TEAM_SET_LEADER, + TEAM_GET_STATUS, + GUILD_CREATE, + GUILD_INVITE, + GUILD_INVITE_RESPONSE, + GUILD_LEAVE, + GUILD_KICK, + GUILD_GET_STATUS, + GUILD_GET_ALL, + SHOW_ALL, + BLUEPRINT_MODERATED, + BLUEPRINT_MODEL_READY, + PROPERTY_READY_FOR_APPROVAL, + PROPERTY_MODERATION_CHANGED, + PROPERTY_BUILDMODE_CHANGED, + PROPERTY_BUILDMODE_CHANGED_REPORT, + MAIL, + WORLD_INSTANCE_LOCATION_REQUEST, + REPUTATION_UPDATE, + SEND_CANNED_TEXT, + GMLEVEL_UPDATE, + CHARACTER_NAME_CHANGE_REQUEST, + CSR_REQUEST, + CSR_REPLY, + GM_KICK, + GM_ANNOUNCE, + GM_MUTE, + ACTIVITY_UPDATE, + WORLD_ROUTE_PACKET, + GET_ZONE_POPULATIONS, + REQUEST_MINIMUM_CHAT_MODE, + REQUEST_MINIMUM_CHAT_MODE_PRIVATE, + MATCH_REQUEST, + UGCMANIFEST_REPORT_MISSING_FILE, + UGCMANIFEST_REPORT_DONE_FILE, + UGCMANIFEST_REPORT_DONE_BLUEPRINT, + UGCC_REQUEST, + WHO, + WORLD_PLAYERS_PET_MODERATED_ACKNOWLEDGE, + ACHIEVEMENT_NOTIFY, + GM_CLOSE_PRIVATE_CHAT_WINDOW, + UNEXPECTED_DISCONNECT, + PLAYER_READY, + GET_DONATION_TOTAL, + UPDATE_DONATION, + PRG_CSR_COMMAND, + HEARTBEAT_REQUEST_FROM_WORLD, + UPDATE_FREE_TRIAL_STATUS +}; + +#endif //!__ECHATMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eCinematicEvent.h b/dCommon/dEnums/eCinematicEvent.h new file mode 100644 index 00000000..7fb82ca7 --- /dev/null +++ b/dCommon/dEnums/eCinematicEvent.h @@ -0,0 +1,12 @@ +#ifndef __ECINEMATICEVENT__H__ +#define __ECINEMATICEVENT__H__ + +#include + +enum class eCinematicEvent : uint32_t { + STARTED, + WAYPOINT, + ENDED, +}; + +#endif //!__ECINEMATICEVENT__H__ diff --git a/dCommon/dEnums/eClientMessageType.h b/dCommon/dEnums/eClientMessageType.h new file mode 100644 index 00000000..aafccc36 --- /dev/null +++ b/dCommon/dEnums/eClientMessageType.h @@ -0,0 +1,76 @@ +#ifndef __ECLIENTMESSAGETYPE__H__ +#define __ECLIENTMESSAGETYPE__H__ + +#include + +enum class eClientMessageType : uint32_t { + LOGIN_RESPONSE = 0, + LOGOUT_RESPONSE, + LOAD_STATIC_ZONE, + CREATE_OBJECT, + CREATE_CHARACTER, + CREATE_CHARACTER_EXTENDED, + CHARACTER_LIST_RESPONSE, + CHARACTER_CREATE_RESPONSE, + CHARACTER_RENAME_RESPONSE, + CHAT_CONNECT_RESPONSE, + AUTH_ACCOUNT_CREATE_RESPONSE, + DELETE_CHARACTER_RESPONSE, + GAME_MSG, + CONNECT_CHAT, + TRANSFER_TO_WORLD, + IMPENDING_RELOAD_NOTIFY, + MAKE_GM_RESPONSE, + HTTP_MONITOR_INFO_RESPONSE, + SLASH_PUSH_MAP_RESPONSE, + SLASH_PULL_MAP_RESPONSE, + SLASH_LOCK_MAP_RESPONSE, + BLUEPRINT_SAVE_RESPONSE, + BLUEPRINT_LUP_SAVE_RESPONSE, + BLUEPRINT_LOAD_RESPONSE_ITEMID, + BLUEPRINT_GET_ALL_DATA_RESPONSE, + MODEL_INSTANTIATE_RESPONSE, + DEBUG_OUTPUT, + ADD_FRIEND_REQUEST, + ADD_FRIEND_RESPONSE, + REMOVE_FRIEND_RESPONSE, + GET_FRIENDS_LIST_RESPONSE, + UPDATE_FRIEND_NOTIFY, + ADD_IGNORE_RESPONSE, + REMOVE_IGNORE_RESPONSE, + GET_IGNORE_LIST_RESPONSE, + TEAM_INVITE, + TEAM_INVITE_INITIAL_RESPONSE, + GUILD_CREATE_RESPONSE, + GUILD_GET_STATUS_RESPONSE, + GUILD_INVITE, + GUILD_INVITE_INITIAL_RESPONSE, + GUILD_INVITE_FINAL_RESPONSE, + GUILD_INVITE_CONFIRM, + GUILD_ADD_PLAYER, + GUILD_REMOVE_PLAYER, + GUILD_LOGIN_LOGOUT, + GUILD_RANK_CHANGE, + GUILD_DATA, + GUILD_STATUS, + MAIL, + DB_PROXY_RESULT, + SHOW_ALL_RESPONSE, + WHO_RESPONSE, + SEND_CANNED_TEXT, + UPDATE_CHARACTER_NAME, + SET_NETWORK_SIMULATOR, + INVALID_CHAT_MESSAGE, + MINIMUM_CHAT_MODE_RESPONSE, + MINIMUM_CHAT_MODE_RESPONSE_PRIVATE, + CHAT_MODERATION_STRING, + UGC_MANIFEST_RESPONSE, + IN_LOGIN_QUEUE, + SERVER_STATES, + GM_CLOSE_TARGET_CHAT_WINDOW, + GENERAL_TEXT_FOR_LOCALIZATION, + UPDATE_FREE_TRIAL_STATUS, + UGC_DOWNLOAD_FAILED = 120 +}; + +#endif //!__ECLIENTMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eConnectionType.h b/dCommon/dEnums/eConnectionType.h new file mode 100644 index 00000000..ce1ff90c --- /dev/null +++ b/dCommon/dEnums/eConnectionType.h @@ -0,0 +1,14 @@ +#ifndef __ECONNECTIONTYPE__H__ +#define __ECONNECTIONTYPE__H__ + +enum class eConnectionType : uint16_t { + SERVER = 0, + AUTH, + CHAT, + CHAT_INTERNAL, + WORLD, + CLIENT, + MASTER +}; + +#endif //!__ECONNECTIONTYPE__H__ diff --git a/dCommon/dEnums/eControlScheme.h b/dCommon/dEnums/eControlScheme.h new file mode 100644 index 00000000..f7585ebb --- /dev/null +++ b/dCommon/dEnums/eControlScheme.h @@ -0,0 +1,18 @@ +#ifndef __ECONTROLSCHEME__H__ +#define __ECONTROLSCHEME__H__ + +#include + +enum class eControlScheme : uint32_t { + SCHEME_A, + SCHEME_D, + SCHEME_GAMEPAD, + SCHEME_E, + SCHEME_FPS, + SCHEME_DRIVING, + SCHEME_TAMING, + SCHEME_MODULAR_BUILD, + SCHEME_WEAR_A_ROBOT //== freecam? +}; + +#endif //!__ECONTROLSCHEME__H__ diff --git a/dCommon/dEnums/eCyclingMode.h b/dCommon/dEnums/eCyclingMode.h new file mode 100644 index 00000000..b5e3248b --- /dev/null +++ b/dCommon/dEnums/eCyclingMode.h @@ -0,0 +1,11 @@ +#ifndef __ECYCLINGMODE__H__ +#define __ECYCLINGMODE__H__ + +#include + +enum class eCyclingMode : uint32_t { + ALLOW_CYCLE_TEAMMATES, + DISALLOW_CYCLING +}; + +#endif //!__ECYCLINGMODE__H__ diff --git a/dCommon/dEnums/eEndBehavior.h b/dCommon/dEnums/eEndBehavior.h new file mode 100644 index 00000000..77afaab4 --- /dev/null +++ b/dCommon/dEnums/eEndBehavior.h @@ -0,0 +1,11 @@ +#ifndef __EENDBEHAVIOR__H__ +#define __EENDBEHAVIOR__H__ + +#include + +enum class eEndBehavior : uint32_t { + RETURN, + WAIT +}; + +#endif //!__EENDBEHAVIOR__H__ diff --git a/dCommon/dEnums/eGameActivity.h b/dCommon/dEnums/eGameActivity.h new file mode 100644 index 00000000..16b75380 --- /dev/null +++ b/dCommon/dEnums/eGameActivity.h @@ -0,0 +1,15 @@ +#ifndef __EGAMEACTIVITY__H__ +#define __EGAMEACTIVITY__H__ + +#include + +enum class eGameActivity : uint32_t { + NONE, + QUICKBUILDING, + SHOOTING_GALLERY, + RACING, + PINBALL, + PET_TAMING +}; + +#endif //!__EGAMEACTIVITY__H__ diff --git a/dCommon/dEnums/eGameMasterLevel.h b/dCommon/dEnums/eGameMasterLevel.h new file mode 100644 index 00000000..a63c1caf --- /dev/null +++ b/dCommon/dEnums/eGameMasterLevel.h @@ -0,0 +1,20 @@ +#ifndef __EGAMEMASTERLEVEL__H__ +#define __EGAMEMASTERLEVEL__H__ + +#include + +enum class eGameMasterLevel : uint8_t { + CIVILIAN = 0, // Normal player. + FORUM_MODERATOR = 1, // No permissions on live servers. + JUNIOR_MODERATOR = 2, // Can kick/mute and pull chat logs. + MODERATOR = 3, // Can return lost items. + SENIOR_MODERATOR = 4, // Can ban. + LEAD_MODERATOR = 5, // Can approve properties. + JUNIOR_DEVELOPER = 6, // Junior developer & future content team. Civilan on live. + INACTIVE_DEVELOPER = 7, // Inactive developer, limited permissions. + DEVELOPER = 8, // Active developer, full permissions on live. + OPERATOR = 9 // Can shutdown server for restarts & updates. +}; + + +#endif //!__EGAMEMASTERLEVEL__H__ diff --git a/dCommon/dEnums/eGameMessageType.h b/dCommon/dEnums/eGameMessageType.h new file mode 100644 index 00000000..d5751b51 --- /dev/null +++ b/dCommon/dEnums/eGameMessageType.h @@ -0,0 +1,1605 @@ +#ifndef __EGAMEMESSAGETYPE__H__ +#define __EGAMEMESSAGETYPE__H__ + +#include + +enum class eGameMessageType : uint16_t { + GET_POSITION = 0, + GET_ROTATION = 1, + GET_LINEAR_VELOCITY = 2, + GET_ANGULAR_VELOCITY = 3, + GET_FORWARD_VELOCITY = 4, + GET_PLAYER_FORWARD = 5, + GET_FORWARD_VECTOR = 6, + SET_POSITION = 7, + SET_LOCAL_POSITION = 8, + SET_ROTATION = 9, + SET_LINEAR_VELOCITY = 10, + MODIFY_LINEAR_VELOCITY = 11, + SET_ANGULAR_VELOCITY = 12, + MODIFY_ANGULAR_VELOCITY = 13, + DEFLECT = 14, + SEND_POSITION_UPDATE = 15, + SET_OBJECT_SCALE = 16, + GET_OBJECT_SCALE = 17, + TIMED_SCALE_FINISHED = 18, + TELEPORT = 19, + TOGGLE_PLAYER_FWD_TO_CAMERA = 20, + LOCK_PLAYER_ROT_TO_CAMERA = 21, + UNLOCK_PLAYER_ROT_FROM_CAMERA = 22, + TOGGLE_PLAYER_ROT_LOCK_TO_MOUSE = 23, + LOCK_PLAYER_ROT_TO_MOUSE = 24, + UNLOCK_PLAYER_ROT_FROM_MOUSE = 25, + SET_PLAYER_CONTROL_SCHEME = 26, + GET_PLAYER_CONTROL_SCHEME = 27, + RESET_PLAYER_CONTROL_SCHEME = 28, + PLAYER_TO_PREVIOUS_CONTROL_SCHEME = 29, + DROP_CLIENT_LOOT = 30, + GET_SPEED = 34, + GET_ROT_SPEED = 35, + IS_DEAD = 36, + DIE = 37, + REQUEST_DIE = 38, + ADD_OBJECT = 39, + PLAY_EMOTE = 41, + PRELOAD_ANIMATION = 42, + PLAY_ANIMATION = 43, + ANIMATION_COMPLETE = 44, + ENABLE_HIGHLIGHT = 45, + DISABLE_HIGHLIGHT = 46, + GET_ANIMATION_NAMES = 47, + CONTROL_BEHAVIORS = 48, + BLEND_PRIMARY_ANIMATION = 52, + SET_OFFSCREEN_ANIMATION = 53, + GET_MOVEMENT_INPUT_VALUES = 54, + SWAP_TEXTURE = 55, + SWAP_COLOR = 56, + ATTACH_HAIR = 57, + GET_ENTITY_STRUCT = 58, + SET_ENTITY_STRUCT = 59, + SET_ATTR = 60, + GET_ATTR = 61, + ON_HIT = 62, + HIT_OR_HEAL_RESULT = 63, + SHOW_ATTACK = 64, + GO_TO = 65, + GET_CONFIG_DATA = 66, + SET_CONFIG_DATA = 68, + GET_INVENTORY_EXTRA_INFO = 69, + GET_DISPLAY_NAME = 70, + GET_NAME = 71, + SET_NAME = 72, + IS_NAME_LOCALIZED = 73, + GET_HAIR_COLOR = 74, + SET_HAIR_COLOR = 75, + GET_HAIR_STYLE = 76, + SET_HAIR_STYLE = 77, + GET_HEAD = 78, + SET_HEAD = 79, + GET_TORSO = 80, + SET_TORSO = 81, + GET_LEGS = 82, + SET_LEGS = 83, + SET_PROXIMITY_RADIUS = 84, + PROXIMITY_UPDATE = 85, + GET_PROXIMITY_OBJECTS = 86, + UNSET_PROXIMITY_RADIUS = 87, + CLEAR_PROXIMITY_RADIUS = 88, + GET_PROXIMITY_DATA = 89, + SET_PROXIMITY_RADIUS_ICON = 90, + TOGGLE_TAC_ARC = 93, + CAST_SKILL = 95, + CAST_LOCAL_SKILL = 96, + ECHO_LOCAL_SKILL = 97, + QUEUE_AI_SKILL = 98, + ADD_THREAT_RATING = 99, + GET_THREAT_RATING = 100, + CLEAR_THREAT_LIST = 103, + GET_TIME_FOR_NPC_SKILL = 111, + ENEMY_HEAL_NOTIFICATION = 112, + RESET_SCRIPTED_AI_STATE = 113, + ENABLE_COMBAT_AI_COMPONENT = 114, + COMBAT_AI_FORCE_TETHER = 115, + SUSPEND_MOVEMENT_AI = 116, + NOTIFY_SCRIPT_VARS_INITIALIZED = 117, + ECHO_START_SKILL = 118, + START_SKILL = 119, + CASTER_DEAD = 120, + VERIFY_ACK = 121, + ADD_PENDING_VERIFY = 122, + MAP_SKILL = 123, + SELECT_SKILL = 124, + CAST_ACTIVE_SKILL = 125, + MODIFY_SKILL_COOLDOWN = 126, + ADD_SKILL = 127, + REMOVE_SKILL = 128, + LOG = 129, + LOG_CHAT = 130, + SET_MAX_CURRENCY = 131, + GET_MAX_CURRENCY = 132, + SET_CURRENCY = 133, + GET_CURRENCY = 134, + ADD_PENDING_CURRENCY = 136, + PICKUP_CURRENCY = 137, + SERVER_DELETE_LOOT_ITEM = 138, + PICKUP_ITEM = 139, + TEAM_PICKUP_ITEM = 140, + CLIENT_DELETE_LOOT_ITEM = 141, + CLIENT_SET_LOOT_ITEM_FFA = 143, + COLLISION_PHANTOM = 144, + OFF_COLLISION_PHANTOM = 145, + COLLISION_PROXIMITY = 146, + OFF_COLLISION_PROXIMITY = 147, + COLLISION = 148, + OFF_COLLISION = 149, + GET_SKILLS = 150, + CLEAR_FX_SINGLE_EFFECT = 152, + GET_FX_EXIST_EFFECT = 153, + PLAY_FX_EFFECT = 154, + STOP_FX_EFFECT = 155, + CLEAR_FX_ALL_CREATE_EFFECTS = 156, + UPDATE_FX_ALL_CREATE_EFFECTS = 157, + REQUEST_RESURRECT = 159, + RESURRECT = 160, + UPDATE_FROM_GHOST = 162, + FETCH_GHOST = 163, + KFM_LOADED = 164, + NIF_LOADED = 165, + HKX_LOADED = 166, + MOVE_TO_DELETE_QUEUE = 167, + RESTORE_FROM_DELETE_QUEUE = 168, + IS_ENEMY = 169, + GET_FACTION = 170, + SET_IMAGINATION = 171, + GET_IMAGINATION = 172, + SET_MAX_IMAGINATION = 173, + GET_MAX_IMAGINATION = 174, + MODIFY_IMAGINATION = 175, + MODIFY_MAX_IMAGINATION = 176, + SET_HEALTH = 177, + GET_HEALTH = 178, + SET_MAX_HEALTH = 179, + GET_MAX_HEALTH = 180, + MODIFY_HEALTH = 181, + MODIFY_MAX_HEALTH = 182, + SET_ARMOR = 183, + GET_ARMOR = 184, + SET_MAX_ARMOR = 185, + GET_MAX_ARMOR = 186, + MODIFY_ARMOR = 187, + MODIFY_MAX_ARMOR = 188, + POP_HEALTH_STATE = 190, + PUSH_EQUIPPED_ITEMS_STATE = 191, + POP_EQUIPPED_ITEMS_STATE = 192, + SET_GM_LEVEL = 193, + GET_GM_LEVEL = 194, + ADD_STATUS_EFFECT = 196, + REMOVE_STATUS_EFFECT = 197, + SET_STUNNED = 198, + GET_STUNNED = 199, + SET_STUN_IMMUNITY = 200, + GET_STUN_IMMUNITY = 201, + KNOCKBACK = 202, + SET_VISIBLE = 203, + GET_VISIBLE = 204, + REPORT_ITEM_INFO = 205, + GET_REBUILD_STATE = 207, + REBUILD_CANCEL = 209, + REBUILD_START = 211, + ENABLE_REBUILD = 213, + SKILL_FAILURE = 214, + IS_ATTACK_STANCE = 216, + SET_OBJECT_RENDER = 217, + REQUEST_MAPPED_SKILLS = 218, + UI_SELECT_MAPPED_SKILL = 219, + GET_INVENTORY_ITEM_IN_SLOT = 220, + GET_FIRST_INVENTORY_ITEM_BY_LOT = 221, + GET_SMALLEST_INVENTORY_STACK_BY_LOT = 222, + MOVE_ITEM_IN_INVENTORY = 224, + ADD_ITEM_TO_INVENTORY_CLIENT_SYNC = 227, + GET_EQUIPPED_ITEMS = 229, + REMOVE_ITEM_FROM_INVENTORY = 230, + EQUIP_INVENTORY = 231, + UN_EQUIP_INVENTORY = 233, + EQUIP_ITEM = 234, + UN_EQUIP_ITEM = 235, + IS_ITEM_RESPOND = 236, + IS_ITEM_EQUIPPED = 237, + ATTACH_ITEM = 238, + DETACH_ITEM = 239, + GET_NODE = 240, + GET_LOT = 241, + IS_ITEM_EQUIPPABLE = 242, + GET_CURRENT_ANIMATION = 243, + GET_INV_ITEM_COUNT = 244, + POST_LOAD_EQUIP = 245, + SET_PHYSICS_ACTIVE_STATE = 246, + GET_CURRENT_SKILL_TAC_ARC = 247, + OFFER_MISSION = 248, + RESPOND_TO_MISSION = 249, + GET_MISSION_STATE = 250, + GET_MISSION_COMPLETE_TIMESTAMP = 251, + NOTIFY_MISSION = 254, + NOTIFY_MISSION_TASK = 255, + ARE_GFX_LOADED = 257, + ADDED_TO_WORLD = 258, + REMOVE_EXTRA_GFX_FROM_PIPE = 259, + HIDE_EQUIPED_WEAPON = 260, + UN_HIDE_EQUIPED_WEAPON = 261, + GET_ITEM_SLOT = 262, + IS_CHARACTER = 263, + SET_IMMUNITY = 264, + TOGGLE_TOOLTIPS = 266, + GET_TOOLTIPS_DISABLED = 267, + GET_BOUNDING_INFO = 268, + OVERRIDE_BOUNDING_RADIUS = 269, + GET_OFFSCREEN = 270, + USE_STATE_MACHINE = 271, + ADD_STATE = 272, + ADD_SUB_STATE = 273, + SET_STATE = 274, + SET_SUB_STATE = 275, + ADD_MESSAGE = 276, + RELOAD_SCRIPT = 277, + RELOAD_ALL_SCRIPTS = 278, + FRIEND_INVITE_MSG = 279, + ADD_FRIEND_REPOSNSE_MSG = 280, + REMOVE_FRIEND_RESPONSE_MSG = 281, + ADD_FRIEND_FROM_UI_MSG = 282, + GET_CACHED_FRIENDS_LIST_MSG = 283, + REQUEST_NEW_FRIENDS_LIST_MSG = 284, + REPOPULATE_FRIENDS_LIST_MSG = 285, + ADD_IGNORE_REPONSE_MSG = 286, + REMOVE_IGNORE_RESPONSE_MSG = 287, + ADD_IGNORE_FROM_UI_MSG = 288, + GET_CACHED_IGNORE_LIST_MSG = 289, + REQUEST_NEW_IGNORE_LIST_MSG = 290, + REMOVE_FRIEND_BY_NAME = 291, + REMOVE_IGNORE_BY_NAME = 292, + IS_PLAYER_IN_IGNORE_LIST_MSG = 293, + REPOPULATE_IGNORE_LIST_MSG = 294, + GET_INVENTORY_LIST = 295, + UPDATE_FRIEND_MSG = 296, + UPDATE_FRIEND_NAME_MSG = 297, + UPDATE_IGNORE_NAME_MSG = 298, + DEPARTED = 299, + ARRIVED = 300, + TEMPLATE_CHANGE_WAYPOINTS = 301, + CANCELLED = 302, + FLUSH_CACHED_GRAPHICS = 303, + FOLLOW_TARGET = 304, + TIMER_DONE = 305, + TIMER_CANCELLED = 306, + SET_TETHER_POINT = 307, + GET_TETHER_POINT = 308, + LEFT_TETHER_RADIUS = 309, + GET_SCRIPT_VARS_PTR = 310, + FACE_TARGET = 311, + ROTATE_BY_DEGREES = 312, + STRING_RENDERED = 313, + RESET_PRIMARY_ANIMATION = 314, + FACE_PLAY_STREAM = 315, + TORSO_PLAY_STREAM = 316, + CAN_PICKUP = 317, + GET_INVENTORY_SIZE = 318, + GET_INVENTORY_COUNT = 319, + GET_OBJECTS_IN_GROUP = 320, + HIDE_ITEM = 321, + IS_OBJECT_IN_FOV = 322, + GET_TYPE = 323, + TEAM_INVITE_MSG = 324, + TEAM_GET_SIZE = 325, + TEAM_REQUEST_SET_LOOT = 326, + TEAM_REMOVE_PLAYER_MSG = 327, + TEAM_UPDATE_PLAYER_NAME_MSG = 328, + SET_UPDATABLE = 329, + REQUEST_TEAM_UI_UPDATE = 330, + SET_COLLISION_GROUP = 331, + GET_COLLISION_GROUP = 332, + GET_ORIGINAL_COLLISION_GROUP = 333, + SET_COLLISION_GROUP_TO_ORIGINAL = 334, + GET_OBJECT_RADIUS = 335, + REBUILD_NOTIFY_STATE = 336, + GET_PLAYER_INTERACTION = 337, + SET_PLAYER_INTERACTION = 338, + FORCE_PLAYER_TO_INTERACT = 339, + GET_SELECTED_POTENTIAL_INTERACTION = 340, + SET_SELECTED_POTENTIAL_INTERACTION = 341, + GET_INTERACTION_DISTANCE = 342, + SET_INTERACTION_DISTANCE = 343, + CALCULATE_INTERACTION_DISTANCE = 344, + INTERACTION_ATTEMPT_FROM_OUT_OF_RANGE = 345, + SET_PICKING_TARGET = 346, + CLIENT_UNUSE = 347, + BEGIN_PET_INTERACTION = 348, + WANTS_INTERACTION_ICON = 349, + PROPERTY_EDIT_ICON_INTERACTION = 350, + PROPERTY_MODEL_INTERACTION = 351, + GET_INTERACTION_DETAILS = 352, + GET_DISABLED_INTERACTION_TYPES = 353, + GET_INTERACTION_INFO = 354, + INTERACTION_GAME_STATE_CHANGE = 355, + TOGGLE_INTERACTION_UPDATES = 356, + TERMINATE_INTERACTION = 357, + SERVER_TERMINATE_INTERACTION = 358, + GET_PLAYERS_TARGET_FOR_SELECTION = 359, + PROCESS_INTERACTION_UNDER_CURSOR = 360, + HANDLE_INTERACT_ACTION = 361, + ATTEMPT_INTERACTION = 362, + HANDLE_INTERACTION_CAMERA = 363, + REQUEST_USE = 364, + CLIENT_USE = 366, + GET_PLAYER_MULTI_INTERACTION = 367, + GET_MULTI_INTERACTION_STATE = 368, + VENDOR_OPEN_WINDOW = 369, + VENDOR_CLOSE_WINDOW = 370, + EMOTE_PLAYED = 371, + EMOTE_RECEIVED = 372, + BUY_FROM_VENDOR = 373, + SELL_TO_VENDOR = 374, + ADD_DONATION_ITEM = 375, + REMOVE_DONATION_ITEM = 376, + CONFIRM_DONATION_ON_PLAYER = 378, + CANCEL_DONATION_ON_PLAYER = 379, + TEAM_GET_LEADER = 380, + TEAM_GET_ON_WORLD_MEMBERS = 381, + TEAM_GET_ALL_MEMBERS = 382, + TEAM_SET_OFF_WORLD_FLAG = 383, + SET_TRANSPARENCY = 385, + GET_PREFERS_FADE = 386, + PROJECTILE_IMPACT = 387, + SET_PROJECTILE_PARAMS = 388, + SET_INVENTORY_SIZE = 389, + ACKNOWLEDGE_POSSESSION = 391, + SET_POSSESSED_OBJECT = 392, + CHANGE_POSSESSOR = 393, + GET_POSSESSION_TYPE = 395, + GET_POSSESSED_OBJECT = 396, + GET_POSSESSOR = 397, + IS_POSSESSED = 398, + ENABLE_ACTIVITY = 399, + SET_SHOOTING_GALLERY_PARAMS = 400, + OPEN_ACTIVITY_START_DIALOG = 401, + REQUEST_ACTIVITY_START_STOP = 402, + REQUEST_ACTIVITY_ENTER = 403, + REQUEST_ACTIVITY_EXIT = 404, + ACTIVITY_ENTER = 405, + ACTIVITY_EXIT = 406, + ACTIVITY_START = 407, + ACTIVITY_STOP = 408, + SHOOTING_GALLERY_CLIENT_AIM_UPDATE = 409, + ROTATE_TO_POINT = 410, + SHOOTING_GALLERY_FIRE = 411, + CALCULATE_FIRING_PARAMETERS = 412, + GET_MUZZLE_OFFSET = 413, + GET_ACTIVITY_POINTS = 414, + TEAM_IS_ON_WORLD_MEMBER = 415, + REQUEST_VENDOR_STATUS_UPDATE = 416, + VENDOR_STATUS_UPDATE = 417, + CANCEL_MISSION = 418, + RESET_MISSIONS = 419, + RENDER_COMPONENT_READY = 420, + SEND_MINIFIG_DECALS = 421, + PHYSICS_COMPONENT_READY = 422, + ENTER_STANDBY_MODE = 423, + LEAVE_STANDBY_MODE = 424, + NOTIFY_CLIENT_SHOOTING_GALLERY_SCORE = 425, + REQUEST_CONSUME_ITEM = 426, + CONSUME_CLIENT_ITEM = 427, + CLIENT_ITEM_CONSUMED = 428, + QUERY_STANDBY_MODE = 429, + GET_NI_BOUND = 430, + MISSION_FAILURE = 431, + GET_ANIMATION_TIME = 432, + GET_CURRENT_ACTIVITY = 434, + SET_EYEBROWS = 435, + GET_EYEBROWS = 436, + SET_EYES = 437, + GET_EYES = 438, + SET_MOUTH = 439, + GET_MOUTH = 440, + IS_OBJECT_SMASHABLE = 441, + SMASHABLE_STATE_CHANGED = 443, + USE_STATE_LOGGER = 444, + ROTATE_SUB_NODE = 445, + GET_SUB_NODE_POSITION = 446, + GET_SUB_NODE = 447, + UPDATE_SHOOTING_GALLERY_ROTATION = 448, + RENDER_FLOATING_TEXT = 449, + REQUEST2_D_TEXT_ELEMENT = 450, + UPDATE2_D_TEXT_ELEMENT = 451, + REMOVE2_D_TEXT_ELEMENT = 452, + SET_COLOR = 453, + GET_COLOR = 454, + HKX_CHARACTER_LOADED = 455, + ACTIVATE_PHYSICS = 457, + SET_ICON_ABOVE_HEAD = 458, + ADD_ICON_COMPOSITE = 459, + CLEAR_ICON_COMPOSITES = 460, + ICON_NIF_LOADED = 461, + ICON_KFM_LOADED = 462, + GET_OVERHEAD_ICON_PROPERTIES_FROM_PARENT = 463, + BOUNCE_PLAYER = 464, + SET_USER_CTRL_COMP_PAUSE = 466, + HAS_COLLIDED = 467, + GET_TOOLTIP_FLAG = 468, + SET_TOOLTIP_FLAG = 469, + GET_FLAG = 470, + SET_FLAG = 471, + NOTIFY_CLIENT_FLAG_CHANGE = 472, + CURSOR_ON = 473, + CURSOR_OFF = 474, + HELP = 475, + VENDOR_TRANSACTION_RESULT = 476, + PERFORM_SPECIAL_DEATH = 477, + GET_SHADER_ID = 478, + GET_RENDER_ENVIRONMENT = 479, + FINISHED_LOADING_SCENE = 480, + GET_SKILL_INFO = 481, + ACTIVITY_CANCEL = 482, + MISSION_USES_OBJECT = 483, + GET_POSITIONAL_ID = 484, + SET_COLLECTIBLE_STATUS = 485, + HAS_BEEN_COLLECTED = 486, + HAS_BEEN_COLLECTED_BY_CLIENT = 487, + GET_POS_UPDATE_STATS = 488, + GET_NUM_VIEWERS_SCOPING_THIS = 489, + GET_ACTIVITY_USER = 490, + GET_ALL_ACTIVITY_USERS = 491, + GET_MISSION_FOR_PLAYER = 492, + SET_FACTION = 493, + SET_PLATFORM_IDLE_STATE = 494, + DISPLAY_CHAT_BUBBLE = 495, + REQUEST_CHAT_BUBBLE_ELEMENT = 496, + GET_MISSION_DATA = 497, + SPAWN_PET = 498, + DESPAWN_PET = 499, + SET_LOCAL_SPACE_STATE = 500, + GET_LOCAL_SPACE_STATE = 501, + SET_POSITION_TO_LOCAL_POSITION = 502, + ALLOW_LOCAL_SPACE_UPDATE = 503, + TOGGLE_FREE_CAM_MODE = 504, + PLAYER_LOADED = 505, + PLAYER_ADDED_TO_WORLD_LOCAL = 506, + OBJECT_LOADED = 507, + GET_PLAYER_READY = 508, + PLAYER_READY = 509, + SET_SMASHABLE_PARAMS = 510, + IS_LOOTABLE_CHEST = 511, + LOOT_OPEN_WINDOW = 512, + LOOT_SELECTION_UPDATE = 513, + TAKE_LOOT_CHEST_ITEM = 514, + REQUEST_LINKED_MISSION = 515, + TRANSFER_TO_ZONE = 516, + TRANSFER_TO_ZONE_CHECKED_IM = 517, + SECURED_TRANSFER_TO_ZONE = 518, + INVALID_ZONE_TRANSFER_LIST = 519, + MISSION_DIALOGUE_OK = 520, + GET_OBJECT_IN_SCOPE = 521, + SET_LAUNCHED_STATE = 522, + P_CREATE_EFFECT_FINISHED = 523, + SMASHED_OBJECT = 524, + CHECK_SMASHCHAIN_OVERRIDE = 525, + DISPLAY_REBUILD_ACTIVATOR = 526, + TRANSFER_TO_LAST_NON_INSTANCE = 527, + SET_ACTIVE_LOCAL_CHARACTER_ID = 528, + DISPLAY_MESSAGE_BOX = 529, + MESSAGE_BOX_RESPOND = 530, + CHOICE_BOX_RESPOND = 531, + SERVER_SET_USER_CTRL_COMP_PAUSE = 532, + SET_CHARACTER_AUTO_RUN = 533, + FOLLOW_WAYPOINTS = 534, + SWAP_DECAL_AND_COLOR = 535, + CONTINUE_WAYPOINTS = 536, + SMASH = 537, + UN_SMASH = 538, + GET_IS_SMASHED = 539, + GET_UP_VECTOR = 540, + SET_GRAVITY_SCALE = 541, + SET_GRAVITY_SCALE_FOR_RIGID_BODY = 542, + STOP_MOVING = 543, + SET_PATHING_SPEED = 544, + SET_SHIELDED = 545, + SET_SHOOTING_GALLERY_RETICULE_EFFECT = 546, + PLACE_MODEL_RESPONSE = 547, + SET_DODGE_INFO = 548, + GET_DODGE_INFO = 549, + SET_SKILL_ATTACK_SPEED = 550, + GET_SKILL_COOLDOWN_GROUP = 551, + GET_INITIAL_SKILL_COOLDOWN = 552, + GET_SKILL_COOLDOWN_REMAINING = 553, + GET_GLOBAL_COOLDOWN = 554, + SET_GLOBAL_COOLDOWN = 555, + RESET_GLOBAL_COOLDOWN = 556, + FINDINVENTORY_ITEM = 558, + PATH_STUCK = 559, + SET_CURRENT_PATH = 560, + SET_JET_PACK_MODE = 561, + SET_JET_PACK_TIME = 562, + PET_FOLLOW_OWNER = 563, + PLAYER_DIED = 564, + REGISTER_PET_ID = 565, + REGISTER_PET_DBID = 566, + GET_PET_ID = 567, + SHOW_ACTIVITY_COUNTDOWN = 568, + DISPLAY_TOOLTIP = 569, + SET_PHANTOM_BASE = 570, + GET_MOTION_STATE = 571, + GET_MOTION_CONFIG = 572, + SET_ACTIVE_PROJECTILE_SKILL = 573, + INITIALIZE_MISSION_VISUALS = 574, + GET_MISSIONS = 575, + START_ACTIVITY_TIME = 576, + ADD_ACTIVITY_TIME = 577, + GUILD_GET_SIZE = 578, + GUILD_CAN_WE_INVITE = 579, + GUILD_CAN_WE_KICK = 580, + SET_CHAR_GUILD_INFO = 581, + GET_CHAR_GUILD_INFO = 582, + GET_CHAR_IS_IN_GUILD = 583, + RE_RENDER_NAME_BILLBOARD = 584, + IS_IN_LOCAL_CHAR_PROXIMITY = 585, + GUILD_SET_STATUS = 586, + GUILD_ADD_PLAYER = 587, + GUILD_REMOVE_PLAYER = 588, + GUILD_UPDATE_PLAYER_NAME = 589, + GUILD_SET_PLAYER_RANK = 590, + GUILD_SET_ONLINE_STATUS = 591, + GUILD_INVITE = 592, + REQUEST_GUILD_DATA = 593, + POPULATE_GUILD_DATA = 594, + GET_CACHED_GUILD_DATA = 595, + GUILD_RENDER_NAME = 596, + GET_IS_SUPPORTED = 600, + CHARACTER_SUPPORT_CHANGED = 601, + ACTIVITY_PAUSE = 602, + USE_NON_EQUIPMENT_ITEM = 603, + REQUEST_USE_ITEM_ON = 604, + REQUEST_USE_ITEM_ON_TARGET = 605, + USE_ITEM_ON = 606, + USE_ITEM_RESULT = 607, + GET_PARENT_OBJ = 608, + SET_PARENT_OBJ = 609, + GET_UPDATES_WITH_PARENT_POSITION = 610, + PARENT_REMOVED = 611, + PARENT_LEFT_SCOPE = 612, + PARENT_ENTERED_SCOPE = 613, + CHILD_LOADED = 614, + CHILD_REMOVED = 615, + CHILD_DETACHED = 616, + CHILD_ENTERED_SCOPE = 617, + CHILD_LEFT_SCOPE = 618, + GET_CHILD_OBJECTS = 619, + ZONE_TRANSFER_FINISHED = 621, + CHAT_CONNECTION_UPDATE = 622, + PLATFORM_AT_LAST_WAYPOINT = 623, + LOOT_TAKE_ALL = 624, + GET_EQUIPPED_ITEM_INFO = 625, + DISPLAY_GUILD_CREATE_BOX = 626, + GET_EDITOR_LEVEL = 627, + GET_ACCOUNT_ID = 628, + GET_LAST_LOGOUT = 629, + GET_LAST_PROP_MOD_DISPLAY_TIME = 630, + SET_LAST_PROP_MOD_DISPLAY_TIME = 631, + SHOW_ACTIVITY_SUMMARY = 632, + CAN_RECEIVE_ALL_REWARDS = 633, + GET_ACTIVITY_REWARD = 634, + LOOT_CLOSE_WINDOW = 635, + GET_BLUEPRINT_ID = 636, + NOTIFY_BLUEPRINT_UPDATE = 637, + FETCH_MODEL_METADATA_REQUEST = 638, + FETCH_MODEL_METADATA_RESPONSE = 639, + COMMAND_PET = 640, + PET_RESPONSE = 641, + GET_ICON_ABOVE_HEAD_STATE = 642, + GET_ICON_ABOVE_HEAD = 643, + ICON_FINISHED_LOADING = 644, + ADD_PET_STATE = 645, + REMOVE_PET_STATE = 646, + SET_PET_STATE = 647, + REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA = 648, + SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA = 649, + SET_ON_TEAM = 650, + GET_PET_HAS_STATE = 651, + FIND_PROPERTY = 652, + SET_PET_MOVEMENT_STATE = 653, + GET_ITEM_TYPE = 654, + GET_ITEM_INFO_KEY = 655, + NOTIFY_OBJECT = 656, + IS_PET_WILD = 657, + CLIENT_NOTIFY_PET = 659, + NOTIFY_PET = 660, + NOTIFY_PET_TAMING_MINIGAME = 661, + START_SERVER_PET_MINIGAME_TIMER = 662, + CLIENT_EXIT_TAMING_MINIGAME = 663, + GET_BUILDMODE_ACTIVE = 664, + GET_PET_TAMING_MINIGAME_ACTIVE = 665, + PET_TAMING_OBJECT_PICKED = 666, + PET_TAMING_MINIGAME_RESULT = 667, + PET_TAMING_TRY_BUILD_RESULT = 668, + SET_PET_TAMING_MODEL = 669, + GET_PET_TAMING_MODEL = 670, + PET_ON_SWITCH = 671, + PET_OFF_SWITCH = 672, + NOTIFY_TAMING_BUILD_SUCCESS = 673, + NOTIFY_TAMING_MODEL_LOADED_ON_SERVER = 674, + NOTIFY_TAMING_PUZZLE_SELECTED = 675, + GET_INSTRUCTION_COUNT = 676, + GET_IS_NPC = 677, + ACTIVATE_BUBBLE_BUFF = 678, + DECTIVATE_BUBBLE_BUFF = 679, // thanks netdevil + EXHIBIT_VOTE = 680, + ADD_PET_TO_PLAYER = 681, + REMOVE_PET_FROM_PLAYER = 682, + REQUEST_SET_PET_NAME = 683, + SET_PET_NAME = 684, + PET_NAME_CHANGED = 686, + GET_PET_AT_INDEX = 687, + GET_LOT_FOR_PET_BY_DBID = 688, + GET_NAME_FOR_PET_BY_DBID = 689, + GET_ACTIVE_PET_OBJ_ID = 690, + GET_ACTIVE_PET_INVENTORY_OBJ_ID = 691, + SHOW_PET_ACTION_BUTTON = 692, + SET_EMOTE_LOCK_STATE = 693, + GET_EMOTE_LOCK_STATE = 694, + LEAVE_TEAM_MSG = 695, + TEAM_KICK_PLAYER_MSG = 697, + TEAM_SET_LEADER_SEND_MSG = 698, + USE_ITEM_ON_CLIENT = 699, + DOES_FORWARD_TARGET_CLICKING = 700, + CHECK_USE_REQUIREMENTS = 701, + USE_REQUIREMENTS_RESPONSE = 702, + USE_ITEM_REQUIREMENTS_RESPONSE = 703, + PET_ADDED_TO_WORLD = 704, + BOUNCER_TRIGGERED = 705, + EXHIBIT_QUERY_CURRENT_MODEL = 706, + EXHIBIT_QUERY_CURRENT_MODEL_RESPONSE = 707, + EXHIBIT_ATTEMPT_VOTE = 708, + EXHIBIT_VOTE_RESPONSE = 709, + EHIBIT_REQUERYMODELS = 710, + IS_SKILL_ACTIVE = 711, + TOGGLE_ACTIVE_SKILL = 712, + PLAY_EMBEDDED_EFFECT_ON_ALL_CLIENTS_NEAR_OBJECT = 713, + EXHIBIT_GET_INFO = 714, + GET_PROPERTY_DATA = 715, + DOWNLOAD_PROPERTY_DATA = 716, + QUERY_PROPERTY_DATA = 717, + MODEL_MODERATION_ACTION = 719, + NOTIFY_SERVER_UGC_REVIEW_READY = 720, + NOTIFY_CLIENT_UGC_REVIEW_READY = 721, + OLD_USE_ITEM_ON = 722, + FIND_PROPERTY_FOR_SALE_RESPONSE = 723, + PROPERTY_EDITOR_BEGIN = 724, + PROPERTY_EDITOR_END = 725, + PROPERTY_EDITOR_SET_MODE = 726, + TOGGLE_TRIGGER = 727, + FIRE_EVENT = 728, + IS_MINIFIG_IN_A_BUBBLE = 729, + GET_ITEM_INFO = 730, + MISSION_NEEDS_LOT = 731, + STOP_PATHING = 732, + START_PATHING = 733, + ACTIVATE_BUBBLE_BUFF_FROM_SERVER = 734, + DEACTIVATE_BUBBLE_BUFF_FROM_SERVER = 735, + HAS_SKILL = 736, + NOTIFY_CLIENT_ZONE_OBJECT = 737, + MOVE_OBJECT = 738, + ROTATE_OBJECT = 739, + GET_SPAWNER_CONFIG_DATA = 740, + UPDATE_SPAWNER_CONFIG_DATA = 741, + TURN_AROUND = 743, + GO_FORWARD = 744, + GO_BACKWARD = 745, + UPDATE_REPUTATION = 746, + GET_REPUTATION = 747, + ADD_REPUTATION = 748, + UPDATE_PROPERTY_DATA = 749, + PROPERTY_RENTAL_RESPONSE = 750, + EXHIBIT_PLACEMENT_RESPONSE = 751, + SQUIRT_WITH_WATERGUN = 752, + GET_VOTES_LEFT = 753, + ADJUST_VOTES_LEFT = 754, + EVADE_TARGET = 755, + STOPPED_EVADING = 756, + GET_PET_HAS_ABILITY = 757, + REQUEST_PLATFORM_RESYNC = 760, + PLATFORM_RESYNC = 761, + PLAY_CINEMATIC = 762, + END_CINEMATIC = 763, + CINEMATIC_UPDATE = 764, + ATTACH_CAMERA_TO_RAIL = 765, + DETACH_CAMERA_FROM_RAIL = 766, + TOGGLE_GHOST_REFERENCE_OVERRIDE = 767, + SET_GHOST_REFERENCE_POSITION = 768, + GET_GHOST_REFERENCE_POSITION = 769, + FIRE_EVENT_SERVER_SIDE = 770, + GET_PET_ABILITY_OBJECT = 771, + TEAM_INVITE_MSG_FROM_UI = 772, + ADD_CAMERA_EFFECT = 773, + REMOVE_CAMERA_EFFECT = 774, + REMOVE_ALL_CAMERA_EFFECTS = 775, + GET_MY_PROPERTIES_IN_THIS_ZONE = 776, + IS_MODEL_WITHIN_PROPERTY_BOUNDS = 777, + PROPERTY_DATA_RESULTS = 778, + ON_UN_SERIALIZE = 779, + SCRIPT_NETWORK_VAR_UPDATE = 781, + ADD_OBJECT_TO_GROUP = 783, + REMOVE_OBJECT_FROM_GROUP = 784, + IS_OBJECT_STATIC = 785, + GET_HAS_MISSION = 786, + GET_MISSION_TARGET_LOT = 787, + GET_MISSION_OFFERER_LOT = 788, + USE_UNIQUE_ITEM = 789, + GET_IS_PET = 790, + DELETE_PROPERTY = 791, + CREATEMODEL_FROM_CLIENT = 792, + UPDATE_MODEL_FROM_CLIENT = 793, + DELETE_MODEL_FROM_CLIENT = 794, + SHOW_PROPERTY_BOUNDS = 795, + SET_PROPERTY_I_DS = 796, + PLAY_FACE_DECAL_ANIMATION = 797, + ADD_ACTIVITY_USER = 798, + REMOVE_ACTIVITY_USER = 799, + GET_NUM_ACTIVITY_USERS = 800, + ACTIVITY_USER_EXISTS = 801, + DO_COMPLETE_ACTIVITY_EVENTS = 805, + SET_ACTIVITY_PARAMS = 806, + SET_ACTIVITY_USER_DATA = 807, + GET_ACTIVITY_USER_DATA = 808, + DO_CALCULATE_ACTIVITY_RATING = 809, + ND_AUDIO_POST_SETUP = 812, + ND_AUDIO_PRE_SHUTDOWN = 813, + SET_ND_AUDION_LISTENER_STANCE = 814, + SET_UP_ND_AUDIO_EMIITTER = 815, + SHUT_DOWN_ND_AUDIO_EMITTER = 816, + METAIFY_ND_AUDIO_EMITTER = 817, + UN_METAIFY_ND_AUDIO_EMITTER = 818, + METAIFY_ND_AUDIO_EMITTERS = 819, + UN_METAIFY_ND_AUDIO_EMITTERS = 820, + PLAY_ND_AUDIO_EMITTER = 821, + STOP_ND_AUDIO_EMITTER = 822, + STOP_ND_AUDIO_EMITTER_ALL = 823, + SET_ND_AUDIO_EMITTER_PARAMETER = 824, + SET_ND_AUDIO_EMITTERS_PARAMETER = 825, + ND_AUDIO_CALLBACK = 826, + ACTIVATE_ND_AUDIO_MUSIC_CUE = 827, + DEACTIVATE_ND_AUDIO_MUSIC_CUE = 828, + FLASH_ND_AUDIO_MUSIC_CUE = 829, + SET_ND_AUDIO_MUSIC_PARAMETER = 830, + PLAY2_D_AMBIENT_SOUND = 831, + STOP2_D_AMBIENT_SOUND = 832, + PLAY3_D_AMBIENT_SOUND = 834, + STOP3_D_AMBIENT_SOUND = 835, + ACTIVATE_ND_AUDIO_MIXER_PROGRAM = 836, + DEACTIVATE_ND_AUDIO_MIXER_PROGRAM = 837, + UPDATE_ACTIVITY_LEADERBOARD = 838, + ACTIVITY_LEADERBOARD_UPDATED = 839, + ENTER_PROPERTY1 = 840, + ENTER_PROPERTY2 = 841, + PROPERTY_ENTRANCE_SYNC = 842, + SEND_PROPERTY_POPULATION_TO_CLIENT = 843, + SEN_PROPERTY_PLAQUE_VIS_UPDATE = 844, + PROPERTY_SELECT_QUERY = 845, + CREATE_POSITION_STRING = 848, + GET_PARALLEL_POSITION = 849, + PARSE_CHAT_MESSAGE = 850, + SET_MISSION_TYPE_STATE = 851, + GET_LOCATIONS_VISITED = 852, + GET_MISSION_TYPE_STATES = 853, + GET_TIME_PLAYED = 854, + SET_MISSION_VIEWED = 855, + SLASH_COMMAND_TEXT_FEEDBACK = 856, + HANDLE_SLASH_COMMAND_KORE_DEBUGGER = 857, + BROADCAST_TEXT_TO_CHATBOX = 858, + OPEN_PROPERTY_MANAGEMENT = 860, + OPEN_PROPERTY_VENDOR = 861, + VOTE_ON_PROPERTY = 862, + UPDATE_PROPERTY_OR_MODEL_FOR_FILTER_CHECK = 863, + NOTIFY_PLAYER_OF_PROPERTY_SUBMISSION = 865, + NOTIFY_PLAYER_OF_MODEL_SUBMISSION = 866, + PHYSICS_SYSTEM_LOADED = 867, + CLIENT_TRADE_REQUEST = 868, + SERVER_TRADE_REQUEST = 869, + SERVER_TRADE_INVITE = 870, + CLIENT_TRADE_REPLY = 871, + SERVER_TRADE_REPLY = 872, + SERVER_TRADE_INITIAL_REPLY = 873, + SERVER_TRADE_FINAL_REPLY = 874, + CLIENT_TRADE_UPDATE = 875, + SERVER_SIDE_TRADE_UPDATE = 876, + SERVER_TRADE_UPDATE = 877, + CLIENT_TRADE_CANCEL = 878, + CLIENT_SIDE_TRADE_CANCEL = 879, + CLIENT_TRADE_ACCEPT = 880, + SERVER_SIDE_TRADE_ACCEPT = 881, + SERVER_SIDE_TRADE_CANCEL = 882, + SERVER_TRADE_CANCEL = 883, + SERVER_TRADE_ACCEPT = 884, + GET_TRADE_INFO = 885, + KF_LOADED = 886, + BRICKS_LOADED = 887, + READY_FOR_UPDATES = 888, + SEND_READY_FOR_UPDATES = 889, + SET_LAST_CUSTOM_BUILD = 890, + GET_LAST_CUSTOM_BUILD = 891, + GET_STATUS_EFFECT_BY_ID = 892, + GET_ALL_STATUS_EFFECTS = 893, + CHILD_RENDER_COMPONENT_READY = 894, + NOTIFY_APPEARANCE_CHANGED_MSG = 895, + SET_PHYSICS_MOTION_STATE = 896, + GET_PHYSICS_MOTION_STATE = 897, + ATTACH_GRAYSCALE_EFFECT = 898, + ATTACH_FADE_EFFECT = 899, + ATTACH_CHANGE_RENDER_ENVIRONMENT_EFFECT = 900, + FORCE_MOVEMENT = 901, + CANCEL_FORCE_MOVEMENT = 902, + SET_IGNORE_PROJECTILE_COLLISION = 903, + GET_IGNORE_PROJECTILE_COLLISION = 904, + ORIENT_TO_OBJECT = 905, + ORIENT_TO_POSITION = 906, + ORIENT_TO_ANGLE = 907, + NOTIFY_CLIENT_UGC_MODEL_READY = 909, + NOTIFY_CLIENT_UGC_ICON_READY = 911, + PROPERTY_BUILD_MODE_CHANGED = 912, + PROPERTY_BUILD_MODE_UPDATE = 913, + PROPERTY_DELETION_ACTION = 914, + PROPERTY_MODERATION_STATUS_ACTION = 915, + PROPERTY_MODERATION_STATUS_ACTION_RESPONSE = 916, + PROPERTY_MODERATION_STATUS_UPDATE = 917, + PROPERTY_NEEDS_GM_ATTENTION = 918, + PROPERTY_MODERATION_CHANGED = 919, + INVENTORY_REFRESH_ITEM_DETAILS = 922, + INVENTORY_LOAD_CUSTOM_ICON = 923, + GET_STATUS_EFFECT_BY_TYPE = 924, + RELEASE_CHARGED_SKILL = 925, + PROPERTY_RELOAD_DB = 926, + SET_PLAYER_TARGET = 927, + GET_PLAYER_TARGET = 928, + LOCK_CAMERA_NETWORKED = 929, + MOVE_CAMERA_NETWORKED = 930, + REBUILD_ACTIVATED = 931, + BOUNCE_NOTIFICATION = 932, + REQUEST_CLIENT_BOUNCE = 934, + GET_RECENT_BOUNCED = 935, + SET_RECENT_BOUNCED = 936, + SET_ACTIVE_STATE = 937, + GET_ACTIVE_STATE = 938, + HAS_COMPONENT_TYPE = 939, + GET_COMPONENT_LIST = 940, + RESPONDS_TO_FACTION = 941, + BOUNCER_ACTIVE_STATUS = 942, + HF_ATTRIBUTES_PUSH = 943, + HF_ATTRIBUTES_PULL = 944, + HF_ATTRIBUTES_PATH_DISPLAY = 945, + HF_CONTROLS_PULL = 946, + HF_OBJECT_SELECTED = 947, + HF_PLACEHOLDER_UPDATE = 948, + HF_PLACEHOLDER_TOGGLE = 949, + HF_GET_ASSOCIATED_PATHS = 950, + HF_GETS_WANT_PATH = 951, + GET_RECENT_MOVEMENT_KEYS = 952, + TRACK_RECENT_MOVEMENT_KEYS = 953, + PHYSICS_MOVEMENT_NOTIFICATION_REQUEST = 954, + PHYSICS_MOVEMENT_NOTIFICATION = 955, + MOVE_INVENTORY_SINGLE = 956, + MOVE_INVENTORY_BATCH = 957, + MINI_GAME_SET_PARAMETERS = 958, + MINI_GAME_GET_TEAM_SKILLS = 961, + MINI_GAME_GET_TEAM_SCORE = 963, + MINI_GAME_GET_PLAYER_SCORE = 967, + MINI_GAME_GET_TEAM_COLOR = 972, + MINI_GAME_GET_TEAM_PLAYERS = 975, + MINI_GAME_UPDATE_CLIENT = 976, + MINI_GAME_GET_TEAM = 977, + MINI_GAME_GET_PARAMETERS = 978, + OBJECT_ACTIVATED_CLIENT = 980, + IS_RESURRECTING = 983, + GET_ITEM_OWNER = 984, + GET_STORED_CONFIG_DATA = 985, + SET_STORED_CONFIG_DATA = 986, + ON_PLAYER_RESSURECTED = 988, + PLAYER_RESURRECTION_FINISHED = 989, + TRANSFORM_CHANGELING_BUILD = 990, + RETURN_CHANGELING_BUILD_ID = 991, + SPEND_BRICK_INVENTORY_FOR_LXFML = 992, + BRICK_INVENTORY_FOR_LXFML_SPENT = 993, + REBUILD_BBB_AUTOSAVE_MSG = 995, + SET_BBB_AUTOSAVE = 996, + USE_BBB_INVENTORY = 998, + UN_USE_BBB_MODEL = 999, + BBB_LOAD_ITEM_REQUEST = 1000, + BBB_SAVE_REQUEST = 1001, + BBBLUP_SAVE_REQUEST = 1002, + BBB_GET_METADATA_SOURCE_ITEM = 1003, + BBB_RESET_METADATA_SOURCE_ITEM = 1004, + BBB_SAVE_RESPONSE = 1005, + PLAYER_EXIT = 1006, + SET_PVP_STATUS = 1008, + GET_PVP_STATUS = 1009, + IS_VALID_PVP_TARGET = 1010, + PVP_RENDER_NAME = 1011, + ATTACH_OBJECT = 1012, + DETACH_OBJECT = 1013, + BOUNCE_SUCCEEDED = 1014, + GET_GAME_OBJECT_POINTER = 1015, + PHANTOM_HKX_LOADED = 1016, + DELAY_CREATE_EFFECT = 1017, + CHOICE_BUILD_SELECTION_CONFIRMED = 1018, + NOTIFY_FADE_UP_VIS_COMPLETE = 1019, + ITEM_HAS_NEW_INFO = 1020, + RESET_SECONDARY_ANIMATION = 1021, + GET_PICK_TYPE = 1022, + SET_PICK_TYPE = 1023, + GET_PRIORITY_PICK_LIST_TYPE = 1024, + REQUEST_PICK_TYPE_UPDATE = 1025, + GET_OVERRIDE_PICK_TYPE = 1026, + REQUEST_DISPLAY_OBJECT_INFO = 1027, + REQUEST_SERVER_OBJECT_INFO = 1028, + REQUEST_OBJECT_INFO_AS_XML = 1029, + GET_OBJECT_REPORT_INFO = 1030, + GET_OBJECT_REPORT_WINDOW_CLOSE = 1031, + GET_OBJECT_REPORT_STATUS = 1032, + GET_MISSION_DATA_FOR_OBJECT_REPORT = 1033, + GET_OBJECT_ROLLOVER_INFO = 1034, + PERFORM_ZONE_ANALYSIS = 1035, + UPDATE_HK_VISUAL_IZATION = 1036, + CLEAR_ITEMS_OWNER = 1037, + APPLY_LINEAR_IMPULSE = 1038, + APPLY_ANGULAR_IMPULSE = 1039, + GET_CONTACT_NORMALS = 1040, + IS_WATCHING_FOR_EMOTE = 1041, + NOTIFY_CLIENT_OBJECT = 1042, + DISPLAY_ZONE_SUMMARY = 1043, + ZONE_SUMMARY_DISMISSED = 1044, + GET_PLAYER_ZONE_STATISTIC = 1045, + MODIFY_PLAYER_ZONE_STATISTIC = 1046, + APPLY_EXTERNAL_FORCE = 1049, + GET_APPLIED_EXTERNAL_FORCE = 1050, + ITEM_EQUIPPED = 1052, + ACTIVITY_STATE_CHANGE_REQUEST = 1053, + OVERRIDE_FRICTION = 1054, + ARRANGE_WITH_ITEM = 1055, + CHECK_CAN_BUILD_WITH_ITEM = 1056, + START_BUILDING_WITH_ITEM = 1057, + START_BUILD_SESSION = 1058, + FINISH_BUILD_SESSION = 1059, + DONE_BUILD_SESSION = 1060, + START_ARRANGING_WITH_ITEM = 1061, + FINISH_ARRANGING_WITH_ITEM = 1062, + DONE_ARRANGING_WITH_ITEM = 1063, + START_ARRANGE_MODE = 1064, + ARRANGE_MODE_WITH_ITEM = 1065, + FINISH_ARRANGE_MODE = 1066, + DONE_ARRANGE_MODE = 1067, + SET_BUILD_MODE = 1068, + BUILD_MODE_SET = 1069, + CONFIRM_BUILD_MODE = 1070, + BUILD_MODE_CONFIRMATION = 1071, + BUILD_EXIT_CONFIRMATION = 1072, + SET_BUILD_MODE_CONFIRMED = 1073, + BUILD_MODE_NOTIFICATION = 1074, + BUILD_MODE_NOTIFICATION_REPORT = 1075, + CLIENT_USE_MODULE_ON = 1076, + SET_MODEL_TO_BUILD = 1077, + SPAWN_MODEL_BRICKS = 1078, + CHECK_PRECONDITION = 1079, + CHECK_ALL_PRECONDITIONS = 1080, + NOTIFY_CLIENT_FAILED_PRECONDITION = 1081, + GET_IS_ITEM_EQUIPPED_BY_LOT = 1082, + GET_IS_ITEM_EQUIPPED_BY_ID = 1083, + GET_OBJECT_DIRECTION_VECTORS = 1084, + GET_CASTABLE_SKILLS = 1085, + CHOICEBUILD_COMPLETE = 1086, + GET_MISSION_CHAT = 1087, + GET_MISSION_AUDIO = 1088, + MODULE_EQUIPPED = 1089, + MODULE_DROPPED = 1090, + MODULE_PICKED_UP = 1091, + MODULE_INFO = 1092, + MOVE_ITEM_BETWEEN_INVENTORY_TYPES = 1093, + MODULAR_BUILD_BEGIN = 1094, + MODULAR_BUILD_END = 1095, + MODULAR_BUILD_MOVE_AND_EQUIP = 1096, + MODULAR_BUILD_FINISH = 1097, + SET_REGISTRATION_FOR_UI_UPDATE = 1114, + GO_TO_WAYPOINT = 1115, + ARRIVED_AT_DESIRED_WAYPOINT = 1116, + CHECK_WITHIN_BOUNDS = 1117, + ATTACH_TO_BUILD_ASSEMBLY = 1118, + SET_BUILD_ASSEMBLY = 1119, + RESET_BUILD_ASSEMBLY = 1120, + GET_INVENTORY_ITEM_INFO = 1125, + GET_ITEM_DETAILS = 1126, + GET_BUILD_ACTIVATOR = 1127, + GET_MISSION_ANIMATION = 1128, + MISSION_DIALOGUE_CANCELLED = 1129, + MODULE_ASSEMBLY_DB_DATA = 1130, + MODULE_ASSEMBLY_DB_DATA_FOR_CLIENT = 1131, + MODULE_ASSEMBLY_QUERY_DATA = 1132, + MODULE_ASSEMBLY_HKX_LOADED = 1133, + MODULE_ASSEMBLY_NIF_LOADED = 1134, + MODULE_ASSEMBLY_MAIN_NIF_LOADED = 1135, + MODULE_ASSEMBLY_KFM_LOADED = 1136, + GET_PRECONDITION_INFO = 1137, + GET_MODEL_LOT = 1138, + ANIMATION_FINISHED_PRELOADING = 1139, + CHILD_BUILD_ASSEMBLY_COMPLETE = 1140, + CHARACTER_UNSERIALIZED = 1141, + CHARACTER_NEEDS_TRANSITION = 1142, + SET_NEEDS_TRANSITION = 1143, + ECHO_SYNC_SKILL = 1144, + SYNC_SKILL = 1145, + GET_BEHAVIOR_HANDLE = 1146, + ADD_OUTSTANDING_BEHAVIOR = 1147, + REQUEST_SERVER_PROJECTILE_IMPACT = 1148, + OFF_WORLD_IMPACT_REQUEST = 1149, + SERVER_IMPACT_REQUEST = 1150, + DO_CLIENT_PROJECTILE_IMPACT = 1151, + MODULE_ASSEMBLY_PART_INFO = 1152, + GET_BUILD_TYPE = 1153, + CHECK_BUILD_TYPE = 1154, + MODULAR_BUILD_CONVERT_MODEL = 1155, + DO_NPC_SHOWCASE_MODEL_SUBMISSION = 1156, + GET_MISSION_I_DS_LIST = 1157, + SET_SHOWCASE_MISSION_NPC_VALS = 1158, + NOTIFY_SHOWCASE_MISSION_NP_COF_SUCCESS = 1159, + SEND_LUA_NOTIFICATION_REQUEST = 1160, + SEND_LUA_NOTIFICATION_CANCEL = 1161, + ACTIVATOR_TOGGLE = 1162, + MAKE_PHYSICS = 1163, + SET_RESPAWN_GROUP = 1164, + SET_PLAYER_ALLOWED_RESPAWN = 1165, + TOGGLE_SENDING_POSITION_UPDATES = 1166, + TOGGLE_RECEIVING_POSITION_UPDATES = 1167, + GET_ENEMY_PRECONDITIONS = 1168, + START_MODEL_VISUALIZATION = 1169, + PLACE_PROPERTY_MODEL = 1170, + PROPERTY_MODEL_PLACED = 1171, + OPEN_EXHIBIT_REPLACE_MODEL_UI = 1172, + REPLACE_SHOWCASEMODEL = 1173, + CLEAR_UI_HOOK_EXHIBIT_REPLACEMENT = 1174, + ATTACH_FLYTO_SCREEN_POS = 1175, + VEHICLE_GET_DEBUG_INFO = 1176, + VEHICLE_GET_MOVEMENT_INPUT_VALUES = 1177, + ACTIVITY_TIMER_SET = 1178, + ACTIVITY_TIMER_UPDATE = 1179, + ACTIVITY_TIMER_GET = 1180, + ACTIVITY_TIMER_STOP = 1181, + ACTIVITY_TIMER_DONE = 1182, + GET_ATTACK_PRIORITY = 1183, + UI_MESSAGE_SERVER_TO_SINGLE_CLIENT = 1184, + UI_MESSAGE_SERVER_TO_ALL_CLIENTS = 1185, + SET_LOSE_COINS_ON_DEATH = 1186, + LOAD_EFFECTS = 1187, + SET_CUSTOM_BUILD = 1188, + ACTIVITY_TIMER_RESET = 1189, + ACTIVITY_TIMER_STOP_ALL_TIMERS = 1190, + ACTIVITY_TIMER_MODIFY = 1191, + SET_KEYFRAM_TRANSFORM = 1192, + ADD_ACTIVITY_OWNER = 1193, + REMOVE_ACTIVITY_OWNER = 1194, + GET_CURRENT_ACTIVITY_OWNERS = 1195, + TOGGLE_SKILL_DEBUGGING = 1196, + PET_TAMING_TRY_BUILD = 1197, + REPORT_BUG = 1198, + REPORT_OFFENSIVE_MODEL = 1199, + REPORT_OFFENSIVE_PROPERTY = 1200, + GET_ACTIVITY_ID = 1201, + REQUEST_SMASH_PLAYER = 1202, + GET_TIMES_REQUESTED_SMASH = 1203, + RESPONSE_SMASH_PLAYER = 1204, + MODIFY_DAMAGE_ABSORPTION = 1205, + UNCAST_SKILL = 1206, + GET_SHOWCASE_MODEL_READY = 1207, + IS_SKILL_NEEDED = 1208, + GET_COMPONENT_DATA = 1209, + VEHICLE_SET_POWERSLIDE_METHOD = 1210, + SHOWS_NAMETAG = 1211, + FIRE_EVENT_CLIENT_SIDE = 1213, + GET_REQUIRES_NAME_RESUBMISSION = 1216, + SET_REQUIRES_NAME_RESUBMISSION = 1217, + TOGGLE_GM_INVIS = 1218, + GET_GM_INVIS = 1219, + KILLED_PLAYER = 1220, + GET_PICKUP_SKILLS = 1221, + GET_FACTION_SKILL = 1222, + CHANGE_OBJECT_WORLD_STATE = 1223, + GET_OBJECT_WORLD_STATE = 1224, + VISIBILITY_CHANGED = 1225, + MOTION_EFFECT_COMPLETE = 1226, + TOGGLE_FREEZE_MODE = 1227, + SHADER_RENDER_MSG_APPLIED = 1228, + PLAYER_RENAME_REQUEST = 1229, + VEHICLE_LOCK_INPUT = 1230, + VEHICLE_UNLOCK_INPUT = 1231, + SET_AIR_MOVEMENT = 1232, + MOVEMENT_STATE_CHANGED = 1233, + SKILL_MOVEMENT_CANCELLED = 1234, + AIR_MOVEMENT_COMPLETE = 1235, + CANCEL_AIR_MOVEMENT = 1236, + FORCE_MINIFIGURE_TEXTURE_UPDATE = 1237, + RESYNC_EQUIPMENT = 1238, + ADD_COMPONENT_TO_OBJECT = 1239, + VEHICLE_GET_MAX_GAME_SPEED = 1240, + VEHICLE_GET_MAX_GAME_SPEED_WITH_BOOST = 1241, + GET_SPEED_FACTOR = 1242, + FREEZE_INVENTORY = 1243, + ADD_STAT_TRIGGER = 1244, + ADD_STAT_TRIGGER_CHILD = 1245, + CHECK_TRIGGERS_AND_FIRE_IF_NEEDED = 1246, + STAT_EVENT_TRIGGERED = 1247, + GET_CURRENT_SPEED = 1248, + RACING_PLAYER_RANK_CHANGED = 1249, + RACING_PLAYER_WRONG_WAY_STATUS_CHANGED = 1250, + RACING_PLAYER_CROSSED_FINISH_LINE = 1251, + RACING_RESET_PLAYER_TO_LAST_RESET = 1252, + RACING_SERVER_SET_PLAYER_LAP_AND_PLANE = 1253, + RACING_SET_PLAYER_RESET_INFO = 1254, + RACING_PLAYER_INFO_RESET_FINISHED = 1255, + RACING_PLAYER_OUT_OF_TRACK_BOUNDS = 1256, + RACING_SYNC_INFO = 1257, + RACING_PLAYER_KEEP_ALIVE = 1258, + RACING_SERVER_KEEP_ALIVE = 1259, + LOCK_NODE_ROTATION = 1260, + GET_PHYSICS_COLLIDABLE = 1261, + SET_PHYSICS_COLOR_FOR_DEBUG = 1262, + GET_PHYSICS_COLOR_FOR_DEBUG = 1263, + SET_PHYSICS_TEXT_AND_STATE_FOR_DEBUG = 1264, + REQUEST_INFO_FOR_PHYSICS_DEBUGGER = 1265, + GET_COLLIDABLE_AT_ADDRESS = 1266, + REQUEST_SERVER_GET_COLLIDABLE_REPORT = 1267, + COLLISION_POINT_ADDED = 1268, + COLLISION_POINT_REMOVED = 1269, + SET_ATTACHED = 1270, + SET_DESTROYABLE_MODEL_BRICKS = 1271, + VEHICLE_SET_POWERSLIDE_LOCK_WHEELS = 1273, + VEHICLE_SET_WHEEL_LOCK_STATE = 1273, + SHOW_HEALTH_BAR = 1274, + GET_SHOWS_HEALTH_BAR = 1275, + NOTIFY_VEHICLE_OF_RACING_OBJECT = 1276, + ENABLE_CLIENT_EQUIP_MODE = 1278, + CLIENT_EQUIP_MODE_WAS_CHANGED = 1279, + VEHICLE_GET_SPAWN_HEIGHT = 1281, + SET_NAME_BILLBOARD_STATE = 1284, + CHECK_TARGETING_REQUIREMENTS = 1285, + VEHICLE_CAN_WRECK = 1286, + ATTACH_RENDER_EFFECT = 1287, + DETACH_RENDER_EFFECT = 1288, + IS_PET_USING_ABILITY = 1289, + SET_BLOCKING = 1290, + GET_BLOCKING = 1291, + UPDATE_BLOCKING = 1292, + CHECK_DAMAGE_RESULTS = 1293, + GET_OBJECT_IS_IN_RENDER_PIPE = 1294, + ATTACH_MOTION_FX_ARC = 1295, + PLAYER_REACHED_RESPAWN_CHECKPOINT = 1296, + GET_LAST_RESPAWN_CHECKPOINT = 1297, + GET_VEHICLE_DEBUG_COLLISIONS = 1298, + VISITING_PROPERTY = 1299, + HANDLE_UGC_POST_DELETE_BASED_ON_EDIT_MODE = 1300, + HANDLE_UGC_POST_CREATE_BASED_ON_EDIT_MODE = 1301, + WORLD_CHECK_RESPONSE = 1302, + ADD_DAMAGE_REDUCTION = 1303, + REMOVE_DAMAGE_REDUCTION = 1304, + PROPERTY_CONTENTS_FROM_CLIENT = 1305, + GET_MODELS_ON_PROPERTY = 1306, + IS_SHOWCASE_DISPLAY_PEDESTAL = 1307, + MATCH_REQUEST = 1308, + MATCH_RESPONSE = 1309, + MATCH_UPDATE = 1310, + IS_DEFAULT_SKILL_ACTIVE = 1311, + PROPERTY_EDITOR_CARRY = 1312, + GET_LOOT_OWNER_ID = 1313, + GET_ENEMY_LOOT_TAG = 1314, + GET_NUM_SPAWNED_BRICKS = 1315, + SET_ITEM_EQUIP_TRANSFORM = 1316, + GET_ITEM_EQUIP_TRANSFORM = 1317, + GET_PROPERTY_BUDGET_INFO = 1318, + CHATBOX_IS_INIT = 1319, + GET_SPAWNED_I_DS = 1320, + GET_IMMUNITY = 1321, + GET_GM_IMMUNITY = 1322, + PROCESS_REMOTE_SLASH_COMMAND = 1323, + IS_FRIEND_MSG = 1324, + RACING_PLAYER_EVENT = 1325, + GET_PROPERTY_EDIT_VALID = 1326, + REFRESH_RENDER_ASSET = 1327, + VEHICLE_APPLY_STAT_CHANGE = 1328, + ZONE_LOADED_INFO = 1329, + B3_INTERFACE_ACTION = 1330, + RACING_STAT_MODIFIERS_FROM_CLIENT = 1332, + GET_RACING_STAT_MODIFIERS = 1333, + SET_RACING_STAT_MODIFIERS = 1334, + GET_RACING_LICENSE_LEVEL = 1335, + ADD_EQUIP_CAST = 1336, + SHOW_BILLBOARD_INTERACT_ICON = 1337, + CHANGE_IDLE_FLAGS = 1338, + GET_ANIMATION_FLAG = 1339, + VEHICLE_ADD_PASSIVE_BOOST_ACTION = 1340, + VEHICLE_REMOVE_PASSIVE_BOOST_ACTION = 1341, + NOTIFY_SERVER_VEHICLE_ADD_PASSIVE_BOOST_ACTION = 1342, + NOTIFY_SERVER_VEHICLE_REMOVE_PASSIVE_BOOST_ACTION = 1343, + VEHICLE_ADD_SLOWDOWN_ACTION = 1344, + VEHICLE_REMOVE_SLOWDOWN_ACTION = 1345, + NOTIFY_SERVER_VEHICLE_ADD_SLOWDOWN_ACTION = 1346, + NOTIFY_SERVER_VEHICLE_REMOVE_SLOWDOWN_ACTION = 1347, + FORCE_UPDATE_ANIMATIONS = 1348, + MATCH_GET_DATA_FOR_PLAYER = 1349, + BUYBACK_FROM_VENDOR = 1350, + SET_INVENTORY_FILTER = 1351, + GET_INVENTORY_FILTER = 1352, + GET_INVENTORY_GROUPS = 1353, + GET_INVENTORY_GROUP = 1354, + UPDATE_INVENTORY_GROUP = 1355, + UPDATE_INVENTORY_UI = 1356, + UPDATE_INVENTORY_GROUP_CONTENTS = 1357, + CAN_REMOVE_ITEM_FROM_INVENTORY = 1362, + DRIVE_THIS_CAR = 1363, + VEHICLE_CAN_ADD_ACTIVE_BOOST = 1364, + VEHICLE_ADD_ACTIVE_BOOST = 1365, + SET_PROPERTY_ACCESS = 1366, + ZONE_PROPERTY_MODEL_PLACED = 1369, + ZONE_PROPERTY_MODEL_ROTATED = 1370, + ZONE_PROPERTY_MODEL_REMOVED_WHILE_EQUIPPED = 1371, + ZONE_PROPERTY_MODEL_EQUIPPED = 1372, + ZONE_PROPERTY_MODEL_PICKED_UP = 1373, + ZONE_PROPERTY_MODEL_REMOVED = 1374, + GET_VERSIONING_INFO = 1381, + OPEN_UG_BEHAVIOR_UI = 1382, + VEHICLE_NOTIFY_HIT_SMASHABLE = 1383, + GET_TETHER_RADIUS = 1384, + VEHICLE_NOTIFY_HIT_EXPLODER = 1385, + CHECK_NEAREST_ROCKET_LAUNCH_PRE_CONDITIONS = 1386, + REQUEST_NEAREST_ROCKET_LAUNCH_PRE_CONDITIONS = 1387, + CONFIGURE_RACING_CONTROL_CLIENT = 1389, + NOTIFY_RACING_CLIENT = 1390, + RACING_PLAYER_HACK_CAR = 1391, + RACING_PLAYER_LOADED = 1392, + RACING_CLIENT_READY = 1393, + POSSESSION_FINISHED_ATTACH = 1394, + UPDATE_CHAT_MODE = 1395, + VEHICLE_NOTIFY_FINISHED_RACE = 1396, + EQUIPPED_ITEM_STARTUP = 1397, + FACTION_TRIGGER_ITEM_EQUIPPED = 1400, + FACTION_TRIGGER_ITEM_UNEQUIPPED = 1401, + TOGGLE_PROPERTY_BEHAVIORS = 1402, + GET_UG_OBJECT_INFO = 1405, + RESET_PROPERTY_BEHAVIORS = 1406, + IS_PROPERTY_MODEL_RESET = 1407, + SET_UG_OBJECT_NAME_AND_DESCRIPTION = 1408, + SET_CONSUMABLE_ITEM = 1409, + VEHICLE_GET_CURRENT_LAP = 1410, + GET_UGID = 1411, + SET_UGID = 1412, + UGID_CHANGED = 1413, + RACING_GET_CURRENT_LAP_FOR_PLAYER = 1414, + SUB_ITEM_UN_EQUIPPED = 1415, + SET_CUSTOM_DROP_SHADOW_TEXTURE = 1416, + GET_PLAYER_KIT_FACTION = 1418, + USED_INFORMATION_PLAQUE = 1419, + RACING_ENABLE_WRONG_WAY_RESET = 1420, + RACING_TOGGLE_RUBBER_BANDING = 1421, + GET_RACING_CONTROL_DEBUG_INFO = 1422, + SET_PROPERTY_BOUNDS_VISIBILITY = 1423, + SET_PROPERTY_VENDOR_VISIBILITY = 1424, + SET_EQUIP_STATE = 1425, + NOTIFY_COMBAT_AI_STATE_CHANGE = 1426, + SET_PROPERTY_MODEL_INTERACTIVE = 1430, + SERVER_STATE_NOTIFY = 1431, + GET_SERVER_STATE = 1432, + GET_ICON_FOR_PROXIMITY = 1433, + GET_LEGO_CLUB_MEMBERSHIP_STATUS = 1434, + SET_STATUS_IMMUNITY = 1435, + GET_STATUS_IMMUNITY = 1436, + TEAM_IS_MEMBER = 1437, + ACTIVATE_BRICK_MODE = 1438, + GET_BUILD_OBJECT_ID = 1439, + SET_ANIMATION_ENABLED = 1444, + PAUSE_COOLDOWNS = 1446, + FORCE_UPDATE_RENDER_NODE = 1447, + SET_PET_NAME_MODERATED = 1448, + TOGGLE_STRAFE_MODE = 1449, + SET_SCHEME_SPEED_SCALE = 1450, + CANCEL_SKILL_CAST = 1451, + CHECK_PLAYER_ASSEMBLY_FOR_UNIQUE_MODULE_BY_LOT = 1454, + MODULE_ASSEMBLY_DB_DATA_TO_LUA = 1455, + IS_ALLY = 1458, + MODIFY_LEGO_SCORE = 1459, + GET_LEGO_SCORE = 1460, + GET_PLAYER_LEVEL = 1461, + NOTIFY_LEGO_SCORE_UPDATE = 1462, + SET_LEGO_SCORE = 1463, + UPDATE_BEHAVIOR_EXECUTION_DETAILS = 1466, + RESTORE_TO_POST_LOAD_STATS = 1468, + PICKUP_OBJECT_ERROR = 1469, + CHECK_AND_SHOW_INVENTORY_FULL_TIP = 1470, + SET_RAIL_MOVEMENT = 1471, + START_RAIL_MOVEMENT = 1472, + SET_UP_VECTOR = 1473, + CANCEL_RAIL_MOVEMENT = 1474, + GET_RAIL_INFO = 1475, + CLIENT_RAIL_MOVEMENT_READY = 1476, + PLAYER_RAIL_ARRIVED_NOTIFICATION = 1477, + NOTIFY_RAIL_ACTOVATOR_STATE_CHANGE = 1478, + REQUEST_RAIL_ACTIVATOR_STATE = 1479, + NOTIFY_REWARD_MAILED = 1480, + UPDATE_PLAYER_STATISTIC = 1481, + IS_IN_COMBAT = 1482, + IS_PRIMITIVE_MODEL_MSG = 1483, + SCALE_PRIMITICE_MODEL_MSG = 1484, + MODIFY_GHOSTING_DISTANCE = 1485, + PRIMITIVE_MODEL_CHANGED_MSG = 1487, + GET_PROPRTY_CLONE_ID = 1488, + REQUEST_LEAVE_PROPERTY = 1489, + REQUERY_PROPERTY_MODELS = 1491, + GET_BEHAVIOR_COUNT = 1492, + UPDATE_BEHAVIOR_CONTROLS = 1493, + MODULE_ASSEMBLY_LXFML_LOADED = 1494, + REQUEST_ASSEMBLED_LXFML = 1495, + ASSEMBLED_LXFML_LOADED = 1496, + GET_REORIENT_UP_VECTOR = 1497, + MODULAR_ASSEMBLY_NIF_COMPLETED = 1498, + CHARACTER_DISCONNECT_BEFORE_CREATE = 1499, + SEND_LAUNCH_TO_PREVIOUS_ZONE_TO_CLIENT = 1500, + ROCKETLAUNCH_REQUEST_DEFAULT_MAP_ID = 1501, + BEGIN_LAUNCH = 1502, + PROCESS_CLAIM_CODES = 1503, + GET_LAST_ZONE_ID = 1504, + ADD_RUN_SPEED_MODIFIER = 1505, + REMOVE_RUN_SPEED_MODIFIER = 1506, + SKILL_EVENT_FIRED = 1507, + SEND_HOT_PROPERTY_DATA = 1510, + GET_HOT_PROPERTY_DATA = 1511, + GET_EQUIPPED_ITEMS_IN_SET = 1512, + IS_ITEM_IN_SET = 1513, + GET_INVENTORY_TYPE_FOR_LOT = 1514, + GET_BANK_TYPE_FOR_LOT = 1515, + NOTIFY_NOT_ENOUGH_INV_SPACE = 1516, + IMPORT_MODEL_TO_BBB = 1517, + SEARCH_NEARBY_OBJECTS = 1518, + SEARCH_NEARBY_OBJECTS_REQUEST_BY_LOT = 1519, + REQUEST_OBJECT_POSITION_BY_ID = 1520, + SEARCH_NEARBY_OBJECTS_REQUEST_BY_COMPONENT = 1521, + SEARCH_NEARBY_OBJECTS_RESPONSE = 1522, + BROADCAST_NON_STANDARD_COLLISIONS = 1523, + GET_REGISTERED_NON_STANDARD_COLLISION_GROUP = 1524, + BROADCAST_CRUSHED_NOTIFICATIONS = 1525, + GET_REGISTERED_CRUSHED_COLLISION_GROUPS = 1526, + IS_BEING_CRUSHED = 1527, + GET_SUPPORTING_OBJECT = 1528, + TREAT_RIGID_BODY_COLLSIONS_AS_FIXED = 1529, + BROADCAST_TELEPORTED_WITHIN_NOTIFICATION = 1530, + GET_REGISTERED_TELEPORTED_WITHIN_OBJECT_GROUP = 1531, + GET_INTERPENTRATING_INFORMATION = 1532, + OBJECT_TELEPORTED_WITHIN = 1533, + SET_PHYSICS_SOLVER_PROPERTIES = 1534, + HAS_BEHAVIORS = 1535, + PLAY_BEHAVIOR_SOUND = 1536, + GET_PLAYER_BEHAVIOR_TIER = 1537, + GET_EMOTE_ANIMATION_TIME = 1538, + GET_CHARACTER_STAT_TRACKING = 1539, + PLAYER_INVENTORY_READY = 1540, + SET_PRECONDITIONS = 1541, + DETACH_SHADOW = 1542, + GET_LOOT_INFO = 1543, + GET_PLAYERS_ON_PROPERTY = 1544, + PROPERTY_SPAWN_BY_BEHAVIOR = 1545, + NOTIFY_PROPERTY_OF_EDIT_MODE = 1546, + UPDATE_PROPERTY_PERFORMANCE_COST = 1547, + GET_PROPERTY_PERFORMANCE_COST = 1548, + GET_INVENTORY_ITEM_WITH_SUBKEY = 1549, + DISPLAY_PROPERTY_SUMMARY_SCREEN = 1550, + VALIDATE_BBB_MODEL = 1551, + BBB_MODEL_VALIDATION = 1552, + PROPERTY_ENTRANCE_BEGIN = 1553, + CHECK_LIST_OF_PRECONDITIONS_FROM_LUA = 1554, + GET_PROPERTYIN_ZONE = 1555, + GET_ZONE_ID_FROM_MULTI_ZONE_ENTRANCE = 1556, + TEAM_SET_LEADER = 1557, + TEAM_INVITE_CONFIRM = 1558, + TEAM_GET_STATUS_RESPONSE = 1559, + MINI_GAME_ENABLE_LOCAL_TEAMS = 1560, + TEAM_INVITE_FINAL_RESPONSE = 1561, + TEAM_ADD_PLAYER = 1562, + TEAM_REMOVE_PLAYER = 1563, + TEAM_CREATE_LOCAL = 1564, + TEAM_GET_LOOT = 1565, + TEAM_SET_LOOT = 1566, + SET_ZERO_IMPULSE_AGAINST_COLLISION_GROUPS = 1567, + SET_CENTER_OF_MASS_TO_PHYSICAL_CENTER = 1568, + SET_INERTIA_INVERSE = 1569, + ADD_REMOVE_CLIMBING_LISTENER = 1570, + GET_INVENTORYITEM_DETAILS = 1571, + PERFORM_CLIENT_SIDE_DEATH = 1572, + LEGO_CLUB_ACCESS_RESULT = 1573, + VEHICLE_GET_IS_REVERSING = 1574, + CHECK_CLAIM_CODE = 1575, + GET_HOLIDAY_EVENT = 1576, + SET_EMOTES_ENABLED = 1577, + GET_EMOTES_ENABLED = 1578, + FREEZE_ANIMATION = 1579, + LOCALIZED_ANNOUNCEMENT_SERVER_TO_SINGLE_CLIENT = 1580, + ANCHOR_FX_NODE = 1581, + WS_GET_FRIEND_LIST_MESSAGE = 1582, + WS_ADD_FRIEND_RESPONSE = 1583, + WS_REMOVE_FRIEND_REPSONSE = 1584, + WS_UPDATE_FRIEND_STATUS = 1585, + WS_UPDATE_FRIEND_NAME = 1586, + IS_BEST_FRIEND = 1587, + TEAM_NOTIFY_UPDATE_MISSION_TASK = 1588, + VEHICLE_ADD_SLIPPERY_ACTION = 1589, + VEHICLE_REMOVE_SLIPPERY_ACTION = 1590, + SET_RESURRECT_RESTORE_VALUES = 1591, + GET_MASS = 1592, + SET_PROPERTY_MODERATION_STATUS = 1593, + UPDATE_PROPERTY_MODEL_DEFAULTS = 1594, + UPDATE_PROPERTYMODEL_COUNT = 1595, + GET_PROPERTY_MODEL_COUNT = 1596, + IS_PLAYER_LOADED = 1597, + ATTACH_RENDER_EFFECT_FROM_LUA = 1598, + DETACH_RENDER_EFFECT_FROM_LUA = 1599, + TEAM_IS_LOCAL = 1600, + CREATE_CAMERA_PARTICLES = 1602, + SET_SMASHABLE_GRAVITY_FACTOR = 1603, + VEHICLE_SET_SURFACE_TYPE_OVERRIDE = 1604, + VEHICLE_NOTIFY_HIT_IMAGINATION = 1605, + VEHICLE_NOTIFY_HIT_IMAGINATION_SERVER = 1606, + GET_SPAWNED_OBJECT_SPAWNER_INFO = 1607, + SAVE_PROPERTY = 1608, + SET_PROPERTY_DIRTY = 1609, + GET_PROPERTY_DIRTY = 1610, + GET_MODEL_LIST_FROM_PLAQUE = 1611, + GET_ORIGINAL_POSITION_AND_ROTATION = 1612, + VEHICLE_SET_MASS_FOR_COLLISION = 1613, + GET_INVENTORY_GROUP_COUNT = 1614, + GET_LATEST_CHAT_CHANNEL_USED = 1615, + SET_SUGGEST_LIST_LANGUAGE = 1616, + VEHICLE_STOP_BOOST = 1617, + START_CELEBRATION_EFFECT = 1618, + LOCK_PLAYER = 1619, + VEHICLE_IS_INPUT_LOCKED = 1620, + GET_MULTI_NODE = 1621, + RENEW_PROPERTY = 1622, + RENEW_PROPERTY_RESULT = 1623, + CHARGE_ACTIVITY_COST = 1624, + CAN_RECEIVE_LOOT = 1625, + JOIN_PLAYER_FACTION = 1626, + SET_PROXIMITY_UPDATE_RATE = 1627, + BBB_MODELS_TO_SAVE = 1628, + BELONGS_TO_FACTION = 1629, + MODIFY_FACTION = 1630, + FACTION_UPDATE = 1631, + CELEBRATION_COMPLETED = 1632, + PLAY_PRIMARY_MODULE_SOUNDS = 1633, + STOP_PRIMARY_MODULE_SOUNDS = 1634, + REQUEST_TEAM_PLAYER_UI_UPDATE = 1635, + SET_LOCAL_TEAM = 1636, + TEAM_GET_WORLD_MEMBERS_IN_RADIUS = 1637, + GET_PARENTAL_LEVEL = 1638, + GET_OBJECTS_MESSAGE_HANDLERS = 1639, + PROPERTY_FEATURED = 1640, + PROPERTY_NOTIFY_MODEL_SPAWNED = 1641, + SERVER_DONE_LOADING_ALL_OBJECTS = 1642, + GET_DONATION_TOTAL = 1643, + UPDATE_DONATION_VALUES = 1644, + DELAYED_DELETE_DROP_EFFECT_BRICK = 1645, + SET_CAMERA_UNLOCK_ROTATION_STATE = 1646, + ADD_BUFF = 1647, + REMOVE_BUFF = 1648, + CHECK_FOR_BUFF = 1649, + TEAM_MEMBERS_DISOWNS_LOOT = 1650, + GET_WHEEL_TEMPLATE = 1651, + ADD_SKILL_IN_PROGRESS = 1652, + REMOVE_SKILL_IN_PROGRESS = 1653, + SET_OVERHEAD_ICON_OFFSET = 1654, + SET_BILLBOARD_OFFSET = 1655, + SET_CHAT_BUBBLE_OFFSET = 1656, + SET_NO_TEAM_INVITES = 1657, + RESET_MODEL_TO_DEFAULTS = 1658, + IS_PROPERTY_IN_EDIT_MODE = 1659, + GET_OBJECTS_IN_PHYSICS_BOUNDS = 1660, + ENABLE_LU_REMOTE = 1661, + SET_IS_USING_FREE_TRIAL = 1662, + GET_IS_USING_FREE_TRIAL = 1663, + GET_ACCOUNT_FREE_TRIAL_MODE = 1664, + TOGGLE_INVENTORY_ITEM_LOCK = 1665, + REQUEST_MOVE_ITEM_BETWEEN_INVENTORY_TYPES = 1666, + RESPONSE_MOVE_ITEM_BETWEEN_INVENTORY_TYPES = 1667, + REMOVE_SUB_COMPONENT = 1668, + TEAM_GET_LOOT_MEMBERS = 1669, + GET_FACTION_TOKEN_TYPE = 1670, + GET_SUBSCRIPTION_PRICING = 1671, + INFORM_AFK = 1672, + OVERHEAD_INDICATOR_CREATED = 1673, + SET_OVERHEAD_INDICATOR_GRID_LOCATION = 1674, + PLAYSTREAM_LOAD_PENDING = 1675, + PLAYER_SET_CAMERA_CYCLING_MODE = 1676, + PLAYER_GET_CAMERA_CYCLING_MODE = 1677, + FORCE_CAMERA_TARGET_CYCLE = 1678, + GET_OBJECT_CONFIG_DATA = 1679, + GET_OBJECT_CONFIG_DATA_NON_CONST = 1680, + SCOPE_CHANGED = 1681, + SET_ALLOW_JUMP_WITHOUT_SUPPORT = 1682, + GET_ALLOW_JUMP_WITHOUT_SUPPORT = 1683, + SET_JUMP_HEIGHT_SCALE = 1684, + GET_JUMP_HEIGHT_SCALE = 1685, + SET_VELOCITY_RESISTANCE = 1686, + GET_VELOCITY_RESISTANCE = 1687, + GATE_RUSH_VEHICLE_HIT_GATE = 1688, + GATE_RUSH_PLAYER_COLLECTED_GATE = 1689, + GATE_RUSH_ADD_GATE = 1690, + GATE_RUSH_REMOVE_GATE = 1691, + NOTIFY_VEHICLE_UPDATED = 1692, + VEHICLE_NOTIFY_HIT_WEAPON_POWERUP = 1693, + VEHICLE_NOTIFY_HIT_WEAPON_POWERUP_SERVER = 1694, + LOCAL_PLAYER_TARGETED = 1696, + SKILL_COUNT_CHANGED = 1697, + DO_YAW_ROTATION = 1698, + DO_PITCH_ROTATION = 1699, + DO_ROLL_ROTATION = 1700, + GET_CURRENT_LOOT_MATRIX = 1701, + SEND_MULTI_MISSION_OFFER_UPDATE_I_DS = 1702, + SET_AIR_SPEED_VALUES = 1703, + USE_LAUNCHER = 1704, + START_LAUNCHER = 1705, + STOP_LAUNCHER = 1706, + CAN_USE_JET_PACK = 1707, + JET_PACK_STATE_CHANGED = 1708, + TURN_OFF_JET_PACK = 1709, + ADD_PLAYER_JET_PACK_PAD = 1710, + SET_JET_PACK_WARNING = 1711, + JET_PACK_DISABLED = 1712, + JET_PACK_PAD_ENTERED = 1713, + UPDATE_RENDER_POSSESSION_FLAG = 1714, + POSSESSABLE_GET_ATTACH_OFFSET = 1715, + ATTEMPT_TO_CRAFT_ITEM = 1718, + CRAFT_ATTEMPT_RESPONSE = 1719, + SET_C_SCORE = 1720, + FILL_IN_RENDERER = 1721, + TOGGLE_CRAFTING_WINDOW = 1722, + REMOVE_TEAM_BUFFS = 1724, + REQUEST_FREE_TRIAL_STATUS_REFRESH = 1725, + REMOVE_BUFFS_APPLIED_BY_OBJECT = 1726, + SET_MOUNT_INVENTORY_ID = 1727, + GET_MOUNT_INVENTORY_ID = 1728, + GET_BUILD_CINEMATIC_TIME_REMAINING = 1730, + JET_PACK_FLYING = 1731, + NOTIFY_SERVER_LEVEL_PROCESSING_COMPLETE = 1734, + NOTIFY_LEVEL_REWARDS = 1735, + CHARACTER_VERSION_CHANGED = 1736, + SET_FREE_TRIAL_RENAME_AVAILABLE = 1737, + SET_PROJECTILE_LAUNCHER_PARAMS = 1738, + RACE_PRECOUNTDOWN_DONE = 1739, + CHECK_INVITE_SPAMMING = 1740, + GET_RESPAWN_VOLUME_INFO = 1741, + INVITE_ACCEPTED = 1742, + TELEPORT_TO_NEAREST_RESPAWN = 1743, + SET_SKILL_CANCEL_ON_MOVE = 1744, + CANCEL_MOVE_SKILL = 1745, + SERVER_CANCEL_MOVE_SKILL = 1746, + CLIENT_CANCEL_MOVE_SKILL = 1747, + END_LAUNCH_SEQUENCE = 1748, + CANCEL_QUEUE = 1749, + UPDATE_PROJECTILE_LAUNCHER_ROTATION = 1750, + GET_CHARACTER_VERSION_INFO = 1751, + GET_CON_INFO = 1753, + GET_SKILLS_FOR_LOT = 1755, + DISMOUNT_COMPLETE = 1756, + MOUNT_FAILURE_RESPONSE = 1757, + CLEAR_BILLBOARD_OFFSET = 1758, + GET_INVENTORY_ITEM_ANIMATION_FLAG = 1759, + SET_JET_PACK_ALLOWED = 1760, + GET_BUILD_TIME_DETAILS = 1761, + USE_SKILL_SET = 1762, + SET_SKILL_SET_POSSESSOR = 1763, + POPULATE_ACTION_BAR = 1764, + GET_COMPONENT_TEMPLATE_ID = 1765, + GET_POSSESSABLE_SKILL_SET = 1766, + MARK_INVENTORY_ITEM_AS_ACTIVE = 1767, + UPDATE_FORGED_ITEM = 1768, + CAN_ITEMS_BE_REFORGED = 1769, + NOTIFY_CLIENT_RAIL_START_FAILED = 1771, + GET_IS_ON_RAIL = 1772 +}; + +#endif //!__EGAMEMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eHelpType.h b/dCommon/dEnums/eHelpType.h new file mode 100644 index 00000000..d1838cc6 --- /dev/null +++ b/dCommon/dEnums/eHelpType.h @@ -0,0 +1,41 @@ + +#ifndef __EHELPTYPE__H__ +#define __EHELPTYPE__H__ + +#include + +enum class eHelpType : int32_t { + NONE = 0, + UNLOCK_MINIMAP = 2, + TOGGLETOOLTIP_OUTOFTIME_REBUILD = 3, + TOGGLETOOLTIP_LEAVELOSE_REBUILD = 4, + TOGGLECONTROLSTUTORIAL_WALKING = 6, + DISPLAYTUTORIAL_PASSPORT_1ST_SMASH = 7, + TOOLTIP_1ST_IMAGINATION_PICKUP = 8, + UNKNOWN9 = 9, + PETTAMINGMINIGAME_TUTORIAL_01 = 15, + PR_BOUNCER_TUTORIAL_03 = 16, + PR_TOOLTIP_1ST_PET_JUMPED_ON_SWITCH = 17, + PR_DIG_TUTORIAL_01 = 18, + PR_DIG_TUTORIAL_03 = 19, + PR_BOUNCER_TUTORIAL_01 = 20, + PR_NO_IMAGINATION_HIBERNATE = 21, + UNKNOWN22 = 22, + TOGGLECONTROLSTUTORIAL_JUMPING = 26, + TOGGLECONTROLSTUTORIAL_DOUBLEJUMPING = 27, + TOGGLECONTROLSTUTORIAL_CAMERA = 28, + TOGGLECONTROLSTUTORIAL_SMASH = 30, + UNKNOWN38 = 38, + UI_MOD_BUILD_PUT_ON_HAT = 40, + UI_MOD_BUILD_EQUIP_FIRST_MODULE = 41, + UNKNOWN42 = 42, + UNKNOWN43 = 43, + UI_MOD_BUILD_GO_LAUNCH_ROCKET = 44, + UI_MOD_BUILD_TALK_TO_SKYLANE = 45, + UNKNOWN53 = 53, + PET_DESPAWN_BY_OWNER_HIBERNATE = 69, + PET_DESPAWN_TAMING_NEW_PET = 70, + UI_INVENTORY_FULL_CANNOT_PICKUP_ITEM = 86 +}; + +#endif //!__EHELPTYPE__H__ diff --git a/dCommon/dEnums/eInventoryType.h b/dCommon/dEnums/eInventoryType.h new file mode 100644 index 00000000..12573aa4 --- /dev/null +++ b/dCommon/dEnums/eInventoryType.h @@ -0,0 +1,59 @@ +#pragma once + +#ifndef __EINVENTORYTYPE__H__ +#define __EINVENTORYTYPE__H__ + +#include +static const uint8_t NUMBER_OF_INVENTORIES = 17; +/** + * Represents the different types of inventories an entity may have + */ +enum eInventoryType : uint32_t { + ITEMS = 0, + VAULT_ITEMS, + BRICKS, + MODELS_IN_BBB, + TEMP_ITEMS, + MODELS, + TEMP_MODELS, + BEHAVIORS, + PROPERTY_DEEDS, + BRICKS_IN_BBB, + VENDOR, + VENDOR_BUYBACK, + QUEST, //Used for mission items + DONATION, + VAULT_MODELS, + ITEM_SETS, //internal, technically this is BankBehaviors. + INVALID // made up, for internal use!!!, Technically this called the ALL inventory. +}; + +class InventoryType { +public: + static const char* InventoryTypeToString(eInventoryType inventory) { + const char* eInventoryTypeTable[NUMBER_OF_INVENTORIES] = { + "ITEMS", + "VAULT_ITEMS", + "BRICKS", + "MODELS_IN_BBB", + "TEMP_ITEMS", + "MODELS", + "TEMP_MODELS", + "BEHAVIORS", + "PROPERTY_DEEDS", + "BRICKS_IN_BBB", + "VENDOR", + "VENDOR_BUYBACK", + "QUEST", //Used for mission items + "DONATION", + "VAULT_MODELS", + "ITEM_SETS", //internal, technically this is BankBehaviors. + "INVALID" // made up, for internal use!!!, Technically this called the ALL inventory. + }; + + if (inventory > NUMBER_OF_INVENTORIES - 1) return nullptr; + return eInventoryTypeTable[inventory]; + }; +}; + +#endif //!__EINVENTORYTYPE__H__ diff --git a/dCommon/dEnums/eItemSetPassiveAbilityID.h b/dCommon/dEnums/eItemSetPassiveAbilityID.h new file mode 100644 index 00000000..8641d0f2 --- /dev/null +++ b/dCommon/dEnums/eItemSetPassiveAbilityID.h @@ -0,0 +1,58 @@ +#pragma once + +#ifndef __EITEMSETPASSIVEABILITYID__H__ +#define __EITEMSETPASSIVEABILITYID__H__ + +enum class eItemSetPassiveAbilityID { + EngineerRank1 = 2, + EngineerRank2 = 3, + EngineerRank3 = 4, + KnightRank1 = 7, + KnightRank2 = 8, + KnightRank3 = 9, + SpaceRangerRank1 = 10, + SpaceRangerRank2 = 11, + SpaceRangerRank3 = 12, + SamuraiRank1 = 13, + SamuraiRank2 = 14, + SamuraiRank3 = 15, + SorcererRank1 = 16, + SorcererRank2 = 17, + SorcererRank3 = 18, + SpaceMarauderRank1 = 19, + SpaceMarauderRank2 = 20, + SpaceMarauderRank3 = 21, + ShinobiRank1 = 22, + ShinobiRank2 = 23, + ShinobiRank3 = 24, + InventorRank1 = 25, + InventorRank2 = 26, + InventorRank3 = 27, + SummonerRank1 = 28, + SummonerRank2 = 29, + SummonerRank3 = 30, + AdventurerRank1 = 31, + AdventurerRank2 = 32, + AdventurerRank3 = 33, + DaredevilRank1 = 34, + DaredevilRank2 = 35, + DaredevilRank3 = 36, + BuccaneerRank1 = 37, + BuccaneerRank2 = 38, + BuccaneerRank3 = 39, + BoneSuit = 40, + ImaginationSpinjitzu = 41, + BatLord = 42, + MosaicJester = 43, + ExplorienBot = 44, + Unnamed1 = 45, + Unnamed2 = 46, + Unnamed3 = 47, + EarthSpinjitzu = 48, + Unnamed4 = 49, + FireSpinjitzu = 50, + IceSpinjitzu = 51, + LightningSpinjitzu = 52 +}; + +#endif //!__EITEMSETPASSIVEABILITYID__H__ diff --git a/dCommon/dEnums/eItemType.h b/dCommon/dEnums/eItemType.h new file mode 100644 index 00000000..41c7765b --- /dev/null +++ b/dCommon/dEnums/eItemType.h @@ -0,0 +1,36 @@ +#pragma once + +#ifndef __EITEMTYPE__H__ +#define __EITEMTYPE__H__ + +#include + +enum class eItemType : int32_t { + UNKNOWN = -1, + BRICK = 1, + HAT, + HAIR, + NECK, + LEFT_HAND, + RIGHT_HAND, + LEGS, + LEFT_TRINKET, + RIGHT_TRINKET, + BEHAVIOR, + PROPERTY, + MODEL, + COLLECTIBLE, + CONSUMABLE, + CHEST, + EGG, + PET_FOOD, + QUEST_OBJECT, + PET_INVENTORY_ITEM, + PACKAGE, + LOOT_MODEL, + VEHICLE, + LUP_MODEL, + MOUNT +}; + +#endif //!__EITEMTYPE__H__ diff --git a/dCommon/dEnums/eKillType.h b/dCommon/dEnums/eKillType.h new file mode 100644 index 00000000..fe3908c8 --- /dev/null +++ b/dCommon/dEnums/eKillType.h @@ -0,0 +1,11 @@ +#ifndef __EKILLTYPE__H__ +#define __EKILLTYPE__H__ + +#include + +enum class eKillType : uint32_t { + VIOLENT, + SILENT +}; + +#endif //!__EKILLTYPE__H__ diff --git a/dCommon/dEnums/eLoginResponse.h b/dCommon/dEnums/eLoginResponse.h new file mode 100644 index 00000000..01bba3d5 --- /dev/null +++ b/dCommon/dEnums/eLoginResponse.h @@ -0,0 +1,24 @@ +#ifndef __ELOGINRESPONSE__H__ +#define __ELOGINRESPONSE__H__ + +#include + +enum class eLoginResponse : uint8_t { + GENERAL_FAILED = 0, + SUCCESS, + BANNED, + // Unused 3 + // Unused 4 + PERMISSIONS_NOT_HIGH_ENOUGH = 5, + INVALID_USER, + ACCOUNT_LOCKED, + WRONG_PASS, + ACCOUNT_ACTIVATION_PENDING, + ACCOUNT_DISABLED, + GAME_TIME_EXPIRED, + FREE_TRIAL_ENDED, + PLAY_SCHEDULE_TIME_UP, + ACCOUNT_NOT_ACTIVATED +}; + +#endif //!__ELOGINRESPONSE__H__ diff --git a/dCommon/dEnums/eLootSourceType.h b/dCommon/dEnums/eLootSourceType.h new file mode 100644 index 00000000..b8ecca4a --- /dev/null +++ b/dCommon/dEnums/eLootSourceType.h @@ -0,0 +1,31 @@ +#ifndef __ELOOTSOURCETYPE__H__ +#define __ELOOTSOURCETYPE__H__ + +#include + +enum class eLootSourceType : uint32_t { + NONE = 0, + CHEST, + MISSION, + MAIL, + CURRENCY, + ACHIEVEMENT, + TRADE, + QUICKBUILD, + DELETION, + VENDOR, + ACTIVITY, + PICKUP, + BRICK, + PROPERTY, + MODERATION, + EXHIBIT, + INVENTORY, + CLAIMCODE, + CONSUMPTION, + CRAFTING, + LEVEL_REWARD, + RELOCATE +}; + +#endif //!__ELOOTSOURCETYPE__H__ diff --git a/dCommon/dEnums/eMasterMessageType.h b/dCommon/dEnums/eMasterMessageType.h new file mode 100644 index 00000000..5c867d70 --- /dev/null +++ b/dCommon/dEnums/eMasterMessageType.h @@ -0,0 +1,36 @@ +#ifndef __EMASTERMESSAGETYPE__H__ +#define __EMASTERMESSAGETYPE__H__ + +#include + +enum class eMasterMessageType : uint32_t { + REQUEST_PERSISTENT_ID = 1, + REQUEST_PERSISTENT_ID_RESPONSE, + REQUEST_ZONE_TRANSFER, + REQUEST_ZONE_TRANSFER_RESPONSE, + SERVER_INFO, + REQUEST_SESSION_KEY, + SET_SESSION_KEY, + SESSION_KEY_RESPONSE, + PLAYER_ADDED, + PLAYER_REMOVED, + + CREATE_PRIVATE_ZONE, + REQUEST_PRIVATE_ZONE, + + WORLD_READY, + PREP_ZONE, + + SHUTDOWN, + SHUTDOWN_RESPONSE, + SHUTDOWN_IMMEDIATE, + + SHUTDOWN_UNIVERSE, + + AFFIRM_TRANSFER_REQUEST, + AFFIRM_TRANSFER_RESPONSE, + + NEW_SESSION_ALERT +}; + +#endif //!__EMASTERMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eMatchUpdate.h b/dCommon/dEnums/eMatchUpdate.h new file mode 100644 index 00000000..a42eecd3 --- /dev/null +++ b/dCommon/dEnums/eMatchUpdate.h @@ -0,0 +1,17 @@ +#ifndef __EMATCHUPDATE__H__ +#define __EMATCHUPDATE__H__ + +#include + +enum class eMatchUpdate : int32_t { + PLAYER_ADDED = 0, + PLAYER_REMOVED, + PHASE_CREATED, + PHASE_WAIT_READY, + PHASE_WAIT_START, + PLAYER_READY, + PLAYER_NOT_READY, + PLAYER_UPDATE +}; + +#endif //!__EMATCHUPDATE__H__ diff --git a/dCommon/dEnums/eMissionLockState.h b/dCommon/dEnums/eMissionLockState.h new file mode 100644 index 00000000..52752767 --- /dev/null +++ b/dCommon/dEnums/eMissionLockState.h @@ -0,0 +1,12 @@ +#pragma once + +#ifndef __EMISSIONLOCKSTATE__H__ +#define __EMISSIONLOCKSTATE__H__ + +enum class eMissionLockState : int { + LOCKED, + NEW, + UNLOCKED, +}; + +#endif //!__EMISSIONLOCKSTATE__H__ diff --git a/dCommon/dEnums/eMissionState.h b/dCommon/dEnums/eMissionState.h new file mode 100644 index 00000000..e080f455 --- /dev/null +++ b/dCommon/dEnums/eMissionState.h @@ -0,0 +1,56 @@ +#pragma once + +#ifndef __MISSIONSTATE__H__ +#define __MISSIONSTATE__H__ + +/** + * Represents the possible states a mission can be in + */ +enum class eMissionState : int { + /** + * The mission state is unknown + */ + UNKNOWN = -1, + + /** + * The mission is yielding rewards + */ + REWARDING = 0, + + /** + * The mission can be accepted + */ + AVAILABLE = 1, + + /** + * The mission has been accepted but not yet completed + */ + ACTIVE = 2, + + /** + * All the tasks for the mission have been completed and the entity can turn the mission in to complete it + */ + READY_TO_COMPLETE = 4, //!< The mission is ready to complete + + /** + * The mission has been completed + */ + COMPLETE = 8, + + /** + * The mission is available again and has been completed before. Used for daily missions. + */ + COMPLETE_AVAILABLE = 9, + + /** + * The mission is active and has been completed before. Used for daily missions. + */ + COMPLETE_ACTIVE = 10, + + /** + * The mission has been completed before and has now been completed again. Used for daily missions. + */ + COMPLETE_READY_TO_COMPLETE = 12 +}; + +#endif //!__MISSIONSTATE__H__ diff --git a/dCommon/dEnums/eMissionTaskType.h b/dCommon/dEnums/eMissionTaskType.h new file mode 100644 index 00000000..2636f88c --- /dev/null +++ b/dCommon/dEnums/eMissionTaskType.h @@ -0,0 +1,43 @@ +#pragma once + +#ifndef __EMISSIONTASKTYPE__H__ +#define __EMISSIONTASKTYPE__H__ + +enum class eMissionTaskType : int { + UNKNOWN = -1, + SMASH, + SCRIPT, + ACTIVITY, + COLLECTION, + TALK_TO_NPC, + EMOTE, + SHASH_CHAIN, + BUY, + SELL, + USE_ITEM, + USE_SKILL, + GATHER, + EXPLORE, + DELIVERY, + PERFORM_ACTIVITY, + INTERACT, + META, + EARN_REPUTATION, + VOTING, + SHOWCASE_DELIVERY, + REVIECE_CAST, + POWERUP, + PET_TAMING, + RACING, + PLAYER_FLAG, + PLACE_MODEL, + REMOVE_MODEL, + ADD_BEHAVIOR, + REMOVE_BEHAVIOR, + CLAIM_PROPERTY, + VISIT_PROPERTY, + TIME_PLAYED, + DONATION +}; + +#endif //!__EMISSIONTASKTYPE__H__ diff --git a/dCommon/dEnums/eMovementPlatformState.h b/dCommon/dEnums/eMovementPlatformState.h new file mode 100644 index 00000000..1df437d8 --- /dev/null +++ b/dCommon/dEnums/eMovementPlatformState.h @@ -0,0 +1,16 @@ +#ifndef __EMOVEMENTPLATFORMSTATE__H__ +#define __EMOVEMENTPLATFORMSTATE__H__ + +#include + +/** + * The different types of platform movement state, supposedly a bitmap + */ +enum class eMovementPlatformState : uint32_t +{ + Moving = 0b00010, + Stationary = 0b11001, + Stopped = 0b01100 +}; + +#endif //!__EMOVEMENTPLATFORMSTATE__H__ diff --git a/dCommon/dEnums/eObjectBits.h b/dCommon/dEnums/eObjectBits.h new file mode 100644 index 00000000..b978aad6 --- /dev/null +++ b/dCommon/dEnums/eObjectBits.h @@ -0,0 +1,13 @@ +#ifndef __EOBJECTBITS__H__ +#define __EOBJECTBITS__H__ + +#include + +enum class eObjectBits : size_t { + PERSISTENT = 32, + CLIENT = 46, + SPAWNED = 58, + CHARACTER = 60 +}; + +#endif //!__EOBJECTBITS__H__ diff --git a/dCommon/dEnums/eObjectWorldState.h b/dCommon/dEnums/eObjectWorldState.h new file mode 100644 index 00000000..9735a072 --- /dev/null +++ b/dCommon/dEnums/eObjectWorldState.h @@ -0,0 +1,12 @@ +#ifndef __EOBJECTWORLDSTATE__H__ +#define __EOBJECTWORLDSTATE__H__ + +#include + +enum class eObjectWorldState : uint32_t { + INWORLD, + ATTACHED, + INVENTORY +}; + +#endif //!__EOBJECTWORLDSTATE__H__ diff --git a/dCommon/dEnums/ePackageType.h b/dCommon/dEnums/ePackageType.h new file mode 100644 index 00000000..2c1c9977 --- /dev/null +++ b/dCommon/dEnums/ePackageType.h @@ -0,0 +1,13 @@ +#ifndef __EPACKAGETYPE__H__ +#define __EPACKAGETYPE__H__ + +enum class ePackageType { + INVALID = -1, + ITEM, + BRICKS, + MODELS, + CAR_MODELS +}; + + +#endif //!__EPACKAGETYPE__H__ diff --git a/dCommon/dEnums/ePermissionMap.h b/dCommon/dEnums/ePermissionMap.h new file mode 100644 index 00000000..d15c9fd3 --- /dev/null +++ b/dCommon/dEnums/ePermissionMap.h @@ -0,0 +1,46 @@ +#pragma once + +#include + +#ifndef __EPERMISSIONMAP__H__ +#define __EPERMISSIONMAP__H__ + +/** + * Bitmap of permissions and restrictions for characters. + */ +enum class ePermissionMap : uint64_t { + /** + * Reserved for future use, bit 0-3. + */ + + /** + * The character has restricted trade acccess, bit 4. + */ + RestrictedTradeAccess = 0x1 << 4, + + /** + * The character has restricted mail access, bit 5. + */ + RestrictedMailAccess = 0x1 << 5, + + /** + * The character has restricted chat access, bit 6. + */ + RestrictedChatAccess = 0x1 << 6, + + // + // Combined permissions + // + + /** + * The character is marked as 'old', restricted from trade and mail. + */ + Old = RestrictedTradeAccess | RestrictedMailAccess, + + /** + * The character is soft banned, restricted from trade, mail, and chat. + */ + SoftBanned = RestrictedTradeAccess | RestrictedMailAccess | RestrictedChatAccess, +}; + +#endif //!__EPERMISSIONMAP__H__ diff --git a/dCommon/dEnums/ePetTamingNotifyType.h b/dCommon/dEnums/ePetTamingNotifyType.h new file mode 100644 index 00000000..564e932f --- /dev/null +++ b/dCommon/dEnums/ePetTamingNotifyType.h @@ -0,0 +1,15 @@ +#ifndef __EPETTAMINGNOTIFYTYPE__H__ +#define __EPETTAMINGNOTIFYTYPE__H__ + +#include + +enum class ePetTamingNotifyType : uint32_t { + SUCCESS, + QUIT, + FAILED, + BEGIN, + READY, + NAMINGPET +}; + +#endif //!__EPETTAMINGNOTIFYTYPE__H__ diff --git a/dCommon/dEnums/ePhysicsEffectType.h b/dCommon/dEnums/ePhysicsEffectType.h new file mode 100644 index 00000000..1aa68b34 --- /dev/null +++ b/dCommon/dEnums/ePhysicsEffectType.h @@ -0,0 +1,15 @@ +#ifndef __EPHYSICSEFFECTTYPE__H__ +#define __EPHYSICSEFFECTTYPE__H__ + + +#include + +enum class ePhysicsEffectType : uint32_t { + PUSH, + ATTRACT, + REPULSE, + GRAVITY_SCALE, + FRICTION +}; + +#endif //!__EPHYSICSEFFECTTYPE__H__ diff --git a/dCommon/dEnums/ePlayerFlag.h b/dCommon/dEnums/ePlayerFlag.h new file mode 100644 index 00000000..fefdfb67 --- /dev/null +++ b/dCommon/dEnums/ePlayerFlag.h @@ -0,0 +1,172 @@ +#ifndef __EPLAYERFLAG__H__ +#define __EPLAYERFLAG__H__ + +#include + +enum ePlayerFlag : int32_t { + BTARR_TESTING = 0, + PLAYER_HAS_ENTERED_PET_RANCH = 1, + MINIMAP_UNLOCKED = 2, + ACTIVITY_REBUILDING_FAIL_TIME = 3, + ACTIVITY_REBUILDING_FAIL_RANGE = 4, + ACTIVITY_SHOOTING_GALLERY_HELP = 5, + HELP_WALKING_CONTROLS = 6, + FIRST_SMASHABLE = 7, + FIRST_IMAGINATION_PICKUP = 8, + FIRST_DAMAGE = 9, + FIRST_ITEM = 10, + FIRST_BRICK = 11, + FIRST_CONSUMABLE = 12, + FIRST_EQUIPPABLE = 13, + CHAT_HELP = 14, + FIRST_PET_TAMING_MINIGAME = 15, + FIRST_PET_ON_SWITCH = 16, + FIRST_PET_JUMPED_ON_SWITCH = 17, + FIRST_PET_FOUND_TREASURE = 18, + FIRST_PET_DUG_TREASURE = 19, + FIRST_PET_OWNER_ON_PET_BOUNCER = 20, + FIRST_PET_DESPAWN_NO_IMAGINATION = 21, + FIRST_PET_SELECTED_ENOUGH_BRICKS = 22, + FIRST_EMOTE_UNLOCKED = 23, + GF_PIRATE_REP = 24, + AG_BOB_CINEMATIC_EVENT = 25, + HELP_JUMPING_CONTROLS = 26, + HELP_DOUBLE_JUMP_CONTROLS = 27, + HELP_CAMERA_CONTROLS = 28, + HELP_ROTATE_CONTROLS = 29, + HELP_SMASH = 30, + MONUMENT_INTRO_MUSIC_PLAYED = 31, + BEGINNING_ZONE_SUMMARY_DISPLAYED = 32, + AG_FINISH_LINE_BUILT = 33, + AG_BOSS_AREA_FOUND = 34, + AG_LANDED_IN_BATTLEFIELD = 35, + GF_PLAYER_HAS_BEEN_TO_THE_RAVINE = 36, + MODULAR_BUILD_STARTED = 37, + MODULAR_BUILD_FINISHED_CLICK_BUTTON = 38, + THINKING_HAT_RECEIVED_GO_TO_MODULAR_BUILD_AREA = 39, + BUILD_AREA_ENTERED_MOD_NOT_ACTIVATED_PUT_ON_HAT = 40, + HAT_ON_INSIDE_OF_MOD_BUILD_EQUIP_A_MODULE_FROM_LEG = 41, + MODULE_EQUIPPED_PLACE_ON_GLOWING_BLUE_SPOT = 42, + FIRST_MODULE_PLACED_CORRECTLY_NOW_DO_THE_REST = 43, + ROCKET_COMPLETE_NOW_LAUNCH_FROM_PAD = 44, + JOINED_A_FACTION = 45, + VENTURE_FACTION = 46, + ASSEMBLY_FACTION = 47, + PARADOX_FACTION = 48, + SENTINEL_FACTION = 49, + LUP_WORLD_ACCESS = 50, + AG_FIRST_FLAG_COLLECTED = 51, + TOOLTIP_TALK_TO_SKYLAND_TO_GET_HAT = 52, + MODULAR_BUILD_PLAYER_PLACES_FIRST_MODEL_IN_SCRATCH = 53, + MODULAR_BUILD_FIRST_ARROW_DISPLAY_FOR_MODULE = 54, + AG_BEACON_QB_SO_THE_PLAYER_CAN_ALWAYS_BUILD_THEM = 55, + GF_PET_DIG_FLAG_1 = 56, + GF_PET_DIG_FLAG_2 = 57, + GF_PET_DIG_FLAG_3 = 58, + SUPPRESS_SPACESHIP_CINEMATIC_FLYTHROUGH = 59, + GF_PLAYER_FALL_DEATH = 60, + GF_PLAYER_CAN_GET_FLAG_1 = 61, + GF_PLAYER_CAN_GET_FLAG_2 = 62, + GF_PLAYER_CAN_GET_FLAG_3 = 63, + ENTER_BBB_FROM_PROPERTY_EDIT_CONFIRMATION_DIALOG = 64, + AG_FIRST_COMBAT_COMPLETE = 65, + AG_COMPLETE_BOB_MISSION = 66, + FIRST_MANUAL_PET_HIBERNATE = 69, + CAGED_SPIDER = 74, + IS_NEWS_SCREEN_VISIBLE = 114, + NJ_GARMADON_CINEMATIC_SEEN = 125, + EQUPPED_TRIAL_FACTION_GEAR = 126, + ELEPHANT_PET_3050 = 801, + CAT_PET_3054 = 803, + TRICERATOPS_PET_3195 = 806, + TERRIER_PET_3254 = 807, + SKUNK_PET_3261 = 811, + BUNNY_PET_3672 = 813, + CROCODILE_PET_3994 = 814, + DOBERMAN_PET_5635 = 815, + BUFFALO_PET_5636 = 816, + ROBOT_DOG_PET_5637 = 818, + RED_DRAGON_PET_5639 = 819, + TORTOISE_PET_5640 = 820, + GREEN_DRAGON_PET_5641 = 821, + PANDA_PET_5643 = 822, + MANTIS_PET_5642 = 823, + WARTHOG_PET_6720 = 824, + LION_PET_3520 = 825, + GOAT_PET_7638 = 818, + CRAB_PET_7694 = 827, + REINDEER_PET_12294 = 829, + STEGOSAURUS_PET_12431 = 830, + SABER_CAT_PET_12432 = 831, + GRYPHON_PET_12433 = 832, + ALINE_PET_12334 = 833, + UNKNOWN_PET = 834, + EARTH_DRAGON_PET_16210 = 836, + SKELETON_DRAGON_PET_13067 = 838, + AG_SPACE_SHIP_BINOC_AT_LAUNCH = 1001, + AG_SPACE_SHIP_BINOC_AT_LAUNCH_PLATFORM = 1002, + AG_SPACE_SHIP_BINOC_ON_PLATFORM_TO_LEFT_OF_START = 1003, + AG_SPACE_SHIP_BINOC_ON_PLATFORM_TO_RIGHT_OF_START = 1004, + AG_SPACE_SHIP_BINOC_AT_BOB = 1005, + AG_BATTLE_BINOC_FOR_TRICERETOPS = 1101, + AG_BATTLE_BINOC_AT_PARADOX = 1102, + AG_BATTLE_BINOC_AT_MISSION_GIVER = 1103, + AG_BATTLE_BINOC_AT_BECK = 1104, + AG_MONUMENT_BINOC_INTRO = 1105, + AG_MONUMENT_BINOC_OUTRO = 1106, + AG_LAUNCH_BINOC_INTRO = 1107, + AG_LAUNCH_BINOC_BISON = 1108, + AG_LAUNCH_BINOC_SHARK = 1109, + NS_BINOC_CONCERT_TRANSITION = 1201, + NS_BINOC_RACE_PLACE_TRANSITION = 1202, + NS_BINOC_BRICK_ANNEX_TRANSITION = 1203, + NS_BINOC_GF_LAUNCH = 1204, + NS_BINOC_FV_LAUNCH = 1205, + NS_BINOC_BRICK_ANNEX_WATER = 1206, + NS_BINOC_AG_LAUNCH_AT_RACE_PLACE = 1207, + NS_BINOC_AG_LAUNCH_AT_BRICK_ANNEX = 1208, + NS_BINOC_AG_LAUNCH_AT_PLAZA = 1209, + NS_BINOC_TBA = 1210, + NS_FLAG_COLLECTABLE_1_BY_JONNY_THUNDER = 1211, + NS_FLAG_COLLECTABLE_2_UNDER_CONCERT_BRIDGE = 1212, + NS_FLAG_COLLECTABLE_3_BY_FV_LAUNCH = 1213, + NS_FLAG_COLLECTABLE_4_IN_PLAZA_BEHIND_BUILDING = 1214, + NS_FLAG_COLLECTABLE_5_BY_GF_LAUNCH = 1215, + NS_FLAG_COLLECTABLE_6_BY_DUCK_SG = 1216, + NS_FLAG_COLLECTABLE_7_BY_LUP_LAUNCH = 1217, + NS_FLAG_COLLECTABLE_8_BY_NT_LUANCH = 1218, + NS_FLAG_COLLECTABLE_9_BY_RACE_BUILD = 1219, + NS_FLAG_COLLECTABLE_10_ON_AG_LAUNCH_PATH = 1220, + PR_BINOC_AT_LAUNCH_PAD = 1251, + PR_BINOC_AT_BEGINNING_OF_ISLAND_B = 1252, + PR_BINOC_AT_FIRST_PET_BOUNCER = 1253, + PR_BINOC_ON_BY_CROWS_NEST = 1254, + PR_PET_DIG_AT_BEGINNING_OF_ISLAND_B = 1261, + PR_PET_DIG_AT_THE_LOCATION_OF_OLD_BOUNCE_BACK = 1262, + PR_PET_DIG_UNDER_QB_BRIDGE = 1263, + PR_PET_DIG_BACK_SIDE_BY_PARTNER_BOUNCE = 1264, + PR_PET_DIG_BY_LAUNCH_PAD = 1265, + PR_PET_DIG_BY_FIRST_PET_BOUNCER = 1266, + GF_BINOC_ON_LANDING_PAD = 1301, + GF_BINOC_AT_RAVINE_START = 1302, + GF_BINOC_ON_TOP_OF_RAVINE_HEAD = 1303, + GF_BINOC_AT_TURTLE_AREA = 1304, + GF_BINOC_IN_TUNNEL_TO_ELEPHANTS = 1305, + GF_BINOC_IN_ELEPHANTS_AREA = 1306, + GF_BINOC_IN_RACING_AREA = 1307, + GF_BINOC_IN_CROC_AREA = 1308, + GF_BINOC_IN_JAIL_AREA = 1309, + GF_BINOC_TELESCOPE_NEXT_TO_CAPTAIN_JACK = 1310, + NT_HEART_OF_DARKNESS = 1911, + NT_PLINTH_REBUILD = 1919, + NT_FACTION_SPY_DUKE = 1974, + NT_FACTION_SPY_OVERBUILD = 1976, + NT_FACTION_SPY_HAEL = 1977, + NJ_EARTH_SPINJITZU = 2030, + NJ_LIGHTNING_SPINJITZU = 2031, + NJ_ICE_SPINJITZU = 2032, + NJ_FIRE_SPINJITZU = 2033, + NJ_WU_SHOW_DAILY_CHEST = 2099 +}; + +#endif //!__EPLAYERFLAG__H__ diff --git a/dCommon/dEnums/eQuickBuildFailReason.h b/dCommon/dEnums/eQuickBuildFailReason.h new file mode 100644 index 00000000..a6144bd5 --- /dev/null +++ b/dCommon/dEnums/eQuickBuildFailReason.h @@ -0,0 +1,13 @@ +#ifndef __EQUICKBUILDFAILREASON__H__ +#define __EQUICKBUILDFAILREASON__H__ + +#include + +enum class eQuickBuildFailReason : uint32_t { + NOT_GIVEN, + OUT_OF_IMAGINATION, + CANCELED_EARLY, + BUILD_ENDED +}; + +#endif //!__EQUICKBUILDFAILREASON__H__ diff --git a/dCommon/dEnums/eRacingTaskParam.h b/dCommon/dEnums/eRacingTaskParam.h new file mode 100644 index 00000000..df50e382 --- /dev/null +++ b/dCommon/dEnums/eRacingTaskParam.h @@ -0,0 +1,25 @@ +#pragma once + +#ifndef __ERACINGTASKPARAM__H__ +#define __ERACINGTASKPARAM__H__ + +#include + +enum class eRacingTaskParam : int32_t { + FINISH_WITH_PLACEMENT = 1, + LAP_TIME, + TOTAL_TRACK_TIME, + COMPLETE_ANY_RACING_TASK, + COMPLETE_TRACK_TASKS, + MODULAR_BUILDING, + SAFE_DRIVER = 10, + SMASHABLES, + COLLECT_IMAGINATION, + COMPETED_IN_RACE, + WIN_RACE_IN_WORLD, + FIRST_PLACE_MULTIPLE_TRACKS, + LAST_PLACE_FINISH, + SMASH_SPECIFIC_SMASHABLE +}; + +#endif //!__ERACINGTASKPARAM__H__ diff --git a/dCommon/dEnums/eRebuildState.h b/dCommon/dEnums/eRebuildState.h new file mode 100644 index 00000000..497bcb77 --- /dev/null +++ b/dCommon/dEnums/eRebuildState.h @@ -0,0 +1,15 @@ +#ifndef __EREBUILDSTATE__H__ +#define __EREBUILDSTATE__H__ + +#include + +enum class eRebuildState : uint32_t { + OPEN, + COMPLETED = 2, + RESETTING = 4, + BUILDING, + INCOMPLETE +}; + + +#endif //!__EREBUILDSTATE__H__ diff --git a/dCommon/dEnums/eRenameResponse.h b/dCommon/dEnums/eRenameResponse.h new file mode 100644 index 00000000..2298e922 --- /dev/null +++ b/dCommon/dEnums/eRenameResponse.h @@ -0,0 +1,15 @@ +#ifndef __ERENAMERESPONSE__H__ +#define __ERENAMERESPONSE__H__ + +#include + +//! An enum for character rename responses +enum class eRenameResponse : uint8_t { + SUCCESS = 0, + UNKNOWN_ERROR, + NAME_UNAVAILABLE, + NAME_IN_USE +}; + + +#endif //!__ERENAMERESPONSE__H__ diff --git a/dCommon/dEnums/eReplicaComponentType.h b/dCommon/dEnums/eReplicaComponentType.h new file mode 100644 index 00000000..2d24c19e --- /dev/null +++ b/dCommon/dEnums/eReplicaComponentType.h @@ -0,0 +1,127 @@ +#ifndef __EREPLICACOMPONENTTYPE__H__ +#define __EREPLICACOMPONENTTYPE__H__ + +#include + +enum class eReplicaComponentType : uint32_t { + INVALID = 0, + CONTROLLABLE_PHYSICS, + RENDER, + SIMPLE_PHYSICS, + CHARACTER, + SCRIPT, + BOUNCER, + BUFF, // buff is really 98, this is DESTROYABLE + GHOST, + SKILL, + SPAWNER, + ITEM, + REBUILD, + REBUILD_START, + REBUILD_ACTIVATOR, + ICON_ONLY, + VENDOR, + INVENTORY, + PROJECTILE_PHYSICS, + SHOOTING_GALLERY, + RIGID_BODY_PHANTOM_PHYSICS, + DROP_EFFECT, + CHEST, + COLLECTIBLE, + BLUEPRINT, + MOVING_PLATFORM, + PET, + PLATFORM_BOUNDARY, + MODULE, + ARCADE, + VEHICLE_PHYSICS, // Havok demo based + MOVEMENT_AI, + EXHIBIT, + OVERHEAD_ICON, + PET_CONTROL, + MINIFIG, + PROPERTY, + PET_CREATOR, + MODEL_BUILDER, + SCRIPTED_ACTIVITY, + PHANTOM_PHYSICS, + SPRINGPAD, + MODEL, + PROPERTY_ENTRANCE, + FX, + PROPERTY_MANAGEMENT, + VEHICLE_PHYSICS_NEW, // internal physics based on havok + PHYSICS_SYSTEM, + QUICK_BUILD, + SWITCH, + ZONE_CONTROL, // Minigame + CHANGLING, + CHOICE_BUILD, + PACKAGE, + SOUND_REPEATER, + SOUND_AMBIENT_2D, + SOUND_AMBIENT_3D, + PRECONDITION, + PLAYER_FLAG, + CUSTOM_BUILD_ASSEMBLY, + BASE_COMBAT_AI, + MODULE_ASSEMBLY, + SHOWCASE_MODEL_HANDLER, + RACING_MODULE, + GENERIC_ACTIVATOR, + PROPERTY_VENDOR, + HF_LIGHT_DIRECTION_GADGET, + ROCKET_LAUNCH, + ROCKET_LANDING, + TRIGGER, + DROPPED_LOOT, + RACING_CONTROL, + FACTION_TRIGGER, + MISSION_OFFER, + RACING_STATS, + LUP_EXHIBIT, + BBB, + SOUND_TRIGGER, + PROXIMITY_MONITOR, + RACING_SOUND_TRIGGER, + CHAT, + FRIENDS_LIST, + GUILD, + LOCAL_SYSTEM, + MISSION, + MUTABLE_MODEL_BEHAVIORS, + PATHFINDING, + PET_TAMING_CONTROL, + PROPERTY_EDITOR, + SKINNED_RENDER, + SLASH_COMMAND, + STATUS_EFFECT, + TEAMS, + TEXT_EFFECT, + TRADE, + USER_CONTROL, + IGNORE_LIST, + ROCKET_LAUNCH_LUP, + BUFF_REAL, // the real buff component, should just be name BUFF + INTERACTION_MANAGER, + DONATION_VENDOR, + COMBAT_MEDIATOR, + COMMENDATION_VENDOR, + GATE_RUSH_CONTROL, + RAIL_ACTIVATOR, + ROLLER, + PLAYER_FORCED_MOVEMENT, + CRAFTING, + POSSESSABLE, + LEVEL_PROGRESSION, + POSSESSOR, + MOUNT_CONTROL, + UNKNOWN_112, + PROPERTY_PLAQUE, + BUILD_BORDER, + UNKNOWN_115, + CULLING_PLANE, + DESTROYABLE = 1000 // Actually 7 +}; + +#endif //!__EREPLICACOMPONENTTYPE__H__ diff --git a/dCommon/dEnums/eReplicaPacketType.h b/dCommon/dEnums/eReplicaPacketType.h new file mode 100644 index 00000000..2650fb30 --- /dev/null +++ b/dCommon/dEnums/eReplicaPacketType.h @@ -0,0 +1,12 @@ +#ifndef __EREPLICAPACKETTYPE__H__ +#define __EREPLICAPACKETTYPE__H__ + +#include + +enum class eReplicaPacketType : uint8_t { + CONSTRUCTION, + SERIALIZATION, + DESTRUCTION +}; + +#endif //!__EREPLICAPACKETTYPE__H__ diff --git a/dCommon/dEnums/eServerDisconnectIdentifiers.h b/dCommon/dEnums/eServerDisconnectIdentifiers.h new file mode 100644 index 00000000..99d2cd44 --- /dev/null +++ b/dCommon/dEnums/eServerDisconnectIdentifiers.h @@ -0,0 +1,24 @@ +#ifndef __ESERVERDISCONNECTIDENTIFIERS__H__ +#define __ESERVERDISCONNECTIDENTIFIERS__H__ + +#include + +enum class eServerDisconnectIdentifiers : uint32_t { + UNKNOWN_SERVER_ERROR = 0, + WRONG_GAME_VERSION, + WRONG_SERVER_VERSION, + CONNECTION_ON_INVALID_PORT, + DUPLICATE_LOGIN, + SERVER_SHUTDOWN, + SERVER_MAP_LOAD_FAILURE, + INVALID_SESSION_KEY, + ACCOUNT_NOT_IN_PENDING_LIST, + CHARACTER_NOT_FOUND, + CHARACTER_CORRUPTED, + KICK, + SAVE_FAILURE, + FREE_TRIAL_EXPIRED, + PLAY_SCHEDULE_TIME_DONE +}; + +#endif //!__ESERVERDISCONNECTIDENTIFIERS__H__ diff --git a/dCommon/dEnums/eServerMessageType.h b/dCommon/dEnums/eServerMessageType.h new file mode 100644 index 00000000..7f211ffb --- /dev/null +++ b/dCommon/dEnums/eServerMessageType.h @@ -0,0 +1,12 @@ +#ifndef __ESERVERMESSAGETYPE__H__ +#define __ESERVERMESSAGETYPE__H__ + +#include +//! The Internal Server Packet Identifiers +enum class eServerMessageType : uint32_t { + VERSION_CONFIRM = 0, + DISCONNECT_NOTIFY, + GENERAL_NOTIFY +}; + +#endif //!__ESERVERMESSAGETYPE__H__ diff --git a/dCommon/dEnums/eSqliteDataType.h b/dCommon/dEnums/eSqliteDataType.h new file mode 100644 index 00000000..26e1233b --- /dev/null +++ b/dCommon/dEnums/eSqliteDataType.h @@ -0,0 +1,16 @@ +#ifndef __ESQLITEDATATYPE__H__ +#define __ESQLITEDATATYPE__H__ + +#include + +enum class eSqliteDataType : int32_t { + NONE = 0, + INT32, + REAL = 3, + TEXT_4, + INT_BOOL, + INT64, + TEXT_8 = 8 +}; + +#endif //!__ESQLITEDATATYPE__H__ diff --git a/dCommon/dEnums/eStateChangeType.h b/dCommon/dEnums/eStateChangeType.h new file mode 100644 index 00000000..e187e265 --- /dev/null +++ b/dCommon/dEnums/eStateChangeType.h @@ -0,0 +1,11 @@ +#ifndef __ESTATECHANGETYPE__H__ +#define __ESTATECHANGETYPE__H__ + +#include + +enum class eStateChangeType : uint32_t { + PUSH, + POP +}; + +#endif //!__ESTATECHANGETYPE__H__ diff --git a/dCommon/dEnums/eTerminateType.h b/dCommon/dEnums/eTerminateType.h new file mode 100644 index 00000000..189734f8 --- /dev/null +++ b/dCommon/dEnums/eTerminateType.h @@ -0,0 +1,12 @@ +#ifndef __ETERMINATETYPE__H__ +#define __ETERMINATETYPE__H__ + +#include + +enum class eTerminateType : uint32_t { + RANGE, + USER, + FROM_INTERACTION +}; + +#endif //!__ETERMINATETYPE__H__ diff --git a/dCommon/dEnums/eTriggerCommandType.h b/dCommon/dEnums/eTriggerCommandType.h new file mode 100644 index 00000000..fadfafb8 --- /dev/null +++ b/dCommon/dEnums/eTriggerCommandType.h @@ -0,0 +1,119 @@ +#ifndef __ETRIGGERCOMMANDTYPE__H__ +#define __ETRIGGERCOMMANDTYPE__H__ + +// For info about Trigger Command see: +// https://docs.lu-dev.net/en/latest/file-structures/lutriggers.html?highlight=trigger#possible-values-commands + +enum class eTriggerCommandType { + INVALID, + ZONE_PLAYER, + FIRE_EVENT, + DESTROY_OBJ, + TOGGLE_TRIGGER, + RESET_REBUILD, + SET_PATH, + SET_PICK_TYPE, + MOVE_OBJECT, + ROTATE_OBJECT, + PUSH_OBJECT, + REPEL_OBJECT, + SET_TIMER, + CANCEL_TIMER, + PLAY_CINEMATIC, + TOGGLE_BBB, + UPDATE_MISSION, + SET_BOUNCER_STATE, + BOUNCE_ALL_ON_BOUNCER, + TURN_AROUND_ON_PATH, + GO_FORWARD_ON_PATH, + GO_BACKWARD_ON_PATH, + STOP_PATHING, + START_PATHING, + LOCK_OR_UNLOCK_CONTROLS, + PLAY_EFFECT, + STOP_EFFECT, + ACTIVATE_MUSIC_CUE, + DEACTIVATE_MUSIC_CUE, + FLASH_MUSIC_CUE, + SET_MUSIC_PARAMETER, + PLAY_2D_AMBIENT_SOUND, + STOP_2D_AMBIENT_SOUND, + PLAY_3D_AMBIENT_SOUND, + STOP_3D_AMBIENT_SOUND, + ACTIVATE_MIXER_PROGRAM, + DEACTIVATE_MIXER_PROGRAM, + CAST_SKILL, + DISPLAY_ZONE_SUMMARY, + SET_PHYSICS_VOLUME_EFFECT, + SET_PHYSICS_VOLUME_STATUS, + SET_MODEL_TO_BUILD, + SPAWN_MODEL_BRICKS, + ACTIVATE_SPAWNER_NETWORK, + DEACTIVATE_SPAWNER_NETWORK, + RESET_SPAWNER_NETWORK, + DESTROY_SPAWNER_NETWORK_OBJECTS, + GO_TO_WAYPOINT, + ACTIVATE_PHYSICS +}; + + +class TriggerCommandType { +public: + static eTriggerCommandType StringToTriggerCommandType(std::string commandString) { + const std::map TriggerCommandMap = { + { "zonePlayer", eTriggerCommandType::ZONE_PLAYER}, + { "fireEvent", eTriggerCommandType::FIRE_EVENT}, + { "destroyObj", eTriggerCommandType::DESTROY_OBJ}, + { "toggleTrigger", eTriggerCommandType::TOGGLE_TRIGGER}, + { "resetRebuild", eTriggerCommandType::RESET_REBUILD}, + { "setPath", eTriggerCommandType::SET_PATH}, + { "setPickType", eTriggerCommandType::SET_PICK_TYPE}, + { "moveObject", eTriggerCommandType::MOVE_OBJECT}, + { "rotateObject", eTriggerCommandType::ROTATE_OBJECT}, + { "pushObject", eTriggerCommandType::PUSH_OBJECT}, + { "repelObject", eTriggerCommandType::REPEL_OBJECT}, + { "setTimer", eTriggerCommandType::SET_TIMER}, + { "cancelTimer", eTriggerCommandType::CANCEL_TIMER}, + { "playCinematic", eTriggerCommandType::PLAY_CINEMATIC}, + { "toggleBBB", eTriggerCommandType::TOGGLE_BBB}, + { "updateMission", eTriggerCommandType::UPDATE_MISSION}, + { "setBouncerState", eTriggerCommandType::SET_BOUNCER_STATE}, + { "bounceAllOnBouncer", eTriggerCommandType::BOUNCE_ALL_ON_BOUNCER}, + { "turnAroundOnPath", eTriggerCommandType::TURN_AROUND_ON_PATH}, + { "goForwardOnPath", eTriggerCommandType::GO_FORWARD_ON_PATH}, + { "goBackwardOnPath", eTriggerCommandType::GO_BACKWARD_ON_PATH}, + { "stopPathing", eTriggerCommandType::STOP_PATHING}, + { "startPathing", eTriggerCommandType::START_PATHING}, + { "LockOrUnlockControls", eTriggerCommandType::LOCK_OR_UNLOCK_CONTROLS}, + { "PlayEffect", eTriggerCommandType::PLAY_EFFECT}, + { "StopEffect", eTriggerCommandType::STOP_EFFECT}, + { "activateMusicCue", eTriggerCommandType::ACTIVATE_MUSIC_CUE}, + { "deactivateMusicCue", eTriggerCommandType::DEACTIVATE_MUSIC_CUE}, + { "flashMusicCue", eTriggerCommandType::FLASH_MUSIC_CUE}, + { "setMusicParameter", eTriggerCommandType::SET_MUSIC_PARAMETER}, + { "play2DAmbientSound", eTriggerCommandType::PLAY_2D_AMBIENT_SOUND}, + { "stop2DAmbientSound", eTriggerCommandType::STOP_2D_AMBIENT_SOUND}, + { "play3DAmbientSound", eTriggerCommandType::PLAY_3D_AMBIENT_SOUND}, + { "stop3DAmbientSound", eTriggerCommandType::STOP_3D_AMBIENT_SOUND}, + { "activateMixerProgram", eTriggerCommandType::ACTIVATE_MIXER_PROGRAM}, + { "deactivateMixerProgram", eTriggerCommandType::DEACTIVATE_MIXER_PROGRAM}, + { "CastSkill", eTriggerCommandType::CAST_SKILL}, + { "displayZoneSummary", eTriggerCommandType::DISPLAY_ZONE_SUMMARY}, + { "SetPhysicsVolumeEffect", eTriggerCommandType::SET_PHYSICS_VOLUME_EFFECT}, + { "SetPhysicsVolumeStatus", eTriggerCommandType::SET_PHYSICS_VOLUME_STATUS}, + { "setModelToBuild", eTriggerCommandType::SET_MODEL_TO_BUILD}, + { "spawnModelBricks", eTriggerCommandType::SPAWN_MODEL_BRICKS}, + { "ActivateSpawnerNetwork", eTriggerCommandType::ACTIVATE_SPAWNER_NETWORK}, + { "DeactivateSpawnerNetwork", eTriggerCommandType::DEACTIVATE_SPAWNER_NETWORK}, + { "ResetSpawnerNetwork", eTriggerCommandType::RESET_SPAWNER_NETWORK}, + { "DestroySpawnerNetworkObjects", eTriggerCommandType::DESTROY_SPAWNER_NETWORK_OBJECTS}, + { "Go_To_Waypoint", eTriggerCommandType::GO_TO_WAYPOINT}, + { "ActivatePhysics", eTriggerCommandType::ACTIVATE_PHYSICS} + }; + + auto intermed = TriggerCommandMap.find(commandString); + return (intermed != TriggerCommandMap.end()) ? intermed->second : eTriggerCommandType::INVALID; + }; +}; + +#endif //!__ETRIGGERCOMMANDTYPE__H__ diff --git a/dCommon/dEnums/eTriggerEventType.h b/dCommon/dEnums/eTriggerEventType.h new file mode 100644 index 00000000..a2daa256 --- /dev/null +++ b/dCommon/dEnums/eTriggerEventType.h @@ -0,0 +1,53 @@ +#ifndef __ETRIGGEREVENTTYPE__H__ +#define __ETRIGGEREVENTTYPE__H__ + +enum class eTriggerEventType { + INVALID, + DESTROY, + CUSTOM_EVENT, + ENTER, + EXIT, + CREATE, + HIT, + TIMER_DONE, + REBUILD_COMPLETE, + ACTIVATED, + DEACTIVATED, + ARRIVED, + ARRIVED_AT_END_OF_PATH, + ZONE_SUMMARY_DISMISSED, + ARRIVED_AT_DESIRED_WAYPOINT, + PET_ON_SWITCH, + PET_OFF_SWITCH, + INTERACT +}; + +class TriggerEventType { +public: + static eTriggerEventType StringToTriggerEventType(std::string commandString) { + const std::map TriggerEventMap = { + {"OnDestroy", eTriggerEventType::DESTROY}, + {"OnCustomEvent", eTriggerEventType::CUSTOM_EVENT}, + {"OnEnter", eTriggerEventType::ENTER}, + {"OnExit", eTriggerEventType::EXIT}, + {"OnCreate", eTriggerEventType::CREATE}, + {"OnHit", eTriggerEventType::HIT}, + {"OnTimerDone", eTriggerEventType::TIMER_DONE}, + {"OnRebuildComplete", eTriggerEventType::REBUILD_COMPLETE}, + {"OnActivated", eTriggerEventType::ACTIVATED}, + {"OnDectivated", eTriggerEventType::DEACTIVATED}, // Dectivated vs Deactivated + {"OnArrived", eTriggerEventType::ARRIVED}, + {"OnArrivedAtEndOfPath", eTriggerEventType::ARRIVED_AT_END_OF_PATH}, + {"OnZoneSummaryDismissed", eTriggerEventType::ZONE_SUMMARY_DISMISSED}, + {"OnArrivedAtDesiredWaypoint", eTriggerEventType::ARRIVED_AT_DESIRED_WAYPOINT}, + {"OnPetOnSwitch", eTriggerEventType::PET_ON_SWITCH}, + {"OnPetOffSwitch", eTriggerEventType::PET_OFF_SWITCH}, + {"OnInteract", eTriggerEventType::INTERACT}, + }; + + auto intermed = TriggerEventMap.find(commandString); + return (intermed != TriggerEventMap.end()) ? intermed->second : eTriggerEventType::INVALID; + }; +}; + +#endif //!__ETRIGGEREVENTTYPE__H__ diff --git a/dCommon/eUnequippableActiveType.h b/dCommon/dEnums/eUnequippableActiveType.h similarity index 100% rename from dCommon/eUnequippableActiveType.h rename to dCommon/dEnums/eUnequippableActiveType.h diff --git a/dCommon/dEnums/eUseItemResponse.h b/dCommon/dEnums/eUseItemResponse.h new file mode 100644 index 00000000..ba0ece7f --- /dev/null +++ b/dCommon/dEnums/eUseItemResponse.h @@ -0,0 +1,12 @@ +#ifndef __EUSEITEMRESPONSE__H__ +#define __EUSEITEMRESPONSE__H__ + +#include + +enum class eUseItemResponse : uint32_t { + NoImaginationForPet = 1, + FailedPrecondition, + MountsNotAllowed +}; + +#endif //!__EUSEITEMRESPONSE__H__ diff --git a/dCommon/dEnums/eWorldMessageType.h b/dCommon/dEnums/eWorldMessageType.h new file mode 100644 index 00000000..2a65fd98 --- /dev/null +++ b/dCommon/dEnums/eWorldMessageType.h @@ -0,0 +1,42 @@ +#ifndef __EWORLDMESSAGETYPE__H__ +#define __EWORLDMESSAGETYPE__H__ + +#include + +enum class eWorldMessageType : uint32_t { + VALIDATION = 1, // Session info + CHARACTER_LIST_REQUEST, + CHARACTER_CREATE_REQUEST, + LOGIN_REQUEST, // Character selected + GAME_MSG, + CHARACTER_DELETE_REQUEST, + CHARACTER_RENAME_REQUEST, + HAPPY_FLOWER_MODE_NOTIFY, + SLASH_RELOAD_MAP, // Reload map cmp + SLASH_PUSH_MAP_REQUEST, // Push map req cmd + SLASH_PUSH_MAP, // Push map cmd + SLASH_PULL_MAP, // Pull map cmd + LOCK_MAP_REQUEST, + GENERAL_CHAT_MESSAGE, // General chat message + HTTP_MONITOR_INFO_REQUEST, + SLASH_DEBUG_SCRIPTS, // Debug scripts cmd + MODELS_CLEAR, + EXHIBIT_INSERT_MODEL, + LEVEL_LOAD_COMPLETE, // Character data request + TMP_GUILD_CREATE, + ROUTE_PACKET, // Social? + POSITION_UPDATE, + MAIL, + WORD_CHECK, // Whitelist word check + STRING_CHECK, // Whitelist string check + GET_PLAYERS_IN_ZONE, + REQUEST_UGC_MANIFEST_INFO, + BLUEPRINT_GET_ALL_DATA_REQUEST, + CANCEL_MAP_QUEUE, + HANDLE_FUNNESS, + FAKE_PRG_CSR_MESSAGE, + REQUEST_FREE_TRIAL_REFRESH, + GM_SET_FREE_TRIAL_STATUS +}; + +#endif //!__EWORLDMESSAGETYPE__H__ diff --git a/dCommon/dLogger.cpp b/dCommon/dLogger.cpp index d4e6a96e..7785a070 100644 --- a/dCommon/dLogger.cpp +++ b/dCommon/dLogger.cpp @@ -89,7 +89,7 @@ void dLogger::Log(const std::string& className, const std::string& message) { void dLogger::LogDebug(const char* className, const char* format, ...) { if (!m_logDebugStatements) return; va_list args; - std::string log = "[" + std::string(className) + "] " + std::string(format); + std::string log = "[" + std::string(className) + "] " + std::string(format) + "\n"; va_start(args, format); vLog(log.c_str(), args); va_end(args); diff --git a/dCommon/eAninmationFlags.h b/dCommon/eAninmationFlags.h deleted file mode 100644 index 9b2ea3fb..00000000 --- a/dCommon/eAninmationFlags.h +++ /dev/null @@ -1,44 +0,0 @@ -#pragma once - -#ifndef __EANINMATIONFLAGS__H__ -#define __EANINMATIONFLAGS__H__ - -#include - -enum class eAnimationFlags : uint32_t { - IDLE_INVALID = 0, // made up, for internal use!!! - IDLE_BASIC, - IDLE_SWIM, - IDLE_CARRY, - IDLE_SWORD, - IDLE_HAMMER, - IDLE_SPEAR, - IDLE_PISTOL, - IDLE_BOW, - IDLE_COMBAT, - IDLE_JETPACK, - IDLE_HORSE, - IDLE_SG, - IDLE_ORGAN, - IDLE_SKATEBOARD, - IDLE_DAREDEVIL, - IDLE_SAMURAI, - IDLE_SUMMONER, - IDLE_BUCCANEER, - IDLE_MISC, - IDLE_NINJA, - IDLE_MISC1, - IDLE_MISC2, - IDLE_MISC3, - IDLE_MISC4, - IDLE_MISC5, - IDLE_MISC6, - IDLE_MISC7, - IDLE_MISC8, - IDLE_MISC9, - IDLE_MISC10, - IDLE_MISC11, - IDLE_MISC12 -}; - -#endif //!__EANINMATIONFLAGS__H__ diff --git a/dCommon/eItemType.h b/dCommon/eItemType.h deleted file mode 100644 index e68ce695..00000000 --- a/dCommon/eItemType.h +++ /dev/null @@ -1,36 +0,0 @@ -#pragma once - -#ifndef __EITEMTYPE__H__ -#define __EITEMTYPE__H__ - -#include - -enum class eItemType : int32_t { - ITEM_TYPE_UNKNOWN = -1, //!< An unknown item type - ITEM_TYPE_BRICK = 1, //!< A brick - ITEM_TYPE_HAT = 2, //!< A hat / head item - ITEM_TYPE_HAIR = 3, //!< A hair item - ITEM_TYPE_NECK = 4, //!< A neck item - ITEM_TYPE_LEFT_HAND = 5, //!< A left handed item - ITEM_TYPE_RIGHT_HAND = 6, //!< A right handed item - ITEM_TYPE_LEGS = 7, //!< A pants item - ITEM_TYPE_LEFT_TRINKET = 8, //!< A left handled trinket item - ITEM_TYPE_RIGHT_TRINKET = 9, //!< A right handed trinket item - ITEM_TYPE_BEHAVIOR = 10, //!< A behavior - ITEM_TYPE_PROPERTY = 11, //!< A property - ITEM_TYPE_MODEL = 12, //!< A model - ITEM_TYPE_COLLECTIBLE = 13, //!< A collectible item - ITEM_TYPE_CONSUMABLE = 14, //!< A consumable item - ITEM_TYPE_CHEST = 15, //!< A chest item - ITEM_TYPE_EGG = 16, //!< An egg - ITEM_TYPE_PET_FOOD = 17, //!< A pet food item - ITEM_TYPE_QUEST_OBJECT = 18, //!< A quest item - ITEM_TYPE_PET_INVENTORY_ITEM = 19, //!< A pet inventory item - ITEM_TYPE_PACKAGE = 20, //!< A package - ITEM_TYPE_LOOT_MODEL = 21, //!< A loot model - ITEM_TYPE_VEHICLE = 22, //!< A vehicle - ITEM_TYPE_CURRENCY = 23, //!< Currency - ITEM_TYPE_MOUNT = 24 //!< A Mount -}; - -#endif //!__EITEMTYPE__H__ diff --git a/dDatabase/CDClientDatabase.cpp b/dDatabase/CDClientDatabase.cpp index bf8485d6..4c2df1d2 100644 --- a/dDatabase/CDClientDatabase.cpp +++ b/dDatabase/CDClientDatabase.cpp @@ -14,6 +14,11 @@ CppSQLite3Query CDClientDatabase::ExecuteQuery(const std::string& query) { return conn->execQuery(query.c_str()); } +//! Updates the CDClient file with Data Manipulation Language (DML) commands. +int CDClientDatabase::ExecuteDML(const std::string& query) { + return conn->execDML(query.c_str()); +} + //! Makes prepared statements CppSQLite3Statement CDClientDatabase::CreatePreppedStmt(const std::string& query) { return conn->compileStatement(query.c_str()); diff --git a/dDatabase/CDClientDatabase.h b/dDatabase/CDClientDatabase.h index 6b254ebb..59f69b7d 100644 --- a/dDatabase/CDClientDatabase.h +++ b/dDatabase/CDClientDatabase.h @@ -16,9 +16,6 @@ // Enable this to cache all entries in each table for fast access, comes with more memory cost //#define CDCLIENT_CACHE_ALL - // Enable this to skip some unused columns in some tables -#define UNUSED(v) - /*! \file CDClientDatabase.hpp \brief An interface between the CDClient.sqlite file and the server @@ -40,6 +37,14 @@ namespace CDClientDatabase { */ CppSQLite3Query ExecuteQuery(const std::string& query); + //! Updates the CDClient file with Data Manipulation Language (DML) commands. + /*! + \param query The DML command to run. DML command can be multiple queries in one string but only + the last one will return its number of updated rows. + \return The number of updated rows. + */ + int ExecuteDML(const std::string& query); + //! Queries the CDClient and parses arguments /*! \param query The query with formatted arguments diff --git a/dDatabase/CDClientManager.cpp b/dDatabase/CDClientManager.cpp index 4b32e749..9df6ff31 100644 --- a/dDatabase/CDClientManager.cpp +++ b/dDatabase/CDClientManager.cpp @@ -1,46 +1,80 @@ #include "CDClientManager.h" +#include "CDActivityRewardsTable.h" +#include "CDAnimationsTable.h" +#include "CDBehaviorParameterTable.h" +#include "CDBehaviorTemplateTable.h" +#include "CDComponentsRegistryTable.h" +#include "CDCurrencyTableTable.h" +#include "CDDestructibleComponentTable.h" +#include "CDEmoteTable.h" +#include "CDInventoryComponentTable.h" +#include "CDItemComponentTable.h" +#include "CDItemSetsTable.h" +#include "CDItemSetSkillsTable.h" +#include "CDLevelProgressionLookupTable.h" +#include "CDLootMatrixTable.h" +#include "CDLootTableTable.h" +#include "CDMissionNPCComponentTable.h" +#include "CDMissionTasksTable.h" +#include "CDMissionsTable.h" +#include "CDObjectSkillsTable.h" +#include "CDObjectsTable.h" +#include "CDPhysicsComponentTable.h" +#include "CDRebuildComponentTable.h" +#include "CDScriptComponentTable.h" +#include "CDSkillBehaviorTable.h" +#include "CDZoneTableTable.h" +#include "CDVendorComponentTable.h" +#include "CDActivitiesTable.h" +#include "CDPackageComponentTable.h" +#include "CDProximityMonitorComponentTable.h" +#include "CDMovementAIComponentTable.h" +#include "CDBrickIDTableTable.h" +#include "CDRarityTableTable.h" +#include "CDMissionEmailTable.h" +#include "CDRewardsTable.h" +#include "CDPropertyEntranceComponentTable.h" +#include "CDPropertyTemplateTable.h" +#include "CDFeatureGatingTable.h" +#include "CDRailActivatorComponent.h" -// Static Variables -CDClientManager* CDClientManager::m_Address = nullptr; - -//! Initializes the manager -void CDClientManager::Initialize(void) { - tables.insert(std::make_pair("ActivityRewards", new CDActivityRewardsTable())); - UNUSED(tables.insert(std::make_pair("Animations", new CDAnimationsTable()))); - tables.insert(std::make_pair("BehaviorParameter", new CDBehaviorParameterTable())); - tables.insert(std::make_pair("BehaviorTemplate", new CDBehaviorTemplateTable())); - tables.insert(std::make_pair("ComponentsRegistry", new CDComponentsRegistryTable())); - tables.insert(std::make_pair("CurrencyTable", new CDCurrencyTableTable())); - tables.insert(std::make_pair("DestructibleComponent", new CDDestructibleComponentTable())); - tables.insert(std::make_pair("EmoteTable", new CDEmoteTableTable())); - tables.insert(std::make_pair("InventoryComponent", new CDInventoryComponentTable())); - tables.insert(std::make_pair("ItemComponent", new CDItemComponentTable())); - tables.insert(std::make_pair("ItemSets", new CDItemSetsTable())); - tables.insert(std::make_pair("ItemSetSkills", new CDItemSetSkillsTable())); - tables.insert(std::make_pair("LevelProgressionLookup", new CDLevelProgressionLookupTable())); - tables.insert(std::make_pair("LootMatrix", new CDLootMatrixTable())); - tables.insert(std::make_pair("LootTable", new CDLootTableTable())); - tables.insert(std::make_pair("MissionNPCComponent", new CDMissionNPCComponentTable())); - tables.insert(std::make_pair("MissionTasks", new CDMissionTasksTable())); - tables.insert(std::make_pair("Missions", new CDMissionsTable())); - tables.insert(std::make_pair("ObjectSkills", new CDObjectSkillsTable())); - tables.insert(std::make_pair("Objects", new CDObjectsTable())); - tables.insert(std::make_pair("PhysicsComponent", new CDPhysicsComponentTable())); - tables.insert(std::make_pair("RebuildComponent", new CDRebuildComponentTable())); - tables.insert(std::make_pair("ScriptComponent", new CDScriptComponentTable())); - tables.insert(std::make_pair("SkillBehavior", new CDSkillBehaviorTable())); - tables.insert(std::make_pair("ZoneTable", new CDZoneTableTable())); - tables.insert(std::make_pair("VendorComponent", new CDVendorComponentTable())); - tables.insert(std::make_pair("Activities", new CDActivitiesTable())); - tables.insert(std::make_pair("PackageComponent", new CDPackageComponentTable())); - tables.insert(std::make_pair("ProximityMonitorComponent", new CDProximityMonitorComponentTable())); - tables.insert(std::make_pair("MovementAIComponent", new CDMovementAIComponentTable())); - tables.insert(std::make_pair("BrickIDTable", new CDBrickIDTableTable())); - tables.insert(std::make_pair("RarityTable", new CDRarityTableTable())); - tables.insert(std::make_pair("MissionEmail", new CDMissionEmailTable())); - tables.insert(std::make_pair("Rewards", new CDRewardsTable())); - tables.insert(std::make_pair("PropertyEntranceComponent", new CDPropertyEntranceComponentTable())); - tables.insert(std::make_pair("PropertyTemplate", new CDPropertyTemplateTable())); - tables.insert(std::make_pair("FeatureGating", new CDFeatureGatingTable())); - tables.insert(std::make_pair("RailActivatorComponent", new CDRailActivatorComponentTable())); +CDClientManager::CDClientManager() { + CDActivityRewardsTable::Instance(); + CDAnimationsTable::Instance(); + CDBehaviorParameterTable::Instance(); + CDBehaviorTemplateTable::Instance(); + CDComponentsRegistryTable::Instance(); + CDCurrencyTableTable::Instance(); + CDDestructibleComponentTable::Instance(); + CDEmoteTableTable::Instance(); + CDInventoryComponentTable::Instance(); + CDItemComponentTable::Instance(); + CDItemSetsTable::Instance(); + CDItemSetSkillsTable::Instance(); + CDLevelProgressionLookupTable::Instance(); + CDLootMatrixTable::Instance(); + CDLootTableTable::Instance(); + CDMissionNPCComponentTable::Instance(); + CDMissionTasksTable::Instance(); + CDMissionsTable::Instance(); + CDObjectSkillsTable::Instance(); + CDObjectsTable::Instance(); + CDPhysicsComponentTable::Instance(); + CDRebuildComponentTable::Instance(); + CDScriptComponentTable::Instance(); + CDSkillBehaviorTable::Instance(); + CDZoneTableTable::Instance(); + CDVendorComponentTable::Instance(); + CDActivitiesTable::Instance(); + CDPackageComponentTable::Instance(); + CDProximityMonitorComponentTable::Instance(); + CDMovementAIComponentTable::Instance(); + CDBrickIDTableTable::Instance(); + CDRarityTableTable::Instance(); + CDMissionEmailTable::Instance(); + CDRewardsTable::Instance(); + CDPropertyEntranceComponentTable::Instance(); + CDPropertyTemplateTable::Instance(); + CDFeatureGatingTable::Instance(); + CDRailActivatorComponentTable::Instance(); } diff --git a/dDatabase/CDClientManager.h b/dDatabase/CDClientManager.h index ea24a373..74069ff4 100644 --- a/dDatabase/CDClientManager.h +++ b/dDatabase/CDClientManager.h @@ -1,96 +1,26 @@ #pragma once -// Custom Classes #include "CDTable.h" -// Tables -#include "CDActivityRewardsTable.h" -#include "CDAnimationsTable.h" -#include "CDBehaviorParameterTable.h" -#include "CDBehaviorTemplateTable.h" -#include "CDComponentsRegistryTable.h" -#include "CDCurrencyTableTable.h" -#include "CDDestructibleComponentTable.h" -#include "CDEmoteTable.h" -#include "CDInventoryComponentTable.h" -#include "CDItemComponentTable.h" -#include "CDItemSetsTable.h" -#include "CDItemSetSkillsTable.h" -#include "CDLevelProgressionLookupTable.h" -#include "CDLootMatrixTable.h" -#include "CDLootTableTable.h" -#include "CDMissionNPCComponentTable.h" -#include "CDMissionTasksTable.h" -#include "CDMissionsTable.h" -#include "CDObjectSkillsTable.h" -#include "CDObjectsTable.h" -#include "CDPhysicsComponentTable.h" -#include "CDRebuildComponentTable.h" -#include "CDScriptComponentTable.h" -#include "CDSkillBehaviorTable.h" -#include "CDZoneTableTable.h" -#include "CDVendorComponentTable.h" -#include "CDActivitiesTable.h" -#include "CDPackageComponentTable.h" -#include "CDProximityMonitorComponentTable.h" -#include "CDMovementAIComponentTable.h" -#include "CDBrickIDTableTable.h" -#include "CDRarityTableTable.h" -#include "CDMissionEmailTable.h" -#include "CDRewardsTable.h" -#include "CDPropertyEntranceComponentTable.h" -#include "CDPropertyTemplateTable.h" -#include "CDFeatureGatingTable.h" -#include "CDRailActivatorComponent.h" +#include "Singleton.h" -// C++ -#include -#include +#define UNUSED_TABLE(v) -/*! - \file CDClientManager.hpp - \brief A manager for the CDClient tables +/** + * Initialize the CDClient tables so they are all loaded into memory. */ - - //! Manages all data from the CDClient -class CDClientManager { -private: - static CDClientManager* m_Address; //!< The singleton address - - std::unordered_map tables; //!< The tables - +class CDClientManager : public Singleton { public: + CDClientManager(); - //! The singleton method - static CDClientManager* Instance() { - if (m_Address == 0) { - m_Address = new CDClientManager; - } - - return m_Address; - } - - //! Initializes the manager - void Initialize(void); - - //! Fetches a CDClient table - /*! - This function uses typename T which must be a subclass of CDTable. - It returns the class that conforms to the class name - - \param tableName The table name - \return The class or nullptr + /** + * Fetch a table from CDClient + * + * @tparam Table type to fetch + * @return A pointer to the requested table. */ template - T* GetTable(const std::string& tableName) { - static_assert(std::is_base_of::value, "T should inherit from CDTable!"); - - for (auto itr = this->tables.begin(); itr != this->tables.end(); ++itr) { - if (itr->first == tableName) { - return dynamic_cast(itr->second); - } - } - - return nullptr; + T* GetTable() { + return &T::Instance(); } }; diff --git a/dDatabase/Database.cpp b/dDatabase/Database.cpp index f955bd43..1ce6966f 100644 --- a/dDatabase/Database.cpp +++ b/dDatabase/Database.cpp @@ -14,8 +14,6 @@ std::string Database::database; void Database::Connect(const string& host, const string& database, const string& username, const string& password) { //To bypass debug issues: - std::string newHost = "tcp://" + host; - const char* szHost = newHost.c_str(); const char* szDatabase = database.c_str(); const char* szUsername = username.c_str(); const char* szPassword = password.c_str(); @@ -23,7 +21,24 @@ void Database::Connect(const string& host, const string& database, const string& driver = sql::mariadb::get_driver_instance(); sql::Properties properties; - properties["hostName"] = szHost; + // The mariadb connector is *supposed* to handle unix:// and pipe:// prefixes to hostName, but there are bugs where + // 1) it tries to parse a database from the connection string (like in tcp://localhost:3001/darkflame) based on the + // presence of a / + // 2) even avoiding that, the connector still assumes you're connecting with a tcp socket + // So, what we do in the presence of a unix socket or pipe is to set the hostname to the protocol and localhost, + // which avoids parsing errors while still ensuring the correct connection type is used, and then setting the appropriate + // property manually (which the URL parsing fails to do) + const std::string UNIX_PROTO = "unix://"; + const std::string PIPE_PROTO = "pipe://"; + if (host.find(UNIX_PROTO) == 0) { + properties["hostName"] = "unix://localhost"; + properties["localSocket"] = host.substr(UNIX_PROTO.length()).c_str(); + } else if (host.find(PIPE_PROTO) == 0) { + properties["hostName"] = "pipe://localhost"; + properties["pipe"] = host.substr(PIPE_PROTO.length()).c_str(); + } else { + properties["hostName"] = host.c_str(); + } properties["user"] = szUsername; properties["password"] = szPassword; properties["autoReconnect"] = "true"; @@ -35,8 +50,14 @@ void Database::Connect(const string& host, const string& database, const string& } void Database::Connect() { - con = driver->connect(Database::props); - con->setSchema(Database::database); + // `connect(const Properties& props)` segfaults in windows debug, but + // `connect(const SQLString& host, const SQLString& user, const SQLString& pwd)` doesn't handle pipes/unix sockets correctly + if (Database::props.find("localSocket") != Database::props.end() || Database::props.find("pipe") != Database::props.end()) { + con = driver->connect(Database::props); + } else { + con = driver->connect(Database::props["hostName"].c_str(), Database::props["user"].c_str(), Database::props["password"].c_str()); + } + con->setSchema(Database::database.c_str()); } void Database::Destroy(std::string source, bool log) { @@ -83,3 +104,15 @@ sql::PreparedStatement* Database::CreatePreppedStmt(const std::string& query) { void Database::Commit() { Database::con->commit(); } + +bool Database::GetAutoCommit() { + // TODO This should not just access a pointer. A future PR should update this + // to check for null and throw an error if the connection is not valid. + return con->getAutoCommit(); +} + +void Database::SetAutoCommit(bool value) { + // TODO This should not just access a pointer. A future PR should update this + // to check for null and throw an error if the connection is not valid. + Database::con->setAutoCommit(value); +} diff --git a/dDatabase/Database.h b/dDatabase/Database.h index a15ba856..f4d13da3 100644 --- a/dDatabase/Database.h +++ b/dDatabase/Database.h @@ -23,6 +23,8 @@ public: static sql::Statement* CreateStmt(); static sql::PreparedStatement* CreatePreppedStmt(const std::string& query); static void Commit(); + static bool GetAutoCommit(); + static void SetAutoCommit(bool value); static std::string GetDatabase() { return database; } static sql::Properties GetProperties() { return props; } diff --git a/dDatabase/MigrationRunner.cpp b/dDatabase/MigrationRunner.cpp index 30a63896..5e70c401 100644 --- a/dDatabase/MigrationRunner.cpp +++ b/dDatabase/MigrationRunner.cpp @@ -1,65 +1,20 @@ #include "MigrationRunner.h" +#include "BrickByBrickFix.h" +#include "CDClientDatabase.h" +#include "Database.h" +#include "Game.h" #include "GeneralUtils.h" +#include "dLogger.h" +#include "BinaryPathFinder.h" -#include -#include -#include +#include -void MigrationRunner::RunMigrations() { - auto stmt = Database::CreatePreppedStmt("CREATE TABLE IF NOT EXISTS migration_history (name TEXT NOT NULL, date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP());"); - stmt->executeQuery(); - delete stmt; - - sql::SQLString finalSQL = ""; - Migration checkMigration{}; - - for (const auto& entry : GeneralUtils::GetFileNamesFromFolder("./migrations/")) { - auto migration = LoadMigration(entry); - - if (migration.data.empty()) { - continue; - } - - checkMigration = migration; - - stmt = Database::CreatePreppedStmt("SELECT name FROM migration_history WHERE name = ?;"); - stmt->setString(1, migration.name); - auto res = stmt->executeQuery(); - bool doExit = res->next(); - delete res; - delete stmt; - if (doExit) continue; - - Game::logger->Log("MigrationRunner", "Running migration: %s", migration.name.c_str()); - - finalSQL.append(migration.data); - finalSQL.append('\n'); - - stmt = Database::CreatePreppedStmt("INSERT INTO migration_history (name) VALUES (?);"); - stmt->setString(1, entry); - stmt->execute(); - delete stmt; - } - - if (!finalSQL.empty()) { - try { - auto simpleStatement = Database::CreateStmt(); - simpleStatement->execute(finalSQL); - delete simpleStatement; - } catch (sql::SQLException e) { - Game::logger->Log("MigrationRunner", "Encountered error running migration: %s", e.what()); - } - } -} - -Migration MigrationRunner::LoadMigration(std::string path) { +Migration LoadMigration(std::string path) { Migration migration{}; - std::ifstream file("./migrations/" + path); + std::ifstream file(BinaryPathFinder::GetBinaryDir() / "migrations/" / path); if (file.is_open()) { - std::hash hash; - std::string line; std::string total = ""; @@ -75,3 +30,129 @@ Migration MigrationRunner::LoadMigration(std::string path) { return migration; } + +void MigrationRunner::RunMigrations() { + auto* stmt = Database::CreatePreppedStmt("CREATE TABLE IF NOT EXISTS migration_history (name TEXT NOT NULL, date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP());"); + stmt->execute(); + delete stmt; + + sql::SQLString finalSQL = ""; + bool runSd0Migrations = false; + for (const auto& entry : GeneralUtils::GetSqlFileNamesFromFolder((BinaryPathFinder::GetBinaryDir() / "./migrations/dlu/").string())) { + auto migration = LoadMigration("dlu/" + entry); + + if (migration.data.empty()) { + continue; + } + + stmt = Database::CreatePreppedStmt("SELECT name FROM migration_history WHERE name = ?;"); + stmt->setString(1, migration.name.c_str()); + auto* res = stmt->executeQuery(); + bool doExit = res->next(); + delete res; + delete stmt; + if (doExit) continue; + + Game::logger->Log("MigrationRunner", "Running migration: %s", migration.name.c_str()); + if (migration.name == "dlu/5_brick_model_sd0.sql") { + runSd0Migrations = true; + } else { + finalSQL.append(migration.data.c_str()); + } + + stmt = Database::CreatePreppedStmt("INSERT INTO migration_history (name) VALUES (?);"); + stmt->setString(1, migration.name.c_str()); + stmt->execute(); + delete stmt; + } + + if (finalSQL.empty() && !runSd0Migrations) { + Game::logger->Log("MigrationRunner", "Server database is up to date."); + return; + } + + if (!finalSQL.empty()) { + auto migration = GeneralUtils::SplitString(static_cast(finalSQL), ';'); + std::unique_ptr simpleStatement(Database::CreateStmt()); + for (auto& query : migration) { + try { + if (query.empty()) continue; + simpleStatement->execute(query.c_str()); + } catch (sql::SQLException& e) { + Game::logger->Log("MigrationRunner", "Encountered error running migration: %s", e.what()); + } + } + } + + // Do this last on the off chance none of the other migrations have been run yet. + if (runSd0Migrations) { + uint32_t numberOfUpdatedModels = BrickByBrickFix::UpdateBrickByBrickModelsToSd0(); + Game::logger->Log("MasterServer", "%i models were updated from zlib to sd0.", numberOfUpdatedModels); + uint32_t numberOfTruncatedModels = BrickByBrickFix::TruncateBrokenBrickByBrickXml(); + Game::logger->Log("MasterServer", "%i models were truncated from the database.", numberOfTruncatedModels); + } +} + +void MigrationRunner::RunSQLiteMigrations() { + auto cdstmt = CDClientDatabase::CreatePreppedStmt("CREATE TABLE IF NOT EXISTS migration_history (name TEXT NOT NULL, date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP);"); + cdstmt.execQuery().finalize(); + cdstmt.finalize(); + + auto* stmt = Database::CreatePreppedStmt("CREATE TABLE IF NOT EXISTS migration_history (name TEXT NOT NULL, date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP());"); + stmt->execute(); + delete stmt; + + for (const auto& entry : GeneralUtils::GetSqlFileNamesFromFolder((BinaryPathFinder::GetBinaryDir() / "migrations/cdserver/").string())) { + auto migration = LoadMigration("cdserver/" + entry); + + if (migration.data.empty()) continue; + + // Check if there is an entry in the migration history table on the cdclient database. + cdstmt = CDClientDatabase::CreatePreppedStmt("SELECT name FROM migration_history WHERE name = ?;"); + cdstmt.bind((int32_t) 1, migration.name.c_str()); + auto cdres = cdstmt.execQuery(); + bool doExit = !cdres.eof(); + cdres.finalize(); + cdstmt.finalize(); + + if (doExit) continue; + + // Check first if there is entry in the migration history table on the main database. + stmt = Database::CreatePreppedStmt("SELECT name FROM migration_history WHERE name = ?;"); + stmt->setString(1, migration.name.c_str()); + auto* res = stmt->executeQuery(); + doExit = res->next(); + delete res; + delete stmt; + if (doExit) { + // Insert into cdclient database if there is an entry in the main database but not the cdclient database. + cdstmt = CDClientDatabase::CreatePreppedStmt("INSERT INTO migration_history (name) VALUES (?);"); + cdstmt.bind((int32_t) 1, migration.name.c_str()); + cdstmt.execQuery().finalize(); + cdstmt.finalize(); + continue; + } + + // Doing these 1 migration at a time since one takes a long time and some may think it is crashing. + // This will at the least guarentee that the full migration needs to be run in order to be counted as "migrated". + Game::logger->Log("MigrationRunner", "Executing migration: %s. This may take a while. Do not shut down server.", migration.name.c_str()); + CDClientDatabase::ExecuteQuery("BEGIN TRANSACTION;"); + for (const auto& dml : GeneralUtils::SplitString(migration.data, ';')) { + if (dml.empty()) continue; + try { + CDClientDatabase::ExecuteDML(dml.c_str()); + } catch (CppSQLite3Exception& e) { + Game::logger->Log("MigrationRunner", "Encountered error running DML command: (%i) : %s", e.errorCode(), e.errorMessage()); + } + } + + // Insert into cdclient database. + cdstmt = CDClientDatabase::CreatePreppedStmt("INSERT INTO migration_history (name) VALUES (?);"); + cdstmt.bind((int32_t) 1, migration.name.c_str()); + cdstmt.execQuery().finalize(); + cdstmt.finalize(); + CDClientDatabase::ExecuteQuery("COMMIT;"); + } + + Game::logger->Log("MigrationRunner", "CDServer database is up to date."); +} diff --git a/dDatabase/MigrationRunner.h b/dDatabase/MigrationRunner.h index f5d1325a..0cb36d53 100644 --- a/dDatabase/MigrationRunner.h +++ b/dDatabase/MigrationRunner.h @@ -1,19 +1,13 @@ #pragma once -#include "Database.h" - -#include "dCommonVars.h" -#include "Game.h" -#include "dCommonVars.h" -#include "dLogger.h" +#include struct Migration { std::string data; std::string name; }; -class MigrationRunner { -public: - static void RunMigrations(); - static Migration LoadMigration(std::string path); +namespace MigrationRunner { + void RunMigrations(); + void RunSQLiteMigrations(); }; diff --git a/dDatabase/Tables/CDActivitiesTable.cpp b/dDatabase/Tables/CDActivitiesTable.cpp index 835ca3d2..e1660d66 100644 --- a/dDatabase/Tables/CDActivitiesTable.cpp +++ b/dDatabase/Tables/CDActivitiesTable.cpp @@ -1,6 +1,5 @@ #include "CDActivitiesTable.h" -//! Constructor CDActivitiesTable::CDActivitiesTable(void) { // First, get the size of the table @@ -21,25 +20,25 @@ CDActivitiesTable::CDActivitiesTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM Activities"); while (!tableData.eof()) { CDActivities entry; - entry.ActivityID = tableData.getIntField(0, -1); - entry.locStatus = tableData.getIntField(1, -1); - entry.instanceMapID = tableData.getIntField(2, -1); - entry.minTeams = tableData.getIntField(3, -1); - entry.maxTeams = tableData.getIntField(4, -1); - entry.minTeamSize = tableData.getIntField(5, -1); - entry.maxTeamSize = tableData.getIntField(6, -1); - entry.waitTime = tableData.getIntField(7, -1); - entry.startDelay = tableData.getIntField(8, -1); - entry.requiresUniqueData = tableData.getIntField(9, -1); - entry.leaderboardType = tableData.getIntField(10, -1); - entry.localize = tableData.getIntField(11, -1); - entry.optionalCostLOT = tableData.getIntField(12, -1); - entry.optionalCostCount = tableData.getIntField(13, -1); - entry.showUIRewards = tableData.getIntField(14, -1); - entry.CommunityActivityFlagID = tableData.getIntField(15, -1); - entry.gate_version = tableData.getStringField(16, ""); - entry.noTeamLootOnDeath = tableData.getIntField(17, -1); - entry.optionalPercentage = tableData.getFloatField(18, -1.0f); + entry.ActivityID = tableData.getIntField("ActivityID", -1); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.instanceMapID = tableData.getIntField("instanceMapID", -1); + entry.minTeams = tableData.getIntField("minTeams", -1); + entry.maxTeams = tableData.getIntField("maxTeams", -1); + entry.minTeamSize = tableData.getIntField("minTeamSize", -1); + entry.maxTeamSize = tableData.getIntField("maxTeamSize", -1); + entry.waitTime = tableData.getIntField("waitTime", -1); + entry.startDelay = tableData.getIntField("startDelay", -1); + entry.requiresUniqueData = tableData.getIntField("requiresUniqueData", -1); + entry.leaderboardType = tableData.getIntField("leaderboardType", -1); + entry.localize = tableData.getIntField("localize", -1); + entry.optionalCostLOT = tableData.getIntField("optionalCostLOT", -1); + entry.optionalCostCount = tableData.getIntField("optionalCostCount", -1); + entry.showUIRewards = tableData.getIntField("showUIRewards", -1); + entry.CommunityActivityFlagID = tableData.getIntField("CommunityActivityFlagID", -1); + entry.gate_version = tableData.getStringField("gate_version", ""); + entry.noTeamLootOnDeath = tableData.getIntField("noTeamLootOnDeath", -1); + entry.optionalPercentage = tableData.getFloatField("optionalPercentage", -1.0f); this->entries.push_back(entry); tableData.nextRow(); @@ -48,15 +47,6 @@ CDActivitiesTable::CDActivitiesTable(void) { tableData.finalize(); } -//! Destructor -CDActivitiesTable::~CDActivitiesTable(void) {} - -//! Returns the table's name -std::string CDActivitiesTable::GetName(void) const { - return "Activities"; -} - -//! Queries the table with a custom "where" clause std::vector CDActivitiesTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -66,7 +56,7 @@ std::vector CDActivitiesTable::Query(std::function CDActivitiesTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDActivitiesTable.h b/dDatabase/Tables/CDActivitiesTable.h index 51c560cf..4b60afbd 100644 --- a/dDatabase/Tables/CDActivitiesTable.h +++ b/dDatabase/Tables/CDActivitiesTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDActivitiesTable.hpp - \brief Contains data for the Activities table - */ - - //! Activities Entry Struct struct CDActivities { unsigned int ActivityID; unsigned int locStatus; @@ -31,36 +25,14 @@ struct CDActivities { float optionalPercentage; }; - -//! Activities table -class CDActivitiesTable : public CDTable { +class CDActivitiesTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDActivitiesTable(void); - - //! Destructor - ~CDActivitiesTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDActivitiesTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDActivityRewardsTable.cpp b/dDatabase/Tables/CDActivityRewardsTable.cpp index f1719204..65ef1101 100644 --- a/dDatabase/Tables/CDActivityRewardsTable.cpp +++ b/dDatabase/Tables/CDActivityRewardsTable.cpp @@ -1,6 +1,5 @@ #include "CDActivityRewardsTable.h" -//! Constructor CDActivityRewardsTable::CDActivityRewardsTable(void) { // First, get the size of the table @@ -21,13 +20,13 @@ CDActivityRewardsTable::CDActivityRewardsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ActivityRewards"); while (!tableData.eof()) { CDActivityRewards entry; - entry.objectTemplate = tableData.getIntField(0, -1); - entry.ActivityRewardIndex = tableData.getIntField(1, -1); - entry.activityRating = tableData.getIntField(2, -1); - entry.LootMatrixIndex = tableData.getIntField(3, -1); - entry.CurrencyIndex = tableData.getIntField(4, -1); - entry.ChallengeRating = tableData.getIntField(5, -1); - entry.description = tableData.getStringField(6, ""); + entry.objectTemplate = tableData.getIntField("objectTemplate", -1); + entry.ActivityRewardIndex = tableData.getIntField("ActivityRewardIndex", -1); + entry.activityRating = tableData.getIntField("activityRating", -1); + entry.LootMatrixIndex = tableData.getIntField("LootMatrixIndex", -1); + entry.CurrencyIndex = tableData.getIntField("CurrencyIndex", -1); + entry.ChallengeRating = tableData.getIntField("ChallengeRating", -1); + entry.description = tableData.getStringField("description", ""); this->entries.push_back(entry); tableData.nextRow(); @@ -36,15 +35,6 @@ CDActivityRewardsTable::CDActivityRewardsTable(void) { tableData.finalize(); } -//! Destructor -CDActivityRewardsTable::~CDActivityRewardsTable(void) {} - -//! Returns the table's name -std::string CDActivityRewardsTable::GetName(void) const { - return "ActivityRewards"; -} - -//! Queries the table with a custom "where" clause std::vector CDActivityRewardsTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -54,7 +44,7 @@ std::vector CDActivityRewardsTable::Query(std::function CDActivityRewardsTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDActivityRewardsTable.h b/dDatabase/Tables/CDActivityRewardsTable.h index 7f1e81a0..b5503fb6 100644 --- a/dDatabase/Tables/CDActivityRewardsTable.h +++ b/dDatabase/Tables/CDActivityRewardsTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDActivityRewardsTable.hpp - \brief Contains data for the ActivityRewards table - */ - - //! ActivityRewards Entry Struct struct CDActivityRewards { unsigned int objectTemplate; //!< The object template (?) unsigned int ActivityRewardIndex; //!< The activity reward index @@ -19,36 +13,15 @@ struct CDActivityRewards { std::string description; //!< The description }; - -//! ActivityRewards table -class CDActivityRewardsTable : public CDTable { +class CDActivityRewardsTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDActivityRewardsTable(void); - - //! Destructor - ~CDActivityRewardsTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDActivityRewardsTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; }; diff --git a/dDatabase/Tables/CDAnimationsTable.cpp b/dDatabase/Tables/CDAnimationsTable.cpp index 399804f8..76ce0e5a 100644 --- a/dDatabase/Tables/CDAnimationsTable.cpp +++ b/dDatabase/Tables/CDAnimationsTable.cpp @@ -1,66 +1,83 @@ #include "CDAnimationsTable.h" +#include "GeneralUtils.h" +#include "Game.h" -//! Constructor -CDAnimationsTable::CDAnimationsTable(void) { +bool CDAnimationsTable::CacheData(CppSQLite3Statement& queryToCache) { + auto tableData = queryToCache.execQuery(); + // If we received a bad lookup, cache it anyways so we do not run the query again. + if (tableData.eof()) return false; - // First, get the size of the table - unsigned int size = 0; - auto tableSize = CDClientDatabase::ExecuteQuery("SELECT COUNT(*) FROM Animations"); - while (!tableSize.eof()) { - size = tableSize.getIntField(0, 0); + do { + std::string animation_type = tableData.getStringField("animation_type", ""); + DluAssert(!animation_type.empty()); + AnimationGroupID animationGroupID = tableData.getIntField("animationGroupID", -1); + DluAssert(animationGroupID != -1); - tableSize.nextRow(); - } + CDAnimation entry; + entry.animation_name = tableData.getStringField("animation_name", ""); + entry.chance_to_play = tableData.getFloatField("chance_to_play", 1.0f); + UNUSED_COLUMN(entry.min_loops = tableData.getIntField("min_loops", 0);) + UNUSED_COLUMN(entry.max_loops = tableData.getIntField("max_loops", 0);) + entry.animation_length = tableData.getFloatField("animation_length", 0.0f); + UNUSED_COLUMN(entry.hideEquip = tableData.getIntField("hideEquip", 0) == 1;) + UNUSED_COLUMN(entry.ignoreUpperBody = tableData.getIntField("ignoreUpperBody", 0) == 1;) + UNUSED_COLUMN(entry.restartable = tableData.getIntField("restartable", 0) == 1;) + UNUSED_COLUMN(entry.face_animation_name = tableData.getStringField("face_animation_name", "");) + UNUSED_COLUMN(entry.priority = tableData.getFloatField("priority", 0.0f);) + UNUSED_COLUMN(entry.blendTime = tableData.getFloatField("blendTime", 0.0f);) - tableSize.finalize(); - - // Reserve the size - this->entries.reserve(size); - - // Now get the data - auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM Animations"); - while (!tableData.eof()) { - CDAnimations entry; - entry.animationGroupID = tableData.getIntField(0, -1); - entry.animation_type = tableData.getStringField(1, ""); - entry.animation_name = tableData.getStringField(2, ""); - entry.chance_to_play = tableData.getFloatField(3, -1.0f); - entry.min_loops = tableData.getIntField(4, -1); - entry.max_loops = tableData.getIntField(5, -1); - entry.animation_length = tableData.getFloatField(6, -1.0f); - entry.hideEquip = tableData.getIntField(7, -1) == 1 ? true : false; - entry.ignoreUpperBody = tableData.getIntField(8, -1) == 1 ? true : false; - entry.restartable = tableData.getIntField(9, -1) == 1 ? true : false; - entry.face_animation_name = tableData.getStringField(10, ""); - entry.priority = tableData.getFloatField(11, -1.0f); - entry.blendTime = tableData.getFloatField(12, -1.0f); - - this->entries.push_back(entry); + this->animations[CDAnimationKey(animation_type, animationGroupID)].push_back(entry); tableData.nextRow(); - } + } while (!tableData.eof()); tableData.finalize(); + + return true; } -//! Destructor -CDAnimationsTable::~CDAnimationsTable(void) {} - -//! Returns the table's name -std::string CDAnimationsTable::GetName(void) const { - return "Animations"; +void CDAnimationsTable::CacheAnimations(const CDAnimationKey animationKey) { + auto query = CDClientDatabase::CreatePreppedStmt("SELECT * FROM Animations WHERE animationGroupID = ? and animation_type = ?"); + query.bind(1, static_cast(animationKey.second)); + query.bind(2, animationKey.first.c_str()); + // If we received a bad lookup, cache it anyways so we do not run the query again. + if (!CacheData(query)) { + this->animations[animationKey]; + } } -//! Queries the table with a custom "where" clause -std::vector CDAnimationsTable::Query(std::function predicate) { +void CDAnimationsTable::CacheAnimationGroup(AnimationGroupID animationGroupID) { + auto animationEntryCached = this->animations.find(CDAnimationKey("", animationGroupID)); + if (animationEntryCached != this->animations.end()) { + return; + } - std::vector data = cpplinq::from(this->entries) - >> cpplinq::where(predicate) - >> cpplinq::to_vector(); + auto query = CDClientDatabase::CreatePreppedStmt("SELECT * FROM Animations WHERE animationGroupID = ?"); + query.bind(1, static_cast(animationGroupID)); - return data; + // Cache the query so we don't run the query again. + CacheData(query); + this->animations[CDAnimationKey("", animationGroupID)]; } -//! Gets all the entries in the table -std::vector CDAnimationsTable::GetEntries(void) const { - return this->entries; +CDAnimationLookupResult CDAnimationsTable::GetAnimation(const AnimationID& animationType, const std::string& previousAnimationName, const AnimationGroupID animationGroupID) { + CDAnimationKey animationKey(animationType, animationGroupID); + auto animationEntryCached = this->animations.find(animationKey); + if (animationEntryCached == this->animations.end()) { + this->CacheAnimations(animationKey); + } + + auto animationEntry = this->animations.find(animationKey); + // If we have only one animation, return it regardless of the chance to play. + if (animationEntry->second.size() == 1) { + return CDAnimationLookupResult(animationEntry->second.front()); + } + auto randomAnimation = GeneralUtils::GenerateRandomNumber(0, 1); + + for (auto& animationEntry : animationEntry->second) { + randomAnimation -= animationEntry.chance_to_play; + // This is how the client gets the random animation. + if (animationEntry.animation_name != previousAnimationName && randomAnimation <= 0.0f) return CDAnimationLookupResult(animationEntry); + } + + return CDAnimationLookupResult(); } diff --git a/dDatabase/Tables/CDAnimationsTable.h b/dDatabase/Tables/CDAnimationsTable.h index 24112985..65f54d98 100644 --- a/dDatabase/Tables/CDAnimationsTable.h +++ b/dDatabase/Tables/CDAnimationsTable.h @@ -1,60 +1,66 @@ #pragma once -// Custom Classes #include "CDTable.h" +#include -/*! - \file CDAnimationsTable.hpp - \brief Contains data for the Animations table - */ - - //! Animations Entry Struct -struct CDAnimations { - unsigned int animationGroupID; //!< The animation group ID - std::string animation_type; //!< The animation type +struct CDAnimation { + // unsigned int animationGroupID; + // std::string animation_type; + // The above two are a pair to represent a primary key in the map. std::string animation_name; //!< The animation name float chance_to_play; //!< The chance to play the animation - unsigned int min_loops; //!< The minimum number of loops - unsigned int max_loops; //!< The maximum number of loops + UNUSED_COLUMN(unsigned int min_loops;) //!< The minimum number of loops + UNUSED_COLUMN(unsigned int max_loops;) //!< The maximum number of loops float animation_length; //!< The animation length - bool hideEquip; //!< Whether or not to hide the equip - bool ignoreUpperBody; //!< Whether or not to ignore the upper body - bool restartable; //!< Whether or not the animation is restartable - std::string face_animation_name; //!< The face animation name - float priority; //!< The priority - float blendTime; //!< The blend time + UNUSED_COLUMN(bool hideEquip;) //!< Whether or not to hide the equip + UNUSED_COLUMN(bool ignoreUpperBody;) //!< Whether or not to ignore the upper body + UNUSED_COLUMN(bool restartable;) //!< Whether or not the animation is restartable + UNUSED_COLUMN(std::string face_animation_name;) //!< The face animation name + UNUSED_COLUMN(float priority;) //!< The priority + UNUSED_COLUMN(float blendTime;) //!< The blend time }; +typedef LookupResult CDAnimationLookupResult; -//! Animations table -class CDAnimationsTable : public CDTable { -private: - std::vector entries; - +class CDAnimationsTable : public CDTable { + typedef int32_t AnimationGroupID; + typedef std::string AnimationID; + typedef std::pair CDAnimationKey; public: - - //! Constructor - CDAnimationsTable(void); - - //! Destructor - ~CDAnimationsTable(void); - - //! Returns the table's name - /*! - \return The table name + /** + * Given an animationType and the previousAnimationName played, return the next animationType to play. + * If there are more than 1 animationTypes that can be played, one is selected at random but also does not allow + * the previousAnimationName to be played twice. + * + * @param animationType The animationID to lookup + * @param previousAnimationName The previously played animation + * @param animationGroupID The animationGroupID to lookup + * @return CDAnimationLookupResult */ - std::string GetName(void) const override; + [[nodiscard]] CDAnimationLookupResult GetAnimation(const AnimationID& animationType, const std::string& previousAnimationName, const AnimationGroupID animationGroupID); - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate + /** + * Cache a full AnimationGroup by its ID. */ - std::vector Query(std::function predicate); + void CacheAnimationGroup(AnimationGroupID animationGroupID); +private: - //! Gets all the entries in the table - /*! - \return The entries + /** + * Cache all animations given a premade key */ - std::vector GetEntries(void) const; + void CacheAnimations(const CDAnimationKey animationKey); + /** + * Run the query responsible for caching the data. + * @param queryToCache + * @return true + * @return false + */ + bool CacheData(CppSQLite3Statement& queryToCache); + + /** + * Each animation is key'd by its animationName and its animationGroupID. Each + * animation has a possible list of animations. This is because there can be animations have a percent chance to play so one is selected at random. + */ + std::map> animations; }; diff --git a/dDatabase/Tables/CDBehaviorParameterTable.cpp b/dDatabase/Tables/CDBehaviorParameterTable.cpp index 96015896..8181245b 100644 --- a/dDatabase/Tables/CDBehaviorParameterTable.cpp +++ b/dDatabase/Tables/CDBehaviorParameterTable.cpp @@ -1,68 +1,54 @@ #include "CDBehaviorParameterTable.h" #include "GeneralUtils.h" -//! Constructor -CDBehaviorParameterTable::CDBehaviorParameterTable(void) { +uint64_t GetHash(const uint32_t behaviorID, const uint32_t parameterID) { + uint64_t hash = behaviorID; + hash <<= 31U; + hash |= parameterID; + + return hash; +} + +CDBehaviorParameterTable::CDBehaviorParameterTable() { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM BehaviorParameter"); - size_t hash = 0; while (!tableData.eof()) { - hash = 0; - CDBehaviorParameter entry; - entry.behaviorID = tableData.getIntField(0, -1); - auto candidateStringToAdd = std::string(tableData.getStringField(1, "")); + uint32_t behaviorID = tableData.getIntField("behaviorID", -1); + auto candidateStringToAdd = std::string(tableData.getStringField("parameterID", "")); auto parameter = m_ParametersList.find(candidateStringToAdd); + uint32_t parameterId; if (parameter != m_ParametersList.end()) { - entry.parameterID = parameter; + parameterId = parameter->second; } else { - entry.parameterID = m_ParametersList.insert(candidateStringToAdd).first; + parameterId = m_ParametersList.insert(std::make_pair(candidateStringToAdd, m_ParametersList.size())).first->second; } - entry.value = tableData.getFloatField(2, -1.0f); + uint64_t hash = GetHash(behaviorID, parameterId); + float value = tableData.getFloatField("value", -1.0f); - GeneralUtils::hash_combine(hash, entry.behaviorID); - GeneralUtils::hash_combine(hash, *entry.parameterID); - - auto it = m_Entries.find(entry.behaviorID); - m_ParametersList.insert(*entry.parameterID); - m_Entries.insert(std::make_pair(hash, entry)); + m_Entries.insert(std::make_pair(hash, value)); tableData.nextRow(); } tableData.finalize(); } -//! Destructor -CDBehaviorParameterTable::~CDBehaviorParameterTable(void) {} - -//! Returns the table's name -std::string CDBehaviorParameterTable::GetName(void) const { - return "BehaviorParameter"; -} - -CDBehaviorParameter CDBehaviorParameterTable::GetEntry(const uint32_t behaviorID, const std::string& name, const float defaultValue) { - CDBehaviorParameter returnValue; - returnValue.behaviorID = 0; - returnValue.parameterID = m_ParametersList.end(); - returnValue.value = defaultValue; - - size_t hash = 0; - GeneralUtils::hash_combine(hash, behaviorID); - GeneralUtils::hash_combine(hash, name); +float CDBehaviorParameterTable::GetValue(const uint32_t behaviorID, const std::string& name, const float defaultValue) { + auto parameterID = this->m_ParametersList.find(name); + if (parameterID == this->m_ParametersList.end()) return defaultValue; + auto hash = GetHash(behaviorID, parameterID->second); // Search for specific parameter - const auto& it = m_Entries.find(hash); - return it != m_Entries.end() ? it->second : returnValue; + auto it = m_Entries.find(hash); + return it != m_Entries.end() ? it->second : defaultValue; } std::map CDBehaviorParameterTable::GetParametersByBehaviorID(uint32_t behaviorID) { - size_t hash; + uint64_t hashBase = behaviorID; std::map returnInfo; - for (auto parameterCandidate : m_ParametersList) { - hash = 0; - GeneralUtils::hash_combine(hash, behaviorID); - GeneralUtils::hash_combine(hash, parameterCandidate); + for (auto& [parameterString, parameterId] : m_ParametersList) { + uint64_t hash = GetHash(hashBase, parameterId); auto infoCandidate = m_Entries.find(hash); if (infoCandidate != m_Entries.end()) { - returnInfo.insert(std::make_pair(*(infoCandidate->second.parameterID), infoCandidate->second.value)); + returnInfo.insert(std::make_pair(parameterString, infoCandidate->second)); } } return returnInfo; diff --git a/dDatabase/Tables/CDBehaviorParameterTable.h b/dDatabase/Tables/CDBehaviorParameterTable.h index f067e7d2..249caacf 100644 --- a/dDatabase/Tables/CDBehaviorParameterTable.h +++ b/dDatabase/Tables/CDBehaviorParameterTable.h @@ -5,38 +5,16 @@ #include #include -/*! - \file CDBehaviorParameterTable.hpp - \brief Contains data for the BehaviorParameter table - */ - - //! BehaviorParameter Entry Struct -struct CDBehaviorParameter { - unsigned int behaviorID; //!< The Behavior ID - std::unordered_set::iterator parameterID; //!< The Parameter ID - float value; //!< The value of the behavior template -}; - -//! BehaviorParameter table -class CDBehaviorParameterTable : public CDTable { +class CDBehaviorParameterTable : public CDTable { private: - std::unordered_map m_Entries; - std::unordered_set m_ParametersList; + typedef uint64_t BehaviorParameterHash; + typedef float BehaviorParameterValue; + std::unordered_map m_Entries; + std::unordered_map m_ParametersList; public: + CDBehaviorParameterTable(); - //! Constructor - CDBehaviorParameterTable(void); - - //! Destructor - ~CDBehaviorParameterTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - CDBehaviorParameter GetEntry(const uint32_t behaviorID, const std::string& name, const float defaultValue = 0); + float GetValue(const uint32_t behaviorID, const std::string& name, const float defaultValue = 0); std::map GetParametersByBehaviorID(uint32_t behaviorID); }; diff --git a/dDatabase/Tables/CDBehaviorTemplateTable.cpp b/dDatabase/Tables/CDBehaviorTemplateTable.cpp index 1628756f..08bc86d1 100644 --- a/dDatabase/Tables/CDBehaviorTemplateTable.cpp +++ b/dDatabase/Tables/CDBehaviorTemplateTable.cpp @@ -1,6 +1,5 @@ #include "CDBehaviorTemplateTable.h" -//! Constructor CDBehaviorTemplateTable::CDBehaviorTemplateTable(void) { // First, get the size of the table @@ -21,9 +20,9 @@ CDBehaviorTemplateTable::CDBehaviorTemplateTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM BehaviorTemplate"); while (!tableData.eof()) { CDBehaviorTemplate entry; - entry.behaviorID = tableData.getIntField(0, -1); - entry.templateID = tableData.getIntField(1, -1); - entry.effectID = tableData.getIntField(2, -1); + entry.behaviorID = tableData.getIntField("behaviorID", -1); + entry.templateID = tableData.getIntField("templateID", -1); + entry.effectID = tableData.getIntField("effectID", -1); auto candidateToAdd = tableData.getStringField(3, ""); auto parameter = m_EffectHandles.find(candidateToAdd); if (parameter != m_EffectHandles.end()) { @@ -40,15 +39,6 @@ CDBehaviorTemplateTable::CDBehaviorTemplateTable(void) { tableData.finalize(); } -//! Destructor -CDBehaviorTemplateTable::~CDBehaviorTemplateTable(void) {} - -//! Returns the table's name -std::string CDBehaviorTemplateTable::GetName(void) const { - return "BehaviorTemplate"; -} - -//! Queries the table with a custom "where" clause std::vector CDBehaviorTemplateTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -58,7 +48,6 @@ std::vector CDBehaviorTemplateTable::Query(std::function CDBehaviorTemplateTable::GetEntries(void) const { return this->entries; } @@ -75,3 +64,4 @@ const CDBehaviorTemplate CDBehaviorTemplateTable::GetByBehaviorID(uint32_t behav return entry->second; } } + diff --git a/dDatabase/Tables/CDBehaviorTemplateTable.h b/dDatabase/Tables/CDBehaviorTemplateTable.h index b2bd2521..f9ac9a09 100644 --- a/dDatabase/Tables/CDBehaviorTemplateTable.h +++ b/dDatabase/Tables/CDBehaviorTemplateTable.h @@ -5,12 +5,6 @@ #include #include -/*! - \file CDBehaviorTemplateTable.hpp - \brief Contains data for the BehaviorTemplate table - */ - - //! BehaviorTemplate Entry Struct struct CDBehaviorTemplate { unsigned int behaviorID; //!< The Behavior ID unsigned int templateID; //!< The Template ID (LOT) @@ -19,36 +13,16 @@ struct CDBehaviorTemplate { }; -//! BehaviorTemplate table -class CDBehaviorTemplateTable : public CDTable { +class CDBehaviorTemplateTable : public CDTable { private: std::vector entries; std::unordered_map entriesMappedByBehaviorID; std::unordered_set m_EffectHandles; public: - - //! Constructor - CDBehaviorTemplateTable(void); - - //! Destructor - ~CDBehaviorTemplateTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDBehaviorTemplateTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; const CDBehaviorTemplate GetByBehaviorID(uint32_t behaviorID); diff --git a/dDatabase/Tables/CDBrickIDTableTable.cpp b/dDatabase/Tables/CDBrickIDTableTable.cpp index 9dbdf2c0..9ad24d39 100644 --- a/dDatabase/Tables/CDBrickIDTableTable.cpp +++ b/dDatabase/Tables/CDBrickIDTableTable.cpp @@ -1,6 +1,5 @@ #include "CDBrickIDTableTable.h" -//! Constructor CDBrickIDTableTable::CDBrickIDTableTable(void) { // First, get the size of the table @@ -21,8 +20,8 @@ CDBrickIDTableTable::CDBrickIDTableTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM BrickIDTable"); while (!tableData.eof()) { CDBrickIDTable entry; - entry.NDObjectID = tableData.getIntField(0, -1); - entry.LEGOBrickID = tableData.getIntField(1, -1); + entry.NDObjectID = tableData.getIntField("NDObjectID", -1); + entry.LEGOBrickID = tableData.getIntField("LEGOBrickID", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -31,15 +30,6 @@ CDBrickIDTableTable::CDBrickIDTableTable(void) { tableData.finalize(); } -//! Destructor -CDBrickIDTableTable::~CDBrickIDTableTable(void) {} - -//! Returns the table's name -std::string CDBrickIDTableTable::GetName(void) const { - return "BrickIDTable"; -} - -//! Queries the table with a custom "where" clause std::vector CDBrickIDTableTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -49,7 +39,7 @@ std::vector CDBrickIDTableTable::Query(std::function CDBrickIDTableTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDBrickIDTableTable.h b/dDatabase/Tables/CDBrickIDTableTable.h index aefe332c..e2084caf 100644 --- a/dDatabase/Tables/CDBrickIDTableTable.h +++ b/dDatabase/Tables/CDBrickIDTableTable.h @@ -16,34 +16,14 @@ struct CDBrickIDTable { //! BrickIDTable table -class CDBrickIDTableTable : public CDTable { +class CDBrickIDTableTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDBrickIDTableTable(void); - - //! Destructor - ~CDBrickIDTableTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDBrickIDTableTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDComponentsRegistryTable.cpp b/dDatabase/Tables/CDComponentsRegistryTable.cpp index 8e5ae108..32012f6c 100644 --- a/dDatabase/Tables/CDComponentsRegistryTable.cpp +++ b/dDatabase/Tables/CDComponentsRegistryTable.cpp @@ -1,8 +1,8 @@ #include "CDComponentsRegistryTable.h" +#include "eReplicaComponentType.h" #define CDCLIENT_CACHE_ALL -//! Constructor CDComponentsRegistryTable::CDComponentsRegistryTable(void) { #ifdef CDCLIENT_CACHE_ALL @@ -24,30 +24,12 @@ CDComponentsRegistryTable::CDComponentsRegistryTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ComponentsRegistry"); while (!tableData.eof()) { CDComponentsRegistry entry; - entry.id = tableData.getIntField(0, -1); - entry.component_type = tableData.getIntField(1, -1); - entry.component_id = tableData.getIntField(2, -1); + entry.id = tableData.getIntField("id", -1); + entry.component_type = static_cast(tableData.getIntField("component_type", 0)); + entry.component_id = tableData.getIntField("component_id", -1); this->mappedEntries.insert_or_assign(((uint64_t)entry.component_type) << 32 | ((uint64_t)entry.id), entry.component_id); - //this->entries.push_back(entry); - - /* - //Darwin's stuff: - const auto& it = this->mappedEntries.find(entry.id); - if (it != mappedEntries.end()) { - const auto& iter = it->second.find(entry.component_type); - if (iter == it->second.end()) { - it->second.insert(std::make_pair(entry.component_type, entry.component_id)); - } - } - else { - std::map map; - map.insert(std::make_pair(entry.component_type, entry.component_id)); - this->mappedEntries.insert(std::make_pair(entry.id, map)); - } - */ - tableData.nextRow(); } @@ -55,15 +37,7 @@ CDComponentsRegistryTable::CDComponentsRegistryTable(void) { #endif } -//! Destructor -CDComponentsRegistryTable::~CDComponentsRegistryTable(void) {} - -//! Returns the table's name -std::string CDComponentsRegistryTable::GetName(void) const { - return "ComponentsRegistry"; -} - -int32_t CDComponentsRegistryTable::GetByIDAndType(uint32_t id, uint32_t componentType, int32_t defaultValue) { +int32_t CDComponentsRegistryTable::GetByIDAndType(uint32_t id, eReplicaComponentType componentType, int32_t defaultValue) { const auto& iter = this->mappedEntries.find(((uint64_t)componentType) << 32 | ((uint64_t)id)); if (iter == this->mappedEntries.end()) { @@ -72,16 +46,6 @@ int32_t CDComponentsRegistryTable::GetByIDAndType(uint32_t id, uint32_t componen return iter->second; - /* - const auto& it = this->mappedEntries.find(id); - if (it != mappedEntries.end()) { - const auto& iter = it->second.find(componentType); - if (iter != it->second.end()) { - return iter->second; - } - } - */ - #ifndef CDCLIENT_CACHE_ALL // Now get the data std::stringstream query; @@ -91,9 +55,9 @@ int32_t CDComponentsRegistryTable::GetByIDAndType(uint32_t id, uint32_t componen auto tableData = CDClientDatabase::ExecuteQuery(query.str()); while (!tableData.eof()) { CDComponentsRegistry entry; - entry.id = tableData.getIntField(0, -1); - entry.component_type = tableData.getIntField(1, -1); - entry.component_id = tableData.getIntField(2, -1); + entry.id = tableData.getIntField("id", -1); + entry.component_type = tableData.getIntField("component_type", -1); + entry.component_id = tableData.getIntField("component_id", -1); //this->entries.push_back(entry); @@ -126,3 +90,4 @@ int32_t CDComponentsRegistryTable::GetByIDAndType(uint32_t id, uint32_t componen return defaultValue; #endif } + diff --git a/dDatabase/Tables/CDComponentsRegistryTable.h b/dDatabase/Tables/CDComponentsRegistryTable.h index c3eb0ed2..990072c9 100644 --- a/dDatabase/Tables/CDComponentsRegistryTable.h +++ b/dDatabase/Tables/CDComponentsRegistryTable.h @@ -3,38 +3,19 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDComponentsRegistryTable.hpp - \brief Contains data for the ComponentsRegistry table - */ - - //! ComponentsRegistry Entry Struct +enum class eReplicaComponentType : uint32_t; struct CDComponentsRegistry { unsigned int id; //!< The LOT is used as the ID - unsigned int component_type; //!< See ComponentTypes enum for values + eReplicaComponentType component_type; //!< See ComponentTypes enum for values unsigned int component_id; //!< The ID used within the component's table (0 may either mean it's non-networked, or that the ID is actually 0 }; -//! ComponentsRegistry table -class CDComponentsRegistryTable : public CDTable { +class CDComponentsRegistryTable : public CDTable { private: - //std::vector entries; std::map mappedEntries; //id, component_type, component_id public: - - //! Constructor - CDComponentsRegistryTable(void); - - //! Destructor - ~CDComponentsRegistryTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - int32_t GetByIDAndType(uint32_t id, uint32_t componentType, int32_t defaultValue = 0); + CDComponentsRegistryTable(); + int32_t GetByIDAndType(uint32_t id, eReplicaComponentType componentType, int32_t defaultValue = 0); }; diff --git a/dDatabase/Tables/CDCurrencyTableTable.cpp b/dDatabase/Tables/CDCurrencyTableTable.cpp index a1923a73..78a716f9 100644 --- a/dDatabase/Tables/CDCurrencyTableTable.cpp +++ b/dDatabase/Tables/CDCurrencyTableTable.cpp @@ -21,11 +21,11 @@ CDCurrencyTableTable::CDCurrencyTableTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM CurrencyTable"); while (!tableData.eof()) { CDCurrencyTable entry; - entry.currencyIndex = tableData.getIntField(0, -1); - entry.npcminlevel = tableData.getIntField(1, -1); - entry.minvalue = tableData.getIntField(2, -1); - entry.maxvalue = tableData.getIntField(3, -1); - entry.id = tableData.getIntField(4, -1); + entry.currencyIndex = tableData.getIntField("currencyIndex", -1); + entry.npcminlevel = tableData.getIntField("npcminlevel", -1); + entry.minvalue = tableData.getIntField("minvalue", -1); + entry.maxvalue = tableData.getIntField("maxvalue", -1); + entry.id = tableData.getIntField("id", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -34,15 +34,6 @@ CDCurrencyTableTable::CDCurrencyTableTable(void) { tableData.finalize(); } -//! Destructor -CDCurrencyTableTable::~CDCurrencyTableTable(void) {} - -//! Returns the table's name -std::string CDCurrencyTableTable::GetName(void) const { - return "CurrencyTable"; -} - -//! Queries the table with a custom "where" clause std::vector CDCurrencyTableTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -52,7 +43,7 @@ std::vector CDCurrencyTableTable::Query(std::function CDCurrencyTableTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDCurrencyTableTable.h b/dDatabase/Tables/CDCurrencyTableTable.h index 5a856395..ec700320 100644 --- a/dDatabase/Tables/CDCurrencyTableTable.h +++ b/dDatabase/Tables/CDCurrencyTableTable.h @@ -18,34 +18,14 @@ struct CDCurrencyTable { }; //! CurrencyTable table -class CDCurrencyTableTable : public CDTable { +class CDCurrencyTableTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDCurrencyTableTable(void); - - //! Destructor - ~CDCurrencyTableTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDCurrencyTableTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDDestructibleComponentTable.cpp b/dDatabase/Tables/CDDestructibleComponentTable.cpp index d7438fc7..4bbc8242 100644 --- a/dDatabase/Tables/CDDestructibleComponentTable.cpp +++ b/dDatabase/Tables/CDDestructibleComponentTable.cpp @@ -21,20 +21,20 @@ CDDestructibleComponentTable::CDDestructibleComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM DestructibleComponent"); while (!tableData.eof()) { CDDestructibleComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.faction = tableData.getIntField(1, -1); - entry.factionList = tableData.getStringField(2, ""); - entry.life = tableData.getIntField(3, -1); - entry.imagination = tableData.getIntField(4, -1); - entry.LootMatrixIndex = tableData.getIntField(5, -1); - entry.CurrencyIndex = tableData.getIntField(6, -1); - entry.level = tableData.getIntField(7, -1); - entry.armor = tableData.getFloatField(8, -1.0f); - entry.death_behavior = tableData.getIntField(9, -1); - entry.isnpc = tableData.getIntField(10, -1) == 1 ? true : false; - entry.attack_priority = tableData.getIntField(11, -1); - entry.isSmashable = tableData.getIntField(12, -1) == 1 ? true : false; - entry.difficultyLevel = tableData.getIntField(13, -1); + entry.id = tableData.getIntField("id", -1); + entry.faction = tableData.getIntField("faction", -1); + entry.factionList = tableData.getStringField("factionList", ""); + entry.life = tableData.getIntField("life", -1); + entry.imagination = tableData.getIntField("imagination", -1); + entry.LootMatrixIndex = tableData.getIntField("LootMatrixIndex", -1); + entry.CurrencyIndex = tableData.getIntField("CurrencyIndex", -1); + entry.level = tableData.getIntField("level", -1); + entry.armor = tableData.getFloatField("armor", -1.0f); + entry.death_behavior = tableData.getIntField("death_behavior", -1); + entry.isnpc = tableData.getIntField("isnpc", -1) == 1 ? true : false; + entry.attack_priority = tableData.getIntField("attack_priority", -1); + entry.isSmashable = tableData.getIntField("isSmashable", -1) == 1 ? true : false; + entry.difficultyLevel = tableData.getIntField("difficultyLevel", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -43,15 +43,6 @@ CDDestructibleComponentTable::CDDestructibleComponentTable(void) { tableData.finalize(); } -//! Destructor -CDDestructibleComponentTable::~CDDestructibleComponentTable(void) {} - -//! Returns the table's name -std::string CDDestructibleComponentTable::GetName(void) const { - return "DestructibleComponent"; -} - -//! Queries the table with a custom "where" clause std::vector CDDestructibleComponentTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -61,7 +52,7 @@ std::vector CDDestructibleComponentTable::Query(std::fu return data; } -//! Gets all the entries in the table std::vector CDDestructibleComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDDestructibleComponentTable.h b/dDatabase/Tables/CDDestructibleComponentTable.h index e89bbff8..e42cf486 100644 --- a/dDatabase/Tables/CDDestructibleComponentTable.h +++ b/dDatabase/Tables/CDDestructibleComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDDestructibleComponentTable.hpp - \brief Contains data for the DestructibleComponent table - */ - - //! ItemComponent Struct struct CDDestructibleComponent { unsigned int id; //!< The component ID from the ComponentsRegistry Table int faction; //!< The Faction ID of the object @@ -26,35 +20,14 @@ struct CDDestructibleComponent { int difficultyLevel; //!< ??? }; -//! ItemComponent table -class CDDestructibleComponentTable : public CDTable { +class CDDestructibleComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDDestructibleComponentTable(void); - - //! Destructor - ~CDDestructibleComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDDestructibleComponentTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDEmoteTable.cpp b/dDatabase/Tables/CDEmoteTable.cpp index e3898422..aacbdd55 100644 --- a/dDatabase/Tables/CDEmoteTable.cpp +++ b/dDatabase/Tables/CDEmoteTable.cpp @@ -5,14 +5,14 @@ CDEmoteTableTable::CDEmoteTableTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM Emotes"); while (!tableData.eof()) { CDEmoteTable* entry = new CDEmoteTable(); - entry->ID = tableData.getIntField(0, -1); - entry->animationName = tableData.getStringField(1, ""); - entry->iconFilename = tableData.getStringField(2, ""); - entry->channel = tableData.getIntField(3, -1); - entry->locked = tableData.getIntField(5, -1) != 0; - entry->localize = tableData.getIntField(6, -1) != 0; - entry->locState = tableData.getIntField(7, -1); - entry->gateVersion = tableData.getIntField(8, -1); + entry->ID = tableData.getIntField("id", -1); + entry->animationName = tableData.getStringField("animationName", ""); + entry->iconFilename = tableData.getStringField("iconFilename", ""); + entry->channel = tableData.getIntField("channel", -1); + entry->locked = tableData.getIntField("locked", -1) != 0; + entry->localize = tableData.getIntField("localize", -1) != 0; + entry->locState = tableData.getIntField("locStatus", -1); + entry->gateVersion = tableData.getStringField("gate_version", ""); entries.insert(std::make_pair(entry->ID, entry)); tableData.nextRow(); @@ -30,11 +30,6 @@ CDEmoteTableTable::~CDEmoteTableTable(void) { entries.clear(); } -//! Returns the table's name -std::string CDEmoteTableTable::GetName(void) const { - return "Emotes"; -} - CDEmoteTable* CDEmoteTableTable::GetEmote(int id) { for (auto e : entries) { if (e.first == id) return e.second; @@ -42,3 +37,4 @@ CDEmoteTable* CDEmoteTableTable::GetEmote(int id) { return nullptr; } + diff --git a/dDatabase/Tables/CDEmoteTable.h b/dDatabase/Tables/CDEmoteTable.h index a22ae23e..be40c86f 100644 --- a/dDatabase/Tables/CDEmoteTable.h +++ b/dDatabase/Tables/CDEmoteTable.h @@ -4,12 +4,6 @@ #include "CDTable.h" #include -/*! - \file CDEmoteTable.hpp - \brief Contains data for the CDEmoteTable table - */ - - //! CDEmoteEntry Struct struct CDEmoteTable { CDEmoteTable() { ID = -1; @@ -19,7 +13,7 @@ struct CDEmoteTable { channel = -1; locked = false; localize = false; - gateVersion = -1; + gateVersion = ""; } int ID; @@ -29,28 +23,16 @@ struct CDEmoteTable { int channel; bool locked; bool localize; - int gateVersion; + std::string gateVersion; }; -//! CDEmoteTable table -class CDEmoteTableTable : public CDTable { +class CDEmoteTableTable : public CDTable { private: std::map entries; public: - - //! Constructor - CDEmoteTableTable(void); - - //! Destructor - ~CDEmoteTableTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Returns an emote by ID + CDEmoteTableTable(); + ~CDEmoteTableTable(); + // Returns an emote by ID CDEmoteTable* GetEmote(int id); }; diff --git a/dDatabase/Tables/CDFeatureGatingTable.cpp b/dDatabase/Tables/CDFeatureGatingTable.cpp index dfc65387..05fe69bf 100644 --- a/dDatabase/Tables/CDFeatureGatingTable.cpp +++ b/dDatabase/Tables/CDFeatureGatingTable.cpp @@ -21,11 +21,11 @@ CDFeatureGatingTable::CDFeatureGatingTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM FeatureGating"); while (!tableData.eof()) { CDFeatureGating entry; - entry.featureName = tableData.getStringField(0, ""); - entry.major = tableData.getIntField(1, -1); - entry.current = tableData.getIntField(2, -1); - entry.minor = tableData.getIntField(3, -1); - entry.description = tableData.getStringField(4, ""); + entry.featureName = tableData.getStringField("featureName", ""); + entry.major = tableData.getIntField("major", -1); + entry.current = tableData.getIntField("current", -1); + entry.minor = tableData.getIntField("minor", -1); + entry.description = tableData.getStringField("description", ""); this->entries.push_back(entry); tableData.nextRow(); @@ -34,15 +34,6 @@ CDFeatureGatingTable::CDFeatureGatingTable(void) { tableData.finalize(); } -//! Destructor -CDFeatureGatingTable::~CDFeatureGatingTable(void) {} - -//! Returns the table's name -std::string CDFeatureGatingTable::GetName(void) const { - return "FeatureGating"; -} - -//! Queries the table with a custom "where" clause std::vector CDFeatureGatingTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -62,7 +53,7 @@ bool CDFeatureGatingTable::FeatureUnlocked(const std::string& feature) const { return false; } -//! Gets all the entries in the table std::vector CDFeatureGatingTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDFeatureGatingTable.h b/dDatabase/Tables/CDFeatureGatingTable.h index 54e6d21b..7f536db5 100644 --- a/dDatabase/Tables/CDFeatureGatingTable.h +++ b/dDatabase/Tables/CDFeatureGatingTable.h @@ -3,11 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDFeatureGatingTable.hpp - */ - - //! ItemComponent Struct struct CDFeatureGating { std::string featureName; int32_t major; @@ -16,37 +11,16 @@ struct CDFeatureGating { std::string description; }; -//! ItemComponent table -class CDFeatureGatingTable : public CDTable { +class CDFeatureGatingTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDFeatureGatingTable(void); - - //! Destructor - ~CDFeatureGatingTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDFeatureGatingTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); bool FeatureUnlocked(const std::string& feature) const; - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDInventoryComponentTable.cpp b/dDatabase/Tables/CDInventoryComponentTable.cpp index 65f09c9b..2dc375ab 100644 --- a/dDatabase/Tables/CDInventoryComponentTable.cpp +++ b/dDatabase/Tables/CDInventoryComponentTable.cpp @@ -21,10 +21,10 @@ CDInventoryComponentTable::CDInventoryComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM InventoryComponent"); while (!tableData.eof()) { CDInventoryComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.itemid = tableData.getIntField(1, -1); - entry.count = tableData.getIntField(2, -1); - entry.equip = tableData.getIntField(3, -1) == 1 ? true : false; + entry.id = tableData.getIntField("id", -1); + entry.itemid = tableData.getIntField("itemid", -1); + entry.count = tableData.getIntField("count", -1); + entry.equip = tableData.getIntField("equip", -1) == 1 ? true : false; this->entries.push_back(entry); tableData.nextRow(); @@ -33,15 +33,6 @@ CDInventoryComponentTable::CDInventoryComponentTable(void) { tableData.finalize(); } -//! Destructor -CDInventoryComponentTable::~CDInventoryComponentTable(void) {} - -//! Returns the table's name -std::string CDInventoryComponentTable::GetName(void) const { - return "InventoryComponent"; -} - -//! Queries the table with a custom "where" clause std::vector CDInventoryComponentTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -51,7 +42,7 @@ std::vector CDInventoryComponentTable::Query(std::function return data; } -//! Gets all the entries in the table std::vector CDInventoryComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDInventoryComponentTable.h b/dDatabase/Tables/CDInventoryComponentTable.h index c6117907..cbc04d99 100644 --- a/dDatabase/Tables/CDInventoryComponentTable.h +++ b/dDatabase/Tables/CDInventoryComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDInventoryComponentTable.hpp - \brief Contains data for the InventoryComponent table - */ - - //! ItemComponent Struct struct CDInventoryComponent { unsigned int id; //!< The component ID for this object unsigned int itemid; //!< The LOT of the object @@ -16,35 +10,14 @@ struct CDInventoryComponent { bool equip; //!< Whether or not to equip the item }; -//! ItemComponent table -class CDInventoryComponentTable : public CDTable { +class CDInventoryComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDInventoryComponentTable(void); - - //! Destructor - ~CDInventoryComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDInventoryComponentTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDItemComponentTable.cpp b/dDatabase/Tables/CDItemComponentTable.cpp index 6ff192fd..54afc417 100644 --- a/dDatabase/Tables/CDItemComponentTable.cpp +++ b/dDatabase/Tables/CDItemComponentTable.cpp @@ -23,48 +23,48 @@ CDItemComponentTable::CDItemComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ItemComponent"); while (!tableData.eof()) { CDItemComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.equipLocation = tableData.getStringField(1, ""); - entry.baseValue = tableData.getIntField(2, -1); - entry.isKitPiece = tableData.getIntField(3, -1) == 1 ? true : false; - entry.rarity = tableData.getIntField(4, 0); - entry.itemType = tableData.getIntField(5, -1); - entry.itemInfo = tableData.getInt64Field(6, -1); - entry.inLootTable = tableData.getIntField(7, -1) == 1 ? true : false; - entry.inVendor = tableData.getIntField(8, -1) == 1 ? true : false; - entry.isUnique = tableData.getIntField(9, -1) == 1 ? true : false; - entry.isBOP = tableData.getIntField(10, -1) == 1 ? true : false; - entry.isBOE = tableData.getIntField(11, -1) == 1 ? true : false; - entry.reqFlagID = tableData.getIntField(12, -1); - entry.reqSpecialtyID = tableData.getIntField(13, -1); - entry.reqSpecRank = tableData.getIntField(14, -1); - entry.reqAchievementID = tableData.getIntField(15, -1); - entry.stackSize = tableData.getIntField(16, -1); - entry.color1 = tableData.getIntField(17, -1); - entry.decal = tableData.getIntField(18, -1); - entry.offsetGroupID = tableData.getIntField(19, -1); - entry.buildTypes = tableData.getIntField(20, -1); - entry.reqPrecondition = tableData.getStringField(21, ""); - entry.animationFlag = tableData.getIntField(22, 0); - entry.equipEffects = tableData.getIntField(23, -1); - entry.readyForQA = tableData.getIntField(24, -1) == 1 ? true : false; - entry.itemRating = tableData.getIntField(25, -1); - entry.isTwoHanded = tableData.getIntField(26, -1) == 1 ? true : false; - entry.minNumRequired = tableData.getIntField(27, -1); - entry.delResIndex = tableData.getIntField(28, -1); - entry.currencyLOT = tableData.getIntField(29, -1); - entry.altCurrencyCost = tableData.getIntField(30, -1); - entry.subItems = tableData.getStringField(31, ""); - entry.audioEventUse = tableData.getStringField(32, ""); - entry.noEquipAnimation = tableData.getIntField(33, -1) == 1 ? true : false; - entry.commendationLOT = tableData.getIntField(34, -1); - entry.commendationCost = tableData.getIntField(35, -1); - entry.audioEquipMetaEventSet = tableData.getStringField(36, ""); - entry.currencyCosts = tableData.getStringField(37, ""); - entry.ingredientInfo = tableData.getStringField(38, ""); - entry.locStatus = tableData.getIntField(39, -1); - entry.forgeType = tableData.getIntField(40, -1); - entry.SellMultiplier = tableData.getFloatField(41, -1.0f); + entry.id = tableData.getIntField("id", -1); + entry.equipLocation = tableData.getStringField("equipLocation", ""); + entry.baseValue = tableData.getIntField("baseValue", -1); + entry.isKitPiece = tableData.getIntField("isKitPiece", -1) == 1 ? true : false; + entry.rarity = tableData.getIntField("rarity", 0); + entry.itemType = tableData.getIntField("itemType", -1); + entry.itemInfo = tableData.getInt64Field("itemInfo", -1); + entry.inLootTable = tableData.getIntField("inLootTable", -1) == 1 ? true : false; + entry.inVendor = tableData.getIntField("inVendor", -1) == 1 ? true : false; + entry.isUnique = tableData.getIntField("isUnique", -1) == 1 ? true : false; + entry.isBOP = tableData.getIntField("isBOP", -1) == 1 ? true : false; + entry.isBOE = tableData.getIntField("isBOE", -1) == 1 ? true : false; + entry.reqFlagID = tableData.getIntField("reqFlagID", -1); + entry.reqSpecialtyID = tableData.getIntField("reqSpecialtyID", -1); + entry.reqSpecRank = tableData.getIntField("reqSpecRank", -1); + entry.reqAchievementID = tableData.getIntField("reqAchievementID", -1); + entry.stackSize = tableData.getIntField("stackSize", -1); + entry.color1 = tableData.getIntField("color1", -1); + entry.decal = tableData.getIntField("decal", -1); + entry.offsetGroupID = tableData.getIntField("offsetGroupID", -1); + entry.buildTypes = tableData.getIntField("buildTypes", -1); + entry.reqPrecondition = tableData.getStringField("reqPrecondition", ""); + entry.animationFlag = tableData.getIntField("animationFlag", 0); + entry.equipEffects = tableData.getIntField("equipEffects", -1); + entry.readyForQA = tableData.getIntField("readyForQA", -1) == 1 ? true : false; + entry.itemRating = tableData.getIntField("itemRating", -1); + entry.isTwoHanded = tableData.getIntField("isTwoHanded", -1) == 1 ? true : false; + entry.minNumRequired = tableData.getIntField("minNumRequired", -1); + entry.delResIndex = tableData.getIntField("delResIndex", -1); + entry.currencyLOT = tableData.getIntField("currencyLOT", -1); + entry.altCurrencyCost = tableData.getIntField("altCurrencyCost", -1); + entry.subItems = tableData.getStringField("subItems", ""); + entry.audioEventUse = tableData.getStringField("audioEventUse", ""); + entry.noEquipAnimation = tableData.getIntField("noEquipAnimation", -1) == 1 ? true : false; + entry.commendationLOT = tableData.getIntField("commendationLOT", -1); + entry.commendationCost = tableData.getIntField("commendationCost", -1); + entry.audioEquipMetaEventSet = tableData.getStringField("audioEquipMetaEventSet", ""); + entry.currencyCosts = tableData.getStringField("currencyCosts", ""); + entry.ingredientInfo = tableData.getStringField("ingredientInfo", ""); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.forgeType = tableData.getIntField("forgeType", -1); + entry.SellMultiplier = tableData.getFloatField("SellMultiplier", -1.0f); this->entries.insert(std::make_pair(entry.id, entry)); tableData.nextRow(); @@ -74,14 +74,6 @@ CDItemComponentTable::CDItemComponentTable(void) { #endif } -//! Destructor -CDItemComponentTable::~CDItemComponentTable(void) {} - -//! Returns the table's name -std::string CDItemComponentTable::GetName(void) const { - return "ItemComponent"; -} - const CDItemComponent& CDItemComponentTable::GetItemComponentByID(unsigned int skillID) { const auto& it = this->entries.find(skillID); if (it != this->entries.end()) { @@ -101,48 +93,48 @@ const CDItemComponent& CDItemComponentTable::GetItemComponentByID(unsigned int s while (!tableData.eof()) { CDItemComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.equipLocation = tableData.getStringField(1, ""); - entry.baseValue = tableData.getIntField(2, -1); - entry.isKitPiece = tableData.getIntField(3, -1) == 1 ? true : false; - entry.rarity = tableData.getIntField(4, 0); - entry.itemType = tableData.getIntField(5, -1); - entry.itemInfo = tableData.getInt64Field(6, -1); - entry.inLootTable = tableData.getIntField(7, -1) == 1 ? true : false; - entry.inVendor = tableData.getIntField(8, -1) == 1 ? true : false; - entry.isUnique = tableData.getIntField(9, -1) == 1 ? true : false; - entry.isBOP = tableData.getIntField(10, -1) == 1 ? true : false; - entry.isBOE = tableData.getIntField(11, -1) == 1 ? true : false; - entry.reqFlagID = tableData.getIntField(12, -1); - entry.reqSpecialtyID = tableData.getIntField(13, -1); - entry.reqSpecRank = tableData.getIntField(14, -1); - entry.reqAchievementID = tableData.getIntField(15, -1); - entry.stackSize = tableData.getIntField(16, -1); - entry.color1 = tableData.getIntField(17, -1); - entry.decal = tableData.getIntField(18, -1); - entry.offsetGroupID = tableData.getIntField(19, -1); - entry.buildTypes = tableData.getIntField(20, -1); - entry.reqPrecondition = tableData.getStringField(21, ""); - entry.animationFlag = tableData.getIntField(22, 0); - entry.equipEffects = tableData.getIntField(23, -1); - entry.readyForQA = tableData.getIntField(24, -1) == 1 ? true : false; - entry.itemRating = tableData.getIntField(25, -1); - entry.isTwoHanded = tableData.getIntField(26, -1) == 1 ? true : false; - entry.minNumRequired = tableData.getIntField(27, -1); - entry.delResIndex = tableData.getIntField(28, -1); - entry.currencyLOT = tableData.getIntField(29, -1); - entry.altCurrencyCost = tableData.getIntField(30, -1); - entry.subItems = tableData.getStringField(31, ""); - UNUSED(entry.audioEventUse = tableData.getStringField(32, "")); - entry.noEquipAnimation = tableData.getIntField(33, -1) == 1 ? true : false; - entry.commendationLOT = tableData.getIntField(34, -1); - entry.commendationCost = tableData.getIntField(35, -1); - UNUSED(entry.audioEquipMetaEventSet = tableData.getStringField(36, "")); - entry.currencyCosts = tableData.getStringField(37, ""); - UNUSED(entry.ingredientInfo = tableData.getStringField(38, "")); - entry.locStatus = tableData.getIntField(39, -1); - entry.forgeType = tableData.getIntField(40, -1); - entry.SellMultiplier = tableData.getFloatField(41, -1.0f); + entry.id = tableData.getIntField("id", -1); + entry.equipLocation = tableData.getStringField("equipLocation", ""); + entry.baseValue = tableData.getIntField("baseValue", -1); + entry.isKitPiece = tableData.getIntField("isKitPiece", -1) == 1 ? true : false; + entry.rarity = tableData.getIntField("rarity", 0); + entry.itemType = tableData.getIntField("itemType", -1); + entry.itemInfo = tableData.getInt64Field("itemInfo", -1); + entry.inLootTable = tableData.getIntField("inLootTable", -1) == 1 ? true : false; + entry.inVendor = tableData.getIntField("inVendor", -1) == 1 ? true : false; + entry.isUnique = tableData.getIntField("isUnique", -1) == 1 ? true : false; + entry.isBOP = tableData.getIntField("isBOP", -1) == 1 ? true : false; + entry.isBOE = tableData.getIntField("isBOE", -1) == 1 ? true : false; + entry.reqFlagID = tableData.getIntField("reqFlagID", -1); + entry.reqSpecialtyID = tableData.getIntField("reqSpecialtyID", -1); + entry.reqSpecRank = tableData.getIntField("reqSpecRank", -1); + entry.reqAchievementID = tableData.getIntField("reqAchievementID", -1); + entry.stackSize = tableData.getIntField("stackSize", -1); + entry.color1 = tableData.getIntField("color1", -1); + entry.decal = tableData.getIntField("decal", -1); + entry.offsetGroupID = tableData.getIntField("offsetGroupID", -1); + entry.buildTypes = tableData.getIntField("buildTypes", -1); + entry.reqPrecondition = tableData.getStringField("reqPrecondition", ""); + entry.animationFlag = tableData.getIntField("animationFlag", 0); + entry.equipEffects = tableData.getIntField("equipEffects", -1); + entry.readyForQA = tableData.getIntField("readyForQA", -1) == 1 ? true : false; + entry.itemRating = tableData.getIntField("itemRating", -1); + entry.isTwoHanded = tableData.getIntField("isTwoHanded", -1) == 1 ? true : false; + entry.minNumRequired = tableData.getIntField("minNumRequired", -1); + entry.delResIndex = tableData.getIntField("delResIndex", -1); + entry.currencyLOT = tableData.getIntField("currencyLOT", -1); + entry.altCurrencyCost = tableData.getIntField("altCurrencyCost", -1); + entry.subItems = tableData.getStringField("subItems", ""); + UNUSED(entry.audioEventUse = tableData.getStringField("audioEventUse", "")); + entry.noEquipAnimation = tableData.getIntField("noEquipAnimation", -1) == 1 ? true : false; + entry.commendationLOT = tableData.getIntField("commendationLOT", -1); + entry.commendationCost = tableData.getIntField("commendationCost", -1); + UNUSED(entry.audioEquipMetaEventSet = tableData.getStringField("audioEquipMetaEventSet", "")); + entry.currencyCosts = tableData.getStringField("currencyCosts", ""); + UNUSED(entry.ingredientInfo = tableData.getStringField("ingredientInfo", "")); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.forgeType = tableData.getIntField("forgeType", -1); + entry.SellMultiplier = tableData.getFloatField("SellMultiplier", -1.0f); this->entries.insert(std::make_pair(entry.id, entry)); tableData.nextRow(); @@ -177,3 +169,4 @@ std::map CDItemComponentTable::ParseCraftingCurrencies(const CDIt return currencies; } + diff --git a/dDatabase/Tables/CDItemComponentTable.h b/dDatabase/Tables/CDItemComponentTable.h index b3a2a5f4..11c34dd6 100644 --- a/dDatabase/Tables/CDItemComponentTable.h +++ b/dDatabase/Tables/CDItemComponentTable.h @@ -4,12 +4,6 @@ #include "CDTable.h" #include "dCommonVars.h" -/*! - \file CDItemComponentTable.hpp - \brief Contains data for the ItemComponent table - */ - - //! ItemComponent Struct struct CDItemComponent { unsigned int id; //!< The Component ID std::string equipLocation; //!< The equip location @@ -55,28 +49,15 @@ struct CDItemComponent { float SellMultiplier; //!< Something to do with early vendors perhaps (but replaced) }; -//! ItemComponent table -class CDItemComponentTable : public CDTable { +class CDItemComponentTable : public CDTable { private: std::map entries; public: - - //! Constructor - CDItemComponentTable(void); - - //! Destructor - ~CDItemComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - + CDItemComponentTable(); static std::map ParseCraftingCurrencies(const CDItemComponent& itemComponent); - //! Gets an entry by ID + // Gets an entry by ID const CDItemComponent& GetItemComponentByID(unsigned int skillID); static CDItemComponent Default; diff --git a/dDatabase/Tables/CDItemSetSkillsTable.cpp b/dDatabase/Tables/CDItemSetSkillsTable.cpp index e94cf527..f6b412ff 100644 --- a/dDatabase/Tables/CDItemSetSkillsTable.cpp +++ b/dDatabase/Tables/CDItemSetSkillsTable.cpp @@ -21,9 +21,9 @@ CDItemSetSkillsTable::CDItemSetSkillsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ItemSetSkills"); while (!tableData.eof()) { CDItemSetSkills entry; - entry.SkillSetID = tableData.getIntField(0, -1); - entry.SkillID = tableData.getIntField(1, -1); - entry.SkillCastType = tableData.getIntField(2, -1); + entry.SkillSetID = tableData.getIntField("SkillSetID", -1); + entry.SkillID = tableData.getIntField("SkillID", -1); + entry.SkillCastType = tableData.getIntField("SkillCastType", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -32,15 +32,6 @@ CDItemSetSkillsTable::CDItemSetSkillsTable(void) { tableData.finalize(); } -//! Destructor -CDItemSetSkillsTable::~CDItemSetSkillsTable(void) {} - -//! Returns the table's name -std::string CDItemSetSkillsTable::GetName(void) const { - return "ItemSetSkills"; -} - -//! Queries the table with a custom "where" clause std::vector CDItemSetSkillsTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -50,7 +41,6 @@ std::vector CDItemSetSkillsTable::Query(std::function CDItemSetSkillsTable::GetEntries(void) const { return this->entries; } diff --git a/dDatabase/Tables/CDItemSetSkillsTable.h b/dDatabase/Tables/CDItemSetSkillsTable.h index bf05eea9..8328c66b 100644 --- a/dDatabase/Tables/CDItemSetSkillsTable.h +++ b/dDatabase/Tables/CDItemSetSkillsTable.h @@ -3,50 +3,22 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDItemSetSkillsTable.hpp - \brief Contains data for the ItemSetSkills table - */ - - //! ZoneTable Struct struct CDItemSetSkills { unsigned int SkillSetID; //!< The skill set ID unsigned int SkillID; //!< The skill ID unsigned int SkillCastType; //!< The skill cast type }; -//! ItemSets table -class CDItemSetSkillsTable : public CDTable { +class CDItemSetSkillsTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDItemSetSkillsTable(void); - - //! Destructor - ~CDItemSetSkillsTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDItemSetSkillsTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; std::vector GetBySkillID(unsigned int SkillSetID); - }; - diff --git a/dDatabase/Tables/CDItemSetsTable.cpp b/dDatabase/Tables/CDItemSetsTable.cpp index 606e0cf9..0632ef13 100644 --- a/dDatabase/Tables/CDItemSetsTable.cpp +++ b/dDatabase/Tables/CDItemSetsTable.cpp @@ -21,21 +21,21 @@ CDItemSetsTable::CDItemSetsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ItemSets"); while (!tableData.eof()) { CDItemSets entry; - entry.setID = tableData.getIntField(0, -1); - entry.locStatus = tableData.getIntField(1, -1); - entry.itemIDs = tableData.getStringField(2, ""); - entry.kitType = tableData.getIntField(3, -1); - entry.kitRank = tableData.getIntField(4, -1); - entry.kitImage = tableData.getIntField(5, -1); - entry.skillSetWith2 = tableData.getIntField(6, -1); - entry.skillSetWith3 = tableData.getIntField(7, -1); - entry.skillSetWith4 = tableData.getIntField(8, -1); - entry.skillSetWith5 = tableData.getIntField(9, -1); - entry.skillSetWith6 = tableData.getIntField(10, -1); - entry.localize = tableData.getIntField(11, -1) == 1 ? true : false; - entry.gate_version = tableData.getStringField(12, ""); - entry.kitID = tableData.getIntField(13, -1); - entry.priority = tableData.getFloatField(14, -1.0f); + entry.setID = tableData.getIntField("setID", -1); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.itemIDs = tableData.getStringField("itemIDs", ""); + entry.kitType = tableData.getIntField("kitType", -1); + entry.kitRank = tableData.getIntField("kitRank", -1); + entry.kitImage = tableData.getIntField("kitImage", -1); + entry.skillSetWith2 = tableData.getIntField("skillSetWith2", -1); + entry.skillSetWith3 = tableData.getIntField("skillSetWith3", -1); + entry.skillSetWith4 = tableData.getIntField("skillSetWith4", -1); + entry.skillSetWith5 = tableData.getIntField("skillSetWith5", -1); + entry.skillSetWith6 = tableData.getIntField("skillSetWith6", -1); + entry.localize = tableData.getIntField("localize", -1) == 1 ? true : false; + entry.gate_version = tableData.getStringField("gate_version", ""); + entry.kitID = tableData.getIntField("kitID", -1); + entry.priority = tableData.getFloatField("priority", -1.0f); this->entries.push_back(entry); tableData.nextRow(); @@ -44,15 +44,6 @@ CDItemSetsTable::CDItemSetsTable(void) { tableData.finalize(); } -//! Destructor -CDItemSetsTable::~CDItemSetsTable(void) {} - -//! Returns the table's name -std::string CDItemSetsTable::GetName(void) const { - return "ItemSets"; -} - -//! Queries the table with a custom "where" clause std::vector CDItemSetsTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -62,7 +53,7 @@ std::vector CDItemSetsTable::Query(std::function p return data; } -//! Gets all the entries in the table std::vector CDItemSetsTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDItemSetsTable.h b/dDatabase/Tables/CDItemSetsTable.h index ef12c7b4..6756e7fa 100644 --- a/dDatabase/Tables/CDItemSetsTable.h +++ b/dDatabase/Tables/CDItemSetsTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDItemSetsTable.hpp - \brief Contains data for the ItemSets table - */ - - //! ZoneTable Struct struct CDItemSets { unsigned int setID; //!< The item set ID unsigned int locStatus; //!< The loc status @@ -27,36 +21,15 @@ struct CDItemSets { float priority; //!< The priority }; -//! ItemSets table -class CDItemSetsTable : public CDTable { +class CDItemSetsTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDItemSetsTable(void); - - //! Destructor - ~CDItemSetsTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDItemSetsTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDLevelProgressionLookupTable.cpp b/dDatabase/Tables/CDLevelProgressionLookupTable.cpp index 121c9f62..47a8fc0e 100644 --- a/dDatabase/Tables/CDLevelProgressionLookupTable.cpp +++ b/dDatabase/Tables/CDLevelProgressionLookupTable.cpp @@ -21,9 +21,9 @@ CDLevelProgressionLookupTable::CDLevelProgressionLookupTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM LevelProgressionLookup"); while (!tableData.eof()) { CDLevelProgressionLookup entry; - entry.id = tableData.getIntField(0, -1); - entry.requiredUScore = tableData.getIntField(1, -1); - entry.BehaviorEffect = tableData.getStringField(2, ""); + entry.id = tableData.getIntField("id", -1); + entry.requiredUScore = tableData.getIntField("requiredUScore", -1); + entry.BehaviorEffect = tableData.getStringField("BehaviorEffect", ""); this->entries.push_back(entry); tableData.nextRow(); @@ -32,14 +32,6 @@ CDLevelProgressionLookupTable::CDLevelProgressionLookupTable(void) { tableData.finalize(); } -//! Destructor -CDLevelProgressionLookupTable::~CDLevelProgressionLookupTable(void) {} - -//! Returns the table's name -std::string CDLevelProgressionLookupTable::GetName(void) const { - return "LevelProgressionLookup"; -} - //! Queries the table with a custom "where" clause std::vector CDLevelProgressionLookupTable::Query(std::function predicate) { @@ -54,3 +46,4 @@ std::vector CDLevelProgressionLookupTable::Query(std:: std::vector CDLevelProgressionLookupTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDLevelProgressionLookupTable.h b/dDatabase/Tables/CDLevelProgressionLookupTable.h index 391c9b37..070b2e0c 100644 --- a/dDatabase/Tables/CDLevelProgressionLookupTable.h +++ b/dDatabase/Tables/CDLevelProgressionLookupTable.h @@ -3,47 +3,21 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDLevelProgressionLookupTable.hpp - \brief Contains data for the LevelProgressionLookup table - */ - - //! LevelProgressionLookup Entry Struct struct CDLevelProgressionLookup { unsigned int id; //!< The Level ID unsigned int requiredUScore; //!< The required LEGO Score std::string BehaviorEffect; //!< The behavior effect attached to this }; -//! LevelProgressionLookup table -class CDLevelProgressionLookupTable : public CDTable { +class CDLevelProgressionLookupTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDLevelProgressionLookupTable(void); - - //! Destructor - ~CDLevelProgressionLookupTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDLevelProgressionLookupTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ + // Gets all the entries in the table std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDLootMatrixTable.cpp b/dDatabase/Tables/CDLootMatrixTable.cpp index d47c081b..8f25e8a3 100644 --- a/dDatabase/Tables/CDLootMatrixTable.cpp +++ b/dDatabase/Tables/CDLootMatrixTable.cpp @@ -21,15 +21,15 @@ CDLootMatrixTable::CDLootMatrixTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM LootMatrix"); while (!tableData.eof()) { CDLootMatrix entry; - entry.LootMatrixIndex = tableData.getIntField(0, -1); - entry.LootTableIndex = tableData.getIntField(1, -1); - entry.RarityTableIndex = tableData.getIntField(2, -1); - entry.percent = tableData.getFloatField(3, -1.0f); - entry.minToDrop = tableData.getIntField(4, -1); - entry.maxToDrop = tableData.getIntField(5, -1); - entry.id = tableData.getIntField(6, -1); - entry.flagID = tableData.getIntField(7, -1); - UNUSED(entry.gate_version = tableData.getStringField(8, "")); + entry.LootMatrixIndex = tableData.getIntField("LootMatrixIndex", -1); + entry.LootTableIndex = tableData.getIntField("LootTableIndex", -1); + entry.RarityTableIndex = tableData.getIntField("RarityTableIndex", -1); + entry.percent = tableData.getFloatField("percent", -1.0f); + entry.minToDrop = tableData.getIntField("minToDrop", -1); + entry.maxToDrop = tableData.getIntField("maxToDrop", -1); + entry.id = tableData.getIntField("id", -1); + entry.flagID = tableData.getIntField("flagID", -1); + UNUSED(entry.gate_version = tableData.getStringField("gate_version", "")); this->entries.push_back(entry); tableData.nextRow(); @@ -38,15 +38,6 @@ CDLootMatrixTable::CDLootMatrixTable(void) { tableData.finalize(); } -//! Destructor -CDLootMatrixTable::~CDLootMatrixTable(void) {} - -//! Returns the table's name -std::string CDLootMatrixTable::GetName(void) const { - return "LootMatrix"; -} - -//! Queries the table with a custom "where" clause std::vector CDLootMatrixTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -56,7 +47,7 @@ std::vector CDLootMatrixTable::Query(std::function& CDLootMatrixTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDLootMatrixTable.h b/dDatabase/Tables/CDLootMatrixTable.h index 7d311b3b..c6035841 100644 --- a/dDatabase/Tables/CDLootMatrixTable.h +++ b/dDatabase/Tables/CDLootMatrixTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDLootMatrixTable.hpp - \brief Contains data for the ObjectSkills table - */ - - //! LootMatrix Struct struct CDLootMatrix { unsigned int LootMatrixIndex; //!< The Loot Matrix Index unsigned int LootTableIndex; //!< The Loot Table Index @@ -21,36 +15,15 @@ struct CDLootMatrix { UNUSED(std::string gate_version); //!< The Gate Version }; -//! MissionNPCComponent table -class CDLootMatrixTable : public CDTable { +class CDLootMatrixTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDLootMatrixTable(void); - - //! Destructor - ~CDLootMatrixTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDLootMatrixTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ const std::vector& GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDLootTableTable.cpp b/dDatabase/Tables/CDLootTableTable.cpp index da8c824b..0a46784a 100644 --- a/dDatabase/Tables/CDLootTableTable.cpp +++ b/dDatabase/Tables/CDLootTableTable.cpp @@ -21,12 +21,12 @@ CDLootTableTable::CDLootTableTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM LootTable"); while (!tableData.eof()) { CDLootTable entry; - entry.id = tableData.getIntField(0, -1); - entry.itemid = tableData.getIntField(0, -1); - entry.LootTableIndex = tableData.getIntField(1, -1); - entry.id = tableData.getIntField(2, -1); - entry.MissionDrop = tableData.getIntField(3, -1) == 1 ? true : false; - entry.sortPriority = tableData.getIntField(4, -1); + entry.id = tableData.getIntField("id", -1); + entry.itemid = tableData.getIntField("itemid", -1); + entry.LootTableIndex = tableData.getIntField("LootTableIndex", -1); + entry.id = tableData.getIntField("id", -1); + entry.MissionDrop = tableData.getIntField("MissionDrop", -1) == 1 ? true : false; + entry.sortPriority = tableData.getIntField("sortPriority", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -35,14 +35,6 @@ CDLootTableTable::CDLootTableTable(void) { tableData.finalize(); } -//! Destructor -CDLootTableTable::~CDLootTableTable(void) {} - -//! Returns the table's name -std::string CDLootTableTable::GetName(void) const { - return "LootTable"; -} - //! Queries the table with a custom "where" clause std::vector CDLootTableTable::Query(std::function predicate) { @@ -57,3 +49,4 @@ std::vector CDLootTableTable::Query(std::function& CDLootTableTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDLootTableTable.h b/dDatabase/Tables/CDLootTableTable.h index 3f1baf60..ba6f207e 100644 --- a/dDatabase/Tables/CDLootTableTable.h +++ b/dDatabase/Tables/CDLootTableTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDLootTableTable.hpp - \brief Contains data for the LootTable table - */ - - //! LootTable Struct struct CDLootTable { unsigned int itemid; //!< The LOT of the item unsigned int LootTableIndex; //!< The Loot Table Index @@ -17,36 +11,15 @@ struct CDLootTable { unsigned int sortPriority; //!< The sorting priority }; -//! LootTable table -class CDLootTableTable : public CDTable { +class CDLootTableTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDLootTableTable(void); - - //! Destructor - ~CDLootTableTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDLootTableTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ const std::vector& GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDMissionEmailTable.cpp b/dDatabase/Tables/CDMissionEmailTable.cpp index a8beb95b..ed855d8a 100644 --- a/dDatabase/Tables/CDMissionEmailTable.cpp +++ b/dDatabase/Tables/CDMissionEmailTable.cpp @@ -21,14 +21,14 @@ CDMissionEmailTable::CDMissionEmailTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM MissionEmail"); while (!tableData.eof()) { CDMissionEmail entry; - entry.ID = tableData.getIntField(0, -1); - entry.messageType = tableData.getIntField(1, -1); - entry.notificationGroup = tableData.getIntField(2, -1); - entry.missionID = tableData.getIntField(3, -1); - entry.attachmentLOT = tableData.getIntField(4, 0); - entry.localize = (bool)tableData.getIntField(5, -1); - entry.locStatus = tableData.getIntField(6, -1); - entry.gate_version = tableData.getStringField(7, ""); + entry.ID = tableData.getIntField("ID", -1); + entry.messageType = tableData.getIntField("messageType", -1); + entry.notificationGroup = tableData.getIntField("notificationGroup", -1); + entry.missionID = tableData.getIntField("missionID", -1); + entry.attachmentLOT = tableData.getIntField("attachmentLOT", 0); + entry.localize = (bool)tableData.getIntField("localize", -1); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.gate_version = tableData.getStringField("gate_version", ""); this->entries.push_back(entry); tableData.nextRow(); @@ -37,14 +37,6 @@ CDMissionEmailTable::CDMissionEmailTable(void) { tableData.finalize(); } -//! Destructor -CDMissionEmailTable::~CDMissionEmailTable(void) {} - -//! Returns the table's name -std::string CDMissionEmailTable::GetName(void) const { - return "MissionEmail"; -} - //! Queries the table with a custom "where" clause std::vector CDMissionEmailTable::Query(std::function predicate) { @@ -59,3 +51,4 @@ std::vector CDMissionEmailTable::Query(std::function CDMissionEmailTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDMissionEmailTable.h b/dDatabase/Tables/CDMissionEmailTable.h index 81a7a793..db2310d4 100644 --- a/dDatabase/Tables/CDMissionEmailTable.h +++ b/dDatabase/Tables/CDMissionEmailTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDMissionEmailTable.hpp - \brief Contains data for the MissionEmail table - */ - - //! MissionEmail Entry Struct struct CDMissionEmail { unsigned int ID; unsigned int messageType; @@ -21,35 +15,14 @@ struct CDMissionEmail { }; -//! MissionEmail table -class CDMissionEmailTable : public CDTable { +class CDMissionEmailTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDMissionEmailTable(void); - - //! Destructor - ~CDMissionEmailTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDMissionEmailTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDMissionNPCComponentTable.cpp b/dDatabase/Tables/CDMissionNPCComponentTable.cpp index e2589890..5672ed67 100644 --- a/dDatabase/Tables/CDMissionNPCComponentTable.cpp +++ b/dDatabase/Tables/CDMissionNPCComponentTable.cpp @@ -21,11 +21,11 @@ CDMissionNPCComponentTable::CDMissionNPCComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM MissionNPCComponent"); while (!tableData.eof()) { CDMissionNPCComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.missionID = tableData.getIntField(1, -1); - entry.offersMission = tableData.getIntField(2, -1) == 1 ? true : false; - entry.acceptsMission = tableData.getIntField(3, -1) == 1 ? true : false; - entry.gate_version = tableData.getStringField(4, ""); + entry.id = tableData.getIntField("id", -1); + entry.missionID = tableData.getIntField("missionID", -1); + entry.offersMission = tableData.getIntField("offersMission", -1) == 1 ? true : false; + entry.acceptsMission = tableData.getIntField("acceptsMission", -1) == 1 ? true : false; + entry.gate_version = tableData.getStringField("gate_version", ""); this->entries.push_back(entry); tableData.nextRow(); @@ -34,14 +34,6 @@ CDMissionNPCComponentTable::CDMissionNPCComponentTable(void) { tableData.finalize(); } -//! Destructor -CDMissionNPCComponentTable::~CDMissionNPCComponentTable(void) {} - -//! Returns the table's name -std::string CDMissionNPCComponentTable::GetName(void) const { - return "MissionNPCComponent"; -} - //! Queries the table with a custom "where" clause std::vector CDMissionNPCComponentTable::Query(std::function predicate) { @@ -56,3 +48,4 @@ std::vector CDMissionNPCComponentTable::Query(std::functi std::vector CDMissionNPCComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDMissionNPCComponentTable.h b/dDatabase/Tables/CDMissionNPCComponentTable.h index 68c94ef0..a7aeb145 100644 --- a/dDatabase/Tables/CDMissionNPCComponentTable.h +++ b/dDatabase/Tables/CDMissionNPCComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDMissionNPCComponentTable.hpp - \brief Contains data for the ObjectSkills table - */ - - //! MissionNPCComponent Struct struct CDMissionNPCComponent { unsigned int id; //!< The ID unsigned int missionID; //!< The Mission ID @@ -17,35 +11,16 @@ struct CDMissionNPCComponent { std::string gate_version; //!< The gate version }; -//! MissionNPCComponent table -class CDMissionNPCComponentTable : public CDTable { +class CDMissionNPCComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDMissionNPCComponentTable(void); - - //! Destructor - ~CDMissionNPCComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDMissionNPCComponentTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ + // Gets all the entries in the table std::vector GetEntries(void) const; }; diff --git a/dDatabase/Tables/CDMissionTasksTable.cpp b/dDatabase/Tables/CDMissionTasksTable.cpp index 81de2088..f32dca1b 100644 --- a/dDatabase/Tables/CDMissionTasksTable.cpp +++ b/dDatabase/Tables/CDMissionTasksTable.cpp @@ -21,19 +21,19 @@ CDMissionTasksTable::CDMissionTasksTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM MissionTasks"); while (!tableData.eof()) { CDMissionTasks entry; - entry.id = tableData.getIntField(0, -1); - UNUSED(entry.locStatus = tableData.getIntField(1, -1)); - entry.taskType = tableData.getIntField(2, -1); - entry.target = tableData.getIntField(3, -1); - entry.targetGroup = tableData.getStringField(4, ""); - entry.targetValue = tableData.getIntField(5, -1); - entry.taskParam1 = tableData.getStringField(6, ""); - UNUSED(entry.largeTaskIcon = tableData.getStringField(7, "")); - UNUSED(entry.IconID = tableData.getIntField(8, -1)); - entry.uid = tableData.getIntField(9, -1); - UNUSED(entry.largeTaskIconID = tableData.getIntField(10, -1)); - UNUSED(entry.localize = tableData.getIntField(11, -1) == 1 ? true : false); - UNUSED(entry.gate_version = tableData.getStringField(12, "")); + entry.id = tableData.getIntField("id", -1); + UNUSED(entry.locStatus = tableData.getIntField("locStatus", -1)); + entry.taskType = tableData.getIntField("taskType", -1); + entry.target = tableData.getIntField("target", -1); + entry.targetGroup = tableData.getStringField("targetGroup", ""); + entry.targetValue = tableData.getIntField("targetValue", -1); + entry.taskParam1 = tableData.getStringField("taskParam1", ""); + UNUSED(entry.largeTaskIcon = tableData.getStringField("largeTaskIcon", "")); + UNUSED(entry.IconID = tableData.getIntField("IconID", -1)); + entry.uid = tableData.getIntField("uid", -1); + UNUSED(entry.largeTaskIconID = tableData.getIntField("largeTaskIconID", -1)); + UNUSED(entry.localize = tableData.getIntField("localize", -1) == 1 ? true : false); + UNUSED(entry.gate_version = tableData.getStringField("gate_version", "")); this->entries.push_back(entry); tableData.nextRow(); @@ -42,15 +42,6 @@ CDMissionTasksTable::CDMissionTasksTable(void) { tableData.finalize(); } -//! Destructor -CDMissionTasksTable::~CDMissionTasksTable(void) {} - -//! Returns the table's name -std::string CDMissionTasksTable::GetName(void) const { - return "MissionTasks"; -} - -//! Queries the table with a custom "where" clause std::vector CDMissionTasksTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -74,7 +65,7 @@ std::vector CDMissionTasksTable::GetByMissionID(uint32_t missio return tasks; } -//! Gets all the entries in the table const std::vector& CDMissionTasksTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDMissionTasksTable.h b/dDatabase/Tables/CDMissionTasksTable.h index 1a4bd361..fa213faf 100644 --- a/dDatabase/Tables/CDMissionTasksTable.h +++ b/dDatabase/Tables/CDMissionTasksTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDMissionTasksTable.hpp - \brief Contains data for the MissionTasks table - */ - - //! ObjectSkills Struct struct CDMissionTasks { unsigned int id; //!< The Mission ID that the task belongs to UNUSED(unsigned int locStatus); //!< ??? @@ -25,37 +19,17 @@ struct CDMissionTasks { UNUSED(std::string gate_version); //!< ??? }; -//! ObjectSkills table -class CDMissionTasksTable : public CDTable { +class CDMissionTasksTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDMissionTasksTable(void); - - //! Destructor - ~CDMissionTasksTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDMissionTasksTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); std::vector GetByMissionID(uint32_t missionID); - //! Gets all the entries in the table - /*! - \return The entries - */ const std::vector& GetEntries(void) const; }; diff --git a/dDatabase/Tables/CDMissionsTable.cpp b/dDatabase/Tables/CDMissionsTable.cpp index 7e59657d..d4ee40ae 100644 --- a/dDatabase/Tables/CDMissionsTable.cpp +++ b/dDatabase/Tables/CDMissionsTable.cpp @@ -23,58 +23,58 @@ CDMissionsTable::CDMissionsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM Missions"); while (!tableData.eof()) { CDMissions entry; - entry.id = tableData.getIntField(0, -1); - entry.defined_type = tableData.getStringField(1, ""); - entry.defined_subtype = tableData.getStringField(2, ""); - entry.UISortOrder = tableData.getIntField(3, -1); - entry.offer_objectID = tableData.getIntField(4, -1); - entry.target_objectID = tableData.getIntField(5, -1); - entry.reward_currency = tableData.getInt64Field(6, -1); - entry.LegoScore = tableData.getIntField(7, -1); - entry.reward_reputation = tableData.getIntField(8, -1); - entry.isChoiceReward = tableData.getIntField(9, -1) == 1 ? true : false; - entry.reward_item1 = tableData.getIntField(10, 0); - entry.reward_item1_count = tableData.getIntField(11, 0); - entry.reward_item2 = tableData.getIntField(12, 0); - entry.reward_item2_count = tableData.getIntField(13, 0); - entry.reward_item3 = tableData.getIntField(14, 0); - entry.reward_item3_count = tableData.getIntField(15, 0); - entry.reward_item4 = tableData.getIntField(16, 0); - entry.reward_item4_count = tableData.getIntField(17, 0); - entry.reward_emote = tableData.getIntField(18, -1); - entry.reward_emote2 = tableData.getIntField(19, -1); - entry.reward_emote3 = tableData.getIntField(20, -1); - entry.reward_emote4 = tableData.getIntField(21, -1); - entry.reward_maximagination = tableData.getIntField(22, -1); - entry.reward_maxhealth = tableData.getIntField(23, -1); - entry.reward_maxinventory = tableData.getIntField(24, -1); - entry.reward_maxmodel = tableData.getIntField(25, -1); - entry.reward_maxwidget = tableData.getIntField(26, -1); - entry.reward_maxwallet = tableData.getIntField(27, -1); - entry.repeatable = tableData.getIntField(28, -1) == 1 ? true : false; - entry.reward_currency_repeatable = tableData.getIntField(29, -1); - entry.reward_item1_repeatable = tableData.getIntField(30, -1); - entry.reward_item1_repeat_count = tableData.getIntField(31, -1); - entry.reward_item2_repeatable = tableData.getIntField(32, -1); - entry.reward_item2_repeat_count = tableData.getIntField(33, -1); - entry.reward_item3_repeatable = tableData.getIntField(34, -1); - entry.reward_item3_repeat_count = tableData.getIntField(35, -1); - entry.reward_item4_repeatable = tableData.getIntField(36, -1); - entry.reward_item4_repeat_count = tableData.getIntField(37, -1); - entry.time_limit = tableData.getIntField(38, -1); - entry.isMission = tableData.getIntField(39, -1) ? true : false; - entry.missionIconID = tableData.getIntField(40, -1); - entry.prereqMissionID = tableData.getStringField(41, ""); - entry.localize = tableData.getIntField(42, -1) == 1 ? true : false; - entry.inMOTD = tableData.getIntField(43, -1) == 1 ? true : false; - entry.cooldownTime = tableData.getInt64Field(44, -1); - entry.isRandom = tableData.getIntField(45, -1) == 1 ? true : false; - entry.randomPool = tableData.getStringField(46, ""); - entry.UIPrereqID = tableData.getIntField(47, -1); - UNUSED(entry.gate_version = tableData.getStringField(48, "")); - UNUSED(entry.HUDStates = tableData.getStringField(49, "")); - UNUSED(entry.locStatus = tableData.getIntField(50, -1)); - entry.reward_bankinventory = tableData.getIntField(51, -1); + entry.id = tableData.getIntField("id", -1); + entry.defined_type = tableData.getStringField("defined_type", ""); + entry.defined_subtype = tableData.getStringField("defined_subtype", ""); + entry.UISortOrder = tableData.getIntField("UISortOrder", -1); + entry.offer_objectID = tableData.getIntField("offer_objectID", -1); + entry.target_objectID = tableData.getIntField("target_objectID", -1); + entry.reward_currency = tableData.getInt64Field("reward_currency", -1); + entry.LegoScore = tableData.getIntField("LegoScore", -1); + entry.reward_reputation = tableData.getIntField("reward_reputation", -1); + entry.isChoiceReward = tableData.getIntField("isChoiceReward", -1) == 1 ? true : false; + entry.reward_item1 = tableData.getIntField("reward_item1", 0); + entry.reward_item1_count = tableData.getIntField("reward_item1_count", 0); + entry.reward_item2 = tableData.getIntField("reward_item2", 0); + entry.reward_item2_count = tableData.getIntField("reward_item2_count", 0); + entry.reward_item3 = tableData.getIntField("reward_item3", 0); + entry.reward_item3_count = tableData.getIntField("reward_item3_count", 0); + entry.reward_item4 = tableData.getIntField("reward_item4", 0); + entry.reward_item4_count = tableData.getIntField("reward_item4_count", 0); + entry.reward_emote = tableData.getIntField("reward_emote", -1); + entry.reward_emote2 = tableData.getIntField("reward_emote2", -1); + entry.reward_emote3 = tableData.getIntField("reward_emote3", -1); + entry.reward_emote4 = tableData.getIntField("reward_emote4", -1); + entry.reward_maximagination = tableData.getIntField("reward_maximagination", -1); + entry.reward_maxhealth = tableData.getIntField("reward_maxhealth", -1); + entry.reward_maxinventory = tableData.getIntField("reward_maxinventory", -1); + entry.reward_maxmodel = tableData.getIntField("reward_maxmodel", -1); + entry.reward_maxwidget = tableData.getIntField("reward_maxwidget", -1); + entry.reward_maxwallet = tableData.getIntField("reward_maxwallet", -1); + entry.repeatable = tableData.getIntField("repeatable", -1) == 1 ? true : false; + entry.reward_currency_repeatable = tableData.getIntField("reward_currency_repeatable", -1); + entry.reward_item1_repeatable = tableData.getIntField("reward_item1_repeatable", -1); + entry.reward_item1_repeat_count = tableData.getIntField("reward_item1_repeat_count", -1); + entry.reward_item2_repeatable = tableData.getIntField("reward_item2_repeatable", -1); + entry.reward_item2_repeat_count = tableData.getIntField("reward_item2_repeat_count", -1); + entry.reward_item3_repeatable = tableData.getIntField("reward_item3_repeatable", -1); + entry.reward_item3_repeat_count = tableData.getIntField("reward_item3_repeat_count", -1); + entry.reward_item4_repeatable = tableData.getIntField("reward_item4_repeatable", -1); + entry.reward_item4_repeat_count = tableData.getIntField("reward_item4_repeat_count", -1); + entry.time_limit = tableData.getIntField("time_limit", -1); + entry.isMission = tableData.getIntField("isMission", -1) ? true : false; + entry.missionIconID = tableData.getIntField("missionIconID", -1); + entry.prereqMissionID = tableData.getStringField("prereqMissionID", ""); + entry.localize = tableData.getIntField("localize", -1) == 1 ? true : false; + entry.inMOTD = tableData.getIntField("inMOTD", -1) == 1 ? true : false; + entry.cooldownTime = tableData.getInt64Field("cooldownTime", -1); + entry.isRandom = tableData.getIntField("isRandom", -1) == 1 ? true : false; + entry.randomPool = tableData.getStringField("randomPool", ""); + entry.UIPrereqID = tableData.getIntField("UIPrereqID", -1); + UNUSED(entry.gate_version = tableData.getStringField("gate_version", "")); + UNUSED(entry.HUDStates = tableData.getStringField("HUDStates", "")); + UNUSED(entry.locStatus = tableData.getIntField("locStatus", -1)); + entry.reward_bankinventory = tableData.getIntField("reward_bankinventory", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -85,15 +85,6 @@ CDMissionsTable::CDMissionsTable(void) { Default.id = -1; } -//! Destructor -CDMissionsTable::~CDMissionsTable(void) {} - -//! Returns the table's name -std::string CDMissionsTable::GetName(void) const { - return "Missions"; -} - -//! Queries the table with a custom "where" clause std::vector CDMissionsTable::Query(std::function predicate) { std::vector data = cpplinq::from(this->entries) @@ -103,7 +94,6 @@ std::vector CDMissionsTable::Query(std::function p return data; } -//! Gets all the entries in the table const std::vector& CDMissionsTable::GetEntries(void) const { return this->entries; } @@ -131,3 +121,4 @@ const CDMissions& CDMissionsTable::GetByMissionID(uint32_t missionID, bool& foun return Default; } + diff --git a/dDatabase/Tables/CDMissionsTable.h b/dDatabase/Tables/CDMissionsTable.h index 6d1c4a8f..e6a44b02 100644 --- a/dDatabase/Tables/CDMissionsTable.h +++ b/dDatabase/Tables/CDMissionsTable.h @@ -5,12 +5,6 @@ #include #include -/*! - \file CDMissionsTable.hpp - \brief Contains data for the Missions table - */ - - //! Missions Struct struct CDMissions { int id; //!< The Mission ID std::string defined_type; //!< The type of mission @@ -66,35 +60,16 @@ struct CDMissions { int reward_bankinventory; //!< The amount of bank space this mission rewards }; -//! Missions table -class CDMissionsTable : public CDTable { +class CDMissionsTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDMissionsTable(void); - - //! Destructor - ~CDMissionsTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDMissionsTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ + // Gets all the entries in the table const std::vector& GetEntries(void) const; const CDMissions* GetPtrByMissionID(uint32_t missionID) const; diff --git a/dDatabase/Tables/CDMovementAIComponentTable.cpp b/dDatabase/Tables/CDMovementAIComponentTable.cpp index a98be8ba..3b9cc4f4 100644 --- a/dDatabase/Tables/CDMovementAIComponentTable.cpp +++ b/dDatabase/Tables/CDMovementAIComponentTable.cpp @@ -21,14 +21,14 @@ CDMovementAIComponentTable::CDMovementAIComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM MovementAIComponent"); while (!tableData.eof()) { CDMovementAIComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.MovementType = tableData.getStringField(1, ""); - entry.WanderChance = tableData.getFloatField(2, -1.0f); - entry.WanderDelayMin = tableData.getFloatField(3, -1.0f); - entry.WanderDelayMax = tableData.getFloatField(4, -1.0f); - entry.WanderSpeed = tableData.getFloatField(5, -1.0f); - entry.WanderRadius = tableData.getFloatField(6, -1.0f); - entry.attachedPath = tableData.getStringField(7, ""); + entry.id = tableData.getIntField("id", -1); + entry.MovementType = tableData.getStringField("MovementType", ""); + entry.WanderChance = tableData.getFloatField("WanderChance", -1.0f); + entry.WanderDelayMin = tableData.getFloatField("WanderDelayMin", -1.0f); + entry.WanderDelayMax = tableData.getFloatField("WanderDelayMax", -1.0f); + entry.WanderSpeed = tableData.getFloatField("WanderSpeed", -1.0f); + entry.WanderRadius = tableData.getFloatField("WanderRadius", -1.0f); + entry.attachedPath = tableData.getStringField("attachedPath", ""); this->entries.push_back(entry); tableData.nextRow(); @@ -37,14 +37,6 @@ CDMovementAIComponentTable::CDMovementAIComponentTable(void) { tableData.finalize(); } -//! Destructor -CDMovementAIComponentTable::~CDMovementAIComponentTable(void) {} - -//! Returns the table's name -std::string CDMovementAIComponentTable::GetName(void) const { - return "MovementAIComponent"; -} - //! Queries the table with a custom "where" clause std::vector CDMovementAIComponentTable::Query(std::function predicate) { @@ -59,3 +51,4 @@ std::vector CDMovementAIComponentTable::Query(std::functi std::vector CDMovementAIComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDMovementAIComponentTable.h b/dDatabase/Tables/CDMovementAIComponentTable.h index 0064a98b..84896e2c 100644 --- a/dDatabase/Tables/CDMovementAIComponentTable.h +++ b/dDatabase/Tables/CDMovementAIComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDMovementAIComponentTable.hpp - \brief Contains data for the MovementAIComponent table - */ - - //! MovementAIComponent Struct struct CDMovementAIComponent { unsigned int id; std::string MovementType; @@ -20,36 +14,15 @@ struct CDMovementAIComponent { std::string attachedPath; }; -//! MovementAIComponent table -class CDMovementAIComponentTable : public CDTable { +class CDMovementAIComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDMovementAIComponentTable(void); - - //! Destructor - ~CDMovementAIComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDMovementAIComponentTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ + // Gets all the entries in the table std::vector GetEntries(void) const; - }; - diff --git a/dDatabase/Tables/CDObjectSkillsTable.cpp b/dDatabase/Tables/CDObjectSkillsTable.cpp index 1a5c4790..2e8b3fb2 100644 --- a/dDatabase/Tables/CDObjectSkillsTable.cpp +++ b/dDatabase/Tables/CDObjectSkillsTable.cpp @@ -21,10 +21,10 @@ CDObjectSkillsTable::CDObjectSkillsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ObjectSkills"); while (!tableData.eof()) { CDObjectSkills entry; - entry.objectTemplate = tableData.getIntField(0, -1); - entry.skillID = tableData.getIntField(1, -1); - entry.castOnType = tableData.getIntField(2, -1); - entry.AICombatWeight = tableData.getIntField(3, -1); + entry.objectTemplate = tableData.getIntField("objectTemplate", -1); + entry.skillID = tableData.getIntField("skillID", -1); + entry.castOnType = tableData.getIntField("castOnType", -1); + entry.AICombatWeight = tableData.getIntField("AICombatWeight", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -33,14 +33,6 @@ CDObjectSkillsTable::CDObjectSkillsTable(void) { tableData.finalize(); } -//! Destructor -CDObjectSkillsTable::~CDObjectSkillsTable(void) {} - -//! Returns the table's name -std::string CDObjectSkillsTable::GetName(void) const { - return "ObjectSkills"; -} - //! Queries the table with a custom "where" clause std::vector CDObjectSkillsTable::Query(std::function predicate) { diff --git a/dDatabase/Tables/CDObjectSkillsTable.h b/dDatabase/Tables/CDObjectSkillsTable.h index c2caf71b..4ceaa447 100644 --- a/dDatabase/Tables/CDObjectSkillsTable.h +++ b/dDatabase/Tables/CDObjectSkillsTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDObjectSkillsTable.hpp - \brief Contains data for the ObjectSkills table - */ - - //! ObjectSkills Struct struct CDObjectSkills { unsigned int objectTemplate; //!< The LOT of the item unsigned int skillID; //!< The Skill ID of the object @@ -16,35 +10,16 @@ struct CDObjectSkills { unsigned int AICombatWeight; //!< ??? }; -//! ObjectSkills table -class CDObjectSkillsTable : public CDTable { +class CDObjectSkillsTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDObjectSkillsTable(void); - - //! Destructor - ~CDObjectSkillsTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDObjectSkillsTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ + // Gets all the entries in the table std::vector GetEntries(void) const; }; diff --git a/dDatabase/Tables/CDObjectsTable.cpp b/dDatabase/Tables/CDObjectsTable.cpp index 03979af2..c68c3e6a 100644 --- a/dDatabase/Tables/CDObjectsTable.cpp +++ b/dDatabase/Tables/CDObjectsTable.cpp @@ -18,20 +18,20 @@ CDObjectsTable::CDObjectsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM Objects"); while (!tableData.eof()) { CDObjects entry; - entry.id = tableData.getIntField(0, -1); - entry.name = tableData.getStringField(1, ""); - entry.placeable = tableData.getIntField(2, -1); - entry.type = tableData.getStringField(3, ""); - entry.description = tableData.getStringField(4, ""); - entry.localize = tableData.getIntField(5, -1); - entry.npcTemplateID = tableData.getIntField(6, -1); - entry.displayName = tableData.getStringField(7, ""); - entry.interactionDistance = tableData.getFloatField(8, -1.0f); - entry.nametag = tableData.getIntField(9, -1); - entry._internalNotes = tableData.getStringField(10, ""); - entry.locStatus = tableData.getIntField(11, -1); - entry.gate_version = tableData.getStringField(12, ""); - entry.HQ_valid = tableData.getIntField(13, -1); + entry.id = tableData.getIntField("id", -1); + entry.name = tableData.getStringField("name", ""); + entry.placeable = tableData.getIntField("placeable", -1); + entry.type = tableData.getStringField("type", ""); + entry.description = tableData.getStringField("description", ""); + entry.localize = tableData.getIntField("localize", -1); + entry.npcTemplateID = tableData.getIntField("npcTemplateID", -1); + entry.displayName = tableData.getStringField("displayName", ""); + entry.interactionDistance = tableData.getFloatField("interactionDistance", -1.0f); + entry.nametag = tableData.getIntField("nametag", -1); + entry._internalNotes = tableData.getStringField("_internalNotes", ""); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.gate_version = tableData.getStringField("gate_version", ""); + entry.HQ_valid = tableData.getIntField("HQ_valid", -1); this->entries.insert(std::make_pair(entry.id, entry)); tableData.nextRow(); @@ -43,14 +43,6 @@ CDObjectsTable::CDObjectsTable(void) { m_default.id = 0; } -//! Destructor -CDObjectsTable::~CDObjectsTable(void) {} - -//! Returns the table's name -std::string CDObjectsTable::GetName(void) const { - return "Objects"; -} - const CDObjects& CDObjectsTable::GetByID(unsigned int LOT) { const auto& it = this->entries.find(LOT); if (it != this->entries.end()) { @@ -71,20 +63,20 @@ const CDObjects& CDObjectsTable::GetByID(unsigned int LOT) { // Now get the data while (!tableData.eof()) { CDObjects entry; - entry.id = tableData.getIntField(0, -1); - entry.name = tableData.getStringField(1, ""); - UNUSED(entry.placeable = tableData.getIntField(2, -1)); - entry.type = tableData.getStringField(3, ""); + entry.id = tableData.getIntField("id", -1); + entry.name = tableData.getStringField("name", ""); + UNUSED(entry.placeable = tableData.getIntField("placeable", -1)); + entry.type = tableData.getStringField("type", ""); UNUSED(ntry.description = tableData.getStringField(4, "")); - UNUSED(entry.localize = tableData.getIntField(5, -1)); - UNUSED(entry.npcTemplateID = tableData.getIntField(6, -1)); - UNUSED(entry.displayName = tableData.getStringField(7, "")); - entry.interactionDistance = tableData.getFloatField(8, -1.0f); - UNUSED(entry.nametag = tableData.getIntField(9, -1)); - UNUSED(entry._internalNotes = tableData.getStringField(10, "")); - UNUSED(entry.locStatus = tableData.getIntField(11, -1)); - UNUSED(entry.gate_version = tableData.getStringField(12, "")); - UNUSED(entry.HQ_valid = tableData.getIntField(13, -1)); + UNUSED(entry.localize = tableData.getIntField("localize", -1)); + UNUSED(entry.npcTemplateID = tableData.getIntField("npcTemplateID", -1)); + UNUSED(entry.displayName = tableData.getStringField("displayName", "")); + entry.interactionDistance = tableData.getFloatField("interactionDistance", -1.0f); + UNUSED(entry.nametag = tableData.getIntField("nametag", -1)); + UNUSED(entry._internalNotes = tableData.getStringField("_internalNotes", "")); + UNUSED(entry.locStatus = tableData.getIntField("locStatus", -1)); + UNUSED(entry.gate_version = tableData.getStringField("gate_version", "")); + UNUSED(entry.HQ_valid = tableData.getIntField("HQ_valid", -1)); this->entries.insert(std::make_pair(entry.id, entry)); tableData.nextRow(); @@ -100,3 +92,4 @@ const CDObjects& CDObjectsTable::GetByID(unsigned int LOT) { return m_default; } + diff --git a/dDatabase/Tables/CDObjectsTable.h b/dDatabase/Tables/CDObjectsTable.h index 333477bd..171eddef 100644 --- a/dDatabase/Tables/CDObjectsTable.h +++ b/dDatabase/Tables/CDObjectsTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDObjectsTable.hpp - \brief Contains data for the Objects table - */ - - //! RebuildComponent Struct struct CDObjects { unsigned int id; //!< The LOT of the object std::string name; //!< The internal name of the object @@ -26,29 +20,14 @@ struct CDObjects { UNUSED(unsigned int HQ_valid); //!< Probably used for the Nexus HQ database on LEGOUniverse.com }; -//! ObjectSkills table -class CDObjectsTable : public CDTable { +class CDObjectsTable : public CDTable { private: - //std::vector entries; std::map entries; CDObjects m_default; public: - - //! Constructor - CDObjectsTable(void); - - //! Destructor - ~CDObjectsTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Gets an entry by ID + CDObjectsTable(); + // Gets an entry by ID const CDObjects& GetByID(unsigned int LOT); - }; diff --git a/dDatabase/Tables/CDPackageComponentTable.cpp b/dDatabase/Tables/CDPackageComponentTable.cpp index 83673c6f..efb85eeb 100644 --- a/dDatabase/Tables/CDPackageComponentTable.cpp +++ b/dDatabase/Tables/CDPackageComponentTable.cpp @@ -21,9 +21,9 @@ CDPackageComponentTable::CDPackageComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM PackageComponent"); while (!tableData.eof()) { CDPackageComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.LootMatrixIndex = tableData.getIntField(1, -1); - entry.packageType = tableData.getIntField(2, -1); + entry.id = tableData.getIntField("id", -1); + entry.LootMatrixIndex = tableData.getIntField("LootMatrixIndex", -1); + entry.packageType = tableData.getIntField("packageType", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -32,14 +32,6 @@ CDPackageComponentTable::CDPackageComponentTable(void) { tableData.finalize(); } -//! Destructor -CDPackageComponentTable::~CDPackageComponentTable(void) {} - -//! Returns the table's name -std::string CDPackageComponentTable::GetName(void) const { - return "PackageComponent"; -} - //! Queries the table with a custom "where" clause std::vector CDPackageComponentTable::Query(std::function predicate) { @@ -54,3 +46,4 @@ std::vector CDPackageComponentTable::Query(std::function CDPackageComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDPackageComponentTable.h b/dDatabase/Tables/CDPackageComponentTable.h index 763acf8c..6c11ab39 100644 --- a/dDatabase/Tables/CDPackageComponentTable.h +++ b/dDatabase/Tables/CDPackageComponentTable.h @@ -3,48 +3,20 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDPackageComponentTable.hpp - \brief Contains data for the PackageComponent table - */ - - //! PackageComponent Entry Struct struct CDPackageComponent { unsigned int id; unsigned int LootMatrixIndex; unsigned int packageType; }; - -//! PackageComponent table -class CDPackageComponentTable : public CDTable { +class CDPackageComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor CDPackageComponentTable(void); - - //! Destructor - ~CDPackageComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDPhysicsComponentTable.cpp b/dDatabase/Tables/CDPhysicsComponentTable.cpp index 8cb7a3b0..bb21ed7f 100644 --- a/dDatabase/Tables/CDPhysicsComponentTable.cpp +++ b/dDatabase/Tables/CDPhysicsComponentTable.cpp @@ -4,22 +4,22 @@ CDPhysicsComponentTable::CDPhysicsComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM PhysicsComponent"); while (!tableData.eof()) { CDPhysicsComponent* entry = new CDPhysicsComponent(); - entry->id = tableData.getIntField(0, -1); - entry->bStatic = tableData.getIntField(1, -1) != 0; - entry->physicsAsset = tableData.getStringField(2, ""); - UNUSED(entry->jump = tableData.getIntField(3, -1) != 0); - UNUSED(entry->doublejump = tableData.getIntField(4, -1) != 0); - entry->speed = tableData.getFloatField(5, -1); - UNUSED(entry->rotSpeed = tableData.getFloatField(6, -1)); - entry->playerHeight = tableData.getFloatField(7); - entry->playerRadius = tableData.getFloatField(8); - entry->pcShapeType = tableData.getIntField(9); - entry->collisionGroup = tableData.getIntField(10); - UNUSED(entry->airSpeed = tableData.getFloatField(11)); - UNUSED(entry->boundaryAsset = tableData.getStringField(12)); - UNUSED(entry->jumpAirSpeed = tableData.getFloatField(13)); - UNUSED(entry->friction = tableData.getFloatField(14)); - UNUSED(entry->gravityVolumeAsset = tableData.getStringField(15)); + entry->id = tableData.getIntField("id", -1); + entry->bStatic = tableData.getIntField("static", -1) != 0; + entry->physicsAsset = tableData.getStringField("physics_asset", ""); + UNUSED(entry->jump = tableData.getIntField("jump", -1) != 0); + UNUSED(entry->doublejump = tableData.getIntField("doublejump", -1) != 0); + entry->speed = tableData.getFloatField("speed", -1); + UNUSED(entry->rotSpeed = tableData.getFloatField("rotSpeed", -1)); + entry->playerHeight = tableData.getFloatField("playerHeight"); + entry->playerRadius = tableData.getFloatField("playerRadius"); + entry->pcShapeType = tableData.getIntField("pcShapeType"); + entry->collisionGroup = tableData.getIntField("collisionGroup"); + UNUSED(entry->airSpeed = tableData.getFloatField("airSpeed")); + UNUSED(entry->boundaryAsset = tableData.getStringField("boundaryAsset")); + UNUSED(entry->jumpAirSpeed = tableData.getFloatField("jumpAirSpeed")); + UNUSED(entry->friction = tableData.getFloatField("friction")); + UNUSED(entry->gravityVolumeAsset = tableData.getStringField("gravityVolumeAsset")); m_entries.insert(std::make_pair(entry->id, entry)); tableData.nextRow(); @@ -28,7 +28,7 @@ CDPhysicsComponentTable::CDPhysicsComponentTable(void) { tableData.finalize(); } -CDPhysicsComponentTable::~CDPhysicsComponentTable(void) { +CDPhysicsComponentTable::~CDPhysicsComponentTable() { for (auto e : m_entries) { if (e.second) delete e.second; } @@ -36,10 +36,6 @@ CDPhysicsComponentTable::~CDPhysicsComponentTable(void) { m_entries.clear(); } -std::string CDPhysicsComponentTable::GetName(void) const { - return "PhysicsComponent"; -} - CDPhysicsComponent* CDPhysicsComponentTable::GetByID(unsigned int componentID) { for (auto e : m_entries) { if (e.first == componentID) return e.second; @@ -47,3 +43,4 @@ CDPhysicsComponent* CDPhysicsComponentTable::GetByID(unsigned int componentID) { return nullptr; } + diff --git a/dDatabase/Tables/CDPhysicsComponentTable.h b/dDatabase/Tables/CDPhysicsComponentTable.h index c5805da3..e63d337d 100644 --- a/dDatabase/Tables/CDPhysicsComponentTable.h +++ b/dDatabase/Tables/CDPhysicsComponentTable.h @@ -21,12 +21,12 @@ struct CDPhysicsComponent { UNUSED(std::string gravityVolumeAsset); }; -class CDPhysicsComponentTable : public CDTable { +class CDPhysicsComponentTable : public CDTable { public: - CDPhysicsComponentTable(void); - ~CDPhysicsComponentTable(void); + CDPhysicsComponentTable(); + ~CDPhysicsComponentTable(); - std::string GetName(void) const override; + static const std::string GetTableName() { return "PhysicsComponent"; }; CDPhysicsComponent* GetByID(unsigned int componentID); private: diff --git a/dDatabase/Tables/CDPropertyEntranceComponentTable.cpp b/dDatabase/Tables/CDPropertyEntranceComponentTable.cpp index 01a2ae37..1fead45e 100644 --- a/dDatabase/Tables/CDPropertyEntranceComponentTable.cpp +++ b/dDatabase/Tables/CDPropertyEntranceComponentTable.cpp @@ -18,11 +18,11 @@ CDPropertyEntranceComponentTable::CDPropertyEntranceComponentTable() { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM PropertyEntranceComponent;"); while (!tableData.eof()) { auto entry = CDPropertyEntranceComponent{ - static_cast(tableData.getIntField(0, -1)), - static_cast(tableData.getIntField(1, -1)), - tableData.getStringField(2, ""), - static_cast(tableData.getIntField(3, false)), - tableData.getStringField(4, "") + static_cast(tableData.getIntField("id", -1)), + static_cast(tableData.getIntField("mapID", -1)), + tableData.getStringField("propertyName", ""), + static_cast(tableData.getIntField("isOnProperty", false)), + tableData.getStringField("groupType", "") }; this->entries.push_back(entry); @@ -32,12 +32,6 @@ CDPropertyEntranceComponentTable::CDPropertyEntranceComponentTable() { tableData.finalize(); } -CDPropertyEntranceComponentTable::~CDPropertyEntranceComponentTable(void) = default; - -std::string CDPropertyEntranceComponentTable::GetName() const { - return "PropertyEntranceComponent"; -} - CDPropertyEntranceComponent CDPropertyEntranceComponentTable::GetByID(uint32_t id) { for (const auto& entry : entries) { if (entry.id == id) @@ -46,3 +40,4 @@ CDPropertyEntranceComponent CDPropertyEntranceComponentTable::GetByID(uint32_t i return defaultEntry; } + diff --git a/dDatabase/Tables/CDPropertyEntranceComponentTable.h b/dDatabase/Tables/CDPropertyEntranceComponentTable.h index fa0603c0..925fd1be 100644 --- a/dDatabase/Tables/CDPropertyEntranceComponentTable.h +++ b/dDatabase/Tables/CDPropertyEntranceComponentTable.h @@ -9,31 +9,13 @@ struct CDPropertyEntranceComponent { std::string groupType; }; -class CDPropertyEntranceComponentTable : public CDTable { +class CDPropertyEntranceComponentTable : public CDTable { public: - //! Constructor CDPropertyEntranceComponentTable(); - - //! Destructor - ~CDPropertyEntranceComponentTable(); - - //! Returns the table's name - /*! - \return The table name - */ - [[nodiscard]] std::string GetName() const override; - - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + // Queries the table with a custom "where" clause CDPropertyEntranceComponent GetByID(uint32_t id); - //! Gets all the entries in the table - /*! - \return The entries - */ + // Gets all the entries in the table [[nodiscard]] std::vector GetEntries() const { return entries; } private: std::vector entries{}; diff --git a/dDatabase/Tables/CDPropertyTemplateTable.cpp b/dDatabase/Tables/CDPropertyTemplateTable.cpp index 7dd118eb..4caa6dc5 100644 --- a/dDatabase/Tables/CDPropertyTemplateTable.cpp +++ b/dDatabase/Tables/CDPropertyTemplateTable.cpp @@ -17,10 +17,10 @@ CDPropertyTemplateTable::CDPropertyTemplateTable() { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM PropertyTemplate;"); while (!tableData.eof()) { auto entry = CDPropertyTemplate{ - static_cast(tableData.getIntField(0, -1)), - static_cast(tableData.getIntField(1, -1)), - static_cast(tableData.getIntField(2, -1)), - tableData.getStringField(3, "") + static_cast(tableData.getIntField("id", -1)), + static_cast(tableData.getIntField("mapID", -1)), + static_cast(tableData.getIntField("vendorMapID", -1)), + tableData.getStringField("spawnName", "") }; this->entries.push_back(entry); @@ -30,12 +30,6 @@ CDPropertyTemplateTable::CDPropertyTemplateTable() { tableData.finalize(); } -CDPropertyTemplateTable::~CDPropertyTemplateTable() = default; - -std::string CDPropertyTemplateTable::GetName() const { - return "PropertyTemplate"; -} - CDPropertyTemplate CDPropertyTemplateTable::GetByMapID(uint32_t mapID) { for (const auto& entry : entries) { if (entry.mapID == mapID) @@ -44,3 +38,4 @@ CDPropertyTemplate CDPropertyTemplateTable::GetByMapID(uint32_t mapID) { return defaultEntry; } + diff --git a/dDatabase/Tables/CDPropertyTemplateTable.h b/dDatabase/Tables/CDPropertyTemplateTable.h index a266932b..cb075dbf 100644 --- a/dDatabase/Tables/CDPropertyTemplateTable.h +++ b/dDatabase/Tables/CDPropertyTemplateTable.h @@ -8,12 +8,11 @@ struct CDPropertyTemplate { std::string spawnName; }; -class CDPropertyTemplateTable : public CDTable { +class CDPropertyTemplateTable : public CDTable { public: CDPropertyTemplateTable(); - ~CDPropertyTemplateTable(); - [[nodiscard]] std::string GetName() const override; + static const std::string GetTableName() { return "PropertyTemplate"; }; CDPropertyTemplate GetByMapID(uint32_t mapID); private: std::vector entries{}; diff --git a/dDatabase/Tables/CDProximityMonitorComponentTable.cpp b/dDatabase/Tables/CDProximityMonitorComponentTable.cpp index 092f3ca1..688de056 100644 --- a/dDatabase/Tables/CDProximityMonitorComponentTable.cpp +++ b/dDatabase/Tables/CDProximityMonitorComponentTable.cpp @@ -21,10 +21,10 @@ CDProximityMonitorComponentTable::CDProximityMonitorComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ProximityMonitorComponent"); while (!tableData.eof()) { CDProximityMonitorComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.Proximities = tableData.getStringField(1, ""); - entry.LoadOnClient = tableData.getIntField(2, -1); - entry.LoadOnServer = tableData.getIntField(3, -1); + entry.id = tableData.getIntField("id", -1); + entry.Proximities = tableData.getStringField("Proximities", ""); + entry.LoadOnClient = tableData.getIntField("LoadOnClient", -1); + entry.LoadOnServer = tableData.getIntField("LoadOnServer", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -33,14 +33,6 @@ CDProximityMonitorComponentTable::CDProximityMonitorComponentTable(void) { tableData.finalize(); } -//! Destructor -CDProximityMonitorComponentTable::~CDProximityMonitorComponentTable(void) {} - -//! Returns the table's name -std::string CDProximityMonitorComponentTable::GetName(void) const { - return "ProximityMonitorComponent"; -} - //! Queries the table with a custom "where" clause std::vector CDProximityMonitorComponentTable::Query(std::function predicate) { @@ -55,3 +47,4 @@ std::vector CDProximityMonitorComponentTable::Query std::vector CDProximityMonitorComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDProximityMonitorComponentTable.h b/dDatabase/Tables/CDProximityMonitorComponentTable.h index 007bb916..38b7d43b 100644 --- a/dDatabase/Tables/CDProximityMonitorComponentTable.h +++ b/dDatabase/Tables/CDProximityMonitorComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDProximityMonitorComponentTable.hpp - \brief Contains data for the ProximityMonitorComponent table - */ - - //! ProximityMonitorComponent Entry Struct struct CDProximityMonitorComponent { unsigned int id; std::string Proximities; @@ -16,36 +10,14 @@ struct CDProximityMonitorComponent { bool LoadOnServer; }; - -//! ProximityMonitorComponent table -class CDProximityMonitorComponentTable : public CDTable { +class CDProximityMonitorComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor CDProximityMonitorComponentTable(void); - - //! Destructor - ~CDProximityMonitorComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDRailActivatorComponent.cpp b/dDatabase/Tables/CDRailActivatorComponent.cpp index 4c141256..2ff8990d 100644 --- a/dDatabase/Tables/CDRailActivatorComponent.cpp +++ b/dDatabase/Tables/CDRailActivatorComponent.cpp @@ -6,35 +6,35 @@ CDRailActivatorComponentTable::CDRailActivatorComponentTable() { while (!tableData.eof()) { CDRailActivatorComponent entry; - entry.id = tableData.getIntField(0); + entry.id = tableData.getIntField("id", 0); - entry.startAnimation = GeneralUtils::ASCIIToUTF16(tableData.getStringField(1, "")); - entry.loopAnimation = GeneralUtils::ASCIIToUTF16(tableData.getStringField(2, "")); - entry.stopAnimation = GeneralUtils::ASCIIToUTF16(tableData.getStringField(3, "")); - entry.startSound = GeneralUtils::ASCIIToUTF16(tableData.getStringField(4, "")); - entry.loopSound = GeneralUtils::ASCIIToUTF16(tableData.getStringField(5, "")); - entry.stopSound = GeneralUtils::ASCIIToUTF16(tableData.getStringField(6, "")); + entry.startAnimation = GeneralUtils::ASCIIToUTF16(tableData.getStringField("startAnim", "")); + entry.loopAnimation = GeneralUtils::ASCIIToUTF16(tableData.getStringField("loopAnim", "")); + entry.stopAnimation = GeneralUtils::ASCIIToUTF16(tableData.getStringField("stopAnim", "")); + entry.startSound = GeneralUtils::ASCIIToUTF16(tableData.getStringField("startSound", "")); + entry.loopSound = GeneralUtils::ASCIIToUTF16(tableData.getStringField("loopSound", "")); + entry.stopSound = GeneralUtils::ASCIIToUTF16(tableData.getStringField("stopSound", "")); - std::string loopEffectString(tableData.getStringField(7, "")); + std::string loopEffectString(tableData.getStringField("effectIDs", "")); entry.loopEffectID = EffectPairFromString(loopEffectString); - entry.preconditions = tableData.getStringField(8, "-1"); + entry.preconditions = tableData.getStringField("preconditions", "-1"); - entry.playerCollision = tableData.getIntField(9, 0); + entry.playerCollision = tableData.getIntField("playerCollision", 0); - entry.cameraLocked = tableData.getIntField(10, 0); + entry.cameraLocked = tableData.getIntField("cameraLocked", 0); - std::string startEffectString(tableData.getStringField(11, "")); + std::string startEffectString(tableData.getStringField("StartEffectID", "")); entry.startEffectID = EffectPairFromString(startEffectString); - std::string stopEffectString(tableData.getStringField(12, "")); + std::string stopEffectString(tableData.getStringField("StopEffectID", "")); entry.stopEffectID = EffectPairFromString(stopEffectString); - entry.damageImmune = tableData.getIntField(13, 0); + entry.damageImmune = tableData.getIntField("DamageImmune", 0); - entry.noAggro = tableData.getIntField(14, 0); + entry.noAggro = tableData.getIntField("NoAggro", 0); - entry.showNameBillboard = tableData.getIntField(15, 0); + entry.showNameBillboard = tableData.getIntField("ShowNameBillboard", 0); m_Entries.push_back(entry); tableData.nextRow(); @@ -43,12 +43,6 @@ CDRailActivatorComponentTable::CDRailActivatorComponentTable() { tableData.finalize(); } -CDRailActivatorComponentTable::~CDRailActivatorComponentTable() = default; - -std::string CDRailActivatorComponentTable::GetName() const { - return "RailActivatorComponent"; -} - CDRailActivatorComponent CDRailActivatorComponentTable::GetEntryByID(int32_t id) const { for (const auto& entry : m_Entries) { if (entry.id == id) @@ -70,3 +64,4 @@ std::pair CDRailActivatorComponentTable::EffectPairFro return {}; } + diff --git a/dDatabase/Tables/CDRailActivatorComponent.h b/dDatabase/Tables/CDRailActivatorComponent.h index b8313e96..03dd0525 100644 --- a/dDatabase/Tables/CDRailActivatorComponent.h +++ b/dDatabase/Tables/CDRailActivatorComponent.h @@ -20,12 +20,10 @@ struct CDRailActivatorComponent { bool showNameBillboard; }; -class CDRailActivatorComponentTable : public CDTable { +class CDRailActivatorComponentTable : public CDTable { public: CDRailActivatorComponentTable(); - ~CDRailActivatorComponentTable(); - - std::string GetName() const override; + static const std::string GetTableName() { return "RailActivatorComponent"; }; [[nodiscard]] CDRailActivatorComponent GetEntryByID(int32_t id) const; [[nodiscard]] std::vector GetEntries() const; private: diff --git a/dDatabase/Tables/CDRarityTableTable.cpp b/dDatabase/Tables/CDRarityTableTable.cpp index 67878cd2..0b1212c0 100644 --- a/dDatabase/Tables/CDRarityTableTable.cpp +++ b/dDatabase/Tables/CDRarityTableTable.cpp @@ -21,10 +21,10 @@ CDRarityTableTable::CDRarityTableTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM RarityTable"); while (!tableData.eof()) { CDRarityTable entry; - entry.id = tableData.getIntField(0, -1); - entry.randmax = tableData.getFloatField(1, -1); - entry.rarity = tableData.getIntField(2, -1); - entry.RarityTableIndex = tableData.getIntField(3, -1); + entry.id = tableData.getIntField("id", -1); + entry.randmax = tableData.getFloatField("randmax", -1); + entry.rarity = tableData.getIntField("rarity", -1); + entry.RarityTableIndex = tableData.getIntField("RarityTableIndex", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -33,14 +33,6 @@ CDRarityTableTable::CDRarityTableTable(void) { tableData.finalize(); } -//! Destructor -CDRarityTableTable::~CDRarityTableTable(void) {} - -//! Returns the table's name -std::string CDRarityTableTable::GetName(void) const { - return "RarityTable"; -} - //! Queries the table with a custom "where" clause std::vector CDRarityTableTable::Query(std::function predicate) { @@ -55,3 +47,4 @@ std::vector CDRarityTableTable::Query(std::function& CDRarityTableTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDRarityTableTable.h b/dDatabase/Tables/CDRarityTableTable.h index 55cfec4b..592346ed 100644 --- a/dDatabase/Tables/CDRarityTableTable.h +++ b/dDatabase/Tables/CDRarityTableTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDRarityTableTable.hpp - \brief Contains data for the RarityTable table - */ - - //! RarityTable Entry Struct struct CDRarityTable { unsigned int id; float randmax; @@ -32,37 +26,15 @@ struct CDRarityTable { } }; - -//! RarityTable table -class CDRarityTableTable : public CDTable { +class CDRarityTableTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDRarityTableTable(void); - - //! Destructor - ~CDRarityTableTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDRarityTableTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ - const std::vector& GetEntries(void) const; - + const std::vector& GetEntries() const; }; diff --git a/dDatabase/Tables/CDRebuildComponentTable.cpp b/dDatabase/Tables/CDRebuildComponentTable.cpp index c4299b98..d5c386d1 100644 --- a/dDatabase/Tables/CDRebuildComponentTable.cpp +++ b/dDatabase/Tables/CDRebuildComponentTable.cpp @@ -21,16 +21,16 @@ CDRebuildComponentTable::CDRebuildComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM RebuildComponent"); while (!tableData.eof()) { CDRebuildComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.reset_time = tableData.getFloatField(1, -1.0f); - entry.complete_time = tableData.getFloatField(2, -1.0f); - entry.take_imagination = tableData.getIntField(3, -1); - entry.interruptible = tableData.getIntField(4, -1) == 1 ? true : false; - entry.self_activator = tableData.getIntField(5, -1) == 1 ? true : false; - entry.custom_modules = tableData.getStringField(6, ""); - entry.activityID = tableData.getIntField(7, -1); - entry.post_imagination_cost = tableData.getIntField(8, -1); - entry.time_before_smash = tableData.getFloatField(9, -1.0f); + entry.id = tableData.getIntField("id", -1); + entry.reset_time = tableData.getFloatField("reset_time", -1.0f); + entry.complete_time = tableData.getFloatField("complete_time", -1.0f); + entry.take_imagination = tableData.getIntField("take_imagination", -1); + entry.interruptible = tableData.getIntField("interruptible", -1) == 1 ? true : false; + entry.self_activator = tableData.getIntField("self_activator", -1) == 1 ? true : false; + entry.custom_modules = tableData.getStringField("custom_modules", ""); + entry.activityID = tableData.getIntField("activityID", -1); + entry.post_imagination_cost = tableData.getIntField("post_imagination_cost", -1); + entry.time_before_smash = tableData.getFloatField("time_before_smash", -1.0f); this->entries.push_back(entry); tableData.nextRow(); @@ -39,14 +39,6 @@ CDRebuildComponentTable::CDRebuildComponentTable(void) { tableData.finalize(); } -//! Destructor -CDRebuildComponentTable::~CDRebuildComponentTable(void) {} - -//! Returns the table's name -std::string CDRebuildComponentTable::GetName(void) const { - return "RebuildComponent"; -} - //! Queries the table with a custom "where" clause std::vector CDRebuildComponentTable::Query(std::function predicate) { @@ -61,3 +53,4 @@ std::vector CDRebuildComponentTable::Query(std::function CDRebuildComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDRebuildComponentTable.h b/dDatabase/Tables/CDRebuildComponentTable.h index bdc2da8a..db70a47d 100644 --- a/dDatabase/Tables/CDRebuildComponentTable.h +++ b/dDatabase/Tables/CDRebuildComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDRebuildComponentTable.hpp - \brief Contains data for the RebuildComponent table - */ - - //! RebuildComponent Struct struct CDRebuildComponent { unsigned int id; //!< The component Id float reset_time; //!< The reset time @@ -22,36 +16,15 @@ struct CDRebuildComponent { float time_before_smash; //!< The time before smash }; -//! ObjectSkills table -class CDRebuildComponentTable : public CDTable { +class CDRebuildComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDRebuildComponentTable(void); - - //! Destructor - ~CDRebuildComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDRebuildComponentTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ - std::vector GetEntries(void) const; - + std::vector GetEntries() const; }; diff --git a/dDatabase/Tables/CDRewardsTable.cpp b/dDatabase/Tables/CDRewardsTable.cpp index 99d56f26..55672add 100644 --- a/dDatabase/Tables/CDRewardsTable.cpp +++ b/dDatabase/Tables/CDRewardsTable.cpp @@ -4,12 +4,12 @@ CDRewardsTable::CDRewardsTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM Rewards"); while (!tableData.eof()) { CDRewards* entry = new CDRewards(); - entry->id = tableData.getIntField(0, -1); - entry->levelID = tableData.getIntField(1, -1); - entry->missionID = tableData.getIntField(2, -1); - entry->rewardType = tableData.getIntField(3, -1); - entry->value = tableData.getIntField(4, -1); - entry->count = tableData.getIntField(5, -1); + entry->id = tableData.getIntField("id", -1); + entry->levelID = tableData.getIntField("LevelID", -1); + entry->missionID = tableData.getIntField("MissionID", -1); + entry->rewardType = tableData.getIntField("RewardType", -1); + entry->value = tableData.getIntField("value", -1); + entry->count = tableData.getIntField("count", -1); m_entries.insert(std::make_pair(entry->id, entry)); tableData.nextRow(); @@ -26,10 +26,6 @@ CDRewardsTable::~CDRewardsTable(void) { m_entries.clear(); } -std::string CDRewardsTable::GetName(void) const { - return "Rewards"; -} - std::vector CDRewardsTable::GetByLevelID(uint32_t levelID) { std::vector result{}; for (const auto& e : m_entries) { @@ -38,3 +34,4 @@ std::vector CDRewardsTable::GetByLevelID(uint32_t levelID) { return result; } + diff --git a/dDatabase/Tables/CDRewardsTable.h b/dDatabase/Tables/CDRewardsTable.h index 2edfd2b3..2e079a83 100644 --- a/dDatabase/Tables/CDRewardsTable.h +++ b/dDatabase/Tables/CDRewardsTable.h @@ -11,12 +11,12 @@ struct CDRewards { int32_t count; }; -class CDRewardsTable : public CDTable { +class CDRewardsTable : public CDTable { public: - CDRewardsTable(void); - ~CDRewardsTable(void); + CDRewardsTable(); + ~CDRewardsTable(); - std::string GetName(void) const override; + static const std::string GetTableName() { return "Rewards"; }; std::vector GetByLevelID(uint32_t levelID); private: diff --git a/dDatabase/Tables/CDScriptComponentTable.cpp b/dDatabase/Tables/CDScriptComponentTable.cpp index b34f96f1..8050c139 100644 --- a/dDatabase/Tables/CDScriptComponentTable.cpp +++ b/dDatabase/Tables/CDScriptComponentTable.cpp @@ -18,9 +18,9 @@ CDScriptComponentTable::CDScriptComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ScriptComponent"); while (!tableData.eof()) { CDScriptComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.script_name = tableData.getStringField(1, ""); - entry.client_script_name = tableData.getStringField(2, ""); + entry.id = tableData.getIntField("id", -1); + entry.script_name = tableData.getStringField("script_name", ""); + entry.client_script_name = tableData.getStringField("client_script_name", ""); this->entries.insert(std::make_pair(entry.id, entry)); tableData.nextRow(); @@ -29,14 +29,6 @@ CDScriptComponentTable::CDScriptComponentTable(void) { tableData.finalize(); } -//! Destructor -CDScriptComponentTable::~CDScriptComponentTable(void) {} - -//! Returns the table's name -std::string CDScriptComponentTable::GetName(void) const { - return "ScriptComponent"; -} - const CDScriptComponent& CDScriptComponentTable::GetByID(unsigned int id) { std::map::iterator it = this->entries.find(id); if (it != this->entries.end()) { @@ -45,3 +37,4 @@ const CDScriptComponent& CDScriptComponentTable::GetByID(unsigned int id) { return m_ToReturnWhenNoneFound; } + diff --git a/dDatabase/Tables/CDScriptComponentTable.h b/dDatabase/Tables/CDScriptComponentTable.h index 92534cc0..77453939 100644 --- a/dDatabase/Tables/CDScriptComponentTable.h +++ b/dDatabase/Tables/CDScriptComponentTable.h @@ -3,44 +3,20 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDScriptComponentTable.hpp - \brief Contains data for the ScriptComponent table - */ - - //! ScriptComponent Struct struct CDScriptComponent { unsigned int id; //!< The component ID std::string script_name; //!< The script name std::string client_script_name; //!< The client script name }; -//! ObjectSkills table -class CDScriptComponentTable : public CDTable { +class CDScriptComponentTable : public CDTable { private: std::map entries; CDScriptComponent m_ToReturnWhenNoneFound; public: - //! Gets an entry by ID + CDScriptComponentTable(); + // Gets an entry by scriptID const CDScriptComponent& GetByID(unsigned int id); - - //! Constructor - CDScriptComponentTable(void); - - //! Destructor - ~CDScriptComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ - }; diff --git a/dDatabase/Tables/CDSkillBehaviorTable.cpp b/dDatabase/Tables/CDSkillBehaviorTable.cpp index e3986578..c5df78ef 100644 --- a/dDatabase/Tables/CDSkillBehaviorTable.cpp +++ b/dDatabase/Tables/CDSkillBehaviorTable.cpp @@ -23,25 +23,25 @@ CDSkillBehaviorTable::CDSkillBehaviorTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM SkillBehavior"); while (!tableData.eof()) { CDSkillBehavior entry; - entry.skillID = tableData.getIntField(0, -1); - UNUSED(entry.locStatus = tableData.getIntField(1, -1)); - entry.behaviorID = tableData.getIntField(2, -1); - entry.imaginationcost = tableData.getIntField(3, -1); - entry.cooldowngroup = tableData.getIntField(4, -1); - entry.cooldown = tableData.getFloatField(5, -1.0f); - UNUSED(entry.isNpcEditor = tableData.getIntField(6, -1) == 1 ? true : false); - UNUSED(entry.skillIcon = tableData.getIntField(7, -1)); - UNUSED(entry.oomSkillID = tableData.getStringField(8, "")); - UNUSED(entry.oomBehaviorEffectID = tableData.getIntField(9, -1)); - UNUSED(entry.castTypeDesc = tableData.getIntField(10, -1)); - UNUSED(entry.imBonusUI = tableData.getIntField(11, -1)); - UNUSED(entry.lifeBonusUI = tableData.getIntField(12, -1)); - UNUSED(entry.armorBonusUI = tableData.getIntField(13, -1)); - UNUSED(entry.damageUI = tableData.getIntField(14, -1)); - UNUSED(entry.hideIcon = tableData.getIntField(15, -1) == 1 ? true : false); - UNUSED(entry.localize = tableData.getIntField(16, -1) == 1 ? true : false); - UNUSED(entry.gate_version = tableData.getStringField(17, "")); - UNUSED(entry.cancelType = tableData.getIntField(18, -1)); + entry.skillID = tableData.getIntField("skillID", -1); + UNUSED(entry.locStatus = tableData.getIntField("locStatus", -1)); + entry.behaviorID = tableData.getIntField("behaviorID", -1); + entry.imaginationcost = tableData.getIntField("imaginationcost", -1); + entry.cooldowngroup = tableData.getIntField("cooldowngroup", -1); + entry.cooldown = tableData.getFloatField("cooldown", -1.0f); + UNUSED(entry.isNpcEditor = tableData.getIntField("isNpcEditor", -1) == 1 ? true : false); + UNUSED(entry.skillIcon = tableData.getIntField("skillIcon", -1)); + UNUSED(entry.oomSkillID = tableData.getStringField("oomSkillID", "")); + UNUSED(entry.oomBehaviorEffectID = tableData.getIntField("oomBehaviorEffectID", -1)); + UNUSED(entry.castTypeDesc = tableData.getIntField("castTypeDesc", -1)); + UNUSED(entry.imBonusUI = tableData.getIntField("imBonusUI", -1)); + UNUSED(entry.lifeBonusUI = tableData.getIntField("lifeBonusUI", -1)); + UNUSED(entry.armorBonusUI = tableData.getIntField("armorBonusUI", -1)); + UNUSED(entry.damageUI = tableData.getIntField("damageUI", -1)); + UNUSED(entry.hideIcon = tableData.getIntField("hideIcon", -1) == 1 ? true : false); + UNUSED(entry.localize = tableData.getIntField("localize", -1) == 1 ? true : false); + UNUSED(entry.gate_version = tableData.getStringField("gate_version", "")); + UNUSED(entry.cancelType = tableData.getIntField("cancelType", -1)); this->entries.insert(std::make_pair(entry.skillID, entry)); //this->entries.push_back(entry); @@ -51,24 +51,8 @@ CDSkillBehaviorTable::CDSkillBehaviorTable(void) { tableData.finalize(); } -//! Destructor -CDSkillBehaviorTable::~CDSkillBehaviorTable(void) {} - -//! Returns the table's name -std::string CDSkillBehaviorTable::GetName(void) const { - return "SkillBehavior"; -} - //! Queries the table with a custom "where" clause std::vector CDSkillBehaviorTable::Query(std::function predicate) { - - /*std::vector data = cpplinq::from(this->entries) - >> cpplinq::where(predicate) - >> cpplinq::to_vector(); - - return data;*/ - - //Logger::LogDebug("CDSkillBehaviorTable", "The 'Query' function is no longer working! Please use GetSkillByID instead!"); std::vector data; //So MSVC shuts up return data; } @@ -82,3 +66,4 @@ const CDSkillBehavior& CDSkillBehaviorTable::GetSkillByID(unsigned int skillID) return m_empty; } + diff --git a/dDatabase/Tables/CDSkillBehaviorTable.h b/dDatabase/Tables/CDSkillBehaviorTable.h index da8676db..eb3094e0 100644 --- a/dDatabase/Tables/CDSkillBehaviorTable.h +++ b/dDatabase/Tables/CDSkillBehaviorTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDSkillBehaviorTable.hpp - \brief Contains data for the SkillBehavior table - */ - - //! ZoneTable Struct struct CDSkillBehavior { unsigned int skillID; //!< The Skill ID of the skill UNUSED(unsigned int locStatus); //!< ?? @@ -31,33 +25,17 @@ struct CDSkillBehavior { UNUSED(unsigned int cancelType); //!< The cancel type (?) }; -//! SkillBehavior table -class CDSkillBehaviorTable : public CDTable { +class CDSkillBehaviorTable : public CDTable { private: std::map entries; CDSkillBehavior m_empty; public: - - //! Constructor - CDSkillBehaviorTable(void); - - //! Destructor - ~CDSkillBehaviorTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDSkillBehaviorTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets an entry by ID + // Gets an entry by skillID const CDSkillBehavior& GetSkillByID(unsigned int skillID); }; diff --git a/dDatabase/Tables/CDTable.h b/dDatabase/Tables/CDTable.h index cd05782d..e4c11fb9 100644 --- a/dDatabase/Tables/CDTable.h +++ b/dDatabase/Tables/CDTable.h @@ -1,9 +1,9 @@ #pragma once -// Custom Classes -#include "../CDClientDatabase.h" +#include "CDClientDatabase.h" +#include "Singleton.h" +#include "DluAssert.h" -// C++ #include #include #include @@ -16,26 +16,29 @@ #endif #include "cpplinq.hpp" +// Used for legacy +#define UNUSED(x) + +// Enable this to skip some unused columns in some tables +#define UNUSED_COLUMN(v) + #pragma warning (disable : 4244) //Disable double to float conversion warnings #pragma warning (disable : 4715) //Disable "not all control paths return a value" -#if defined(__unix) || defined(__APPLE__) -//For Linux: -typedef __int64_t __int64; -#endif - -/*! - \file CDTable.hpp - \brief A virtual class for CDClient Tables - */ - - //! The base class for all CD tables -class CDTable { -public: - - //! Returns the table's name - /*! - \return The table name - */ - virtual std::string GetName() const = 0; +template +class CDTable : public Singleton { +protected: + virtual ~CDTable() = default; +}; + +template +class LookupResult { + typedef std::pair DataType; +public: + LookupResult() { m_data.first = T(); m_data.second = false; }; + LookupResult(T& data) { m_data.first = data; m_data.second = true; }; + inline const T& Data() { return m_data.first; }; + inline const bool& FoundData() { return m_data.second; }; +private: + DataType m_data; }; diff --git a/dDatabase/Tables/CDVendorComponentTable.cpp b/dDatabase/Tables/CDVendorComponentTable.cpp index f1dd96db..17989dfb 100644 --- a/dDatabase/Tables/CDVendorComponentTable.cpp +++ b/dDatabase/Tables/CDVendorComponentTable.cpp @@ -21,11 +21,11 @@ CDVendorComponentTable::CDVendorComponentTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM VendorComponent"); while (!tableData.eof()) { CDVendorComponent entry; - entry.id = tableData.getIntField(0, -1); - entry.buyScalar = tableData.getFloatField(1, -1.0f); - entry.sellScalar = tableData.getFloatField(2, -1.0f); - entry.refreshTimeSeconds = tableData.getFloatField(3, -1.0f); - entry.LootMatrixIndex = tableData.getIntField(4, -1); + entry.id = tableData.getIntField("id", -1); + entry.buyScalar = tableData.getFloatField("buyScalar", 0.0f); + entry.sellScalar = tableData.getFloatField("sellScalar", -1.0f); + entry.refreshTimeSeconds = tableData.getFloatField("refreshTimeSeconds", -1.0f); + entry.LootMatrixIndex = tableData.getIntField("LootMatrixIndex", -1); this->entries.push_back(entry); tableData.nextRow(); @@ -34,14 +34,6 @@ CDVendorComponentTable::CDVendorComponentTable(void) { tableData.finalize(); } -//! Destructor -CDVendorComponentTable::~CDVendorComponentTable(void) {} - -//! Returns the table's name -std::string CDVendorComponentTable::GetName(void) const { - return "VendorComponent"; -} - //! Queries the table with a custom "where" clause std::vector CDVendorComponentTable::Query(std::function predicate) { @@ -56,3 +48,4 @@ std::vector CDVendorComponentTable::Query(std::function CDVendorComponentTable::GetEntries(void) const { return this->entries; } + diff --git a/dDatabase/Tables/CDVendorComponentTable.h b/dDatabase/Tables/CDVendorComponentTable.h index 2e0fb9e4..f2666d7e 100644 --- a/dDatabase/Tables/CDVendorComponentTable.h +++ b/dDatabase/Tables/CDVendorComponentTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDVendorComponentTable.hpp - \brief Contains data for the VendorComponent table - */ - - //! VendorComponent Struct struct CDVendorComponent { unsigned int id; //!< The Component ID float buyScalar; //!< Buy Scalar (what does that mean?) @@ -17,36 +11,15 @@ struct CDVendorComponent { unsigned int LootMatrixIndex; //!< LootMatrixIndex of the vendor's items }; -//! VendorComponent table -class CDVendorComponentTable : public CDTable { +class CDVendorComponentTable : public CDTable { private: std::vector entries; public: - - //! Constructor - CDVendorComponentTable(void); - - //! Destructor - ~CDVendorComponentTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a custom "where" clause - /*! - \param predicate The predicate - */ + CDVendorComponentTable(); + // Queries the table with a custom "where" clause std::vector Query(std::function predicate); - //! Gets all the entries in the table - /*! - \return The entries - */ std::vector GetEntries(void) const; - }; diff --git a/dDatabase/Tables/CDZoneTableTable.cpp b/dDatabase/Tables/CDZoneTableTable.cpp index e172f79a..bafbf8fe 100644 --- a/dDatabase/Tables/CDZoneTableTable.cpp +++ b/dDatabase/Tables/CDZoneTableTable.cpp @@ -18,33 +18,33 @@ CDZoneTableTable::CDZoneTableTable(void) { auto tableData = CDClientDatabase::ExecuteQuery("SELECT * FROM ZoneTable"); while (!tableData.eof()) { CDZoneTable entry; - entry.zoneID = tableData.getIntField(0, -1); - entry.locStatus = tableData.getIntField(1, -1); - entry.zoneName = tableData.getStringField(2, ""); - entry.scriptID = tableData.getIntField(3, -1); - entry.ghostdistance_min = tableData.getFloatField(4, -1.0f); - entry.ghostdistance = tableData.getFloatField(5, -1.0f); - entry.population_soft_cap = tableData.getIntField(6, -1); - entry.population_hard_cap = tableData.getIntField(7, -1); - UNUSED(entry.DisplayDescription = tableData.getStringField(8, "")); - UNUSED(entry.mapFolder = tableData.getStringField(9, "")); - entry.smashableMinDistance = tableData.getFloatField(10, -1.0f); - entry.smashableMaxDistance = tableData.getFloatField(11, -1.0f); - UNUSED(entry.mixerProgram = tableData.getStringField(12, "")); - UNUSED(entry.clientPhysicsFramerate = tableData.getStringField(13, "")); - UNUSED(entry.serverPhysicsFramerate = tableData.getStringField(14, "")); - entry.zoneControlTemplate = tableData.getIntField(15, -1); - entry.widthInChunks = tableData.getIntField(16, -1); - entry.heightInChunks = tableData.getIntField(17, -1); - entry.petsAllowed = tableData.getIntField(18, -1) == 1 ? true : false; - entry.localize = tableData.getIntField(19, -1) == 1 ? true : false; - entry.fZoneWeight = tableData.getFloatField(20, -1.0f); - UNUSED(entry.thumbnail = tableData.getStringField(21, "")); - entry.PlayerLoseCoinsOnDeath = tableData.getIntField(22, -1) == 1 ? true : false; - UNUSED(entry.disableSaveLoc = tableData.getIntField(23, -1) == 1 ? true : false); - entry.teamRadius = tableData.getFloatField(24, -1.0f); - UNUSED(entry.gate_version = tableData.getStringField(25, "")); - UNUSED(entry.mountsAllowed = tableData.getIntField(26, -1) == 1 ? true : false); + entry.zoneID = tableData.getIntField("zoneID", -1); + entry.locStatus = tableData.getIntField("locStatus", -1); + entry.zoneName = tableData.getStringField("zoneName", ""); + entry.scriptID = tableData.getIntField("scriptID", -1); + entry.ghostdistance_min = tableData.getFloatField("ghostdistance_min", -1.0f); + entry.ghostdistance = tableData.getFloatField("ghostdistance", -1.0f); + entry.population_soft_cap = tableData.getIntField("population_soft_cap", -1); + entry.population_hard_cap = tableData.getIntField("population_hard_cap", -1); + UNUSED(entry.DisplayDescription = tableData.getStringField("DisplayDescription", "")); + UNUSED(entry.mapFolder = tableData.getStringField("mapFolder", "")); + entry.smashableMinDistance = tableData.getFloatField("smashableMinDistance", -1.0f); + entry.smashableMaxDistance = tableData.getFloatField("smashableMaxDistance", -1.0f); + UNUSED(entry.mixerProgram = tableData.getStringField("mixerProgram", "")); + UNUSED(entry.clientPhysicsFramerate = tableData.getStringField("clientPhysicsFramerate", "")); + UNUSED(entry.serverPhysicsFramerate = tableData.getStringField("serverPhysicsFramerate", "")); + entry.zoneControlTemplate = tableData.getIntField("zoneControlTemplate", -1); + entry.widthInChunks = tableData.getIntField("widthInChunks", -1); + entry.heightInChunks = tableData.getIntField("heightInChunks", -1); + entry.petsAllowed = tableData.getIntField("petsAllowed", -1) == 1 ? true : false; + entry.localize = tableData.getIntField("localize", -1) == 1 ? true : false; + entry.fZoneWeight = tableData.getFloatField("fZoneWeight", -1.0f); + UNUSED(entry.thumbnail = tableData.getStringField("thumbnail", "")); + entry.PlayerLoseCoinsOnDeath = tableData.getIntField("PlayerLoseCoinsOnDeath", -1) == 1 ? true : false; + UNUSED(entry.disableSaveLoc = tableData.getIntField("disableSaveLoc", -1) == 1 ? true : false); + entry.teamRadius = tableData.getFloatField("teamRadius", -1.0f); + UNUSED(entry.gate_version = tableData.getStringField("gate_version", "")); + UNUSED(entry.mountsAllowed = tableData.getIntField("mountsAllowed", -1) == 1 ? true : false); this->m_Entries.insert(std::make_pair(entry.zoneID, entry)); tableData.nextRow(); @@ -53,14 +53,6 @@ CDZoneTableTable::CDZoneTableTable(void) { tableData.finalize(); } -//! Destructor -CDZoneTableTable::~CDZoneTableTable(void) {} - -//! Returns the table's name -std::string CDZoneTableTable::GetName(void) const { - return "ZoneTable"; -} - //! Queries the table with a zoneID to find. const CDZoneTable* CDZoneTableTable::Query(unsigned int zoneID) { const auto& iter = m_Entries.find(zoneID); @@ -71,3 +63,4 @@ const CDZoneTable* CDZoneTableTable::Query(unsigned int zoneID) { return nullptr; } + diff --git a/dDatabase/Tables/CDZoneTableTable.h b/dDatabase/Tables/CDZoneTableTable.h index c115a38a..f844fd25 100644 --- a/dDatabase/Tables/CDZoneTableTable.h +++ b/dDatabase/Tables/CDZoneTableTable.h @@ -3,12 +3,6 @@ // Custom Classes #include "CDTable.h" -/*! - \file CDZoneTableTable.hpp - \brief Contains data for the ZoneTable table - */ - - //! ZoneTable Struct struct CDZoneTable { unsigned int zoneID; //!< The Zone ID of the object unsigned int locStatus; //!< The Locale Status(?) @@ -39,28 +33,13 @@ struct CDZoneTable { UNUSED(bool mountsAllowed); //!< Whether or not mounts are allowed }; -//! ZoneTable table -class CDZoneTableTable : public CDTable { +class CDZoneTableTable : public CDTable { private: std::map m_Entries; public: + CDZoneTableTable(); - //! Constructor - CDZoneTableTable(void); - - //! Destructor - ~CDZoneTableTable(void); - - //! Returns the table's name - /*! - \return The table name - */ - std::string GetName(void) const override; - - //! Queries the table with a zoneID to find. - /*! - \param id The zoneID - */ + // Queries the table with a zoneID to find. const CDZoneTable* Query(unsigned int zoneID); }; diff --git a/dGame/CMakeLists.txt b/dGame/CMakeLists.txt index 6b31802e..80f16042 100644 --- a/dGame/CMakeLists.txt +++ b/dGame/CMakeLists.txt @@ -44,13 +44,19 @@ foreach(file ${DGAME_DMISSION_SOURCES}) set(DGAME_SOURCES ${DGAME_SOURCES} "dMission/${file}") endforeach() +add_subdirectory(dPropertyBehaviors) + +foreach(file ${DGAME_DPROPERTYBEHAVIORS_SOURCES}) + set(DGAME_SOURCES ${DGAME_SOURCES} "dPropertyBehaviors/${file}") +endforeach() + add_subdirectory(dUtilities) foreach(file ${DGAME_DUTILITIES_SOURCES}) set(DGAME_SOURCES ${DGAME_SOURCES} "dUtilities/${file}") endforeach() -foreach(file ${DSCRIPT_SOURCES}) +foreach(file ${DSCRIPTS_SOURCES}) set(DGAME_SOURCES ${DGAME_SOURCES} "${PROJECT_SOURCE_DIR}/dScripts/${file}") endforeach() diff --git a/dGame/Character.cpp b/dGame/Character.cpp index 440c30c1..c5602bf2 100644 --- a/dGame/Character.cpp +++ b/dGame/Character.cpp @@ -15,6 +15,12 @@ #include "Zone.h" #include "ChatPackets.h" #include "Inventory.h" +#include "InventoryComponent.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" +#include "eObjectBits.h" +#include "eGameMasterLevel.h" +#include "ePlayerFlag.h" Character::Character(uint32_t id, User* parentUser) { //First load the name, etc: @@ -33,7 +39,7 @@ Character::Character(uint32_t id, User* parentUser) { m_UnapprovedName = res->getString(2).c_str(); m_NameRejected = res->getBoolean(3); m_PropertyCloneID = res->getUInt(4); - m_PermissionMap = static_cast(res->getUInt64(5)); + m_PermissionMap = static_cast(res->getUInt64(5)); } delete res; @@ -65,8 +71,8 @@ Character::Character(uint32_t id, User* parentUser) { //Set our objectID: m_ObjectID = m_ID; - m_ObjectID = GeneralUtils::SetBit(m_ObjectID, OBJECT_BIT_CHARACTER); - m_ObjectID = GeneralUtils::SetBit(m_ObjectID, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(m_ObjectID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(m_ObjectID, eObjectBits::PERSISTENT); m_ParentUser = parentUser; m_OurEntity = nullptr; @@ -92,7 +98,7 @@ void Character::UpdateFromDatabase() { m_UnapprovedName = res->getString(2).c_str(); m_NameRejected = res->getBoolean(3); m_PropertyCloneID = res->getUInt(4); - m_PermissionMap = static_cast(res->getUInt64(5)); + m_PermissionMap = static_cast(res->getUInt64(5)); } delete res; @@ -124,8 +130,8 @@ void Character::UpdateFromDatabase() { //Set our objectID: m_ObjectID = m_ID; - m_ObjectID = GeneralUtils::SetBit(m_ObjectID, OBJECT_BIT_CHARACTER); - m_ObjectID = GeneralUtils::SetBit(m_ObjectID, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(m_ObjectID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(m_ObjectID, eObjectBits::PERSISTENT); m_OurEntity = nullptr; m_BuildMode = false; @@ -201,7 +207,9 @@ void Character::DoQuickXMLDataParse() { tinyxml2::XMLElement* character = m_Doc->FirstChildElement("obj")->FirstChildElement("char"); if (character) { character->QueryAttribute("cc", &m_Coins); - character->QueryAttribute("gm", &m_GMLevel); + int32_t gm_level = 0; + character->QueryAttribute("gm", &gm_level); + m_GMLevel = static_cast(gm_level); uint64_t lzidConcat = 0; if (character->FindAttribute("lzid")) { @@ -233,7 +241,7 @@ void Character::DoQuickXMLDataParse() { //To try and fix the AG landing into: if (m_ZoneID == 1000 && Game::server->GetZoneID() == 1100) { //sneakily insert our position: - auto pos = dZoneManager::Instance()->GetZone()->GetSpawnPos(); + auto pos = Game::zoneManager->GetZone()->GetSpawnPos(); character->SetAttribute("lzx", pos.x); character->SetAttribute("lzy", pos.y); character->SetAttribute("lzz", pos.z); @@ -264,14 +272,17 @@ void Character::DoQuickXMLDataParse() { if (flags) { auto* currentChild = flags->FirstChildElement(); while (currentChild) { - uint32_t index = 0; - uint64_t value = 0; const auto* temp = currentChild->Attribute("v"); + const auto* id = currentChild->Attribute("id"); + if (temp && id) { + uint32_t index = 0; + uint64_t value = 0; - index = std::stoul(currentChild->Attribute("id")); - value = std::stoull(temp); + index = std::stoul(id); + value = std::stoull(temp); - m_PlayerFlags.insert(std::make_pair(index, value)); + m_PlayerFlags.insert(std::make_pair(index, value)); + } currentChild = currentChild->NextSiblingElement(); } } @@ -279,13 +290,13 @@ void Character::DoQuickXMLDataParse() { void Character::UnlockEmote(int emoteID) { m_UnlockedEmotes.push_back(emoteID); - GameMessages::SendSetEmoteLockState(EntityManager::Instance()->GetEntity(m_ObjectID), false, emoteID); + GameMessages::SendSetEmoteLockState(Game::entityManager->GetEntity(m_ObjectID), false, emoteID); } void Character::SetBuildMode(bool buildMode) { m_BuildMode = buildMode; - auto* controller = dZoneManager::Instance()->GetZoneControlObject(); + auto* controller = Game::zoneManager->GetZoneControlObject(); controller->OnFireEventServerSide(m_OurEntity, buildMode ? "OnBuildModeEnter" : "OnBuildModeLeave"); } @@ -298,10 +309,10 @@ void Character::SaveXMLToDatabase() { tinyxml2::XMLElement* character = m_Doc->FirstChildElement("obj")->FirstChildElement("char"); if (character) { - character->SetAttribute("gm", m_GMLevel); + character->SetAttribute("gm", static_cast(m_GMLevel)); character->SetAttribute("cc", m_Coins); - auto zoneInfo = dZoneManager::Instance()->GetZone()->GetZoneID(); + auto zoneInfo = Game::zoneManager->GetZone()->GetZoneID(); // lzid garbage, binary concat of zoneID, zoneInstance and zoneClone if (zoneInfo.GetMapID() != 0 && zoneInfo.GetCloneID() == 0) { uint64_t lzidConcat = zoneInfo.GetCloneID(); @@ -351,16 +362,48 @@ void Character::SaveXMLToDatabase() { flags->LinkEndChild(f); } + // Prevents the news feed from showing up on world transfers + if (GetPlayerFlag(ePlayerFlag::IS_NEWS_SCREEN_VISIBLE)) { + auto* s = m_Doc->NewElement("s"); + s->SetAttribute("si", ePlayerFlag::IS_NEWS_SCREEN_VISIBLE); + flags->LinkEndChild(s); + } + SaveXmlRespawnCheckpoints(); //Call upon the entity to update our xmlDoc: if (!m_OurEntity) { - Game::logger->Log("Character", "We didn't have an entity set while saving! CHARACTER WILL NOT BE SAVED!"); + Game::logger->Log("Character", "%i:%s didn't have an entity set while saving! CHARACTER WILL NOT BE SAVED!", this->GetID(), this->GetName().c_str()); return; } m_OurEntity->UpdateXMLDoc(m_Doc); + WriteToDatabase(); + + //For metrics, log the time it took to save: + auto end = std::chrono::system_clock::now(); + std::chrono::duration elapsed = end - start; + Game::logger->Log("Character", "%i:%s Saved character to Database in: %fs", this->GetID(), this->GetName().c_str(), elapsed.count()); +} + +void Character::SetIsNewLogin() { + // If we dont have a flag element, then we cannot have a s element as a child of flag. + auto* flags = m_Doc->FirstChildElement("obj")->FirstChildElement("flag"); + if (!flags) return; + + auto* currentChild = flags->FirstChildElement(); + while (currentChild) { + if (currentChild->Attribute("si")) { + flags->DeleteChild(currentChild); + Game::logger->Log("Character", "Removed isLoggedIn flag from character %i:%s, saving character to database", GetID(), GetName().c_str()); + WriteToDatabase(); + } + currentChild = currentChild->NextSiblingElement(); + } +} + +void Character::WriteToDatabase() { //Dump our xml into m_XMLData: auto* printer = new tinyxml2::XMLPrinter(0, true, 0); m_Doc->Print(printer); @@ -372,12 +415,6 @@ void Character::SaveXMLToDatabase() { stmt->setUInt(2, m_ID); stmt->execute(); delete stmt; - - //For metrics, log the time it took to save: - auto end = std::chrono::system_clock::now(); - std::chrono::duration elapsed = end - start; - Game::logger->Log("Character", "Saved character to Database in: %fs", elapsed.count()); - delete printer; } @@ -387,13 +424,13 @@ void Character::SetPlayerFlag(const uint32_t flagId, const bool value) { if (value) { // Update the mission component: - auto* player = EntityManager::Instance()->GetEntity(m_ObjectID); + auto* player = Game::entityManager->GetEntity(m_ObjectID); if (player != nullptr) { auto* missionComponent = player->GetComponent(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_PLAYER_FLAG, flagId); + missionComponent->Progress(eMissionTaskType::PLAYER_FLAG, flagId); } } } @@ -445,8 +482,8 @@ bool Character::GetPlayerFlag(const uint32_t flagId) const { void Character::SetRetroactiveFlags() { // Retroactive check for if player has joined a faction to set their 'joined a faction' flag to true. - if (GetPlayerFlag(ePlayerFlags::VENTURE_FACTION) || GetPlayerFlag(ePlayerFlags::ASSEMBLY_FACTION) || GetPlayerFlag(ePlayerFlags::PARADOX_FACTION) || GetPlayerFlag(ePlayerFlags::SENTINEL_FACTION)) { - SetPlayerFlag(ePlayerFlags::JOINED_A_FACTION, true); + if (GetPlayerFlag(ePlayerFlag::VENTURE_FACTION) || GetPlayerFlag(ePlayerFlag::ASSEMBLY_FACTION) || GetPlayerFlag(ePlayerFlag::PARADOX_FACTION) || GetPlayerFlag(ePlayerFlag::SENTINEL_FACTION)) { + SetPlayerFlag(ePlayerFlag::JOINED_A_FACTION, true); } } @@ -505,24 +542,24 @@ void Character::OnZoneLoad() { if (missionComponent != nullptr) { // Fix the monument race flag - if (missionComponent->GetMissionState(319) >= MissionState::MISSION_STATE_READY_TO_COMPLETE) { - SetPlayerFlag(33, true); + if (missionComponent->GetMissionState(319) >= eMissionState::READY_TO_COMPLETE) { + SetPlayerFlag(ePlayerFlag::AG_FINISH_LINE_BUILT, true); } } const auto maxGMLevel = m_ParentUser->GetMaxGMLevel(); // This does not apply to the GMs - if (maxGMLevel > GAME_MASTER_LEVEL_CIVILIAN) { + if (maxGMLevel > eGameMasterLevel::CIVILIAN) { return; } /** * Restrict old character to 1 million coins */ - if (HasPermission(PermissionMap::Old)) { + if (HasPermission(ePermissionMap::Old)) { if (GetCoins() > 1000000) { - SetCoins(1000000, eLootSourceType::LOOT_SOURCE_NONE); + SetCoins(1000000, eLootSourceType::NONE); } } @@ -538,11 +575,11 @@ void Character::OnZoneLoad() { } } -PermissionMap Character::GetPermissionMap() const { +ePermissionMap Character::GetPermissionMap() const { return m_PermissionMap; } -bool Character::HasPermission(PermissionMap permission) const { +bool Character::HasPermission(ePermissionMap permission) const { return (static_cast(m_PermissionMap) & static_cast(permission)) != 0; } @@ -565,7 +602,7 @@ void Character::SetCoins(int64_t newCoins, eLootSourceType lootSource) { m_Coins = newCoins; - GameMessages::SendSetCurrency(EntityManager::Instance()->GetEntity(m_ObjectID), m_Coins, 0, 0, 0, 0, true, lootSource); + GameMessages::SendSetCurrency(Game::entityManager->GetEntity(m_ObjectID), m_Coins, 0, 0, 0, 0, true, lootSource); } bool Character::HasBeenToWorld(LWOMAPID mapID) const { @@ -589,3 +626,21 @@ void Character::SendMuteNotice() const { ChatPackets::SendSystemMessage(GetEntity()->GetSystemAddress(), u"You are muted until " + timeStr); } + +void Character::SetBillboardVisible(bool visible) { + if (m_BillboardVisible == visible) return; + m_BillboardVisible = visible; + + GameMessages::SendSetNamebillboardState(UNASSIGNED_SYSTEM_ADDRESS, m_OurEntity->GetObjectID()); + + if (!visible) return; + + // The GameMessage we send for turning the nameplate off just deletes the BillboardSubcomponent from the parent component. + // Because that same message does not allow for custom parameters, we need to create the BillboardSubcomponent a different way + // This workaround involves sending an unrelated GameMessage that does not apply to player entites, + // but forces the client to create the necessary SubComponent that controls the billboard. + GameMessages::SendShowBillboardInteractIcon(UNASSIGNED_SYSTEM_ADDRESS, m_OurEntity->GetObjectID()); + + // Now turn off the billboard for the owner. + GameMessages::SendSetNamebillboardState(m_OurEntity->GetSystemAddress(), m_OurEntity->GetObjectID()); +} diff --git a/dGame/Character.h b/dGame/Character.h index 52bff83d..79ce0c0c 100644 --- a/dGame/Character.h +++ b/dGame/Character.h @@ -9,11 +9,14 @@ #include "NiPoint3.h" #include "NiQuaternion.h" -#include "PermissionMap.h" +#include "ePermissionMap.h" class User; struct Packet; class Entity; +enum class ePermissionMap : uint64_t; +enum class eGameMasterLevel : uint8_t; +enum class eLootSourceType : uint32_t; /** * Meta information about a character, like their name and style @@ -23,6 +26,10 @@ public: Character(uint32_t id, User* parentUser); ~Character(); + /** + * Write the current m_Doc to the database for saving. + */ + void WriteToDatabase(); void SaveXMLToDatabase(); void UpdateFromDatabase(); @@ -32,6 +39,15 @@ public: const std::string& GetXMLData() const { return m_XMLData; } tinyxml2::XMLDocument* GetXMLDoc() const { return m_Doc; } + /** + * Out of abundance of safety and clarity of what this saves, this is its own function. + * + * Clears the s element from the flag element and saves the xml to the database. Used to prevent the news + * feed from showing up on world transfers. + * + */ + void SetIsNewLogin(); + /** * Gets the database ID of the character * @return the database ID of the character @@ -294,13 +310,13 @@ public: * Gets the GM level of the character * @return the GM level */ - int32_t GetGMLevel() const { return m_GMLevel; } + eGameMasterLevel GetGMLevel() const { return m_GMLevel; } /** * Sets the GM level of the character * @param value the GM level to set */ - void SetGMLevel(uint8_t value) { m_GMLevel = value; } + void SetGMLevel(eGameMasterLevel value) { m_GMLevel = value; } /** * Gets the current amount of coins of the character @@ -372,14 +388,14 @@ public: * Gets the permissions of the character, determining what actions a character may do * @return the permissions for this character */ - PermissionMap GetPermissionMap() const; + ePermissionMap GetPermissionMap() const; /** * Check if this character has a certain permission * @param permission the ID of the permission to check for * @return whether the character has the specified permission */ - bool HasPermission(PermissionMap permission) const; + bool HasPermission(ePermissionMap permission) const; /** * Gets all the emotes this character has unlocked so far @@ -427,7 +443,7 @@ public: /** * @brief Get the flying state - * @return value of the flying state + * @return value of the flying state */ bool GetIsFlying() { return m_IsFlying; } @@ -437,6 +453,10 @@ public: */ void SetIsFlying(bool isFlying) { m_IsFlying = isFlying; } + bool GetBillboardVisible() { return m_BillboardVisible; } + + void SetBillboardVisible(bool visible); + private: /** * The ID of this character. First 32 bits of the ObjectID. @@ -463,12 +483,12 @@ private: * * @see eGameMasterLevel */ - int32_t m_GMLevel; + eGameMasterLevel m_GMLevel; /** * Bitmap of permission attributes this character has. */ - PermissionMap m_PermissionMap; + ePermissionMap m_PermissionMap; /** * The default name of this character @@ -638,6 +658,11 @@ private: */ bool m_IsFlying = false; + /** + * True if billboard (referred to as nameplate for end users) is visible, false otherwise + */ + bool m_BillboardVisible = true; + /** * Queries the character XML and updates all the fields of this object * NOTE: quick as there's no DB lookups diff --git a/dGame/Entity.cpp b/dGame/Entity.cpp index 52eddd06..e3bacb8b 100644 --- a/dGame/Entity.cpp +++ b/dGame/Entity.cpp @@ -17,6 +17,14 @@ #include "UserManager.h" #include "dpWorld.h" #include "Player.h" +#include "LUTriggers.h" +#include "User.h" +#include "EntityTimer.h" +#include "EntityCallbackTimer.h" +#include "Loot.h" +#include "eMissionTaskType.h" +#include "eTriggerEventType.h" +#include "eObjectBits.h" //Component includes: #include "Component.h" @@ -43,6 +51,7 @@ #include "BuildBorderComponent.h" #include "MovementAIComponent.h" #include "VendorComponent.h" +#include "DonationVendorComponent.h" #include "RocketLaunchpadControlComponent.h" #include "PropertyComponent.h" #include "BaseCombatAIComponent.h" @@ -62,15 +71,30 @@ #include "ShootingGalleryComponent.h" #include "RailActivatorComponent.h" #include "LUPExhibitComponent.h" +#include "TriggerComponent.h" +#include "eGameMasterLevel.h" +#include "eReplicaComponentType.h" +#include "eReplicaPacketType.h" + +// Table includes +#include "CDComponentsRegistryTable.h" +#include "CDCurrencyTableTable.h" +#include "CDMovementAIComponentTable.h" +#include "CDProximityMonitorComponentTable.h" +#include "CDRebuildComponentTable.h" +#include "CDObjectSkillsTable.h" +#include "CDObjectsTable.h" +#include "CDScriptComponentTable.h" +#include "CDSkillBehaviorTable.h" +#include "CDZoneTableTable.h" Entity::Entity(const LWOOBJID& objectID, EntityInfo info, Entity* parentEntity) { m_ObjectID = objectID; m_TemplateID = info.lot; m_ParentEntity = parentEntity; m_Character = nullptr; - m_GMLevel = 0; + m_GMLevel = eGameMasterLevel::CIVILIAN; m_CollectibleID = 0; - m_Trigger = nullptr; //new LUTriggers::Trigger(); m_NetworkID = 0; m_Groups = {}; m_OwnerOverride = LWOOBJID_EMPTY; @@ -126,30 +150,9 @@ void Entity::Initialize() { * Setup trigger */ - const auto triggerName = GetVarAsString(u"trigger_id"); + const auto triggerInfo = GetVarAsString(u"trigger_id"); - if (!triggerName.empty()) { - std::stringstream ss(triggerName); - std::vector tokens; - std::string token; - while (std::getline(ss, token, ':')) { - tokens.push_back(token); - } - - uint32_t sceneID = std::stoi(tokens[0]); - uint32_t triggerID = std::stoi(tokens[1]); - - if (m_Trigger != nullptr) { - delete m_Trigger; - m_Trigger = nullptr; - } - - m_Trigger = dZoneManager::Instance()->GetZone()->GetTrigger(sceneID, triggerID); - - if (m_Trigger == nullptr) { - m_Trigger = new LUTriggers::Trigger(); - } - } + if (!triggerInfo.empty()) m_Components.emplace(eReplicaComponentType::TRIGGER, new TriggerComponent(this, triggerInfo)); /** * Setup groups @@ -171,30 +174,30 @@ void Entity::Initialize() { } // Get the registry table - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); /** * Special case for BBB models. They have components not corresponding to the registry. */ if (m_TemplateID == 14) { - const auto simplePhysicsComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SIMPLE_PHYSICS); + const auto simplePhysicsComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SIMPLE_PHYSICS); SimplePhysicsComponent* comp = new SimplePhysicsComponent(simplePhysicsComponentID, this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_SIMPLE_PHYSICS, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::SIMPLE_PHYSICS, comp)); ModelComponent* modelcomp = new ModelComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_MODEL, modelcomp)); + m_Components.insert(std::make_pair(eReplicaComponentType::MODEL, modelcomp)); RenderComponent* render = new RenderComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_RENDER, render)); + m_Components.insert(std::make_pair(eReplicaComponentType::RENDER, render)); auto destroyableComponent = new DestroyableComponent(this); destroyableComponent->SetHealth(1); destroyableComponent->SetMaxHealth(1.0f); destroyableComponent->SetFaction(-1, true); destroyableComponent->SetIsSmashable(true); - m_Components.insert(std::make_pair(COMPONENT_TYPE_DESTROYABLE, destroyableComponent)); + m_Components.insert(std::make_pair(eReplicaComponentType::DESTROYABLE, destroyableComponent)); // We have all our components. return; } @@ -207,47 +210,47 @@ void Entity::Initialize() { if (GetParentUser()) { auto missions = new MissionComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_MISSION, missions)); + m_Components.insert(std::make_pair(eReplicaComponentType::MISSION, missions)); missions->LoadFromXml(m_Character->GetXMLDoc()); } - uint32_t petComponentId = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PET); + uint32_t petComponentId = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PET); if (petComponentId > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_PET, new PetComponent(this, petComponentId))); + m_Components.insert(std::make_pair(eReplicaComponentType::PET, new PetComponent(this, petComponentId))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_ZONE_CONTROL) > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_ZONE_CONTROL, nullptr)); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::ZONE_CONTROL) > 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::ZONE_CONTROL, nullptr)); } - uint32_t possessableComponentId = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_POSSESSABLE); + uint32_t possessableComponentId = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::POSSESSABLE); if (possessableComponentId > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_POSSESSABLE, new PossessableComponent(this, possessableComponentId))); + m_Components.insert(std::make_pair(eReplicaComponentType::POSSESSABLE, new PossessableComponent(this, possessableComponentId))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_MODULE_ASSEMBLY) > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_MODULE_ASSEMBLY, new ModuleAssemblyComponent(this))); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::MODULE_ASSEMBLY) > 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::MODULE_ASSEMBLY, new ModuleAssemblyComponent(this))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_RACING_STATS) > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_RACING_STATS, nullptr)); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::RACING_STATS) > 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::RACING_STATS, nullptr)); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_EXHIBIT, -1) >= 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_EXHIBIT, new LUPExhibitComponent(this))); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::LUP_EXHIBIT, -1) >= 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::LUP_EXHIBIT, new LUPExhibitComponent(this))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_RACING_CONTROL) > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_RACING_CONTROL, new RacingControlComponent(this))); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::RACING_CONTROL) > 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::RACING_CONTROL, new RacingControlComponent(this))); } - const auto propertyEntranceComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PROPERTY_ENTRANCE); + const auto propertyEntranceComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PROPERTY_ENTRANCE); if (propertyEntranceComponentID > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_PROPERTY_ENTRANCE, + m_Components.insert(std::make_pair(eReplicaComponentType::PROPERTY_ENTRANCE, new PropertyEntranceComponent(propertyEntranceComponentID, this))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_CONTROLLABLE_PHYSICS) > 0) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::CONTROLLABLE_PHYSICS) > 0) { ControllablePhysicsComponent* controllablePhysics = new ControllablePhysicsComponent(this); if (m_Character) { @@ -261,17 +264,17 @@ void Entity::Initialize() { NiQuaternion rot; const auto& targetSceneName = m_Character->GetTargetScene(); - auto* targetScene = EntityManager::Instance()->GetSpawnPointEntity(targetSceneName); + auto* targetScene = Game::entityManager->GetSpawnPointEntity(targetSceneName); if (m_Character->HasBeenToWorld(mapID) && targetSceneName.empty()) { pos = m_Character->GetRespawnPoint(mapID); - rot = dZoneManager::Instance()->GetZone()->GetSpawnRot(); + rot = Game::zoneManager->GetZone()->GetSpawnRot(); } else if (targetScene != nullptr) { pos = targetScene->GetPosition(); rot = targetScene->GetRotation(); } else { - pos = dZoneManager::Instance()->GetZone()->GetSpawnPos(); - rot = dZoneManager::Instance()->GetZone()->GetSpawnRot(); + pos = Game::zoneManager->GetZone()->GetSpawnPos(); + rot = Game::zoneManager->GetZone()->GetSpawnRot(); } controllablePhysics->SetPosition(pos); @@ -282,77 +285,68 @@ void Entity::Initialize() { controllablePhysics->SetRotation(m_DefaultRotation); } - m_Components.insert(std::make_pair(COMPONENT_TYPE_CONTROLLABLE_PHYSICS, controllablePhysics)); + m_Components.insert(std::make_pair(eReplicaComponentType::CONTROLLABLE_PHYSICS, controllablePhysics)); } // If an entity is marked a phantom, simple physics is made into phantom phyics. bool markedAsPhantom = GetVar(u"markedAsPhantom"); - const auto simplePhysicsComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SIMPLE_PHYSICS); + const auto simplePhysicsComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SIMPLE_PHYSICS); if (!markedAsPhantom && simplePhysicsComponentID > 0) { SimplePhysicsComponent* comp = new SimplePhysicsComponent(simplePhysicsComponentID, this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_SIMPLE_PHYSICS, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::SIMPLE_PHYSICS, comp)); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_RIGID_BODY_PHANTOM_PHYSICS) > 0) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::RIGID_BODY_PHANTOM_PHYSICS) > 0) { RigidbodyPhantomPhysicsComponent* comp = new RigidbodyPhantomPhysicsComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_RIGID_BODY_PHANTOM_PHYSICS, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::RIGID_BODY_PHANTOM_PHYSICS, comp)); } - if (markedAsPhantom || compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PHANTOM_PHYSICS) > 0) { + if (markedAsPhantom || compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PHANTOM_PHYSICS) > 0) { PhantomPhysicsComponent* phantomPhysics = new PhantomPhysicsComponent(this); phantomPhysics->SetPhysicsEffectActive(false); - m_Components.insert(std::make_pair(COMPONENT_TYPE_PHANTOM_PHYSICS, phantomPhysics)); + m_Components.insert(std::make_pair(eReplicaComponentType::PHANTOM_PHYSICS, phantomPhysics)); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_VEHICLE_PHYSICS) > 0) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::VEHICLE_PHYSICS) > 0) { VehiclePhysicsComponent* vehiclePhysicsComponent = new VehiclePhysicsComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_VEHICLE_PHYSICS, vehiclePhysicsComponent)); + m_Components.insert(std::make_pair(eReplicaComponentType::VEHICLE_PHYSICS, vehiclePhysicsComponent)); vehiclePhysicsComponent->SetPosition(m_DefaultPosition); vehiclePhysicsComponent->SetRotation(m_DefaultRotation); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SOUND_TRIGGER, -1) != -1) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SOUND_TRIGGER, -1) != -1) { auto* comp = new SoundTriggerComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_SOUND_TRIGGER, comp)); - } - - //Check to see if we have a moving platform component: - //Which, for some reason didn't get added to the ComponentsRegistry so we have to check for a path manually here. - std::string attachedPath = GetVarAsString(u"attached_path"); - - if (!attachedPath.empty() || compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_MOVING_PLATFORM, -1) != -1) { - MovingPlatformComponent* plat = new MovingPlatformComponent(this, attachedPath); - m_Components.insert(std::make_pair(COMPONENT_TYPE_MOVING_PLATFORM, plat)); + m_Components.insert(std::make_pair(eReplicaComponentType::SOUND_TRIGGER, comp)); } //Also check for the collectible id: m_CollectibleID = GetVarAs(u"collectible_id"); - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_BUFF) > 0) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::BUFF) > 0) { BuffComponent* comp = new BuffComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_BUFF, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::BUFF, comp)); } - int collectibleComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_COLLECTIBLE); + int collectibleComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::COLLECTIBLE); if (collectibleComponentID > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_COLLECTIBLE, nullptr)); + m_Components.insert(std::make_pair(eReplicaComponentType::COLLECTIBLE, nullptr)); } /** * Multiple components require the destructible component. */ - int buffComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_BUFF); - int rebuildComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_REBUILD); + int buffComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::BUFF); + int rebuildComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::QUICK_BUILD); int componentID = 0; if (collectibleComponentID > 0) componentID = collectibleComponentID; if (rebuildComponentID > 0) componentID = rebuildComponentID; if (buffComponentID > 0) componentID = buffComponentID; - CDDestructibleComponentTable* destCompTable = CDClientManager::Instance()->GetTable("DestructibleComponent"); + CDDestructibleComponentTable* destCompTable = CDClientManager::Instance().GetTable(); std::vector destCompData = destCompTable->Query([=](CDDestructibleComponent entry) { return (entry.id == componentID); }); if (buffComponentID > 0 || collectibleComponentID > 0) { @@ -364,7 +358,7 @@ void Entity::Initialize() { std::vector destCompData = destCompTable->Query([=](CDDestructibleComponent entry) { return (entry.id == componentID); }); if (destCompData.size() > 0) { - if (HasComponent(COMPONENT_TYPE_RACING_STATS)) { + if (HasComponent(eReplicaComponentType::RACING_STATS)) { destCompData[0].imagination = 60; } @@ -384,7 +378,7 @@ void Entity::Initialize() { uint32_t npcMinLevel = destCompData[0].level; uint32_t currencyIndex = destCompData[0].CurrencyIndex; - CDCurrencyTableTable* currencyTable = CDClientManager::Instance()->GetTable("CurrencyTable"); + CDCurrencyTableTable* currencyTable = CDClientManager::Instance().GetTable(); std::vector currencyValues = currencyTable->Query([=](CDCurrencyTable entry) { return (entry.currencyIndex == currencyIndex && entry.npcminlevel == npcMinLevel); }); if (currencyValues.size() > 0) { @@ -393,8 +387,8 @@ void Entity::Initialize() { comp->SetMaxCoins(currencyValues[0].maxvalue); } - // extraInfo overrides - comp->SetIsSmashable(GetVarAs(u"is_smashable") != 0); + // extraInfo overrides. Client ORs the database smashable and the luz smashable. + comp->SetIsSmashable(comp->GetIsSmashable() | (GetVarAs(u"is_smashable") != 0)); } } else { comp->SetHealth(1); @@ -408,8 +402,8 @@ void Entity::Initialize() { comp->AddFaction(6); //Smashables // A race car has 60 imagination, other entities defaults to 0. - comp->SetImagination(HasComponent(COMPONENT_TYPE_RACING_STATS) ? 60 : 0); - comp->SetMaxImagination(HasComponent(COMPONENT_TYPE_RACING_STATS) ? 60 : 0); + comp->SetImagination(HasComponent(eReplicaComponentType::RACING_STATS) ? 60 : 0); + comp->SetMaxImagination(HasComponent(eReplicaComponentType::RACING_STATS) ? 60 : 0); } } @@ -427,43 +421,43 @@ void Entity::Initialize() { } } - m_Components.insert(std::make_pair(COMPONENT_TYPE_DESTROYABLE, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::DESTROYABLE, comp)); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_CHARACTER) > 0 || m_Character) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::CHARACTER) > 0 || m_Character) { // Character Component always has a possessor, level, and forced movement components - m_Components.insert(std::make_pair(COMPONENT_TYPE_POSSESSOR, new PossessorComponent(this))); + m_Components.insert(std::make_pair(eReplicaComponentType::POSSESSOR, new PossessorComponent(this))); // load in the xml for the level auto* levelComp = new LevelProgressionComponent(this); levelComp->LoadFromXml(m_Character->GetXMLDoc()); - m_Components.insert(std::make_pair(COMPONENT_TYPE_LEVEL_PROGRESSION, levelComp)); + m_Components.insert(std::make_pair(eReplicaComponentType::LEVEL_PROGRESSION, levelComp)); - m_Components.insert(std::make_pair(COMPONENT_TYPE_PLAYER_FORCED_MOVEMENT, new PlayerForcedMovementComponent(this))); + m_Components.insert(std::make_pair(eReplicaComponentType::PLAYER_FORCED_MOVEMENT, new PlayerForcedMovementComponent(this))); CharacterComponent* charComp = new CharacterComponent(this, m_Character); charComp->LoadFromXml(m_Character->GetXMLDoc()); - m_Components.insert(std::make_pair(COMPONENT_TYPE_CHARACTER, charComp)); + m_Components.insert(std::make_pair(eReplicaComponentType::CHARACTER, charComp)); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_INVENTORY) > 0 || m_Character) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::INVENTORY) > 0 || m_Character) { InventoryComponent* comp = nullptr; if (m_Character) comp = new InventoryComponent(this, m_Character->GetXMLDoc()); else comp = new InventoryComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_INVENTORY, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::INVENTORY, comp)); } // if this component exists, then we initialize it. it's value is always 0 - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_ROCKET_LAUNCH_LUP, -1) != -1) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::ROCKET_LAUNCH_LUP, -1) != -1) { auto comp = new RocketLaunchLupComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_ROCKET_LAUNCH_LUP, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::ROCKET_LAUNCH_LUP, comp)); } /** * This is a bit of a mess */ - CDScriptComponentTable* scriptCompTable = CDClientManager::Instance()->GetTable("ScriptComponent"); - int scriptComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SCRIPT); + CDScriptComponentTable* scriptCompTable = CDClientManager::Instance().GetTable(); + int32_t scriptComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SCRIPT, -1); std::string scriptName = ""; bool client = false; @@ -505,14 +499,14 @@ void Entity::Initialize() { scriptName = customScriptServer; } - if (!scriptName.empty() || client || m_Character) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_SCRIPT, new ScriptComponent(this, scriptName, true, client && scriptName.empty()))); + if (!scriptName.empty() || client || m_Character || scriptComponentID >= 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::SCRIPT, new ScriptComponent(this, scriptName, true, client && scriptName.empty()))); } // ZoneControl script if (m_TemplateID == 2365) { - CDZoneTableTable* zoneTable = CDClientManager::Instance()->GetTable("ZoneTable"); - const auto zoneID = dZoneManager::Instance()->GetZoneID(); + CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable(); + const auto zoneID = Game::zoneManager->GetZoneID(); const CDZoneTable* zoneData = zoneTable->Query(zoneID.GetMapID()); if (zoneData != nullptr) { @@ -520,26 +514,26 @@ void Entity::Initialize() { CDScriptComponent zoneScriptData = scriptCompTable->GetByID(zoneScriptID); ScriptComponent* comp = new ScriptComponent(this, zoneScriptData.script_name, true); - m_Components.insert(std::make_pair(COMPONENT_TYPE_SCRIPT, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::SCRIPT, comp)); } } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SKILL, -1) != -1 || m_Character) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SKILL, -1) != -1 || m_Character) { SkillComponent* comp = new SkillComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_SKILL, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::SKILL, comp)); } - const auto combatAiId = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_BASE_COMBAT_AI); + const auto combatAiId = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::BASE_COMBAT_AI); if (combatAiId > 0) { BaseCombatAIComponent* comp = new BaseCombatAIComponent(this, combatAiId); - m_Components.insert(std::make_pair(COMPONENT_TYPE_BASE_COMBAT_AI, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::BASE_COMBAT_AI, comp)); } - if (int componentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_REBUILD) > 0) { + if (int componentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::QUICK_BUILD) > 0) { RebuildComponent* comp = new RebuildComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_REBUILD, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::QUICK_BUILD, comp)); - CDRebuildComponentTable* rebCompTable = CDClientManager::Instance()->GetTable("RebuildComponent"); + CDRebuildComponentTable* rebCompTable = CDClientManager::Instance().GetTable(); std::vector rebCompData = rebCompTable->Query([=](CDRebuildComponent entry) { return (entry.id == rebuildComponentID); }); if (rebCompData.size() > 0) { @@ -577,89 +571,93 @@ void Entity::Initialize() { } } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SWITCH, -1) != -1) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SWITCH, -1) != -1) { SwitchComponent* comp = new SwitchComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_SWITCH, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::SWITCH, comp)); } - if ((compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_VENDOR) > 0)) { + if ((compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::VENDOR) > 0)) { VendorComponent* comp = new VendorComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_VENDOR, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::VENDOR, comp)); + } else if ((compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::DONATION_VENDOR, -1) != -1)) { + DonationVendorComponent* comp = new DonationVendorComponent(this); + m_Components.insert(std::make_pair(eReplicaComponentType::DONATION_VENDOR, comp)); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PROPERTY_VENDOR, -1) != -1) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PROPERTY_VENDOR, -1) != -1) { auto* component = new PropertyVendorComponent(this); - m_Components.insert_or_assign(COMPONENT_TYPE_PROPERTY_VENDOR, component); + m_Components.insert_or_assign(eReplicaComponentType::PROPERTY_VENDOR, component); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PROPERTY_MANAGEMENT, -1) != -1) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PROPERTY_MANAGEMENT, -1) != -1) { auto* component = new PropertyManagementComponent(this); - m_Components.insert_or_assign(COMPONENT_TYPE_PROPERTY_MANAGEMENT, component); + m_Components.insert_or_assign(eReplicaComponentType::PROPERTY_MANAGEMENT, component); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_BOUNCER, -1) != -1) { // you have to determine it like this because all bouncers have a componentID of 0 + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::BOUNCER, -1) != -1) { // you have to determine it like this because all bouncers have a componentID of 0 BouncerComponent* comp = new BouncerComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_BOUNCER, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::BOUNCER, comp)); } - if ((compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_RENDER) > 0 && m_TemplateID != 2365) || m_Character) { - RenderComponent* render = new RenderComponent(this); - m_Components.insert(std::make_pair(COMPONENT_TYPE_RENDER, render)); + int32_t renderComponentId = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::RENDER); + if ((renderComponentId > 0 && m_TemplateID != 2365) || m_Character) { + RenderComponent* render = new RenderComponent(this, renderComponentId); + m_Components.insert(std::make_pair(eReplicaComponentType::RENDER, render)); } - if ((compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_MISSION_OFFER) > 0) || m_Character) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_MISSION_OFFER, new MissionOfferComponent(this, m_TemplateID))); + if ((compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::MISSION_OFFER) > 0) || m_Character) { + m_Components.insert(std::make_pair(eReplicaComponentType::MISSION_OFFER, new MissionOfferComponent(this, m_TemplateID))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_BUILD_BORDER, -1) != -1) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_BUILD_BORDER, new BuildBorderComponent(this))); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::BUILD_BORDER, -1) != -1) { + m_Components.insert(std::make_pair(eReplicaComponentType::BUILD_BORDER, new BuildBorderComponent(this))); } // Scripted activity component - int scriptedActivityID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SCRIPTED_ACTIVITY); + int scriptedActivityID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SCRIPTED_ACTIVITY); if ((scriptedActivityID > 0)) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_SCRIPTED_ACTIVITY, new ScriptedActivityComponent(this, scriptedActivityID))); + m_Components.insert(std::make_pair(eReplicaComponentType::SCRIPTED_ACTIVITY, new ScriptedActivityComponent(this, scriptedActivityID))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_MODEL, -1) != -1 && !GetComponent()) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_MODEL, new ModelComponent(this))); - if (m_Components.find(COMPONENT_TYPE_DESTROYABLE) == m_Components.end()) { + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::MODEL, -1) != -1 && !GetComponent()) { + m_Components.insert(std::make_pair(eReplicaComponentType::MODEL, new ModelComponent(this))); + if (m_Components.find(eReplicaComponentType::DESTROYABLE) == m_Components.end()) { auto destroyableComponent = new DestroyableComponent(this); destroyableComponent->SetHealth(1); destroyableComponent->SetMaxHealth(1.0f); destroyableComponent->SetFaction(-1, true); destroyableComponent->SetIsSmashable(true); - m_Components.insert(std::make_pair(COMPONENT_TYPE_DESTROYABLE, destroyableComponent)); + m_Components.insert(std::make_pair(eReplicaComponentType::DESTROYABLE, destroyableComponent)); } } PetComponent* petComponent; - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_ITEM) > 0 && !TryGetComponent(COMPONENT_TYPE_PET, petComponent) && !HasComponent(COMPONENT_TYPE_MODEL)) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_ITEM, nullptr)); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::ITEM) > 0 && !TryGetComponent(eReplicaComponentType::PET, petComponent) && !HasComponent(eReplicaComponentType::MODEL)) { + m_Components.insert(std::make_pair(eReplicaComponentType::ITEM, nullptr)); } // Shooting gallery component - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_SHOOTING_GALLERY) > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_SHOOTING_GALLERY, new ShootingGalleryComponent(this))); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::SHOOTING_GALLERY) > 0) { + m_Components.insert(std::make_pair(eReplicaComponentType::SHOOTING_GALLERY, new ShootingGalleryComponent(this))); } - if (compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PROPERTY, -1) != -1) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_PROPERTY, new PropertyComponent(this))); + if (compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PROPERTY, -1) != -1) { + m_Components.insert(std::make_pair(eReplicaComponentType::PROPERTY, new PropertyComponent(this))); } - const int rocketId = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_ROCKET_LAUNCH); + const int rocketId = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::ROCKET_LAUNCH); if ((rocketId > 0)) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_ROCKET_LAUNCH, new RocketLaunchpadControlComponent(this, rocketId))); + m_Components.insert(std::make_pair(eReplicaComponentType::ROCKET_LAUNCH, new RocketLaunchpadControlComponent(this, rocketId))); } - const int32_t railComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_RAIL_ACTIVATOR); + const int32_t railComponentID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::RAIL_ACTIVATOR); if (railComponentID > 0) { - m_Components.insert(std::make_pair(COMPONENT_TYPE_RAIL_ACTIVATOR, new RailActivatorComponent(this, railComponentID))); + m_Components.insert(std::make_pair(eReplicaComponentType::RAIL_ACTIVATOR, new RailActivatorComponent(this, railComponentID))); } - int movementAIID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_MOVEMENT_AI); + int movementAIID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::MOVEMENT_AI); if (movementAIID > 0) { - CDMovementAIComponentTable* moveAITable = CDClientManager::Instance()->GetTable("MovementAIComponent"); + CDMovementAIComponentTable* moveAITable = CDClientManager::Instance().GetTable(); std::vector moveAIComp = moveAITable->Query([=](CDMovementAIComponent entry) {return (entry.id == movementAIID); }); if (moveAIComp.size() > 0) { @@ -682,7 +680,7 @@ void Entity::Initialize() { } } - m_Components.insert(std::make_pair(COMPONENT_TYPE_MOVEMENT_AI, new MovementAIComponent(this, moveInfo))); + m_Components.insert(std::make_pair(eReplicaComponentType::MOVEMENT_AI, new MovementAIComponent(this, moveInfo))); } } else if (petComponentId > 0 || combatAiId > 0 && GetComponent()->GetTetherSpeed() > 0) { MovementAIInfo moveInfo = MovementAIInfo(); @@ -693,17 +691,44 @@ void Entity::Initialize() { moveInfo.wanderDelayMax = 5; moveInfo.wanderDelayMin = 2; - m_Components.insert(std::make_pair(COMPONENT_TYPE_MOVEMENT_AI, new MovementAIComponent(this, moveInfo))); + m_Components.insert(std::make_pair(eReplicaComponentType::MOVEMENT_AI, new MovementAIComponent(this, moveInfo))); } - int proximityMonitorID = compRegistryTable->GetByIDAndType(m_TemplateID, COMPONENT_TYPE_PROXIMITY_MONITOR); + std::string pathName = GetVarAsString(u"attached_path"); + const Path* path = Game::zoneManager->GetZone()->GetPath(pathName); + + //Check to see if we have an attached path and add the appropiate component to handle it: + if (path){ + // if we have a moving platform path, then we need a moving platform component + if (path->pathType == PathType::MovingPlatform) { + MovingPlatformComponent* plat = new MovingPlatformComponent(this, pathName); + m_Components.insert(std::make_pair(eReplicaComponentType::MOVING_PLATFORM, plat)); + // else if we are a movement path + } /*else if (path->pathType == PathType::Movement) { + auto movementAIcomp = GetComponent(); + if (movementAIcomp){ + // TODO: set path in existing movementAIComp + } else { + // TODO: create movementAIcomp and set path + } + }*/ + } else { + // else we still need to setup moving platform if it has a moving platform comp but no path + int32_t movingPlatformComponentId = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::MOVING_PLATFORM, -1); + if (movingPlatformComponentId >= 0) { + MovingPlatformComponent* plat = new MovingPlatformComponent(this, pathName); + m_Components.insert(std::make_pair(eReplicaComponentType::MOVING_PLATFORM, plat)); + } + } + + int proximityMonitorID = compRegistryTable->GetByIDAndType(m_TemplateID, eReplicaComponentType::PROXIMITY_MONITOR); if (proximityMonitorID > 0) { - CDProximityMonitorComponentTable* proxCompTable = CDClientManager::Instance()->GetTable("ProximityMonitorComponent"); + CDProximityMonitorComponentTable* proxCompTable = CDClientManager::Instance().GetTable(); std::vector proxCompData = proxCompTable->Query([=](CDProximityMonitorComponent entry) { return (entry.id == proximityMonitorID); }); if (proxCompData.size() > 0) { std::vector proximityStr = GeneralUtils::SplitString(proxCompData[0].Proximities, ','); ProximityMonitorComponent* comp = new ProximityMonitorComponent(this, std::stoi(proximityStr[0]), std::stoi(proximityStr[1])); - m_Components.insert(std::make_pair(COMPONENT_TYPE_PROXIMITY_MONITOR, comp)); + m_Components.insert(std::make_pair(eReplicaComponentType::PROXIMITY_MONITOR, comp)); } } @@ -714,9 +739,9 @@ void Entity::Initialize() { } }); - if (!m_Character && EntityManager::Instance()->GetGhostingEnabled()) { + if (!m_Character && Game::entityManager->GetGhostingEnabled()) { // Don't ghost what is likely large scene elements - if (m_Components.size() == 2 && HasComponent(COMPONENT_TYPE_SIMPLE_PHYSICS) && HasComponent(COMPONENT_TYPE_RENDER)) { + if (HasComponent(eReplicaComponentType::SIMPLE_PHYSICS) && HasComponent(eReplicaComponentType::RENDER) && (m_Components.size() == 2 || (HasComponent(eReplicaComponentType::TRIGGER) && m_Components.size() == 3))) { goto no_ghosting; } @@ -728,14 +753,14 @@ void Entity::Initialize() { */ if ( !EntityManager::IsExcludedFromGhosting(GetLOT()) && - !HasComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY) && - !HasComponent(COMPONENT_TYPE_MOVING_PLATFORM) && - !HasComponent(COMPONENT_TYPE_PHANTOM_PHYSICS) && - !HasComponent(COMPONENT_TYPE_PROPERTY) && - !HasComponent(COMPONENT_TYPE_RACING_CONTROL) && - !HasComponent(COMPONENT_TYPE_VEHICLE_PHYSICS) + !HasComponent(eReplicaComponentType::SCRIPTED_ACTIVITY) && + !HasComponent(eReplicaComponentType::MOVING_PLATFORM) && + !HasComponent(eReplicaComponentType::PHANTOM_PHYSICS) && + !HasComponent(eReplicaComponentType::PROPERTY) && + !HasComponent(eReplicaComponentType::RACING_CONTROL) && + !HasComponent(eReplicaComponentType::VEHICLE_PHYSICS) ) - //if (HasComponent(COMPONENT_TYPE_BASE_COMBAT_AI)) + //if (HasComponent(eReplicaComponentType::BASE_COMBAT_AI)) { m_IsGhostingCandidate = true; } @@ -745,21 +770,21 @@ void Entity::Initialize() { } // Special case for collectibles in Ninjago - if (HasComponent(COMPONENT_TYPE_COLLECTIBLE) && Game::server->GetZoneID() == 2000) { + if (HasComponent(eReplicaComponentType::COLLECTIBLE) && Game::server->GetZoneID() == 2000) { m_IsGhostingCandidate = true; } } no_ghosting: - TriggerEvent("OnCreate"); + TriggerEvent(eTriggerEventType::CREATE, this); if (m_Character) { auto* controllablePhysicsComponent = GetComponent(); auto* levelComponent = GetComponent(); - if (controllablePhysicsComponent != nullptr && levelComponent->GetLevel() >= 20) { - controllablePhysicsComponent->SetSpeedMultiplier(525.0f / 500.0f); + if (controllablePhysicsComponent && levelComponent) { + controllablePhysicsComponent->SetSpeedMultiplier(levelComponent->GetSpeedBase() / 500.0f); } } } @@ -780,7 +805,7 @@ User* Entity::GetParentUser() const { return static_cast(this)->GetParentUser(); } -Component* Entity::GetComponent(int32_t componentID) const { +Component* Entity::GetComponent(eReplicaComponentType componentID) const { const auto& index = m_Components.find(componentID); if (index == m_Components.end()) { @@ -790,11 +815,11 @@ Component* Entity::GetComponent(int32_t componentID) const { return index->second; } -bool Entity::HasComponent(const int32_t componentId) const { +bool Entity::HasComponent(const eReplicaComponentType componentId) const { return m_Components.find(componentId) != m_Components.end(); } -void Entity::AddComponent(const int32_t componentId, Component* component) { +void Entity::AddComponent(const eReplicaComponentType componentId, Component* component) { if (HasComponent(componentId)) { return; } @@ -804,8 +829,8 @@ void Entity::AddComponent(const int32_t componentId, Component* component) { std::vector Entity::GetScriptComponents() { std::vector comps; - for (std::pair p : m_Components) { - if (p.first == COMPONENT_TYPE_SCRIPT) { + for (std::pair p : m_Components) { + if (p.first == eReplicaComponentType::SCRIPT) { comps.push_back(static_cast(p.second)); } } @@ -813,11 +838,27 @@ std::vector Entity::GetScriptComponents() { return comps; } +void Entity::Subscribe(LWOOBJID scriptObjId, CppScripts::Script* scriptToAdd, const std::string& notificationName) { + if (notificationName == "HitOrHealResult" || notificationName == "Hit") { + auto* destroyableComponent = GetComponent(); + if (!destroyableComponent) return; + destroyableComponent->Subscribe(scriptObjId, scriptToAdd); + } +} + +void Entity::Unsubscribe(LWOOBJID scriptObjId, const std::string& notificationName) { + if (notificationName == "HitOrHealResult" || notificationName == "Hit") { + auto* destroyableComponent = GetComponent(); + if (!destroyableComponent) return; + destroyableComponent->Unsubscribe(scriptObjId); + } +} + void Entity::SetProximityRadius(float proxRadius, std::string name) { ProximityMonitorComponent* proxMon = GetComponent(); if (!proxMon) { proxMon = new ProximityMonitorComponent(this); - m_Components.insert_or_assign(COMPONENT_TYPE_PROXIMITY_MONITOR, proxMon); + m_Components.insert_or_assign(eReplicaComponentType::PROXIMITY_MONITOR, proxMon); } proxMon->SetProximityRadius(proxRadius, name); } @@ -826,12 +867,12 @@ void Entity::SetProximityRadius(dpEntity* entity, std::string name) { ProximityMonitorComponent* proxMon = GetComponent(); if (!proxMon) { proxMon = new ProximityMonitorComponent(this); - m_Components.insert_or_assign(COMPONENT_TYPE_PROXIMITY_MONITOR, proxMon); + m_Components.insert_or_assign(eReplicaComponentType::PROXIMITY_MONITOR, proxMon); } proxMon->SetProximityRadius(entity, name); } -void Entity::SetGMLevel(uint8_t value) { +void Entity::SetGMLevel(eGameMasterLevel value) { m_GMLevel = value; if (GetParentUser()) { Character* character = GetParentUser()->GetLastUsedChar(); @@ -848,7 +889,7 @@ void Entity::SetGMLevel(uint8_t value) { } void Entity::WriteBaseReplicaData(RakNet::BitStream* outBitStream, eReplicaPacketType packetType) { - if (packetType == PACKET_TYPE_CONSTRUCTION) { + if (packetType == eReplicaPacketType::CONSTRUCTION) { outBitStream->Write(m_ObjectID); outBitStream->Write(m_TemplateID); @@ -913,17 +954,21 @@ void Entity::WriteBaseReplicaData(RakNet::BitStream* outBitStream, eReplicaPacke outBitStream->Write0(); //No ldf data } - if (m_Trigger != nullptr && m_Trigger->events.size() > 0) { - outBitStream->Write1(); - } else { + TriggerComponent* triggerComponent; + if (TryGetComponent(eReplicaComponentType::TRIGGER, triggerComponent)) { + // has trigger component, check to see if we have events to handle + auto* trigger = triggerComponent->GetTrigger(); + outBitStream->Write(trigger && trigger->events.size() > 0); + } else { // no trigger componenet, so definitely no triggers outBitStream->Write0(); } + if (m_ParentEntity != nullptr || m_SpawnerID != 0) { outBitStream->Write1(); - if (m_ParentEntity != nullptr) outBitStream->Write(GeneralUtils::SetBit(m_ParentEntity->GetObjectID(), OBJECT_BIT_CLIENT)); + if (m_ParentEntity != nullptr) outBitStream->Write(GeneralUtils::SetBit(m_ParentEntity->GetObjectID(), static_cast(eObjectBits::CLIENT))); else if (m_Spawner != nullptr && m_Spawner->m_Info.isNetwork) outBitStream->Write(m_SpawnerID); - else outBitStream->Write(GeneralUtils::SetBit(m_SpawnerID, OBJECT_BIT_CLIENT)); + else outBitStream->Write(GeneralUtils::SetBit(m_SpawnerID, static_cast(eObjectBits::CLIENT))); } else outBitStream->Write0(); outBitStream->Write(m_HasSpawnerNodeID); @@ -939,15 +984,15 @@ void Entity::WriteBaseReplicaData(RakNet::BitStream* outBitStream, eReplicaPacke outBitStream->Write0(); //ObjectWorldState - if (m_GMLevel != 0) { + if (m_GMLevel != eGameMasterLevel::CIVILIAN) { outBitStream->Write1(); outBitStream->Write(m_GMLevel); } else outBitStream->Write0(); //No GM Level } // Only serialize parent / child info should the info be dirty (changed) or if this is the construction of the entity. - outBitStream->Write(m_IsParentChildDirty || packetType == PACKET_TYPE_CONSTRUCTION); - if (m_IsParentChildDirty || packetType == PACKET_TYPE_CONSTRUCTION) { + outBitStream->Write(m_IsParentChildDirty || packetType == eReplicaPacketType::CONSTRUCTION); + if (m_IsParentChildDirty || packetType == eReplicaPacketType::CONSTRUCTION) { m_IsParentChildDirty = false; outBitStream->Write(m_ParentEntity != nullptr); if (m_ParentEntity) { @@ -972,63 +1017,63 @@ void Entity::WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType bool destroyableSerialized = false; bool bIsInitialUpdate = false; - if (packetType == PACKET_TYPE_CONSTRUCTION) bIsInitialUpdate = true; + if (packetType == eReplicaPacketType::CONSTRUCTION) bIsInitialUpdate = true; unsigned int flags = 0; PossessableComponent* possessableComponent; - if (TryGetComponent(COMPONENT_TYPE_POSSESSABLE, possessableComponent)) { + if (TryGetComponent(eReplicaComponentType::POSSESSABLE, possessableComponent)) { possessableComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } ModuleAssemblyComponent* moduleAssemblyComponent; - if (TryGetComponent(COMPONENT_TYPE_MODULE_ASSEMBLY, moduleAssemblyComponent)) { + if (TryGetComponent(eReplicaComponentType::MODULE_ASSEMBLY, moduleAssemblyComponent)) { moduleAssemblyComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } ControllablePhysicsComponent* controllablePhysicsComponent; - if (TryGetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS, controllablePhysicsComponent)) { + if (TryGetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS, controllablePhysicsComponent)) { controllablePhysicsComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } SimplePhysicsComponent* simplePhysicsComponent; - if (TryGetComponent(COMPONENT_TYPE_SIMPLE_PHYSICS, simplePhysicsComponent)) { + if (TryGetComponent(eReplicaComponentType::SIMPLE_PHYSICS, simplePhysicsComponent)) { simplePhysicsComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } RigidbodyPhantomPhysicsComponent* rigidbodyPhantomPhysics; - if (TryGetComponent(COMPONENT_TYPE_RIGID_BODY_PHANTOM_PHYSICS, rigidbodyPhantomPhysics)) { + if (TryGetComponent(eReplicaComponentType::RIGID_BODY_PHANTOM_PHYSICS, rigidbodyPhantomPhysics)) { rigidbodyPhantomPhysics->Serialize(outBitStream, bIsInitialUpdate, flags); } VehiclePhysicsComponent* vehiclePhysicsComponent; - if (TryGetComponent(COMPONENT_TYPE_VEHICLE_PHYSICS, vehiclePhysicsComponent)) { + if (TryGetComponent(eReplicaComponentType::VEHICLE_PHYSICS, vehiclePhysicsComponent)) { vehiclePhysicsComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } PhantomPhysicsComponent* phantomPhysicsComponent; - if (TryGetComponent(COMPONENT_TYPE_PHANTOM_PHYSICS, phantomPhysicsComponent)) { + if (TryGetComponent(eReplicaComponentType::PHANTOM_PHYSICS, phantomPhysicsComponent)) { phantomPhysicsComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } SoundTriggerComponent* soundTriggerComponent; - if (TryGetComponent(COMPONENT_TYPE_SOUND_TRIGGER, soundTriggerComponent)) { + if (TryGetComponent(eReplicaComponentType::SOUND_TRIGGER, soundTriggerComponent)) { soundTriggerComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } BuffComponent* buffComponent; - if (TryGetComponent(COMPONENT_TYPE_BUFF, buffComponent)) { + if (TryGetComponent(eReplicaComponentType::BUFF, buffComponent)) { buffComponent->Serialize(outBitStream, bIsInitialUpdate, flags); DestroyableComponent* destroyableComponent; - if (TryGetComponent(COMPONENT_TYPE_DESTROYABLE, destroyableComponent)) { + if (TryGetComponent(eReplicaComponentType::DESTROYABLE, destroyableComponent)) { destroyableComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } destroyableSerialized = true; } - if (HasComponent(COMPONENT_TYPE_COLLECTIBLE)) { + if (HasComponent(eReplicaComponentType::COLLECTIBLE)) { DestroyableComponent* destroyableComponent; - if (TryGetComponent(COMPONENT_TYPE_DESTROYABLE, destroyableComponent) && !destroyableSerialized) { + if (TryGetComponent(eReplicaComponentType::DESTROYABLE, destroyableComponent) && !destroyableSerialized) { destroyableComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } destroyableSerialized = true; @@ -1036,15 +1081,15 @@ void Entity::WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType } PetComponent* petComponent; - if (TryGetComponent(COMPONENT_TYPE_PET, petComponent)) { + if (TryGetComponent(eReplicaComponentType::PET, petComponent)) { petComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } CharacterComponent* characterComponent; - if (TryGetComponent(COMPONENT_TYPE_CHARACTER, characterComponent)) { + if (TryGetComponent(eReplicaComponentType::CHARACTER, characterComponent)) { PossessorComponent* possessorComponent; - if (TryGetComponent(COMPONENT_TYPE_POSSESSOR, possessorComponent)) { + if (TryGetComponent(eReplicaComponentType::POSSESSOR, possessorComponent)) { possessorComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } else { // Should never happen, but just to be safe @@ -1052,7 +1097,7 @@ void Entity::WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType } LevelProgressionComponent* levelProgressionComponent; - if (TryGetComponent(COMPONENT_TYPE_LEVEL_PROGRESSION, levelProgressionComponent)) { + if (TryGetComponent(eReplicaComponentType::LEVEL_PROGRESSION, levelProgressionComponent)) { levelProgressionComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } else { // Should never happen, but just to be safe @@ -1060,7 +1105,7 @@ void Entity::WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType } PlayerForcedMovementComponent* playerForcedMovementComponent; - if (TryGetComponent(COMPONENT_TYPE_PLAYER_FORCED_MOVEMENT, playerForcedMovementComponent)) { + if (TryGetComponent(eReplicaComponentType::PLAYER_FORCED_MOVEMENT, playerForcedMovementComponent)) { playerForcedMovementComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } else { // Should never happen, but just to be safe @@ -1070,34 +1115,34 @@ void Entity::WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType characterComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } - if (HasComponent(COMPONENT_TYPE_ITEM)) { + if (HasComponent(eReplicaComponentType::ITEM)) { outBitStream->Write0(); } InventoryComponent* inventoryComponent; - if (TryGetComponent(COMPONENT_TYPE_INVENTORY, inventoryComponent)) { + if (TryGetComponent(eReplicaComponentType::INVENTORY, inventoryComponent)) { inventoryComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } ScriptComponent* scriptComponent; - if (TryGetComponent(COMPONENT_TYPE_SCRIPT, scriptComponent)) { + if (TryGetComponent(eReplicaComponentType::SCRIPT, scriptComponent)) { scriptComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } SkillComponent* skillComponent; - if (TryGetComponent(COMPONENT_TYPE_SKILL, skillComponent)) { + if (TryGetComponent(eReplicaComponentType::SKILL, skillComponent)) { skillComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } BaseCombatAIComponent* baseCombatAiComponent; - if (TryGetComponent(COMPONENT_TYPE_BASE_COMBAT_AI, baseCombatAiComponent)) { + if (TryGetComponent(eReplicaComponentType::BASE_COMBAT_AI, baseCombatAiComponent)) { baseCombatAiComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } RebuildComponent* rebuildComponent; - if (TryGetComponent(COMPONENT_TYPE_REBUILD, rebuildComponent)) { + if (TryGetComponent(eReplicaComponentType::QUICK_BUILD, rebuildComponent)) { DestroyableComponent* destroyableComponent; - if (TryGetComponent(COMPONENT_TYPE_DESTROYABLE, destroyableComponent) && !destroyableSerialized) { + if (TryGetComponent(eReplicaComponentType::DESTROYABLE, destroyableComponent) && !destroyableSerialized) { destroyableComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } destroyableSerialized = true; @@ -1105,64 +1150,69 @@ void Entity::WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType } MovingPlatformComponent* movingPlatformComponent; - if (TryGetComponent(COMPONENT_TYPE_MOVING_PLATFORM, movingPlatformComponent)) { + if (TryGetComponent(eReplicaComponentType::MOVING_PLATFORM, movingPlatformComponent)) { movingPlatformComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } SwitchComponent* switchComponent; - if (TryGetComponent(COMPONENT_TYPE_SWITCH, switchComponent)) { + if (TryGetComponent(eReplicaComponentType::SWITCH, switchComponent)) { switchComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } VendorComponent* vendorComponent; - if (TryGetComponent(COMPONENT_TYPE_VENDOR, vendorComponent)) { + if (TryGetComponent(eReplicaComponentType::VENDOR, vendorComponent)) { vendorComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } + DonationVendorComponent* donationVendorComponent; + if (TryGetComponent(eReplicaComponentType::DONATION_VENDOR, donationVendorComponent)) { + donationVendorComponent->Serialize(outBitStream, bIsInitialUpdate, flags); + } + BouncerComponent* bouncerComponent; - if (TryGetComponent(COMPONENT_TYPE_BOUNCER, bouncerComponent)) { + if (TryGetComponent(eReplicaComponentType::BOUNCER, bouncerComponent)) { bouncerComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } ScriptedActivityComponent* scriptedActivityComponent; - if (TryGetComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY, scriptedActivityComponent)) { + if (TryGetComponent(eReplicaComponentType::SCRIPTED_ACTIVITY, scriptedActivityComponent)) { scriptedActivityComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } ShootingGalleryComponent* shootingGalleryComponent; - if (TryGetComponent(COMPONENT_TYPE_SHOOTING_GALLERY, shootingGalleryComponent)) { + if (TryGetComponent(eReplicaComponentType::SHOOTING_GALLERY, shootingGalleryComponent)) { shootingGalleryComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } RacingControlComponent* racingControlComponent; - if (TryGetComponent(COMPONENT_TYPE_RACING_CONTROL, racingControlComponent)) { + if (TryGetComponent(eReplicaComponentType::RACING_CONTROL, racingControlComponent)) { racingControlComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } LUPExhibitComponent* lupExhibitComponent; - if (TryGetComponent(COMPONENT_TYPE_EXHIBIT, lupExhibitComponent)) { + if (TryGetComponent(eReplicaComponentType::LUP_EXHIBIT, lupExhibitComponent)) { lupExhibitComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } ModelComponent* modelComponent; - if (TryGetComponent(COMPONENT_TYPE_MODEL, modelComponent)) { + if (TryGetComponent(eReplicaComponentType::MODEL, modelComponent)) { modelComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } RenderComponent* renderComponent; - if (TryGetComponent(COMPONENT_TYPE_RENDER, renderComponent)) { + if (TryGetComponent(eReplicaComponentType::RENDER, renderComponent)) { renderComponent->Serialize(outBitStream, bIsInitialUpdate, flags); } if (modelComponent) { DestroyableComponent* destroyableComponent; - if (TryGetComponent(COMPONENT_TYPE_DESTROYABLE, destroyableComponent) && !destroyableSerialized) { + if (TryGetComponent(eReplicaComponentType::DESTROYABLE, destroyableComponent) && !destroyableSerialized) { destroyableComponent->Serialize(outBitStream, bIsInitialUpdate, flags); destroyableSerialized = true; } } - if (HasComponent(COMPONENT_TYPE_ZONE_CONTROL)) { + if (HasComponent(eReplicaComponentType::ZONE_CONTROL)) { outBitStream->Write(0x40000000); } @@ -1201,6 +1251,7 @@ void Entity::Update(const float deltaTime) { for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { script->OnTimerDone(this, timerName); } + TriggerEvent(eTriggerEventType::TIMER_DONE, this); } else { timerPosition++; } @@ -1242,12 +1293,12 @@ void Entity::Update(const float deltaTime) { } if (m_ShouldDestroyAfterUpdate) { - EntityManager::Instance()->DestroyEntity(this->GetObjectID()); + Game::entityManager->DestroyEntity(this->GetObjectID()); } } void Entity::OnCollisionProximity(LWOOBJID otherEntity, const std::string& proxName, const std::string& status) { - Entity* other = EntityManager::Instance()->GetEntity(otherEntity); + Entity* other = Game::entityManager->GetEntity(otherEntity); if (!other) return; for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { @@ -1261,7 +1312,7 @@ void Entity::OnCollisionProximity(LWOOBJID otherEntity, const std::string& proxN } void Entity::OnCollisionPhantom(const LWOOBJID otherEntity) { - auto* other = EntityManager::Instance()->GetEntity(otherEntity); + auto* other = Game::entityManager->GetEntity(otherEntity); if (!other) return; for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { @@ -1277,7 +1328,7 @@ void Entity::OnCollisionPhantom(const LWOOBJID otherEntity) { switchComp->EntityEnter(other); } - TriggerEvent("OnEnter", other); + TriggerEvent(eTriggerEventType::ENTER, other); // POI system const auto& poi = GetVar(u"POI"); @@ -1286,7 +1337,7 @@ void Entity::OnCollisionPhantom(const LWOOBJID otherEntity) { auto* missionComponent = other->GetComponent(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_LOCATION, 0, 0, GeneralUtils::UTF16ToWTF8(poi)); + missionComponent->Progress(eMissionTaskType::EXPLORE, 0, 0, GeneralUtils::UTF16ToWTF8(poi)); } } @@ -1308,10 +1359,14 @@ void Entity::OnCollisionPhantom(const LWOOBJID otherEntity) { } void Entity::OnCollisionLeavePhantom(const LWOOBJID otherEntity) { - auto* other = EntityManager::Instance()->GetEntity(otherEntity); + auto* other = Game::entityManager->GetEntity(otherEntity); if (!other) return; - TriggerEvent("OnLeave", other); + for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { + script->OnOffCollisionPhantom(this, other); + } + + TriggerEvent(eTriggerEventType::EXIT, other); SwitchComponent* switchComp = GetComponent(); if (switchComp) { @@ -1359,7 +1414,7 @@ void Entity::OnEmoteReceived(const int32_t emote, Entity* target) { } void Entity::OnUse(Entity* originator) { - TriggerEvent("OnInteract"); + TriggerEvent(eTriggerEventType::INTERACT, originator); for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { script->OnUse(this, originator); @@ -1381,6 +1436,7 @@ void Entity::OnHitOrHealResult(Entity* attacker, int32_t damage) { } void Entity::OnHit(Entity* attacker) { + TriggerEvent(eTriggerEventType::HIT, attacker); for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { script->OnHit(this, attacker); } @@ -1446,18 +1502,24 @@ void Entity::OnChoiceBoxResponse(Entity* sender, int32_t button, const std::u16s } } +void Entity::RequestActivityExit(Entity* sender, LWOOBJID player, bool canceled) { + for (CppScripts::Script* script : CppScripts::GetEntityScripts(this)) { + script->OnRequestActivityExit(sender, player, canceled); + } +} + void Entity::Smash(const LWOOBJID source, const eKillType killType, const std::u16string& deathType) { if (!m_PlayerIsReadyForUpdates) return; auto* destroyableComponent = GetComponent(); if (destroyableComponent == nullptr) { - Kill(EntityManager::Instance()->GetEntity(source)); + Kill(Game::entityManager->GetEntity(source)); return; } auto* possessorComponent = GetComponent(); if (possessorComponent) { if (possessorComponent->GetPossessable() != LWOOBJID_EMPTY) { - auto* mount = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); + auto* mount = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); if (mount) possessorComponent->Dismount(mount, true); } } @@ -1485,20 +1547,20 @@ void Entity::Kill(Entity* murderer) { } if (!IsPlayer()) { - EntityManager::Instance()->DestroyEntity(this); + Game::entityManager->DestroyEntity(this); } const auto& grpNameQBShowBricks = GetVar(u"grpNameQBShowBricks"); if (!grpNameQBShowBricks.empty()) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(grpNameQBShowBricks); + auto spawners = Game::zoneManager->GetSpawnersByName(grpNameQBShowBricks); Spawner* spawner = nullptr; if (!spawners.empty()) { spawner = spawners[0]; } else { - spawners = dZoneManager::Instance()->GetSpawnersInGroup(grpNameQBShowBricks); + spawners = Game::zoneManager->GetSpawnersInGroup(grpNameQBShowBricks); if (!spawners.empty()) { spawner = spawners[0]; @@ -1558,7 +1620,7 @@ void Entity::PickupItem(const LWOOBJID& objectID) { InventoryComponent* inv = GetComponent(); if (!inv) return; - CDObjectsTable* objectsTable = CDClientManager::Instance()->GetTable("Objects"); + CDObjectsTable* objectsTable = CDClientManager::Instance().GetTable(); auto& droppedLoot = static_cast(this)->GetDroppedLoot(); @@ -1571,10 +1633,10 @@ void Entity::PickupItem(const LWOOBJID& objectID) { const CDObjects& object = objectsTable->GetByID(p.second.lot); if (object.id != 0 && object.type == "Powerup") { - CDObjectSkillsTable* skillsTable = CDClientManager::Instance()->GetTable("ObjectSkills"); + CDObjectSkillsTable* skillsTable = CDClientManager::Instance().GetTable(); std::vector skills = skillsTable->Query([=](CDObjectSkills entry) {return (entry.objectTemplate == p.second.lot); }); for (CDObjectSkills skill : skills) { - CDSkillBehaviorTable* skillBehTable = CDClientManager::Instance()->GetTable("SkillBehavior"); + CDSkillBehaviorTable* skillBehTable = CDClientManager::Instance().GetTable(); CDSkillBehavior behaviorData = skillBehTable->GetSkillByID(skill.skillID); SkillComponent::HandleUnmanaged(behaviorData.behaviorID, GetObjectID()); @@ -1582,11 +1644,11 @@ void Entity::PickupItem(const LWOOBJID& objectID) { auto* missionComponent = GetComponent(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_POWERUP, skill.skillID); + missionComponent->Progress(eMissionTaskType::POWERUP, skill.skillID); } } } else { - inv->AddItem(p.second.lot, p.second.count, eLootSourceType::LOOT_SOURCE_PICKUP, eInventoryType::INVALID, {}, LWOOBJID_EMPTY, true, false, LWOOBJID_EMPTY, eInventoryType::INVALID, 1); + inv->AddItem(p.second.lot, p.second.count, eLootSourceType::PICKUP, eInventoryType::INVALID, {}, LWOOBJID_EMPTY, true, false, LWOOBJID_EMPTY, eInventoryType::INVALID, 1); } } } @@ -1666,7 +1728,7 @@ void Entity::CancelCallbackTimers() { void Entity::ScheduleKillAfterUpdate(Entity* murderer) { //if (m_Info.spawner) m_Info.spawner->ScheduleKill(this); - EntityManager::Instance()->ScheduleForKill(this); + Game::entityManager->ScheduleForKill(this); if (murderer) m_ScheduleKiller = murderer; } @@ -1703,99 +1765,14 @@ bool Entity::IsPlayer() const { return m_TemplateID == 1 && GetSystemAddress() != UNASSIGNED_SYSTEM_ADDRESS; } -void Entity::TriggerEvent(std::string eventID, Entity* optionalTarget) { - if (m_Trigger != nullptr && m_Trigger->enabled) { - for (LUTriggers::Event* triggerEvent : m_Trigger->events) { - if (triggerEvent->eventID == eventID) { - for (LUTriggers::Command* cmd : triggerEvent->commands) { - HandleTriggerCommand(cmd->id, cmd->target, cmd->targetName, cmd->args, optionalTarget); - } - } - } - } -} - -// This should probably get it's own triggers class at some point... -void Entity::HandleTriggerCommand(std::string id, std::string target, std::string targetName, std::string args, Entity* optionalTarget) { - std::vector argArray; - // Parse args - std::stringstream ssData(args); - std::string token; - char deliminator = ','; - - while (std::getline(ssData, token, deliminator)) { - std::string lowerToken; - for (char character : token) { - lowerToken.push_back(std::tolower(character)); // make lowercase to ensure it works - } - argArray.push_back(lowerToken); - } - - std::vector targetEntities; - if (target == "self") targetEntities.push_back(this); - if (target == "objGroup") targetEntities = EntityManager::Instance()->GetEntitiesInGroup(targetName); - if (optionalTarget) targetEntities.push_back(optionalTarget); - if (targetEntities.size() == 0) return; - for (Entity* targetEntity : targetEntities) { - if (!targetEntity) continue; - - if (id == "SetPhysicsVolumeEffect") { - PhantomPhysicsComponent* phanPhys = GetComponent(); - if (!phanPhys) return; - - phanPhys->SetPhysicsEffectActive(true); - uint32_t effectType = 0; - if (argArray[0] == "push") effectType = 0; - else if (argArray[0] == "attract") effectType = 1; - else if (argArray[0] == "repulse") effectType = 2; - else if (argArray[0] == "gravity") effectType = 3; - else if (argArray[0] == "friction") effectType = 4; - - phanPhys->SetEffectType(effectType); - phanPhys->SetDirectionalMultiplier(std::stof(argArray[1])); - if (argArray.size() > 4) { - NiPoint3 direction = NiPoint3::ZERO; - GeneralUtils::TryParse(argArray[2], direction.x); - GeneralUtils::TryParse(argArray[3], direction.y); - GeneralUtils::TryParse(argArray[4], direction.z); - phanPhys->SetDirection(direction); - } - if (argArray.size() > 5) { - phanPhys->SetMin(std::stoi(argArray[6])); - phanPhys->SetMax(std::stoi(argArray[7])); - } - - if (target == "self") { - EntityManager::Instance()->ConstructEntity(this); - } - } else if (id == "updateMission") { - CDMissionTasksTable* missionTasksTable = CDClientManager::Instance()->GetTable("MissionTasks"); - std::vector missionTasks = missionTasksTable->Query([=](CDMissionTasks entry) { - std::string lowerTargetGroup; - for (char character : entry.targetGroup) { - lowerTargetGroup.push_back(std::tolower(character)); // make lowercase to ensure it works - } - - return (lowerTargetGroup == argArray[4]); - }); - - for (const CDMissionTasks& task : missionTasks) { - MissionComponent* missionComponent = targetEntity->GetComponent(); - if (!missionComponent) continue; - - missionComponent->ForceProgress(task.id, task.uid, std::stoi(argArray[2])); - } - } else if (id == "fireEvent") { - for (CppScripts::Script* script : CppScripts::GetEntityScripts(targetEntity)) { - script->OnFireEventServerSide(targetEntity, this, args, 0, 0, 0); - } - } - } +void Entity::TriggerEvent(eTriggerEventType event, Entity* optionalTarget) { + auto* triggerComponent = GetComponent(); + if (triggerComponent) triggerComponent->TriggerEvent(event, optionalTarget); } Entity* Entity::GetOwner() const { if (m_OwnerOverride != LWOOBJID_EMPTY) { - auto* other = EntityManager::Instance()->GetEntity(m_OwnerOverride); + auto* other = Game::entityManager->GetEntity(m_OwnerOverride); if (other != nullptr) { return other->GetOwner(); @@ -1859,8 +1836,6 @@ bool Entity::IsSleeping() const { const NiPoint3& Entity::GetPosition() const { - if (!this) return NiPoint3::ZERO; - auto* controllable = GetComponent(); if (controllable != nullptr) { @@ -1941,7 +1916,7 @@ void Entity::SetPosition(NiPoint3 position) { vehicel->SetPosition(position); } - EntityManager::Instance()->SerializeEntity(this); + Game::entityManager->SerializeEntity(this); } void Entity::SetRotation(NiQuaternion rotation) { @@ -1969,7 +1944,7 @@ void Entity::SetRotation(NiQuaternion rotation) { vehicel->SetRotation(rotation); } - EntityManager::Instance()->SerializeEntity(this); + Game::entityManager->SerializeEntity(this); } bool Entity::GetBoolean(const std::u16string& name) const { @@ -2021,7 +1996,7 @@ std::vector& Entity::GetTargetsInPhantom() { for (auto i = 0u; i < m_TargetsInPhantom.size(); ++i) { const auto id = m_TargetsInPhantom.at(i); - auto* entity = EntityManager::Instance()->GetEntity(id); + auto* entity = Game::entityManager->GetEntity(id); if (entity == nullptr) { continue; diff --git a/dGame/Entity.h b/dGame/Entity.h index 6c0968f8..ca007912 100644 --- a/dGame/Entity.h +++ b/dGame/Entity.h @@ -4,29 +4,42 @@ #include #include #include +#include #include -#include "../thirdparty/raknet/Source/Replica.h" -#include "../thirdparty/raknet/Source/ReplicaManager.h" - -#include "dCommonVars.h" -#include "User.h" #include "NiPoint3.h" #include "NiQuaternion.h" #include "LDFFormat.h" -#include "Loot.h" -#include "Zone.h" +#include "eKillType.h" -#include "EntityTimer.h" -#include "EntityCallbackTimer.h" -#include "EntityInfo.h" +namespace Loot { + class Info; +}; + +namespace tinyxml2 { + class XMLDocument; +}; class Player; +class EntityInfo; +class User; class Spawner; class ScriptComponent; class dpEntity; +class EntityTimer; class Component; +class Item; class Character; +class EntityCallbackTimer; +enum class eTriggerEventType; +enum class eGameMasterLevel : uint8_t; +enum class eReplicaComponentType : uint32_t; +enum class eReplicaPacketType : uint8_t; +enum class eCinematicEvent : uint32_t; + +namespace CppScripts { + class Script; +}; /** * An entity in the world. Has multiple components. @@ -51,14 +64,12 @@ public: Character* GetCharacter() const { return m_Character; } - uint8_t GetGMLevel() const { return m_GMLevel; } + eGameMasterLevel GetGMLevel() const { return m_GMLevel; } uint8_t GetCollectibleID() const { return uint8_t(m_CollectibleID); } Entity* GetParentEntity() const { return m_ParentEntity; } - LUTriggers::Trigger* GetTrigger() const { return m_Trigger; } - std::vector& GetGroups() { return m_Groups; }; Spawner* GetSpawner() const { return m_Spawner; } @@ -74,6 +85,7 @@ public: bool GetPlayerReadyForUpdates() const { return m_PlayerIsReadyForUpdates; } bool GetIsGhostingCandidate() const; + void SetIsGhostingCandidate(bool value) { m_IsGhostingCandidate = value; }; int8_t GetObservers() const; @@ -101,7 +113,7 @@ public: void SetCharacter(Character* value) { m_Character = value; } - void SetGMLevel(uint8_t value); + void SetGMLevel(eGameMasterLevel value); void SetOwnerOverride(LWOOBJID value); @@ -125,20 +137,23 @@ public: * Component management */ - Component* GetComponent(int32_t componentID) const; + Component* GetComponent(eReplicaComponentType componentID) const; template T* GetComponent() const; template - bool TryGetComponent(int32_t componentId, T*& component) const; + bool TryGetComponent(eReplicaComponentType componentId, T*& component) const; - bool HasComponent(int32_t componentId) const; + bool HasComponent(eReplicaComponentType componentId) const; - void AddComponent(int32_t componentId, Component* component); + void AddComponent(eReplicaComponentType componentId, Component* component); std::vector GetScriptComponents(); + void Subscribe(LWOOBJID scriptObjId, CppScripts::Script* scriptToAdd, const std::string& notificationName); + void Unsubscribe(LWOOBJID scriptObjId, const std::string& notificationName); + void SetProximityRadius(float proxRadius, std::string name); void SetProximityRadius(dpEntity* entity, std::string name); @@ -155,7 +170,7 @@ public: void AddToGroup(const std::string& group); bool IsPlayer() const; - std::unordered_map& GetComponents() { return m_Components; } // TODO: Remove + std::unordered_map& GetComponents() { return m_Components; } // TODO: Remove void WriteBaseReplicaData(RakNet::BitStream* outBitStream, eReplicaPacketType packetType); void WriteComponents(RakNet::BitStream* outBitStream, eReplicaPacketType packetType); @@ -193,6 +208,7 @@ public: void OnMessageBoxResponse(Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData); void OnChoiceBoxResponse(Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier); + void RequestActivityExit(Entity* sender, LWOOBJID player, bool canceled); void Smash(const LWOOBJID source = LWOOBJID_EMPTY, const eKillType killType = eKillType::VIOLENT, const std::u16string& deathType = u""); void Kill(Entity* murderer = nullptr); @@ -208,9 +224,8 @@ public: void RegisterCoinDrop(uint64_t count); void ScheduleKillAfterUpdate(Entity* murderer = nullptr); - void TriggerEvent(std::string eveneventtID, Entity* optionalTarget = nullptr); + void TriggerEvent(eTriggerEventType event, Entity* optionalTarget = nullptr); void ScheduleDestructionAfterUpdate() { m_ShouldDestroyAfterUpdate = true; } - void HandleTriggerCommand(std::string id, std::string target, std::string targetName, std::string args, Entity* optionalTarget); virtual NiPoint3 GetRespawnPosition() const { return NiPoint3::ZERO; } virtual NiQuaternion GetRespawnRotation() const { return NiQuaternion::IDENTITY; } @@ -295,20 +310,18 @@ protected: bool m_HasSpawnerNodeID; uint32_t m_SpawnerNodeID; - LUTriggers::Trigger* m_Trigger; - Character* m_Character; Entity* m_ParentEntity; //For spawners and the like std::vector m_ChildEntities; - uint8_t m_GMLevel; + eGameMasterLevel m_GMLevel; uint16_t m_CollectibleID; std::vector m_Groups; uint16_t m_NetworkID; std::vector> m_DieCallbacks; std::vector> m_PhantomCollisionCallbacks; - std::unordered_map m_Components; //The int is the ID of the component + std::unordered_map m_Components; std::vector m_Timers; std::vector m_PendingTimers; std::vector m_CallbackTimers; @@ -338,7 +351,7 @@ protected: */ template -bool Entity::TryGetComponent(const int32_t componentId, T*& component) const { +bool Entity::TryGetComponent(const eReplicaComponentType componentId, T*& component) const { const auto& index = m_Components.find(componentId); if (index == m_Components.end()) { diff --git a/dGame/EntityManager.cpp b/dGame/EntityManager.cpp index 02b8b59e..28b5f526 100644 --- a/dGame/EntityManager.cpp +++ b/dGame/EntityManager.cpp @@ -17,8 +17,13 @@ #include "MissionComponent.h" #include "Game.h" #include "dLogger.h" - -EntityManager* EntityManager::m_Address = nullptr; +#include "MessageIdentifiers.h" +#include "dConfig.h" +#include "eTriggerEventType.h" +#include "eObjectBits.h" +#include "eGameMasterLevel.h" +#include "eReplicaComponentType.h" +#include "eReplicaPacketType.h" // Configure which zones have ghosting disabled, mostly small worlds. std::vector EntityManager::m_GhostingExcludedZones = { @@ -55,11 +60,22 @@ void EntityManager::Initialize() { m_GhostingEnabled = std::find( m_GhostingExcludedZones.begin(), m_GhostingExcludedZones.end(), - dZoneManager::Instance()->GetZoneID().GetMapID() + Game::zoneManager->GetZoneID().GetMapID() ) == m_GhostingExcludedZones.end(); -} -EntityManager::~EntityManager() { + // grab hardcore mode settings and load them with sane defaults + auto hcmode = Game::config->GetValue("hardcore_mode"); + m_HardcoreMode = hcmode.empty() ? false : (hcmode == "1"); + auto hcUscorePercent = Game::config->GetValue("hardcore_lose_uscore_on_death_percent"); + m_HardcoreLoseUscoreOnDeathPercent = hcUscorePercent.empty() ? 10 : std::stoi(hcUscorePercent); + auto hcUscoreMult = Game::config->GetValue("hardcore_uscore_enemies_multiplier"); + m_HardcoreUscoreEnemiesMultiplier = hcUscoreMult.empty() ? 2 : std::stoi(hcUscoreMult); + auto hcDropInv = Game::config->GetValue("hardcore_dropinventory_on_death"); + m_HardcoreDropinventoryOnDeath = hcDropInv.empty() ? false : (hcDropInv == "1"); + + // If cloneID is not zero, then hardcore mode is disabled + // aka minigames and props + if (Game::zoneManager->GetZoneID().GetCloneID() != 0) m_HardcoreMode = false; } Entity* EntityManager::CreateEntity(EntityInfo info, User* user, Entity* parentEntity, const bool controller, const LWOOBJID explicitId) { @@ -89,11 +105,11 @@ Entity* EntityManager::CreateEntity(EntityInfo info, User* user, Entity* parentE if (!controller && info.lot != 14) { // The client flags means the client should render the entity - id = GeneralUtils::SetBit(id, OBJECT_BIT_CLIENT); + GeneralUtils::SetBit(id, eObjectBits::CLIENT); // Spawned entities require the spawned flag to render if (info.spawnerID != 0) { - id = GeneralUtils::SetBit(id, OBJECT_BIT_SPAWNED); + GeneralUtils::SetBit(id, eObjectBits::SPAWNED); } } } @@ -140,9 +156,9 @@ void EntityManager::DestroyEntity(const LWOOBJID& objectID) { } void EntityManager::DestroyEntity(Entity* entity) { - if (entity == nullptr) { - return; - } + if (!entity) return; + + entity->TriggerEvent(eTriggerEventType::DESTROY, entity); const auto id = entity->GetObjectID(); @@ -159,15 +175,11 @@ void EntityManager::DestroyEntity(Entity* entity) { ScheduleForDeletion(id); } -void EntityManager::UpdateEntities(const float deltaTime) { - for (const auto& e : m_Entities) { - e.second->Update(deltaTime); - } - +void EntityManager::SerializeEntities() { for (auto entry = m_EntitiesToSerialize.begin(); entry != m_EntitiesToSerialize.end(); entry++) { auto* entity = GetEntity(*entry); - if (entity == nullptr) continue; + if (!entity) continue; m_SerializationCounter++; @@ -175,8 +187,8 @@ void EntityManager::UpdateEntities(const float deltaTime) { stream.Write(static_cast(ID_REPLICA_MANAGER_SERIALIZE)); stream.Write(static_cast(entity->GetNetworkId())); - entity->WriteBaseReplicaData(&stream, PACKET_TYPE_SERIALIZATION); - entity->WriteComponents(&stream, PACKET_TYPE_SERIALIZATION); + entity->WriteBaseReplicaData(&stream, eReplicaPacketType::SERIALIZATION); + entity->WriteComponents(&stream, eReplicaPacketType::SERIALIZATION); if (entity->GetIsGhostingCandidate()) { for (auto* player : Player::GetAllPlayers()) { @@ -189,45 +201,59 @@ void EntityManager::UpdateEntities(const float deltaTime) { } } m_EntitiesToSerialize.clear(); +} +void EntityManager::KillEntities() { for (auto entry = m_EntitiesToKill.begin(); entry != m_EntitiesToKill.end(); entry++) { auto* entity = GetEntity(*entry); - if (!entity) continue; + if (!entity) { + Game::logger->Log("EntityManager", "Attempting to kill null entity %llu", *entry); + continue; + } if (entity->GetScheduledKiller()) { - entity->Smash(entity->GetScheduledKiller()->GetObjectID(), SILENT); + entity->Smash(entity->GetScheduledKiller()->GetObjectID(), eKillType::SILENT); } else { - entity->Smash(LWOOBJID_EMPTY, SILENT); + entity->Smash(LWOOBJID_EMPTY, eKillType::SILENT); } } m_EntitiesToKill.clear(); +} +void EntityManager::DeleteEntities() { for (auto entry = m_EntitiesToDelete.begin(); entry != m_EntitiesToDelete.end(); entry++) { - - // Get all this info first before we delete the player. auto entityToDelete = GetEntity(*entry); - auto networkIdToErase = entityToDelete->GetNetworkId(); - const auto& ghostingToDelete = std::find(m_EntitiesToGhost.begin(), m_EntitiesToGhost.end(), entityToDelete); - if (entityToDelete) { - // If we are a player run through the player destructor. - if (entityToDelete->IsPlayer()) { - delete dynamic_cast(entityToDelete); - } else { - delete entityToDelete; - } + // Get all this info first before we delete the player. + auto networkIdToErase = entityToDelete->GetNetworkId(); + const auto& ghostingToDelete = std::find(m_EntitiesToGhost.begin(), m_EntitiesToGhost.end(), entityToDelete); + + delete entityToDelete; + entityToDelete = nullptr; + if (networkIdToErase != 0) m_LostNetworkIds.push(networkIdToErase); + + if (ghostingToDelete != m_EntitiesToGhost.end()) m_EntitiesToGhost.erase(ghostingToDelete); + } else { + Game::logger->Log("EntityManager", "Attempted to delete non-existent entity %llu", *entry); } - - if (ghostingToDelete != m_EntitiesToGhost.end()) m_EntitiesToGhost.erase(ghostingToDelete); - m_Entities.erase(*entry); } m_EntitiesToDelete.clear(); } +void EntityManager::UpdateEntities(const float deltaTime) { + for (const auto& e : m_Entities) { + e.second->Update(deltaTime); + } + + SerializeEntities(); + KillEntities(); + DeleteEntities(); +} + Entity* EntityManager::GetEntity(const LWOOBJID& objectId) const { const auto& index = m_Entities.find(objectId); @@ -251,10 +277,10 @@ std::vector EntityManager::GetEntitiesInGroup(const std::string& group) return entitiesInGroup; } -std::vector EntityManager::GetEntitiesByComponent(const int componentType) const { +std::vector EntityManager::GetEntitiesByComponent(const eReplicaComponentType componentType) const { std::vector withComp; for (const auto& entity : m_Entities) { - if (componentType != -1 && !entity.second->HasComponent(componentType)) continue; + if (componentType != eReplicaComponentType::INVALID && !entity.second->HasComponent(componentType)) continue; withComp.push_back(entity.second); } @@ -293,6 +319,11 @@ const std::unordered_map& EntityManager::GetSpawnPointEnt } void EntityManager::ConstructEntity(Entity* entity, const SystemAddress& sysAddr, const bool skipChecks) { + if (!entity) { + Game::logger->Log("EntityManager", "Attempted to construct null entity"); + return; + } + if (entity->GetNetworkId() == 0) { uint16_t networkId; @@ -330,8 +361,8 @@ void EntityManager::ConstructEntity(Entity* entity, const SystemAddress& sysAddr stream.Write(true); stream.Write(static_cast(entity->GetNetworkId())); - entity->WriteBaseReplicaData(&stream, PACKET_TYPE_CONSTRUCTION); - entity->WriteComponents(&stream, PACKET_TYPE_CONSTRUCTION); + entity->WriteBaseReplicaData(&stream, eReplicaPacketType::CONSTRUCTION); + entity->WriteComponents(&stream, eReplicaPacketType::CONSTRUCTION); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) { if (skipChecks) { @@ -352,7 +383,7 @@ void EntityManager::ConstructEntity(Entity* entity, const SystemAddress& sysAddr // PacketUtils::SavePacket("[24]_"+std::to_string(entity->GetObjectID()) + "_" + std::to_string(m_SerializationCounter) + ".bin", (char*)stream.GetData(), stream.GetNumberOfBytesUsed()); if (entity->IsPlayer()) { - if (entity->GetGMLevel() > GAME_MASTER_LEVEL_CIVILIAN) { + if (entity->GetGMLevel() > eGameMasterLevel::CIVILIAN) { GameMessages::SendToggleGMInvis(entity->GetObjectID(), true, sysAddr); } } @@ -372,9 +403,7 @@ void EntityManager::ConstructAllEntities(const SystemAddress& sysAddr) { } void EntityManager::DestructEntity(Entity* entity, const SystemAddress& sysAddr) { - if (entity->GetNetworkId() == 0) { - return; - } + if (!entity || entity->GetNetworkId() == 0) return; RakNet::BitStream stream; @@ -391,9 +420,7 @@ void EntityManager::DestructEntity(Entity* entity, const SystemAddress& sysAddr) } void EntityManager::SerializeEntity(Entity* entity) { - if (entity->GetNetworkId() == 0) { - return; - } + if (!entity || entity->GetNetworkId() == 0) return; if (std::find(m_EntitiesToSerialize.begin(), m_EntitiesToSerialize.end(), entity->GetObjectID()) == m_EntitiesToSerialize.end()) { m_EntitiesToSerialize.push_back(entity->GetObjectID()); @@ -569,7 +596,7 @@ void EntityManager::ScheduleForKill(Entity* entity) { SwitchComponent* switchComp = entity->GetComponent(); if (switchComp) { - entity->TriggerEvent("OnDectivated"); + entity->TriggerEvent(eTriggerEventType::DEACTIVATED, entity); } const auto objectId = entity->GetObjectID(); diff --git a/dGame/EntityManager.h b/dGame/EntityManager.h index 40a98076..9bea0618 100644 --- a/dGame/EntityManager.h +++ b/dGame/EntityManager.h @@ -1,39 +1,32 @@ #ifndef ENTITYMANAGER_H #define ENTITYMANAGER_H -#include "dCommonVars.h" -#include "../thirdparty/raknet/Source/Replica.h" #include #include - -#include "Entity.h" #include +#include + +#include "dCommonVars.h" + +class Entity; +class EntityInfo; +class Player; +class User; +enum class eReplicaComponentType : uint32_t; struct SystemAddress; -class User; class EntityManager { public: - static EntityManager* Instance() { - if (!m_Address) { - m_Address = new EntityManager(); - m_Address->Initialize(); - } - - return m_Address; - } - void Initialize(); - ~EntityManager(); - void UpdateEntities(float deltaTime); Entity* CreateEntity(EntityInfo info, User* user = nullptr, Entity* parentEntity = nullptr, bool controller = false, LWOOBJID explicitId = LWOOBJID_EMPTY); void DestroyEntity(const LWOOBJID& objectID); void DestroyEntity(Entity* entity); Entity* GetEntity(const LWOOBJID& objectId) const; std::vector GetEntitiesInGroup(const std::string& group); - std::vector GetEntitiesByComponent(int componentType) const; + std::vector GetEntitiesByComponent(eReplicaComponentType componentType) const; std::vector GetEntitiesByLOT(const LOT& lot) const; Entity* GetZoneControlEntity() const; @@ -76,8 +69,16 @@ public: static bool IsExcludedFromGhosting(LOT lot); + const bool GetHardcoreMode() { return m_HardcoreMode; }; + const uint32_t GetHardcoreLoseUscoreOnDeathPercent() { return m_HardcoreLoseUscoreOnDeathPercent; }; + const bool GetHardcoreDropinventoryOnDeath() { return m_HardcoreDropinventoryOnDeath; }; + const uint32_t GetHardcoreUscoreEnemiesMultiplier() { return m_HardcoreUscoreEnemiesMultiplier; }; + private: - static EntityManager* m_Address; //For singleton method + void SerializeEntities(); + void KillEntities(); + void DeleteEntities(); + static std::vector m_GhostingExcludedZones; static std::vector m_GhostingExcludedLOTs; @@ -100,6 +101,12 @@ private: // Map of spawnname to entity object ID std::unordered_map m_SpawnPoints; + + // hardcore mode vars + bool m_HardcoreMode; + uint32_t m_HardcoreLoseUscoreOnDeathPercent; + bool m_HardcoreDropinventoryOnDeath; + uint32_t m_HardcoreUscoreEnemiesMultiplier; }; #endif // ENTITYMANAGER_H diff --git a/dGame/LeaderboardManager.cpp b/dGame/LeaderboardManager.cpp index b069e761..0e64b6b3 100644 --- a/dGame/LeaderboardManager.cpp +++ b/dGame/LeaderboardManager.cpp @@ -1,5 +1,8 @@ #include "LeaderboardManager.h" + +#include #include + #include "Database.h" #include "EntityManager.h" #include "Character.h" @@ -7,459 +10,403 @@ #include "GameMessages.h" #include "dLogger.h" #include "dConfig.h" +#include "CDClientManager.h" +#include "GeneralUtils.h" +#include "Entity.h" +#include "LDFFormat.h" +#include "DluAssert.h" -Leaderboard::Leaderboard(uint32_t gameID, uint32_t infoType, bool weekly, std::vector entries, - LWOOBJID relatedPlayer, LeaderboardType leaderboardType) { - this->relatedPlayer = relatedPlayer; +#include "CDActivitiesTable.h" +#include "Metrics.hpp" + +namespace LeaderboardManager { + std::map leaderboardCache; +} + +Leaderboard::Leaderboard(const GameID gameID, const Leaderboard::InfoType infoType, const bool weekly, LWOOBJID relatedPlayer, const Leaderboard::Type leaderboardType) { this->gameID = gameID; this->weekly = weekly; this->infoType = infoType; - this->entries = std::move(entries); this->leaderboardType = leaderboardType; + this->relatedPlayer = relatedPlayer; } -std::u16string Leaderboard::ToString() const { - std::string leaderboard; +Leaderboard::~Leaderboard() { + Clear(); +} - leaderboard += "ADO.Result=7:1\n"; - leaderboard += "Result.Count=1:1\n"; - leaderboard += "Result[0].Index=0:RowNumber\n"; - leaderboard += "Result[0].RowCount=1:" + std::to_string(entries.size()) + "\n"; +void Leaderboard::Clear() { + for (auto& entry : entries) for (auto ldfData : entry) delete ldfData; +} - auto index = 0; - for (const auto& entry : entries) { - leaderboard += "Result[0].Row[" + std::to_string(index) + "].LastPlayed=8:" + std::to_string(entry.lastPlayed) + "\n"; - leaderboard += "Result[0].Row[" + std::to_string(index) + "].CharacterID=8:" + std::to_string(entry.playerID) + "\n"; - leaderboard += "Result[0].Row[" + std::to_string(index) + "].NumPlayed=1:1\n"; - leaderboard += "Result[0].Row[" + std::to_string(index) + "].RowNumber=8:" + std::to_string(entry.placement) + "\n"; - leaderboard += "Result[0].Row[" + std::to_string(index) + "].Time=1:" + std::to_string(entry.time) + "\n"; +inline void WriteLeaderboardRow(std::ostringstream& leaderboard, const uint32_t& index, LDFBaseData* data) { + leaderboard << "\nResult[0].Row[" << index << "]." << data->GetString(); +} - // Only these minigames have a points system - if (leaderboardType == Survival || leaderboardType == ShootingGallery) { - leaderboard += "Result[0].Row[" + std::to_string(index) + "].Points=1:" + std::to_string(entry.score) + "\n"; - } else if (leaderboardType == SurvivalNS) { - leaderboard += "Result[0].Row[" + std::to_string(index) + "].Wave=1:" + std::to_string(entry.score) + "\n"; +void Leaderboard::Serialize(RakNet::BitStream* bitStream) const { + bitStream->Write(gameID); + bitStream->Write(infoType); + + std::ostringstream leaderboard; + + leaderboard << "ADO.Result=7:1"; // Unused in 1.10.64, but is in captures + leaderboard << "\nResult.Count=1:1"; // number of results, always 1 + if (!this->entries.empty()) leaderboard << "\nResult[0].Index=0:RowNumber"; // "Primary key". Live doesn't include this if there are no entries. + leaderboard << "\nResult[0].RowCount=1:" << entries.size(); + + int32_t rowNumber = 0; + for (auto& entry : entries) { + for (auto* data : entry) { + WriteLeaderboardRow(leaderboard, rowNumber, data); } - - leaderboard += "Result[0].Row[" + std::to_string(index) + "].name=0:" + entry.playerName + "\n"; - index++; + rowNumber++; } - return GeneralUtils::UTF8ToUTF16(leaderboard); + // Serialize the thing to a BitStream + uint32_t leaderboardSize = leaderboard.tellp(); + bitStream->Write(leaderboardSize); + // Doing this all in 1 call so there is no possbility of a dangling pointer. + bitStream->WriteAlignedBytes(reinterpret_cast(GeneralUtils::ASCIIToUTF16(leaderboard.str()).c_str()), leaderboardSize * sizeof(char16_t)); + if (leaderboardSize > 0) bitStream->Write(0); + bitStream->Write0(); + bitStream->Write0(); } -std::vector Leaderboard::GetEntries() { - return entries; +void Leaderboard::QueryToLdf(std::unique_ptr& rows) { + Clear(); + if (rows->rowsCount() == 0) return; + + this->entries.reserve(rows->rowsCount()); + while (rows->next()) { + constexpr int32_t MAX_NUM_DATA_PER_ROW = 9; + this->entries.push_back(std::vector()); + auto& entry = this->entries.back(); + entry.reserve(MAX_NUM_DATA_PER_ROW); + entry.push_back(new LDFData(u"CharacterID", rows->getInt("character_id"))); + entry.push_back(new LDFData(u"LastPlayed", rows->getUInt64("lastPlayed"))); + entry.push_back(new LDFData(u"NumPlayed", rows->getInt("timesPlayed"))); + entry.push_back(new LDFData(u"name", GeneralUtils::ASCIIToUTF16(rows->getString("name").c_str()))); + entry.push_back(new LDFData(u"RowNumber", rows->getInt("ranking"))); + switch (leaderboardType) { + case Type::ShootingGallery: + entry.push_back(new LDFData(u"Score", rows->getInt("primaryScore"))); + // Score:1 + entry.push_back(new LDFData(u"Streak", rows->getInt("secondaryScore"))); + // Streak:1 + entry.push_back(new LDFData(u"HitPercentage", (rows->getInt("tertiaryScore") / 100.0f))); + // HitPercentage:3 between 0 and 1 + break; + case Type::Racing: + entry.push_back(new LDFData(u"BestTime", rows->getDouble("primaryScore"))); + // BestLapTime:3 + entry.push_back(new LDFData(u"BestLapTime", rows->getDouble("secondaryScore"))); + // BestTime:3 + entry.push_back(new LDFData(u"License", 1)); + // License:1 - 1 if player has completed mission 637 and 0 otherwise + entry.push_back(new LDFData(u"NumWins", rows->getInt("numWins"))); + // NumWins:1 + break; + case Type::UnusedLeaderboard4: + entry.push_back(new LDFData(u"Points", rows->getInt("primaryScore"))); + // Points:1 + break; + case Type::MonumentRace: + entry.push_back(new LDFData(u"Time", rows->getInt("primaryScore"))); + // Time:1(?) + break; + case Type::FootRace: + entry.push_back(new LDFData(u"Time", rows->getInt("primaryScore"))); + // Time:1 + break; + case Type::Survival: + entry.push_back(new LDFData(u"Points", rows->getInt("primaryScore"))); + // Points:1 + entry.push_back(new LDFData(u"Time", rows->getInt("secondaryScore"))); + // Time:1 + break; + case Type::SurvivalNS: + entry.push_back(new LDFData(u"Wave", rows->getInt("primaryScore"))); + // Wave:1 + entry.push_back(new LDFData(u"Time", rows->getInt("secondaryScore"))); + // Time:1 + break; + case Type::Donations: + entry.push_back(new LDFData(u"Score", rows->getInt("primaryScore"))); + // Score:1 + break; + case Type::None: + // This type is included here simply to resolve a compiler warning on mac about unused enum types + break; + default: + break; + } + } } -uint32_t Leaderboard::GetGameID() const { - return gameID; +const std::string_view Leaderboard::GetOrdering(Leaderboard::Type leaderboardType) { + // Use a switch case and return desc for all 3 columns if higher is better and asc if lower is better + switch (leaderboardType) { + case Type::Racing: + case Type::MonumentRace: + return "primaryScore ASC, secondaryScore ASC, tertiaryScore ASC"; + case Type::Survival: + return Game::config->GetValue("classic_survival_scoring") == "1" ? + "secondaryScore DESC, primaryScore DESC, tertiaryScore DESC" : + "primaryScore DESC, secondaryScore DESC, tertiaryScore DESC"; + case Type::SurvivalNS: + return "primaryScore DESC, secondaryScore ASC, tertiaryScore DESC"; + case Type::ShootingGallery: + case Type::FootRace: + case Type::UnusedLeaderboard4: + case Type::Donations: + case Type::None: + default: + return "primaryScore DESC, secondaryScore DESC, tertiaryScore DESC"; + } } -uint32_t Leaderboard::GetInfoType() const { - return infoType; +void Leaderboard::SetupLeaderboard(bool weekly, uint32_t resultStart, uint32_t resultEnd) { + resultStart++; + resultEnd++; + // We need everything except 1 column so i'm selecting * from leaderboard + const std::string queryBase = + R"QUERY( + WITH leaderboardsRanked AS ( + SELECT leaderboard.*, charinfo.name, + RANK() OVER + ( + ORDER BY %s, UNIX_TIMESTAMP(last_played) ASC, id DESC + ) AS ranking + FROM leaderboard JOIN charinfo on charinfo.id = leaderboard.character_id + WHERE game_id = ? %s + ), + myStanding AS ( + SELECT + ranking as myRank + FROM leaderboardsRanked + WHERE id = ? + ), + lowestRanking AS ( + SELECT MAX(ranking) AS lowestRank + FROM leaderboardsRanked + ) + SELECT leaderboardsRanked.*, character_id, UNIX_TIMESTAMP(last_played) as lastPlayed, leaderboardsRanked.name, leaderboardsRanked.ranking FROM leaderboardsRanked, myStanding, lowestRanking + WHERE leaderboardsRanked.ranking + BETWEEN + LEAST(GREATEST(CAST(myRank AS SIGNED) - 5, %i), lowestRanking.lowestRank - 9) + AND + LEAST(GREATEST(myRank + 5, %i), lowestRanking.lowestRank) + ORDER BY ranking ASC; + )QUERY"; + + std::string friendsFilter = + R"QUERY( + AND ( + character_id IN ( + SELECT fr.requested_player FROM ( + SELECT CASE + WHEN player_id = ? THEN friend_id + WHEN friend_id = ? THEN player_id + END AS requested_player + FROM friends + ) AS fr + JOIN charinfo AS ci + ON ci.id = fr.requested_player + WHERE fr.requested_player IS NOT NULL + ) + OR character_id = ? + ) + )QUERY"; + + std::string weeklyFilter = " AND UNIX_TIMESTAMP(last_played) BETWEEN UNIX_TIMESTAMP(date_sub(now(),INTERVAL 1 WEEK)) AND UNIX_TIMESTAMP(now()) "; + + std::string filter; + // Setup our filter based on the query type + if (this->infoType == InfoType::Friends) filter += friendsFilter; + if (this->weekly) filter += weeklyFilter; + const auto orderBase = GetOrdering(this->leaderboardType); + + // For top query, we want to just rank all scores, but for all others we need the scores around a specific player + std::string baseLookup; + if (this->infoType == InfoType::Top) { + baseLookup = "SELECT id, last_played FROM leaderboard WHERE game_id = ? " + (this->weekly ? weeklyFilter : std::string("")) + " ORDER BY "; + baseLookup += orderBase.data(); + } else { + baseLookup = "SELECT id, last_played FROM leaderboard WHERE game_id = ? " + (this->weekly ? weeklyFilter : std::string("")) + " AND character_id = "; + baseLookup += std::to_string(static_cast(this->relatedPlayer)); + } + baseLookup += " LIMIT 1"; + Game::logger->LogDebug("LeaderboardManager", "query is %s", baseLookup.c_str()); + std::unique_ptr baseQuery(Database::CreatePreppedStmt(baseLookup)); + baseQuery->setInt(1, this->gameID); + std::unique_ptr baseResult(baseQuery->executeQuery()); + + if (!baseResult->next()) return; // In this case, there are no entries in the leaderboard for this game. + + uint32_t relatedPlayerLeaderboardId = baseResult->getInt("id"); + + // Create and execute the actual save here. Using a heap allocated buffer to avoid stack overflow + constexpr uint16_t STRING_LENGTH = 4096; + std::unique_ptr lookupBuffer = std::make_unique(STRING_LENGTH); + int32_t res = snprintf(lookupBuffer.get(), STRING_LENGTH, queryBase.c_str(), orderBase.data(), filter.c_str(), resultStart, resultEnd); + DluAssert(res != -1); + std::unique_ptr query(Database::CreatePreppedStmt(lookupBuffer.get())); + Game::logger->LogDebug("LeaderboardManager", "Query is %s vars are %i %i %i", lookupBuffer.get(), this->gameID, this->relatedPlayer, relatedPlayerLeaderboardId); + query->setInt(1, this->gameID); + if (this->infoType == InfoType::Friends) { + query->setInt(2, this->relatedPlayer); + query->setInt(3, this->relatedPlayer); + query->setInt(4, this->relatedPlayer); + query->setInt(5, relatedPlayerLeaderboardId); + } else { + query->setInt(2, relatedPlayerLeaderboardId); + } + std::unique_ptr result(query->executeQuery()); + QueryToLdf(result); } -void Leaderboard::Send(LWOOBJID targetID) const { - auto* player = EntityManager::Instance()->GetEntity(relatedPlayer); +void Leaderboard::Send(const LWOOBJID targetID) const { + auto* player = Game::entityManager->GetEntity(relatedPlayer); if (player != nullptr) { GameMessages::SendActivitySummaryLeaderboardData(targetID, this, player->GetSystemAddress()); } } -void LeaderboardManager::SaveScore(LWOOBJID playerID, uint32_t gameID, uint32_t score, uint32_t time) { - const auto* player = EntityManager::Instance()->GetEntity(playerID); - if (player == nullptr) - return; +std::string FormatInsert(const Leaderboard::Type& type, const Score& score, const bool useUpdate) { + std::string insertStatement; + if (useUpdate) { + insertStatement = + R"QUERY( + UPDATE leaderboard + SET primaryScore = %f, secondaryScore = %f, tertiaryScore = %f, + timesPlayed = timesPlayed + 1 WHERE character_id = ? AND game_id = ?; + )QUERY"; + } else { + insertStatement = + R"QUERY( + INSERT leaderboard SET + primaryScore = %f, secondaryScore = %f, tertiaryScore = %f, + character_id = ?, game_id = ?; + )QUERY"; + } - auto* character = player->GetCharacter(); - if (character == nullptr) - return; + constexpr uint16_t STRING_LENGTH = 400; + // Then fill in our score + char finishedQuery[STRING_LENGTH]; + int32_t res = snprintf(finishedQuery, STRING_LENGTH, insertStatement.c_str(), score.GetPrimaryScore(), score.GetSecondaryScore(), score.GetTertiaryScore()); + DluAssert(res != -1); + return finishedQuery; +} - auto* select = Database::CreatePreppedStmt("SELECT time, score FROM leaderboard WHERE character_id = ? AND game_id = ?;"); +void LeaderboardManager::SaveScore(const LWOOBJID& playerID, const GameID activityId, const float primaryScore, const float secondaryScore, const float tertiaryScore) { + const Leaderboard::Type leaderboardType = GetLeaderboardType(activityId); - select->setUInt64(1, character->GetID()); - select->setInt(2, gameID); - - auto any = false; - auto* result = select->executeQuery(); - auto leaderboardType = GetLeaderboardType(gameID); - - // Check if the new score is a high score - while (result->next()) { - any = true; - - const auto storedTime = result->getInt(1); - const auto storedScore = result->getInt(2); - auto highscore = true; - bool classicSurvivalScoring = Game::config->GetValue("classic_survival_scoring") == "1"; + std::unique_ptr query(Database::CreatePreppedStmt("SELECT * FROM leaderboard WHERE character_id = ? AND game_id = ?;")); + query->setInt(1, playerID); + query->setInt(2, activityId); + std::unique_ptr myScoreResult(query->executeQuery()); + std::string saveQuery("UPDATE leaderboard SET timesPlayed = timesPlayed + 1 WHERE character_id = ? AND game_id = ?;"); + Score newScore(primaryScore, secondaryScore, tertiaryScore); + if (myScoreResult->next()) { + Score oldScore; + bool lowerScoreBetter = false; switch (leaderboardType) { - case ShootingGallery: - if (score <= storedScore) - highscore = false; + // Higher score better + case Leaderboard::Type::ShootingGallery: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + oldScore.SetSecondaryScore(myScoreResult->getInt("secondaryScore")); + oldScore.SetTertiaryScore(myScoreResult->getInt("tertiaryScore")); break; - case Racing: - if (time >= storedTime) - highscore = false; - break; - case MonumentRace: - if (time >= storedTime) - highscore = false; - break; - case FootRace: - if (time <= storedTime) - highscore = false; - break; - case Survival: - if (classicSurvivalScoring) { - if (time <= storedTime) { // Based on time (LU live) - highscore = false; - } - } else { - if (score <= storedScore) // Based on score (DLU) - highscore = false; - } - break; - case SurvivalNS: - if (!(score > storedScore || (time < storedTime && score >= storedScore))) - highscore = false; - break; - default: - highscore = false; } + case Leaderboard::Type::FootRace: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + break; + } + case Leaderboard::Type::Survival: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + oldScore.SetSecondaryScore(myScoreResult->getInt("secondaryScore")); + break; + } + case Leaderboard::Type::SurvivalNS: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + oldScore.SetSecondaryScore(myScoreResult->getInt("secondaryScore")); + break; + } + case Leaderboard::Type::UnusedLeaderboard4: + case Leaderboard::Type::Donations: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + newScore.SetPrimaryScore(oldScore.GetPrimaryScore() + newScore.GetPrimaryScore()); + break; + } + case Leaderboard::Type::Racing: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + oldScore.SetSecondaryScore(myScoreResult->getInt("secondaryScore")); - if (!highscore) { - delete select; - delete result; + // For wins we dont care about the score, just the time, so zero out the tertiary. + // Wins are updated later. + oldScore.SetTertiaryScore(0); + newScore.SetTertiaryScore(0); + lowerScoreBetter = true; + break; + } + case Leaderboard::Type::MonumentRace: { + oldScore.SetPrimaryScore(myScoreResult->getInt("primaryScore")); + lowerScoreBetter = true; + // Do score checking here + break; + } + case Leaderboard::Type::None: + default: + Game::logger->Log("LeaderboardManager", "Unknown leaderboard type %i for game %i. Cannot save score!", leaderboardType, activityId); return; } - } - - delete select; - delete result; - - if (any) { - auto* statement = Database::CreatePreppedStmt("UPDATE leaderboard SET time = ?, score = ?, last_played=SYSDATE() WHERE character_id = ? AND game_id = ?;"); - statement->setInt(1, time); - statement->setInt(2, score); - statement->setUInt64(3, character->GetID()); - statement->setInt(4, gameID); - statement->execute(); - - delete statement; + bool newHighScore = lowerScoreBetter ? newScore < oldScore : newScore > oldScore; + // Nimbus station has a weird leaderboard where we need a custom scoring system + if (leaderboardType == Leaderboard::Type::SurvivalNS) { + newHighScore = newScore.GetPrimaryScore() > oldScore.GetPrimaryScore() || + (newScore.GetPrimaryScore() == oldScore.GetPrimaryScore() && newScore.GetSecondaryScore() < oldScore.GetSecondaryScore()); + } else if (leaderboardType == Leaderboard::Type::Survival && Game::config->GetValue("classic_survival_scoring") == "1") { + Score oldScoreFlipped(oldScore.GetSecondaryScore(), oldScore.GetPrimaryScore()); + Score newScoreFlipped(newScore.GetSecondaryScore(), newScore.GetPrimaryScore()); + newHighScore = newScoreFlipped > oldScoreFlipped; + } + if (newHighScore) { + saveQuery = FormatInsert(leaderboardType, newScore, true); + } } else { - // Note: last_played will be set to SYSDATE() by default when inserting into leaderboard - auto* statement = Database::CreatePreppedStmt("INSERT INTO leaderboard (character_id, game_id, time, score) VALUES (?, ?, ?, ?);"); - statement->setUInt64(1, character->GetID()); - statement->setInt(2, gameID); - statement->setInt(3, time); - statement->setInt(4, score); - statement->execute(); + saveQuery = FormatInsert(leaderboardType, newScore, false); + } + Game::logger->Log("LeaderboardManager", "save query %s %i %i", saveQuery.c_str(), playerID, activityId); + std::unique_ptr saveStatement(Database::CreatePreppedStmt(saveQuery)); + saveStatement->setInt(1, playerID); + saveStatement->setInt(2, activityId); + saveStatement->execute(); - delete statement; + // track wins separately + if (leaderboardType == Leaderboard::Type::Racing && tertiaryScore != 0.0f) { + std::unique_ptr winUpdate(Database::CreatePreppedStmt("UPDATE leaderboard SET numWins = numWins + 1 WHERE character_id = ? AND game_id = ?;")); + winUpdate->setInt(1, playerID); + winUpdate->setInt(2, activityId); + winUpdate->execute(); } } -Leaderboard* LeaderboardManager::GetLeaderboard(uint32_t gameID, InfoType infoType, bool weekly, LWOOBJID playerID) { - auto leaderboardType = GetLeaderboardType(gameID); - - std::string query; - bool classicSurvivalScoring = Game::config->GetValue("classic_survival_scoring") == "1"; - switch (infoType) { - case InfoType::Standings: - switch (leaderboardType) { - case ShootingGallery: - query = standingsScoreQuery; // Shooting gallery is based on the highest score. - break; - case FootRace: - query = standingsTimeQuery; // The higher your time, the better for FootRace. - break; - case Survival: - query = classicSurvivalScoring ? standingsTimeQuery : standingsScoreQuery; - break; - case SurvivalNS: - query = standingsScoreQueryAsc; // BoNS is scored by highest wave (score) first, then time. - break; - default: - query = standingsTimeQueryAsc; // MonumentRace and Racing are based on the shortest time. - } - break; - case InfoType::Friends: - switch (leaderboardType) { - case ShootingGallery: - query = friendsScoreQuery; // Shooting gallery is based on the highest score. - break; - case FootRace: - query = friendsTimeQuery; // The higher your time, the better for FootRace. - break; - case Survival: - query = classicSurvivalScoring ? friendsTimeQuery : friendsScoreQuery; - break; - case SurvivalNS: - query = friendsScoreQueryAsc; // BoNS is scored by highest wave (score) first, then time. - break; - default: - query = friendsTimeQueryAsc; // MonumentRace and Racing are based on the shortest time. - } - break; - - default: - switch (leaderboardType) { - case ShootingGallery: - query = topPlayersScoreQuery; // Shooting gallery is based on the highest score. - break; - case FootRace: - query = topPlayersTimeQuery; // The higher your time, the better for FootRace. - break; - case Survival: - query = classicSurvivalScoring ? topPlayersTimeQuery : topPlayersScoreQuery; - break; - case SurvivalNS: - query = topPlayersScoreQueryAsc; // BoNS is scored by highest wave (score) first, then time. - break; - default: - query = topPlayersTimeQueryAsc; // MonumentRace and Racing are based on the shortest time. - } - } - - auto* statement = Database::CreatePreppedStmt(query); - statement->setUInt(1, gameID); - - // Only the standings and friends leaderboards require the character ID to be set - if (infoType == Standings || infoType == Friends) { - auto characterID = 0; - - const auto* player = EntityManager::Instance()->GetEntity(playerID); - if (player != nullptr) { - auto* character = player->GetCharacter(); - if (character != nullptr) - characterID = character->GetID(); - } - - statement->setUInt64(2, characterID); - } - - auto* res = statement->executeQuery(); - - std::vector entries{}; - - uint32_t index = 0; - while (res->next()) { - LeaderboardEntry entry; - entry.playerID = res->getUInt64(4); - entry.playerName = res->getString(5); - entry.time = res->getUInt(1); - entry.score = res->getUInt(2); - entry.placement = res->getUInt(3); - entry.lastPlayed = res->getUInt(6); - - entries.push_back(entry); - index++; - } - - delete res; - delete statement; - - return new Leaderboard(gameID, infoType, weekly, entries, playerID, leaderboardType); +void LeaderboardManager::SendLeaderboard(const GameID gameID, const Leaderboard::InfoType infoType, const bool weekly, const LWOOBJID playerID, const LWOOBJID targetID, const uint32_t resultStart, const uint32_t resultEnd) { + Leaderboard leaderboard(gameID, infoType, weekly, playerID, GetLeaderboardType(gameID)); + leaderboard.SetupLeaderboard(weekly, resultStart, resultEnd); + leaderboard.Send(targetID); } -void LeaderboardManager::SendLeaderboard(uint32_t gameID, InfoType infoType, bool weekly, LWOOBJID targetID, - LWOOBJID playerID) { - const auto* leaderboard = LeaderboardManager::GetLeaderboard(gameID, infoType, weekly, playerID); - leaderboard->Send(targetID); - delete leaderboard; -} +Leaderboard::Type LeaderboardManager::GetLeaderboardType(const GameID gameID) { + auto lookup = leaderboardCache.find(gameID); + if (lookup != leaderboardCache.end()) return lookup->second; -LeaderboardType LeaderboardManager::GetLeaderboardType(uint32_t gameID) { - auto* activitiesTable = CDClientManager::Instance()->GetTable("Activities"); - std::vector activities = activitiesTable->Query([=](const CDActivities& entry) { - return (entry.ActivityID == gameID); + auto* activitiesTable = CDClientManager::Instance().GetTable(); + std::vector activities = activitiesTable->Query([gameID](const CDActivities& entry) { + return entry.ActivityID == gameID; }); - - for (const auto& activity : activities) { - return static_cast(activity.leaderboardType); - } - - return LeaderboardType::None; + auto type = !activities.empty() ? static_cast(activities.at(0).leaderboardType) : Leaderboard::Type::None; + leaderboardCache.insert_or_assign(gameID, type); + return type; } - -const std::string LeaderboardManager::topPlayersScoreQuery = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -"RANK() OVER ( ORDER BY l.score DESC, l.time DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -"INNER JOIN charinfo c ON l.character_id = c.id " -"WHERE l.game_id = ? " -"ORDER BY leaderboard_rank) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales LIMIT 11;"; - -const std::string LeaderboardManager::friendsScoreQuery = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, " -" RANK() OVER ( ORDER BY l.score DESC, l.time DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" INNER JOIN friends f ON f.player_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -" personal_values AS ( " -" SELECT id as related_player_id, " -" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);"; - -const std::string LeaderboardManager::standingsScoreQuery = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -" RANK() OVER ( ORDER BY l.score DESC, l.time DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -"personal_values AS ( " -" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;"; - -const std::string LeaderboardManager::topPlayersScoreQueryAsc = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -"RANK() OVER ( ORDER BY l.score DESC, l.time ASC, last_played ) leaderboard_rank " -" FROM leaderboard l " -"INNER JOIN charinfo c ON l.character_id = c.id " -"WHERE l.game_id = ? " -"ORDER BY leaderboard_rank) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales LIMIT 11;"; - -const std::string LeaderboardManager::friendsScoreQueryAsc = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, " -" RANK() OVER ( ORDER BY l.score DESC, l.time ASC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" INNER JOIN friends f ON f.player_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -" personal_values AS ( " -" SELECT id as related_player_id, " -" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);"; - -const std::string LeaderboardManager::standingsScoreQueryAsc = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -" RANK() OVER ( ORDER BY l.score DESC, l.time ASC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -"personal_values AS ( " -" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;"; - -const std::string LeaderboardManager::topPlayersTimeQuery = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -"RANK() OVER ( ORDER BY l.time DESC, l.score DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -"INNER JOIN charinfo c ON l.character_id = c.id " -"WHERE l.game_id = ? " -"ORDER BY leaderboard_rank) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales LIMIT 11;"; - -const std::string LeaderboardManager::friendsTimeQuery = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, " -" RANK() OVER ( ORDER BY l.time DESC, l.score DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" INNER JOIN friends f ON f.player_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -" personal_values AS ( " -" SELECT id as related_player_id, " -" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);"; - -const std::string LeaderboardManager::standingsTimeQuery = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -" RANK() OVER ( ORDER BY l.time DESC, l.score DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -"personal_values AS ( " -" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;"; - -const std::string LeaderboardManager::topPlayersTimeQueryAsc = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -"RANK() OVER ( ORDER BY l.time ASC, l.score DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -"INNER JOIN charinfo c ON l.character_id = c.id " -"WHERE l.game_id = ? " -"ORDER BY leaderboard_rank) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales LIMIT 11;"; - -const std::string LeaderboardManager::friendsTimeQueryAsc = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, " -" RANK() OVER ( ORDER BY l.time ASC, l.score DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" INNER JOIN friends f ON f.player_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -" personal_values AS ( " -" SELECT id as related_player_id, " -" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);"; - -const std::string LeaderboardManager::standingsTimeQueryAsc = -"WITH leaderboard_vales AS ( " -" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, " -" RANK() OVER ( ORDER BY l.time ASC, l.score DESC, last_played ) leaderboard_rank " -" FROM leaderboard l " -" INNER JOIN charinfo c ON l.character_id = c.id " -" WHERE l.game_id = ? " -" ORDER BY leaderboard_rank), " -"personal_values AS ( " -" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, " -" GREATEST(leaderboard_rank + 5, 11) AS max_rank " -" FROM leaderboard_vales WHERE id = ? LIMIT 1) " -"SELECT time, score, leaderboard_rank, id, name, last_played " -"FROM leaderboard_vales, personal_values " -"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;"; diff --git a/dGame/LeaderboardManager.h b/dGame/LeaderboardManager.h index cabdf2d6..e2ce3f97 100644 --- a/dGame/LeaderboardManager.h +++ b/dGame/LeaderboardManager.h @@ -1,80 +1,134 @@ -#pragma once +#ifndef __LEADERBOARDMANAGER__H__ +#define __LEADERBOARDMANAGER__H__ + +#include +#include +#include #include -#include + +#include "Singleton.h" #include "dCommonVars.h" +#include "LDFFormat.h" -struct LeaderboardEntry { - uint64_t playerID; - std::string playerName; - uint32_t time; - uint32_t score; - uint32_t placement; - time_t lastPlayed; +namespace sql { + class ResultSet; }; -enum InfoType : uint32_t { - Top, // Top 11 all time players - Standings, // Ranking of the current player - Friends // Ranking between friends +namespace RakNet { + class BitStream; }; -enum LeaderboardType : uint32_t { - ShootingGallery, - Racing, - MonumentRace, - FootRace, - Survival = 5, - SurvivalNS = 6, - None = UINT_MAX +class Score { +public: + Score() { + primaryScore = 0; + secondaryScore = 0; + tertiaryScore = 0; + } + Score(const float primaryScore, const float secondaryScore = 0, const float tertiaryScore = 0) { + this->primaryScore = primaryScore; + this->secondaryScore = secondaryScore; + this->tertiaryScore = tertiaryScore; + } + bool operator<(const Score& rhs) const { + return primaryScore < rhs.primaryScore || (primaryScore == rhs.primaryScore && secondaryScore < rhs.secondaryScore) || (primaryScore == rhs.primaryScore && secondaryScore == rhs.secondaryScore && tertiaryScore < rhs.tertiaryScore); + } + bool operator>(const Score& rhs) const { + return primaryScore > rhs.primaryScore || (primaryScore == rhs.primaryScore && secondaryScore > rhs.secondaryScore) || (primaryScore == rhs.primaryScore && secondaryScore == rhs.secondaryScore && tertiaryScore > rhs.tertiaryScore); + } + void SetPrimaryScore(const float score) { primaryScore = score; } + float GetPrimaryScore() const { return primaryScore; } + + void SetSecondaryScore(const float score) { secondaryScore = score; } + float GetSecondaryScore() const { return secondaryScore; } + + void SetTertiaryScore(const float score) { tertiaryScore = score; } + float GetTertiaryScore() const { return tertiaryScore; } +private: + float primaryScore; + float secondaryScore; + float tertiaryScore; }; +using GameID = uint32_t; + class Leaderboard { public: - Leaderboard(uint32_t gameID, uint32_t infoType, bool weekly, std::vector entries, - LWOOBJID relatedPlayer = LWOOBJID_EMPTY, LeaderboardType = None); - std::vector GetEntries(); - [[nodiscard]] std::u16string ToString() const; - [[nodiscard]] uint32_t GetGameID() const; - [[nodiscard]] uint32_t GetInfoType() const; - void Send(LWOOBJID targetID) const; + + // Enums for leaderboards + enum InfoType : uint32_t { + Top, // Top 11 all time players + MyStanding, // Ranking of the current player + Friends // Ranking between friends + }; + + enum Type : uint32_t { + ShootingGallery, + Racing, + MonumentRace, + FootRace, + UnusedLeaderboard4, // There is no 4 defined anywhere in the cdclient, but it takes a Score. + Survival, + SurvivalNS, + Donations, + None + }; + Leaderboard() = delete; + Leaderboard(const GameID gameID, const Leaderboard::InfoType infoType, const bool weekly, LWOOBJID relatedPlayer, const Leaderboard::Type = None); + + ~Leaderboard(); + + /** + * @brief Resets the leaderboard state and frees its allocated memory + * + */ + void Clear(); + + /** + * Serialize the Leaderboard to a BitStream + * + * Expensive! Leaderboards are very string intensive so be wary of performatnce calling this method. + */ + void Serialize(RakNet::BitStream* bitStream) const; + + /** + * Builds the leaderboard from the database based on the associated gameID + * + * @param resultStart The index to start the leaderboard at. Zero indexed. + * @param resultEnd The index to end the leaderboard at. Zero indexed. + */ + void SetupLeaderboard(bool weekly, uint32_t resultStart = 0, uint32_t resultEnd = 10); + + /** + * Sends the leaderboard to the client specified by targetID. + */ + void Send(const LWOOBJID targetID) const; + + // Helper function to get the columns, ordering and insert format for a leaderboard + static const std::string_view GetOrdering(Type leaderboardType); private: - std::vector entries{}; + // Takes the resulting query from a leaderboard lookup and converts it to the LDF we need + // to send it to a client. + void QueryToLdf(std::unique_ptr& rows); + + using LeaderboardEntry = std::vector; + using LeaderboardEntries = std::vector; + + LeaderboardEntries entries; LWOOBJID relatedPlayer; - uint32_t gameID; - uint32_t infoType; - LeaderboardType leaderboardType; + GameID gameID; + InfoType infoType; + Leaderboard::Type leaderboardType; bool weekly; }; -class LeaderboardManager { -public: - static LeaderboardManager* Instance() { - if (address == nullptr) - address = new LeaderboardManager; - return address; - } - static void SendLeaderboard(uint32_t gameID, InfoType infoType, bool weekly, LWOOBJID targetID, - LWOOBJID playerID = LWOOBJID_EMPTY); - static Leaderboard* GetLeaderboard(uint32_t gameID, InfoType infoType, bool weekly, LWOOBJID playerID = LWOOBJID_EMPTY); - static void SaveScore(LWOOBJID playerID, uint32_t gameID, uint32_t score, uint32_t time); - static LeaderboardType GetLeaderboardType(uint32_t gameID); -private: - static LeaderboardManager* address; +namespace LeaderboardManager { + void SendLeaderboard(const GameID gameID, const Leaderboard::InfoType infoType, const bool weekly, const LWOOBJID playerID, const LWOOBJID targetID, const uint32_t resultStart = 0, const uint32_t resultEnd = 10); - // Modified 12/12/2021: Existing queries were renamed to be more descriptive. - static const std::string topPlayersScoreQuery; - static const std::string friendsScoreQuery; - static const std::string standingsScoreQuery; - static const std::string topPlayersScoreQueryAsc; - static const std::string friendsScoreQueryAsc; - static const std::string standingsScoreQueryAsc; + void SaveScore(const LWOOBJID& playerID, const GameID activityId, const float primaryScore, const float secondaryScore = 0, const float tertiaryScore = 0); - // Added 12/12/2021: Queries dictated by time are needed for certain minigames. - static const std::string topPlayersTimeQuery; - static const std::string friendsTimeQuery; - static const std::string standingsTimeQuery; - static const std::string topPlayersTimeQueryAsc; - static const std::string friendsTimeQueryAsc; - static const std::string standingsTimeQueryAsc; + Leaderboard::Type GetLeaderboardType(const GameID gameID); + extern std::map leaderboardCache; }; +#endif //!__LEADERBOARDMANAGER__H__ diff --git a/dGame/Player.cpp b/dGame/Player.cpp index 5306584e..48b983aa 100644 --- a/dGame/Player.cpp +++ b/dGame/Player.cpp @@ -13,7 +13,10 @@ #include "dZoneManager.h" #include "CharacterComponent.h" #include "Mail.h" +#include "User.h" #include "CppScripts.h" +#include "Loot.h" +#include "eReplicaComponentType.h" std::vector Player::m_Players = {}; @@ -59,7 +62,7 @@ void Player::SetSystemAddress(const SystemAddress& value) { void Player::SetRespawnPos(const NiPoint3 position) { m_respawnPos = position; - m_Character->SetRespawnPoint(dZoneManager::Instance()->GetZone()->GetWorldID(), position); + m_Character->SetRespawnPoint(Game::zoneManager->GetZone()->GetWorldID(), position); } void Player::SetRespawnRot(const NiQuaternion rotation) { @@ -82,7 +85,7 @@ void Player::SendToZone(LWOMAPID zoneId, LWOCLONEID cloneId) { const auto objid = GetObjectID(); ZoneInstanceManager::Instance()->RequestZoneTransfer(Game::server, zoneId, cloneId, false, [objid](bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, std::string serverIP, uint16_t serverPort) { - auto* entity = EntityManager::Instance()->GetEntity(objid); + auto* entity = Game::entityManager->GetEntity(objid); if (entity == nullptr) { return; @@ -105,7 +108,7 @@ void Player::SendToZone(LWOMAPID zoneId, LWOCLONEID cloneId) { WorldPackets::SendTransferToWorld(sysAddr, serverIP, serverPort, mythranShift); - EntityManager::Instance()->DestructEntity(entity); + Game::entityManager->DestructEntity(entity); return; }); } @@ -132,13 +135,13 @@ void Player::RemoveLimboConstruction(LWOOBJID objectId) { void Player::ConstructLimboEntities() { for (const auto objectId : m_LimboConstructions) { - auto* entity = EntityManager::Instance()->GetEntity(objectId); + auto* entity = Game::entityManager->GetEntity(objectId); if (entity == nullptr) { continue; } - EntityManager::Instance()->ConstructEntity(entity, m_SystemAddress); + Game::entityManager->ConstructEntity(entity, m_SystemAddress); } m_LimboConstructions.clear(); @@ -221,7 +224,7 @@ Player* Player::GetPlayer(const SystemAddress& sysAddr) { } Player* Player::GetPlayer(const std::string& name) { - const auto characters = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_CHARACTER); + const auto characters = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::CHARACTER); for (auto* character : characters) { if (!character->IsPlayer()) continue; @@ -266,7 +269,7 @@ Player::~Player() { continue; } - auto* entity = EntityManager::Instance()->GetGhostCandidate(id); + auto* entity = Game::entityManager->GetGhostCandidate(id); if (entity != nullptr) { entity->SetObservers(entity->GetObservers() - 1); @@ -282,12 +285,12 @@ Player::~Player() { } if (IsPlayer()) { - Entity* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); + Entity* zoneControl = Game::entityManager->GetZoneControlEntity(); for (CppScripts::Script* script : CppScripts::GetEntityScripts(zoneControl)) { script->OnPlayerExit(zoneControl, this); } - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPTED_ACTIVITY); for (Entity* scriptEntity : scriptedActs) { if (scriptEntity->GetObjectID() != zoneControl->GetObjectID()) { // Don't want to trigger twice on instance worlds for (CppScripts::Script* script : CppScripts::GetEntityScripts(scriptEntity)) { diff --git a/dGame/TradingManager.cpp b/dGame/TradingManager.cpp index e49cca70..09d452b3 100644 --- a/dGame/TradingManager.cpp +++ b/dGame/TradingManager.cpp @@ -9,6 +9,7 @@ #include "Character.h" #include "CharacterComponent.h" #include "MissionComponent.h" +#include "eMissionTaskType.h" TradingManager* TradingManager::m_Address = nullptr; @@ -39,11 +40,11 @@ LWOOBJID Trade::GetParticipantB() const { } Entity* Trade::GetParticipantAEntity() const { - return EntityManager::Instance()->GetEntity(m_ParticipantA); + return Game::entityManager->GetEntity(m_ParticipantA); } Entity* Trade::GetParticipantBEntity() const { - return EntityManager::Instance()->GetEntity(m_ParticipantB); + return Game::entityManager->GetEntity(m_ParticipantB); } void Trade::SetCoins(LWOOBJID participant, uint64_t coins) { @@ -103,6 +104,7 @@ void Trade::SetAccepted(LWOOBJID participant, bool value) { } Complete(); + TradingManager::Instance()->CancelTrade(m_TradeId); } } @@ -121,33 +123,59 @@ void Trade::Complete() { if (inventoryA == nullptr || inventoryB == nullptr || characterA == nullptr || characterB == nullptr || missionsA == nullptr || missionsB == nullptr) return; - characterA->SetCoins(characterA->GetCoins() - m_CoinsA + m_CoinsB, eLootSourceType::LOOT_SOURCE_TRADE); - characterB->SetCoins(characterB->GetCoins() - m_CoinsB + m_CoinsA, eLootSourceType::LOOT_SOURCE_TRADE); - - for (const auto& tradeItem : m_ItemsA) { - inventoryA->RemoveItem(tradeItem.itemLot, tradeItem.itemCount, INVALID, true); - - missionsA->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, tradeItem.itemLot, LWOOBJID_EMPTY, "", -tradeItem.itemCount); - } - - for (const auto& tradeItem : m_ItemsB) { - inventoryB->RemoveItem(tradeItem.itemLot, tradeItem.itemCount, INVALID, true); - - missionsB->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, tradeItem.itemLot, LWOOBJID_EMPTY, "", -tradeItem.itemCount); + // First verify both players have the coins and items requested for the trade. + if (characterA->GetCoins() < m_CoinsA || characterB->GetCoins() < m_CoinsB) { + Game::logger->Log("TradingManager", "Possible coin trade cheating attempt! Aborting trade."); + return; } for (const auto& tradeItem : m_ItemsA) { - inventoryB->AddItem(tradeItem.itemLot, tradeItem.itemCount, eLootSourceType::LOOT_SOURCE_TRADE); + auto* itemToRemove = inventoryA->FindItemById(tradeItem.itemId); + if (itemToRemove) { + if (itemToRemove->GetCount() < tradeItem.itemCount) { + Game::logger->Log("TradingManager", "Possible cheating attempt from %s in trading!!! Aborting trade", characterA->GetName().c_str()); + return; + } + } else { + Game::logger->Log("TradingManager", "Possible cheating attempt from %s in trading due to item not being available!!!", characterA->GetName().c_str()); + return; + } } for (const auto& tradeItem : m_ItemsB) { - inventoryA->AddItem(tradeItem.itemLot, tradeItem.itemCount, eLootSourceType::LOOT_SOURCE_TRADE); + auto* itemToRemove = inventoryB->FindItemById(tradeItem.itemId); + if (itemToRemove) { + if (itemToRemove->GetCount() < tradeItem.itemCount) { + Game::logger->Log("TradingManager", "Possible cheating attempt from %s in trading!!! Aborting trade", characterB->GetName().c_str()); + return; + } + } else { + Game::logger->Log("TradingManager", "Possible cheating attempt from %s in trading due to item not being available!!! Aborting trade", characterB->GetName().c_str()); + return; + } } - TradingManager::Instance()->CancelTrade(m_TradeId); + // Now actually do the trade. + characterA->SetCoins(characterA->GetCoins() - m_CoinsA + m_CoinsB, eLootSourceType::TRADE); + characterB->SetCoins(characterB->GetCoins() - m_CoinsB + m_CoinsA, eLootSourceType::TRADE); + + for (const auto& tradeItem : m_ItemsA) { + auto* itemToRemove = inventoryA->FindItemById(tradeItem.itemId); + if (itemToRemove) itemToRemove->SetCount(itemToRemove->GetCount() - tradeItem.itemCount); + missionsA->Progress(eMissionTaskType::GATHER, tradeItem.itemLot, LWOOBJID_EMPTY, "", -tradeItem.itemCount); + inventoryB->AddItem(tradeItem.itemLot, tradeItem.itemCount, eLootSourceType::TRADE); + } + + for (const auto& tradeItem : m_ItemsB) { + auto* itemToRemove = inventoryB->FindItemById(tradeItem.itemId); + if (itemToRemove) itemToRemove->SetCount(itemToRemove->GetCount() - tradeItem.itemCount); + missionsB->Progress(eMissionTaskType::GATHER, tradeItem.itemLot, LWOOBJID_EMPTY, "", -tradeItem.itemCount); + inventoryA->AddItem(tradeItem.itemLot, tradeItem.itemCount, eLootSourceType::TRADE); + } characterA->SaveXMLToDatabase(); characterB->SaveXMLToDatabase(); + return; } void Trade::Cancel() { diff --git a/dGame/User.cpp b/dGame/User.cpp index 20cc3ab4..55bbcc09 100644 --- a/dGame/User.cpp +++ b/dGame/User.cpp @@ -5,13 +5,15 @@ #include "dLogger.h" #include "Game.h" #include "dZoneManager.h" +#include "eServerDisconnectIdentifiers.h" +#include "eGameMasterLevel.h" User::User(const SystemAddress& sysAddr, const std::string& username, const std::string& sessionKey) { m_AccountID = 0; m_Username = ""; m_SessionKey = ""; - m_MaxGMLevel = 0; //The max GM level this account can assign to it's characters + m_MaxGMLevel = eGameMasterLevel::CIVILIAN; //The max GM level this account can assign to it's characters m_LastCharID = 0; m_SessionKey = sessionKey; @@ -32,7 +34,7 @@ User::User(const SystemAddress& sysAddr, const std::string& username, const std: sql::ResultSet* res = stmt->executeQuery(); while (res->next()) { m_AccountID = res->getUInt(1); - m_MaxGMLevel = res->getInt(2); + m_MaxGMLevel = static_cast(res->getInt(2)); m_MuteExpire = 0; //res->getUInt64(3); } @@ -126,6 +128,6 @@ void User::UserOutOfSync() { if (m_AmountOfTimesOutOfSync > m_MaxDesyncAllowed) { //YEET Game::logger->Log("User", "User %s was out of sync %i times out of %i, disconnecting for suspected speedhacking.", m_Username.c_str(), m_AmountOfTimesOutOfSync, m_MaxDesyncAllowed); - Game::server->Disconnect(this->m_SystemAddress, SERVER_DISCON_KICK); + Game::server->Disconnect(this->m_SystemAddress, eServerDisconnectIdentifiers::PLAY_SCHEDULE_TIME_DONE); } } diff --git a/dGame/User.h b/dGame/User.h index 59416c4c..3201538e 100644 --- a/dGame/User.h +++ b/dGame/User.h @@ -9,6 +9,7 @@ #include class Character; +enum class eGameMasterLevel : uint8_t; struct BehaviorParams { uint32_t behavior; @@ -29,7 +30,7 @@ public: std::string& GetSessionKey() { return m_SessionKey; } SystemAddress& GetSystemAddress() { return m_SystemAddress; } - uint32_t GetMaxGMLevel() { return m_MaxGMLevel; } + eGameMasterLevel GetMaxGMLevel() { return m_MaxGMLevel; } uint32_t GetLastCharID() { return m_LastCharID; } void SetLastCharID(uint32_t newCharID) { m_LastCharID = newCharID; } @@ -61,7 +62,7 @@ private: std::string m_SessionKey; SystemAddress m_SystemAddress; - uint32_t m_MaxGMLevel; //The max GM level this account can assign to it's characters + eGameMasterLevel m_MaxGMLevel; //The max GM level this account can assign to it's characters uint32_t m_LastCharID; std::vector m_Characters; LWOOBJID m_LoggedInCharID; diff --git a/dGame/UserManager.cpp b/dGame/UserManager.cpp index 7da26b9a..9e409019 100644 --- a/dGame/UserManager.cpp +++ b/dGame/UserManager.cpp @@ -20,6 +20,14 @@ #include "Entity.h" #include "EntityManager.h" #include "SkillComponent.h" +#include "AssetManager.h" +#include "CDClientDatabase.h" +#include "eObjectBits.h" +#include "eGameMasterLevel.h" +#include "eCharacterCreationResponse.h" +#include "eRenameResponse.h" +#include "eConnectionType.h" +#include "eChatInternalMessageType.h" UserManager* UserManager::m_Address = nullptr; @@ -32,43 +40,59 @@ inline void StripCR(std::string& str) { } void UserManager::Initialize() { - std::string firstNamePath = "./res/names/minifigname_first.txt"; - std::string middleNamePath = "./res/names/minifigname_middle.txt"; - std::string lastNamePath = "./res/names/minifigname_last.txt"; std::string line; - std::fstream fnFile(firstNamePath, std::ios::in); - std::fstream mnFile(middleNamePath, std::ios::in); - std::fstream lnFile(lastNamePath, std::ios::in); - - while (std::getline(fnFile, line, '\n')) { + AssetMemoryBuffer fnBuff = Game::assetManager->GetFileAsBuffer("names/minifigname_first.txt"); + if (!fnBuff.m_Success) { + Game::logger->Log("UserManager", "Failed to load %s", (Game::assetManager->GetResPath() / "names/minifigname_first.txt").string().c_str()); + throw std::runtime_error("Aborting initialization due to missing minifigure name file."); + } + std::istream fnStream = std::istream(&fnBuff); + while (std::getline(fnStream, line, '\n')) { std::string name = line; StripCR(name); m_FirstNames.push_back(name); } + fnBuff.close(); - while (std::getline(mnFile, line, '\n')) { + AssetMemoryBuffer mnBuff = Game::assetManager->GetFileAsBuffer("names/minifigname_middle.txt"); + if (!mnBuff.m_Success) { + Game::logger->Log("UserManager", "Failed to load %s", (Game::assetManager->GetResPath() / "names/minifigname_middle.txt").string().c_str()); + throw std::runtime_error("Aborting initialization due to missing minifigure name file."); + } + std::istream mnStream = std::istream(&mnBuff); + while (std::getline(mnStream, line, '\n')) { std::string name = line; StripCR(name); m_MiddleNames.push_back(name); } + mnBuff.close(); - while (std::getline(lnFile, line, '\n')) { + AssetMemoryBuffer lnBuff = Game::assetManager->GetFileAsBuffer("names/minifigname_last.txt"); + if (!lnBuff.m_Success) { + Game::logger->Log("UserManager", "Failed to load %s", (Game::assetManager->GetResPath() / "names/minifigname_last.txt").string().c_str()); + throw std::runtime_error("Aborting initialization due to missing minifigure name file."); + } + std::istream lnStream = std::istream(&lnBuff); + while (std::getline(lnStream, line, '\n')) { std::string name = line; StripCR(name); m_LastNames.push_back(name); } - - fnFile.close(); - mnFile.close(); - lnFile.close(); + lnBuff.close(); //Load our pre-approved names: - std::fstream chatList("./res/chatplus_en_us.txt", std::ios::in); - while (std::getline(chatList, line, '\n')) { + AssetMemoryBuffer chatListBuff = Game::assetManager->GetFileAsBuffer("chatplus_en_us.txt"); + if (!chatListBuff.m_Success) { + Game::logger->Log("UserManager", "Failed to load %s", (Game::assetManager->GetResPath() / "chatplus_en_us.txt").string().c_str()); + throw std::runtime_error("Aborting initialization due to missing chat whitelist file."); + } + std::istream chatListStream = std::istream(&chatListBuff); + while (std::getline(chatListStream, line, '\n')) { StripCR(line); m_PreapprovedNames.push_back(line); } + chatListBuff.close(); } UserManager::~UserManager() { @@ -196,7 +220,7 @@ void UserManager::RequestCharacterList(const SystemAddress& sysAddr) { skillComponent->Reset(); } - EntityManager::Instance()->DestroyEntity(chars[i]->GetEntity()); + Game::entityManager->DestroyEntity(chars[i]->GetEntity()); chars[i]->SaveXMLToDatabase(); @@ -210,6 +234,7 @@ void UserManager::RequestCharacterList(const SystemAddress& sysAddr) { while (res->next()) { LWOOBJID objID = res->getUInt64(1); Character* character = new Character(uint32_t(objID), u); + character->SetIsNewLogin(); chars.push_back(character); } } @@ -247,13 +272,13 @@ void UserManager::CreateCharacter(const SystemAddress& sysAddr, Packet* packet) if (name != "" && !UserManager::IsNameAvailable(name)) { Game::logger->Log("UserManager", "AccountID: %i chose unavailable name: %s", u->GetAccountID(), name.c_str()); - WorldPackets::SendCharacterCreationResponse(sysAddr, CREATION_RESPONSE_CUSTOM_NAME_IN_USE); + WorldPackets::SendCharacterCreationResponse(sysAddr, eCharacterCreationResponse::CUSTOM_NAME_IN_USE); return; } if (!IsNameAvailable(predefinedName)) { Game::logger->Log("UserManager", "AccountID: %i chose unavailable predefined name: %s", u->GetAccountID(), predefinedName.c_str()); - WorldPackets::SendCharacterCreationResponse(sysAddr, CREATION_RESPONSE_PREDEFINED_NAME_IN_USE); + WorldPackets::SendCharacterCreationResponse(sysAddr, eCharacterCreationResponse::PREDEFINED_NAME_IN_USE); return; } @@ -272,7 +297,7 @@ void UserManager::CreateCharacter(const SystemAddress& sysAddr, Packet* packet) if (overlapResult->next()) { Game::logger->Log("UserManager", "Character object id unavailable, check objectidtracker!"); - WorldPackets::SendCharacterCreationResponse(sysAddr, CREATION_RESPONSE_OBJECT_ID_UNAVAILABLE); + WorldPackets::SendCharacterCreationResponse(sysAddr, eCharacterCreationResponse::OBJECT_ID_UNAVAILABLE); return; } @@ -292,16 +317,16 @@ void UserManager::CreateCharacter(const SystemAddress& sysAddr, Packet* packet) std::stringstream xml2; LWOOBJID lwoidforshirt = idforshirt; - lwoidforshirt = GeneralUtils::SetBit(lwoidforshirt, OBJECT_BIT_CHARACTER); - lwoidforshirt = GeneralUtils::SetBit(lwoidforshirt, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(lwoidforshirt, eObjectBits::CHARACTER); + GeneralUtils::SetBit(lwoidforshirt, eObjectBits::PERSISTENT); xml2 << xmlSave1 << ""; std::string xmlSave2 = xml2.str(); ObjectIDManager::Instance()->RequestPersistentID([=](uint32_t idforpants) { LWOOBJID lwoidforpants = idforpants; - lwoidforpants = GeneralUtils::SetBit(lwoidforpants, OBJECT_BIT_CHARACTER); - lwoidforpants = GeneralUtils::SetBit(lwoidforpants, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(lwoidforpants, eObjectBits::CHARACTER); + GeneralUtils::SetBit(lwoidforpants, eObjectBits::PERSISTENT); std::stringstream xml3; xml3 << xmlSave2 << ""; @@ -310,7 +335,7 @@ void UserManager::CreateCharacter(const SystemAddress& sysAddr, Packet* packet) //Check to see if our name was pre-approved: bool nameOk = IsNamePreapproved(name); - if (!nameOk && u->GetMaxGMLevel() > 1) nameOk = true; + if (!nameOk && u->GetMaxGMLevel() > eGameMasterLevel::FORUM_MODERATOR) nameOk = true; if (name != "") { sql::PreparedStatement* stmt = Database::CreatePreppedStmt("INSERT INTO `charinfo`(`id`, `account_id`, `name`, `pending_name`, `needs_rename`, `last_login`) VALUES (?,?,?,?,?,?)"); @@ -348,7 +373,7 @@ void UserManager::CreateCharacter(const SystemAddress& sysAddr, Packet* packet) stmt->execute(); delete stmt; - WorldPackets::SendCharacterCreationResponse(sysAddr, CREATION_RESPONSE_SUCCESS); + WorldPackets::SendCharacterCreationResponse(sysAddr, eCharacterCreationResponse::SUCCESS); UserManager::RequestCharacterList(sysAddr); }); }); @@ -398,7 +423,7 @@ void UserManager::DeleteCharacter(const SystemAddress& sysAddr, Packet* packet) stmt->execute(); delete stmt; CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_PLAYER_REMOVED_NOTIFICATION); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::PLAYER_REMOVED_NOTIFICATION); bitStream.Write(objectID); Game::chatServer->Send(&bitStream, SYSTEM_PRIORITY, RELIABLE, 0, Game::chatSysAddr, false); } @@ -459,8 +484,8 @@ void UserManager::RenameCharacter(const SystemAddress& sysAddr, Packet* packet) } LWOOBJID objectID = PacketUtils::ReadPacketS64(8, packet); - objectID = GeneralUtils::ClearBit(objectID, OBJECT_BIT_CHARACTER); - objectID = GeneralUtils::ClearBit(objectID, OBJECT_BIT_PERSISTENT); + GeneralUtils::ClearBit(objectID, eObjectBits::CHARACTER); + GeneralUtils::ClearBit(objectID, eObjectBits::PERSISTENT); uint32_t charID = static_cast(objectID); Game::logger->Log("UserManager", "Received char rename request for ID: %llu (%u)", objectID, charID); @@ -478,10 +503,10 @@ void UserManager::RenameCharacter(const SystemAddress& sysAddr, Packet* packet) if (!hasCharacter || !character) { Game::logger->Log("UserManager", "User %i tried to rename a character that it does not own!", u->GetAccountID()); - WorldPackets::SendCharacterRenameResponse(sysAddr, RENAME_RESPONSE_UNKNOWN_ERROR); + WorldPackets::SendCharacterRenameResponse(sysAddr, eRenameResponse::UNKNOWN_ERROR); } else if (hasCharacter && character) { if (newName == character->GetName()) { - WorldPackets::SendCharacterRenameResponse(sysAddr, RENAME_RESPONSE_NAME_UNAVAILABLE); + WorldPackets::SendCharacterRenameResponse(sysAddr, eRenameResponse::NAME_UNAVAILABLE); return; } @@ -495,7 +520,7 @@ void UserManager::RenameCharacter(const SystemAddress& sysAddr, Packet* packet) delete stmt; Game::logger->Log("UserManager", "Character %s now known as %s", character->GetName().c_str(), newName.c_str()); - WorldPackets::SendCharacterRenameResponse(sysAddr, RENAME_RESPONSE_SUCCESS); + WorldPackets::SendCharacterRenameResponse(sysAddr, eRenameResponse::SUCCESS); UserManager::RequestCharacterList(sysAddr); } else { sql::PreparedStatement* stmt = Database::CreatePreppedStmt("UPDATE charinfo SET pending_name=?, needs_rename=0, last_login=? WHERE id=? LIMIT 1"); @@ -506,15 +531,15 @@ void UserManager::RenameCharacter(const SystemAddress& sysAddr, Packet* packet) delete stmt; Game::logger->Log("UserManager", "Character %s has been renamed to %s and is pending approval by a moderator.", character->GetName().c_str(), newName.c_str()); - WorldPackets::SendCharacterRenameResponse(sysAddr, RENAME_RESPONSE_SUCCESS); + WorldPackets::SendCharacterRenameResponse(sysAddr, eRenameResponse::SUCCESS); UserManager::RequestCharacterList(sysAddr); } } else { - WorldPackets::SendCharacterRenameResponse(sysAddr, RENAME_RESPONSE_NAME_IN_USE); + WorldPackets::SendCharacterRenameResponse(sysAddr, eRenameResponse::NAME_IN_USE); } } else { Game::logger->Log("UserManager", "Unknown error occurred when renaming character, either hasCharacter or character variable != true."); - WorldPackets::SendCharacterRenameResponse(sysAddr, RENAME_RESPONSE_UNKNOWN_ERROR); + WorldPackets::SendCharacterRenameResponse(sysAddr, eRenameResponse::UNKNOWN_ERROR); } } diff --git a/dGame/dBehaviors/AirMovementBehavior.cpp b/dGame/dBehaviors/AirMovementBehavior.cpp index 469ac6e4..a6d749d3 100644 --- a/dGame/dBehaviors/AirMovementBehavior.cpp +++ b/dGame/dBehaviors/AirMovementBehavior.cpp @@ -2,13 +2,18 @@ #include "BehaviorBranchContext.h" #include "BehaviorContext.h" #include "EntityManager.h" +#include "Game.h" +#include "dLogger.h" void AirMovementBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - uint32_t handle; + uint32_t handle{}; - bitStream->Read(handle); + if (!bitStream->Read(handle)) { + Game::logger->Log("AirMovementBehavior", "Unable to read handle from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + } - context->RegisterSyncBehavior(handle, this, branch); + context->RegisterSyncBehavior(handle, this, branch, this->m_Timeout); } void AirMovementBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { @@ -17,23 +22,30 @@ void AirMovementBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream->Write(handle); } -void AirMovementBehavior::Sync(BehaviorContext* context, RakNet::BitStream* bit_stream, BehaviorBranchContext branch) { - uint32_t behaviorId; +void AirMovementBehavior::Sync(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + uint32_t behaviorId{}; - bit_stream->Read(behaviorId); + if (!bitStream->Read(behaviorId)) { + Game::logger->Log("AirMovementBehavior", "Unable to read behaviorId from bitStream, aborting Sync! %i", bitStream->GetNumberOfUnreadBits()); + return; + } - LWOOBJID target; + LWOOBJID target{}; - bit_stream->Read(target); + if (!bitStream->Read(target)) { + Game::logger->Log("AirMovementBehavior", "Unable to read target from bitStream, aborting Sync! %i", bitStream->GetNumberOfUnreadBits()); + return; + } auto* behavior = CreateBehavior(behaviorId); - if (EntityManager::Instance()->GetEntity(target) != nullptr) { + if (Game::entityManager->GetEntity(target) != nullptr) { branch.target = target; } - behavior->Handle(context, bit_stream, branch); + behavior->Handle(context, bitStream, branch); } void AirMovementBehavior::Load() { + this->m_Timeout = (GetFloat("timeout_ms") / 1000.0f); } diff --git a/dGame/dBehaviors/AirMovementBehavior.h b/dGame/dBehaviors/AirMovementBehavior.h index 323edf37..9d51ef03 100644 --- a/dGame/dBehaviors/AirMovementBehavior.h +++ b/dGame/dBehaviors/AirMovementBehavior.h @@ -4,19 +4,15 @@ class AirMovementBehavior final : public Behavior { public: - - /* - * Inherited - */ - - explicit AirMovementBehavior(const uint32_t behavior_id) : Behavior(behavior_id) { - } + explicit AirMovementBehavior(const uint32_t behavior_id) : Behavior(behavior_id) {} void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; - void Sync(BehaviorContext* context, RakNet::BitStream* bit_stream, BehaviorBranchContext branch) override; + void Sync(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; void Load() override; +private: + float m_Timeout; }; diff --git a/dGame/dBehaviors/ApplyBuffBehavior.cpp b/dGame/dBehaviors/ApplyBuffBehavior.cpp index 35b0f269..c94762aa 100644 --- a/dGame/dBehaviors/ApplyBuffBehavior.cpp +++ b/dGame/dBehaviors/ApplyBuffBehavior.cpp @@ -6,7 +6,7 @@ void ApplyBuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target == LWOOBJID_EMPTY ? context->originator : branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target == LWOOBJID_EMPTY ? context->originator : branch.target); if (entity == nullptr) return; @@ -19,7 +19,7 @@ void ApplyBuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitS } void ApplyBuffBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); if (entity == nullptr) return; diff --git a/dGame/dBehaviors/AreaOfEffectBehavior.cpp b/dGame/dBehaviors/AreaOfEffectBehavior.cpp index 898d1f99..e43d542d 100644 --- a/dGame/dBehaviors/AreaOfEffectBehavior.cpp +++ b/dGame/dBehaviors/AreaOfEffectBehavior.cpp @@ -9,11 +9,16 @@ #include "BehaviorContext.h" #include "RebuildComponent.h" #include "DestroyableComponent.h" +#include "Game.h" +#include "dLogger.h" void AreaOfEffectBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - uint32_t targetCount; + uint32_t targetCount{}; - bitStream->Read(targetCount); + if (!bitStream->Read(targetCount)) { + Game::logger->Log("AreaOfEffectBehavior", "Unable to read targetCount from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + } if (targetCount > this->m_maxTargets) { return; @@ -24,9 +29,12 @@ void AreaOfEffectBehavior::Handle(BehaviorContext* context, RakNet::BitStream* b targets.reserve(targetCount); for (auto i = 0u; i < targetCount; ++i) { - LWOOBJID target; + LWOOBJID target{}; - bitStream->Read(target); + if (!bitStream->Read(target)) { + Game::logger->Log("AreaOfEffectBehavior", "failed to read in target %i from bitStream, aborting target Handle!", i); + return; + }; targets.push_back(target); } @@ -39,10 +47,9 @@ void AreaOfEffectBehavior::Handle(BehaviorContext* context, RakNet::BitStream* b } void AreaOfEffectBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* self = EntityManager::Instance()->GetEntity(context->caster); - + auto* self = Game::entityManager->GetEntity(context->caster); if (self == nullptr) { - Game::logger->Log("TacArcBehavior", "Invalid self for (%llu)!", context->originator); + Game::logger->Log("AreaOfEffectBehavior", "Invalid self for (%llu)!", context->originator); return; } @@ -51,7 +58,7 @@ void AreaOfEffectBehavior::Calculate(BehaviorContext* context, RakNet::BitStream std::vector targets; - auto* presetTarget = EntityManager::Instance()->GetEntity(branch.target); + auto* presetTarget = Game::entityManager->GetEntity(branch.target); if (presetTarget != nullptr) { if (this->m_radius * this->m_radius >= Vector3::DistanceSquared(reference, presetTarget->GetPosition())) { @@ -68,10 +75,10 @@ void AreaOfEffectBehavior::Calculate(BehaviorContext* context, RakNet::BitStream // Gets all of the valid targets, passing in if should target enemies and friends for (auto validTarget : context->GetValidTargets(m_ignoreFaction, includeFaction, m_TargetSelf == 1, m_targetEnemy == 1, m_targetFriend == 1)) { - auto* entity = EntityManager::Instance()->GetEntity(validTarget); + auto* entity = Game::entityManager->GetEntity(validTarget); if (entity == nullptr) { - Game::logger->Log("TacArcBehavior", "Invalid target (%llu) for (%llu)!", validTarget, context->originator); + Game::logger->Log("AreaOfEffectBehavior", "Invalid target (%llu) for (%llu)!", validTarget, context->originator); continue; } diff --git a/dGame/dBehaviors/AttackDelayBehavior.cpp b/dGame/dBehaviors/AttackDelayBehavior.cpp index 450741ec..3f12f662 100644 --- a/dGame/dBehaviors/AttackDelayBehavior.cpp +++ b/dGame/dBehaviors/AttackDelayBehavior.cpp @@ -5,12 +5,15 @@ #include "dLogger.h" void AttackDelayBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - uint32_t handle; + uint32_t handle{}; - bitStream->Read(handle); + if (!bitStream->Read(handle)) { + Game::logger->Log("AttackDelayBehavior", "Unable to read handle from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; for (auto i = 0u; i < this->m_numIntervals; ++i) { - context->RegisterSyncBehavior(handle, this, branch); + context->RegisterSyncBehavior(handle, this, branch, this->m_delay * i, m_ignoreInterrupts); } } diff --git a/dGame/dBehaviors/BasicAttackBehavior.cpp b/dGame/dBehaviors/BasicAttackBehavior.cpp index fe773f36..0a21383b 100644 --- a/dGame/dBehaviors/BasicAttackBehavior.cpp +++ b/dGame/dBehaviors/BasicAttackBehavior.cpp @@ -5,146 +5,251 @@ #include "EntityManager.h" #include "DestroyableComponent.h" #include "BehaviorContext.h" - +#include "eBasicAttackSuccessTypes.h" void BasicAttackBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { if (context->unmanaged) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); auto* destroyableComponent = entity->GetComponent(); if (destroyableComponent != nullptr) { PlayFx(u"onhit", entity->GetObjectID()); - destroyableComponent->Damage(this->m_maxDamage, context->originator, context->skillID); + destroyableComponent->Damage(this->m_MaxDamage, context->originator, context->skillID); } - this->m_onSuccess->Handle(context, bitStream, branch); + this->m_OnSuccess->Handle(context, bitStream, branch); return; } bitStream->AlignReadToByteBoundary(); - uint16_t allocatedBits; - bitStream->Read(allocatedBits); - + uint16_t allocatedBits{}; + if (!bitStream->Read(allocatedBits) || allocatedBits == 0) { + Game::logger->LogDebug("BasicAttackBehavior", "No allocated bits"); + return; + } + Game::logger->LogDebug("BasicAttackBehavior", "Number of allocated bits %i", allocatedBits); const auto baseAddress = bitStream->GetReadOffset(); - if (bitStream->ReadBit()) { // Blocked - return; - } - if (bitStream->ReadBit()) { // Immune - return; - } - - if (bitStream->ReadBit()) { // Success - uint32_t unknown; - bitStream->Read(unknown); - - uint32_t damageDealt; - bitStream->Read(damageDealt); - - // A value that's too large may be a cheating attempt, so we set it to MIN too - if (damageDealt > this->m_maxDamage || damageDealt < this->m_minDamage) { - damageDealt = this->m_minDamage; - } - - auto* entity = EntityManager::Instance()->GetEntity(branch.target); - bool died; - bitStream->Read(died); - - if (entity != nullptr) { - auto* destroyableComponent = entity->GetComponent(); - if (destroyableComponent != nullptr) { - PlayFx(u"onhit", entity->GetObjectID()); - destroyableComponent->Damage(damageDealt, context->originator, context->skillID); - } - } - } - - uint8_t successState; - bitStream->Read(successState); - - switch (successState) { - case 1: - this->m_onSuccess->Handle(context, bitStream, branch); - break; - default: - Game::logger->Log("BasicAttackBehavior", "Unknown success state (%i)!", successState); - break; - } + DoHandleBehavior(context, bitStream, branch); bitStream->SetReadOffset(baseAddress + allocatedBits); } -void BasicAttackBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* self = EntityManager::Instance()->GetEntity(context->originator); - if (self == nullptr) { - Game::logger->Log("BasicAttackBehavior", "Invalid self entity (%llu)!", context->originator); +void BasicAttackBehavior::DoHandleBehavior(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + auto* targetEntity = Game::entityManager->GetEntity(branch.target); + if (!targetEntity) { + Game::logger->Log("BasicAttackBehavior", "Target targetEntity %llu not found.", branch.target); return; } + auto* destroyableComponent = targetEntity->GetComponent(); + if (!destroyableComponent) { + Game::logger->Log("BasicAttackBehavior", "No destroyable found on the obj/lot %llu/%i", branch.target, targetEntity->GetLOT()); + return; + } + + bool isBlocked{}; + bool isImmune{}; + bool isSuccess{}; + + if (!bitStream->Read(isBlocked)) { + Game::logger->Log("BasicAttackBehavior", "Unable to read isBlocked"); + return; + } + + if (isBlocked) { + destroyableComponent->SetAttacksToBlock(std::min(destroyableComponent->GetAttacksToBlock() - 1, 0U)); + Game::entityManager->SerializeEntity(targetEntity); + this->m_OnFailBlocked->Handle(context, bitStream, branch); + return; + } + + if (!bitStream->Read(isImmune)) { + Game::logger->Log("BasicAttackBehavior", "Unable to read isImmune"); + return; + } + + if (isImmune) { + this->m_OnFailImmune->Handle(context, bitStream, branch); + return; + } + + if (!bitStream->Read(isSuccess)) { + Game::logger->Log("BasicAttackBehavior", "failed to read success from bitstream"); + return; + } + + if (isSuccess) { + uint32_t armorDamageDealt{}; + if (!bitStream->Read(armorDamageDealt)) { + Game::logger->Log("BasicAttackBehavior", "Unable to read armorDamageDealt"); + return; + } + + uint32_t healthDamageDealt{}; + if (!bitStream->Read(healthDamageDealt)) { + Game::logger->Log("BasicAttackBehavior", "Unable to read healthDamageDealt"); + return; + } + + uint32_t totalDamageDealt = armorDamageDealt + healthDamageDealt; + + // A value that's too large may be a cheating attempt, so we set it to MIN + if (totalDamageDealt > this->m_MaxDamage) { + totalDamageDealt = this->m_MinDamage; + } + + bool died{}; + if (!bitStream->Read(died)) { + Game::logger->Log("BasicAttackBehavior", "Unable to read died"); + return; + } + auto previousArmor = destroyableComponent->GetArmor(); + auto previousHealth = destroyableComponent->GetHealth(); + PlayFx(u"onhit", targetEntity->GetObjectID()); + destroyableComponent->Damage(totalDamageDealt, context->originator, context->skillID); + } + + uint8_t successState{}; + if (!bitStream->Read(successState)) { + Game::logger->Log("BasicAttackBehavior", "Unable to read success state"); + return; + } + + switch (static_cast(successState)) { + case eBasicAttackSuccessTypes::SUCCESS: + this->m_OnSuccess->Handle(context, bitStream, branch); + break; + case eBasicAttackSuccessTypes::FAILARMOR: + this->m_OnFailArmor->Handle(context, bitStream, branch); + break; + default: + if (static_cast(successState) != eBasicAttackSuccessTypes::FAILIMMUNE) { + Game::logger->Log("BasicAttackBehavior", "Unknown success state (%i)!", successState); + return; + } + this->m_OnFailImmune->Handle(context, bitStream, branch); + break; + } +} + +void BasicAttackBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { bitStream->AlignWriteToByteBoundary(); const auto allocatedAddress = bitStream->GetWriteOffset(); - bitStream->Write(uint16_t(0)); + bitStream->Write(0); const auto startAddress = bitStream->GetWriteOffset(); - bitStream->Write0(); // Blocked - bitStream->Write0(); // Immune - bitStream->Write1(); // Success - - if (true) { - uint32_t unknown3 = 0; - bitStream->Write(unknown3); - - auto damage = this->m_minDamage; - auto* entity = EntityManager::Instance()->GetEntity(branch.target); - - if (entity == nullptr) { - damage = 0; - bitStream->Write(damage); - bitStream->Write(false); - } else { - bitStream->Write(damage); - bitStream->Write(true); - - auto* destroyableComponent = entity->GetComponent(); - if (damage != 0 && destroyableComponent != nullptr) { - PlayFx(u"onhit", entity->GetObjectID(), 1); - destroyableComponent->Damage(damage, context->originator, context->skillID, false); - context->ScheduleUpdate(branch.target); - } - } - } - - uint8_t successState = 1; - bitStream->Write(successState); - - switch (successState) { - case 1: - this->m_onSuccess->Calculate(context, bitStream, branch); - break; - default: - Game::logger->Log("BasicAttackBehavior", "Unknown success state (%i)!", successState); - break; - } + DoBehaviorCalculation(context, bitStream, branch); const auto endAddress = bitStream->GetWriteOffset(); - const uint16_t allocate = endAddress - startAddress + 1; + const uint16_t allocate = endAddress - startAddress; bitStream->SetWriteOffset(allocatedAddress); bitStream->Write(allocate); bitStream->SetWriteOffset(startAddress + allocate); } -void BasicAttackBehavior::Load() { - this->m_minDamage = GetInt("min damage"); - if (this->m_minDamage == 0) this->m_minDamage = 1; +void BasicAttackBehavior::DoBehaviorCalculation(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + auto* targetEntity = Game::entityManager->GetEntity(branch.target); + if (!targetEntity) { + Game::logger->Log("BasicAttackBehavior", "Target entity %llu is null!", branch.target); + return; + } - this->m_maxDamage = GetInt("max damage"); - if (this->m_maxDamage == 0) this->m_maxDamage = 1; + auto* destroyableComponent = targetEntity->GetComponent(); + if (!destroyableComponent || !destroyableComponent->GetParent()) { + Game::logger->Log("BasicAttackBehavior", "No destroyable component on %llu", branch.target); + return; + } - this->m_onSuccess = GetAction("on_success"); + const bool isBlocking = destroyableComponent->GetAttacksToBlock() > 0; + + bitStream->Write(isBlocking); + + if (isBlocking) { + destroyableComponent->SetAttacksToBlock(destroyableComponent->GetAttacksToBlock() - 1); + Game::entityManager->SerializeEntity(targetEntity); + this->m_OnFailBlocked->Calculate(context, bitStream, branch); + return; + } + + const bool isImmune = destroyableComponent->IsImmune(); + + bitStream->Write(isImmune); + + if (isImmune) { + this->m_OnFailImmune->Calculate(context, bitStream, branch); + return; + } + + bool isSuccess = false; + const uint32_t previousHealth = destroyableComponent->GetHealth(); + const uint32_t previousArmor = destroyableComponent->GetArmor(); + + const auto damage = this->m_MinDamage; + + PlayFx(u"onhit", targetEntity->GetObjectID(), 1); + destroyableComponent->Damage(damage, context->originator, context->skillID, false); + context->ScheduleUpdate(branch.target); + + const uint32_t armorDamageDealt = previousArmor - destroyableComponent->GetArmor(); + const uint32_t healthDamageDealt = previousHealth - destroyableComponent->GetHealth(); + isSuccess = armorDamageDealt > 0 || healthDamageDealt > 0 || (armorDamageDealt + healthDamageDealt) > 0; + + bitStream->Write(isSuccess); + + eBasicAttackSuccessTypes successState = eBasicAttackSuccessTypes::FAILIMMUNE; + if (isSuccess) { + if (healthDamageDealt >= 1) { + successState = eBasicAttackSuccessTypes::SUCCESS; + } else if (armorDamageDealt >= 1) { + successState = this->m_OnFailArmor->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY ? eBasicAttackSuccessTypes::FAILIMMUNE : eBasicAttackSuccessTypes::FAILARMOR; + } + + bitStream->Write(armorDamageDealt); + bitStream->Write(healthDamageDealt); + bitStream->Write(targetEntity->GetIsDead()); + } + + bitStream->Write(successState); + + switch (static_cast(successState)) { + case eBasicAttackSuccessTypes::SUCCESS: + this->m_OnSuccess->Calculate(context, bitStream, branch); + break; + case eBasicAttackSuccessTypes::FAILARMOR: + this->m_OnFailArmor->Calculate(context, bitStream, branch); + break; + default: + if (static_cast(successState) != eBasicAttackSuccessTypes::FAILIMMUNE) { + Game::logger->Log("BasicAttackBehavior", "Unknown success state (%i)!", successState); + break; + } + this->m_OnFailImmune->Calculate(context, bitStream, branch); + break; + } +} + +void BasicAttackBehavior::Load() { + this->m_MinDamage = GetInt("min damage"); + if (this->m_MinDamage == 0) this->m_MinDamage = 1; + + this->m_MaxDamage = GetInt("max damage"); + if (this->m_MaxDamage == 0) this->m_MaxDamage = 1; + + // The client sets the minimum damage to maximum, so we'll do the same. These are usually the same value anyways. + if (this->m_MinDamage < this->m_MaxDamage) this->m_MinDamage = this->m_MaxDamage; + + this->m_OnSuccess = GetAction("on_success"); + + this->m_OnFailArmor = GetAction("on_fail_armor"); + + this->m_OnFailImmune = GetAction("on_fail_immune"); + + this->m_OnFailBlocked = GetAction("on_fail_blocked"); } diff --git a/dGame/dBehaviors/BasicAttackBehavior.h b/dGame/dBehaviors/BasicAttackBehavior.h index 8e23d443..f6e3fa28 100644 --- a/dGame/dBehaviors/BasicAttackBehavior.h +++ b/dGame/dBehaviors/BasicAttackBehavior.h @@ -4,18 +4,59 @@ class BasicAttackBehavior final : public Behavior { public: - uint32_t m_minDamage; - - uint32_t m_maxDamage; - - Behavior* m_onSuccess; - explicit BasicAttackBehavior(const uint32_t behaviorId) : Behavior(behaviorId) { } + /** + * @brief Reads a 16bit short from the bitStream and when the actual behavior handling finishes with all of its branches, the bitStream + * is then offset to after the allocated bits for this stream. + * + */ + void DoHandleBehavior(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch); + + /** + * @brief Handles a client initialized Basic Attack Behavior cast to be deserialized and verified on the server. + * + * @param context The Skill's Behavior context. All behaviors in the same tree share the same context + * @param bitStream The bitStream to deserialize. BitStreams will always check their bounds before reading in a behavior + * and will fail gracefully if an overread is detected. + * @param branch The context of this specific branch of the Skill Behavior. Changes based on which branch you are going down. + */ void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + /** + * @brief Writes a 16bit short to the bitStream and when the actual behavior calculation finishes with all of its branches, the number + * of bits used is then written to where the 16bit short initially was. + * + */ void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + /** + * @brief Calculates a server initialized Basic Attack Behavior cast to be serialized to the client + * + * @param context The Skill's Behavior context. All behaviors in the same tree share the same context + * @param bitStream The bitStream to serialize to. + * @param branch The context of this specific branch of the Skill Behavior. Changes based on which branch you are going down. + */ + void DoBehaviorCalculation(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch); + + /** + * @brief Loads this Behaviors parameters from the database. For this behavior specifically: + * max and min damage will always be the same. If min is less than max, they are both set to max. + * If an action is not in the database, then no action is taken for that result. + * + */ void Load() override; +private: + uint32_t m_MinDamage; + + uint32_t m_MaxDamage; + + Behavior* m_OnSuccess; + + Behavior* m_OnFailArmor; + + Behavior* m_OnFailImmune; + + Behavior* m_OnFailBlocked; }; diff --git a/dGame/dBehaviors/Behavior.cpp b/dGame/dBehaviors/Behavior.cpp index 046df117..6fe84a9f 100644 --- a/dGame/dBehaviors/Behavior.cpp +++ b/dGame/dBehaviors/Behavior.cpp @@ -42,11 +42,13 @@ #include "SkillCastFailedBehavior.h" #include "SpawnBehavior.h" #include "ForceMovementBehavior.h" +#include "RemoveBuffBehavior.h" #include "ImmunityBehavior.h" #include "InterruptBehavior.h" #include "PlayEffectBehavior.h" #include "DamageAbsorptionBehavior.h" #include "VentureVisionBehavior.h" +#include "PropertyTeleportBehavior.h" #include "BlockBehavior.h" #include "ClearTargetBehavior.h" #include "PullToPointBehavior.h" @@ -59,6 +61,9 @@ #include "SpeedBehavior.h" #include "DamageReductionBehavior.h" #include "JetPackBehavior.h" +#include "FallSpeedBehavior.h" +#include "ChangeIdleFlagsBehavior.h" +#include "DarkInspirationBehavior.h" //CDClient includes #include "CDBehaviorParameterTable.h" @@ -69,13 +74,14 @@ #include "EntityManager.h" #include "RenderComponent.h" #include "DestroyableComponent.h" +#include "CDBehaviorTemplateTable.h" std::unordered_map Behavior::Cache = {}; CDBehaviorParameterTable* Behavior::BehaviorParameterTable = nullptr; Behavior* Behavior::GetBehavior(const uint32_t behaviorId) { if (BehaviorParameterTable == nullptr) { - BehaviorParameterTable = CDClientManager::Instance()->GetTable("BehaviorParameter"); + BehaviorParameterTable = CDClientManager::Instance().GetTable(); } const auto pair = Cache.find(behaviorId); @@ -159,7 +165,9 @@ Behavior* Behavior::CreateBehavior(const uint32_t behaviorId) { case BehaviorTemplates::BEHAVIOR_CAR_BOOST: behavior = new CarBoostBehavior(behaviorId); break; - case BehaviorTemplates::BEHAVIOR_FALL_SPEED: break; + case BehaviorTemplates::BEHAVIOR_FALL_SPEED: + behavior = new FallSpeedBehavior(behaviorId); + break; case BehaviorTemplates::BEHAVIOR_SHIELD: break; case BehaviorTemplates::BEHAVIOR_REPAIR_ARMOR: behavior = new RepairBehavior(behaviorId); @@ -167,7 +175,9 @@ Behavior* Behavior::CreateBehavior(const uint32_t behaviorId) { case BehaviorTemplates::BEHAVIOR_SPEED: behavior = new SpeedBehavior(behaviorId); break; - case BehaviorTemplates::BEHAVIOR_DARK_INSPIRATION: break; + case BehaviorTemplates::BEHAVIOR_DARK_INSPIRATION: + behavior = new DarkInspirationBehavior(behaviorId); + break; case BehaviorTemplates::BEHAVIOR_LOOT_BUFF: behavior = new LootBuffBehavior(behaviorId); break; @@ -195,7 +205,9 @@ Behavior* Behavior::CreateBehavior(const uint32_t behaviorId) { behavior = new SkillCastFailedBehavior(behaviorId); break; case BehaviorTemplates::BEHAVIOR_IMITATION_SKUNK_STINK: break; - case BehaviorTemplates::BEHAVIOR_CHANGE_IDLE_FLAGS: break; + case BehaviorTemplates::BEHAVIOR_CHANGE_IDLE_FLAGS: + behavior = new ChangeIdleFlagsBehavior(behaviorId); + break; case BehaviorTemplates::BEHAVIOR_APPLY_BUFF: behavior = new ApplyBuffBehavior(behaviorId); break; @@ -226,7 +238,9 @@ Behavior* Behavior::CreateBehavior(const uint32_t behaviorId) { break; case BehaviorTemplates::BEHAVIOR_ALTER_CHAIN_DELAY: break; case BehaviorTemplates::BEHAVIOR_CAMERA: break; - case BehaviorTemplates::BEHAVIOR_REMOVE_BUFF: break; + case BehaviorTemplates::BEHAVIOR_REMOVE_BUFF: + behavior = new RemoveBuffBehavior(behaviorId); + break; case BehaviorTemplates::BEHAVIOR_GRAB: break; case BehaviorTemplates::BEHAVIOR_MODULAR_BUILD: break; case BehaviorTemplates::BEHAVIOR_NPC_COMBAT_SKILL: @@ -254,7 +268,9 @@ Behavior* Behavior::CreateBehavior(const uint32_t behaviorId) { case BehaviorTemplates::BEHAVIOR_DAMAGE_REDUCTION: behavior = new DamageReductionBehavior(behaviorId); break; - case BehaviorTemplates::BEHAVIOR_PROPERTY_TELEPORT: break; + case BehaviorTemplates::BEHAVIOR_PROPERTY_TELEPORT: + behavior = new PropertyTeleportBehavior(behaviorId); + break; case BehaviorTemplates::BEHAVIOR_PROPERTY_CLEAR_TARGET: behavior = new ClearTargetBehavior(behaviorId); break; @@ -278,7 +294,7 @@ Behavior* Behavior::CreateBehavior(const uint32_t behaviorId) { } BehaviorTemplates Behavior::GetBehaviorTemplate(const uint32_t behaviorId) { - auto behaviorTemplateTable = CDClientManager::Instance()->GetTable("BehaviorTemplate"); + auto behaviorTemplateTable = CDClientManager::Instance().GetTable(); BehaviorTemplates templateID = BehaviorTemplates::BEHAVIOR_EMPTY; // Find behavior template by its behavior id. Default to 0. @@ -298,7 +314,7 @@ BehaviorTemplates Behavior::GetBehaviorTemplate(const uint32_t behaviorId) { // For use with enemies, to display the correct damage animations on the players void Behavior::PlayFx(std::u16string type, const LWOOBJID target, const LWOOBJID secondary) { - auto* targetEntity = EntityManager::Instance()->GetEntity(target); + auto* targetEntity = Game::entityManager->GetEntity(target); if (targetEntity == nullptr) { return; @@ -386,7 +402,7 @@ void Behavior::PlayFx(std::u16string type, const LWOOBJID target, const LWOOBJID } Behavior::Behavior(const uint32_t behaviorId) { - auto behaviorTemplateTable = CDClientManager::Instance()->GetTable("BehaviorTemplate"); + auto behaviorTemplateTable = CDClientManager::Instance().GetTable(); CDBehaviorTemplate templateInDatabase{}; @@ -429,8 +445,8 @@ Behavior::Behavior(const uint32_t behaviorId) { float Behavior::GetFloat(const std::string& name, const float defaultValue) const { // Get the behavior parameter entry and return its value. - if (!BehaviorParameterTable) BehaviorParameterTable = CDClientManager::Instance()->GetTable("BehaviorParameter"); - return BehaviorParameterTable->GetEntry(this->m_behaviorId, name, defaultValue).value; + if (!BehaviorParameterTable) BehaviorParameterTable = CDClientManager::Instance().GetTable(); + return BehaviorParameterTable->GetValue(this->m_behaviorId, name, defaultValue); } @@ -457,7 +473,7 @@ Behavior* Behavior::GetAction(float value) const { std::map Behavior::GetParameterNames() const { std::map templatesInDatabase; // Find behavior template by its behavior id. - if (!BehaviorParameterTable) BehaviorParameterTable = CDClientManager::Instance()->GetTable("BehaviorParameter"); + if (!BehaviorParameterTable) BehaviorParameterTable = CDClientManager::Instance().GetTable(); if (BehaviorParameterTable) { templatesInDatabase = BehaviorParameterTable->GetParametersByBehaviorID(this->m_behaviorId); } diff --git a/dGame/dBehaviors/BehaviorBranchContext.h b/dGame/dBehaviors/BehaviorBranchContext.h index 5b2ec595..2e56cd35 100644 --- a/dGame/dBehaviors/BehaviorBranchContext.h +++ b/dGame/dBehaviors/BehaviorBranchContext.h @@ -15,6 +15,8 @@ struct BehaviorBranchContext uint32_t start = 0; + bool isSync = false; + BehaviorBranchContext(); BehaviorBranchContext(LWOOBJID target, float duration = 0, const NiPoint3& referencePosition = NiPoint3(0, 0, 0)); diff --git a/dGame/dBehaviors/BehaviorContext.cpp b/dGame/dBehaviors/BehaviorContext.cpp index 4cd9e415..43ba8457 100644 --- a/dGame/dBehaviors/BehaviorContext.cpp +++ b/dGame/dBehaviors/BehaviorContext.cpp @@ -10,10 +10,12 @@ #include - #include "DestroyableComponent.h" +#include "EchoSyncSkill.h" #include "PhantomPhysicsComponent.h" #include "RebuildComponent.h" +#include "eReplicaComponentType.h" +#include "eConnectionType.h" BehaviorSyncEntry::BehaviorSyncEntry() { } @@ -25,7 +27,7 @@ BehaviorEndEntry::BehaviorEndEntry() { } uint32_t BehaviorContext::GetUniqueSkillId() const { - auto* entity = EntityManager::Instance()->GetEntity(this->originator); + auto* entity = Game::entityManager->GetEntity(this->originator); if (entity == nullptr) { Game::logger->Log("BehaviorContext", "Invalid entity for (%llu)!", this->originator); @@ -45,12 +47,17 @@ uint32_t BehaviorContext::GetUniqueSkillId() const { } -void BehaviorContext::RegisterSyncBehavior(const uint32_t syncId, Behavior* behavior, const BehaviorBranchContext& branchContext) { +void BehaviorContext::RegisterSyncBehavior(const uint32_t syncId, Behavior* behavior, const BehaviorBranchContext& branchContext, const float duration, bool ignoreInterrupts) { auto entry = BehaviorSyncEntry(); entry.handle = syncId; entry.behavior = behavior; entry.branchContext = branchContext; + entry.branchContext.isSync = true; + entry.ignoreInterrupts = ignoreInterrupts; + // Add 10 seconds + duration time to account for lag and give clients time to send their syncs to the server. + constexpr float lagTime = 10.0f; + entry.time = lagTime + duration; this->syncEntries.push_back(entry); } @@ -87,11 +94,11 @@ void BehaviorContext::ScheduleUpdate(const LWOOBJID id) { void BehaviorContext::ExecuteUpdates() { for (const auto& id : this->scheduledUpdates) { - auto* entity = EntityManager::Instance()->GetEntity(id); + auto* entity = Game::entityManager->GetEntity(id); if (entity == nullptr) continue; - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } this->scheduledUpdates.clear(); @@ -179,6 +186,21 @@ void BehaviorContext::SyncCalculation(const uint32_t syncId, const float time, B this->syncEntries.push_back(entry); } +void BehaviorContext::UpdatePlayerSyncs(float deltaTime) { + uint32_t i = 0; + while (i < this->syncEntries.size()) { + auto& entry = this->syncEntries.at(i); + + entry.time -= deltaTime; + + if (entry.time >= 0.0f) { + i++; + continue; + } + this->syncEntries.erase(this->syncEntries.begin() + i); + } +} + void BehaviorContext::InvokeEnd(const uint32_t id) { std::vector entries; @@ -214,7 +236,7 @@ bool BehaviorContext::CalculateUpdate(const float deltaTime) { } // Echo sync - GameMessages::EchoSyncSkill echo; + EchoSyncSkill echo; echo.bDone = true; echo.uiBehaviorHandle = entry.handle; @@ -231,7 +253,7 @@ bool BehaviorContext::CalculateUpdate(const float deltaTime) { // Write message RakNet::BitStream message; - PacketUtils::WriteHeader(message, CLIENT, MSG_CLIENT_GAME_MSG); + PacketUtils::WriteHeader(message, eConnectionType::CLIENT, eClientMessageType::GAME_MSG); message.Write(this->originator); echo.Serialize(&message); @@ -286,7 +308,7 @@ void BehaviorContext::Reset() { } std::vector BehaviorContext::GetValidTargets(int32_t ignoreFaction, int32_t includeFaction, bool targetSelf, bool targetEnemy, bool targetFriend) const { - auto* entity = EntityManager::Instance()->GetEntity(this->caster); + auto* entity = Game::entityManager->GetEntity(this->caster); std::vector targets; @@ -298,7 +320,7 @@ std::vector BehaviorContext::GetValidTargets(int32_t ignoreFaction, in if (!ignoreFaction && !includeFaction) { for (auto entry : entity->GetTargetsInPhantom()) { - auto* instance = EntityManager::Instance()->GetEntity(entry); + auto* instance = Game::entityManager->GetEntity(entry); if (instance == nullptr) { continue; @@ -308,13 +330,13 @@ std::vector BehaviorContext::GetValidTargets(int32_t ignoreFaction, in } } - if (ignoreFaction || includeFaction || (!entity->HasComponent(COMPONENT_TYPE_PHANTOM_PHYSICS) && targets.empty())) { + if (ignoreFaction || includeFaction || (!entity->HasComponent(eReplicaComponentType::PHANTOM_PHYSICS) && targets.empty())) { DestroyableComponent* destroyableComponent; - if (!entity->TryGetComponent(COMPONENT_TYPE_DESTROYABLE, destroyableComponent)) { + if (!entity->TryGetComponent(eReplicaComponentType::DESTROYABLE, destroyableComponent)) { return targets; } - auto entities = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS); + auto entities = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS); for (auto* candidate : entities) { const auto id = candidate->GetObjectID(); diff --git a/dGame/dBehaviors/BehaviorContext.h b/dGame/dBehaviors/BehaviorContext.h index 0462d97f..117f328d 100644 --- a/dGame/dBehaviors/BehaviorContext.h +++ b/dGame/dBehaviors/BehaviorContext.h @@ -80,7 +80,9 @@ struct BehaviorContext uint32_t GetUniqueSkillId() const; - void RegisterSyncBehavior(uint32_t syncId, Behavior* behavior, const BehaviorBranchContext& branchContext); + void UpdatePlayerSyncs(float deltaTime); + + void RegisterSyncBehavior(uint32_t syncId, Behavior* behavior, const BehaviorBranchContext& branchContext, const float duration, bool ignoreInterrupts = false); void RegisterTimerBehavior(Behavior* behavior, const BehaviorBranchContext& branchContext, LWOOBJID second = LWOOBJID_EMPTY); diff --git a/dGame/dBehaviors/BlockBehavior.cpp b/dGame/dBehaviors/BlockBehavior.cpp index cdbb3d80..19db0267 100644 --- a/dGame/dBehaviors/BlockBehavior.cpp +++ b/dGame/dBehaviors/BlockBehavior.cpp @@ -10,7 +10,7 @@ void BlockBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { const auto target = context->originator; - auto* entity = EntityManager::Instance()->GetEntity(target); + auto* entity = Game::entityManager->GetEntity(target); if (entity == nullptr) { Game::logger->Log("DamageAbsorptionBehavior", "Failed to find target (%llu)!", branch.target); @@ -40,7 +40,7 @@ void BlockBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitSt void BlockBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { const auto target = context->originator; - auto* entity = EntityManager::Instance()->GetEntity(target); + auto* entity = Game::entityManager->GetEntity(target); if (entity == nullptr) { Game::logger->Log("DamageAbsorptionBehavior", "Failed to find target (%llu)!", branch.target); diff --git a/dGame/dBehaviors/BuffBehavior.cpp b/dGame/dBehaviors/BuffBehavior.cpp index a39fd165..2c06dbd4 100644 --- a/dGame/dBehaviors/BuffBehavior.cpp +++ b/dGame/dBehaviors/BuffBehavior.cpp @@ -10,7 +10,7 @@ void BuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { const auto target = branch.target != LWOOBJID_EMPTY ? branch.target : context->originator; - auto* entity = EntityManager::Instance()->GetEntity(target); + auto* entity = Game::entityManager->GetEntity(target); if (entity == nullptr) { Game::logger->Log("BuffBehavior", "Invalid target (%llu)!", target); @@ -30,7 +30,7 @@ void BuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream component->SetMaxArmor(component->GetMaxArmor() + this->m_armor); component->SetMaxImagination(component->GetMaxImagination() + this->m_imagination); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); if (!context->unmanaged) { if (branch.duration > 0) { @@ -44,7 +44,7 @@ void BuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream void BuffBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { const auto target = branch.target != LWOOBJID_EMPTY ? branch.target : context->originator; - auto* entity = EntityManager::Instance()->GetEntity(target); + auto* entity = Game::entityManager->GetEntity(target); if (entity == nullptr) { Game::logger->Log("BuffBehavior", "Invalid target (%llu)!", target); @@ -64,7 +64,7 @@ void BuffBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch component->SetMaxArmor(component->GetMaxArmor() - this->m_armor); component->SetMaxImagination(component->GetMaxImagination() - this->m_imagination); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } void BuffBehavior::Timer(BehaviorContext* context, const BehaviorBranchContext branch, LWOOBJID second) { diff --git a/dGame/dBehaviors/CMakeLists.txt b/dGame/dBehaviors/CMakeLists.txt index fe8e89b8..8a9368b9 100644 --- a/dGame/dBehaviors/CMakeLists.txt +++ b/dGame/dBehaviors/CMakeLists.txt @@ -12,14 +12,17 @@ set(DGAME_DBEHAVIORS_SOURCES "AirMovementBehavior.cpp" "BuffBehavior.cpp" "CarBoostBehavior.cpp" "ChainBehavior.cpp" + "ChangeIdleFlagsBehavior.cpp" "ChangeOrientationBehavior.cpp" "ChargeUpBehavior.cpp" "ClearTargetBehavior.cpp" "DamageAbsorptionBehavior.cpp" "DamageReductionBehavior.cpp" + "DarkInspirationBehavior.cpp" "DurationBehavior.cpp" "EmptyBehavior.cpp" "EndBehavior.cpp" + "FallSpeedBehavior.cpp" "ForceMovementBehavior.cpp" "HealBehavior.cpp" "ImaginationBehavior.cpp" @@ -33,7 +36,9 @@ set(DGAME_DBEHAVIORS_SOURCES "AirMovementBehavior.cpp" "OverTimeBehavior.cpp" "PlayEffectBehavior.cpp" "ProjectileAttackBehavior.cpp" + "PropertyTeleportBehavior.cpp" "PullToPointBehavior.cpp" + "RemoveBuffBehavior.cpp" "RepairBehavior.cpp" "SkillCastFailedBehavior.cpp" "SkillEventBehavior.cpp" diff --git a/dGame/dBehaviors/CarBoostBehavior.cpp b/dGame/dBehaviors/CarBoostBehavior.cpp index 1ab0af95..e2929863 100644 --- a/dGame/dBehaviors/CarBoostBehavior.cpp +++ b/dGame/dBehaviors/CarBoostBehavior.cpp @@ -11,7 +11,7 @@ void CarBoostBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { GameMessages::SendVehicleAddPassiveBoostAction(branch.target, UNASSIGNED_SYSTEM_ADDRESS); - auto* entity = EntityManager::Instance()->GetEntity(context->originator); + auto* entity = Game::entityManager->GetEntity(context->originator); if (entity == nullptr) { return; @@ -22,7 +22,7 @@ void CarBoostBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitSt auto* possessableComponent = entity->GetComponent(); if (possessableComponent != nullptr) { - auto* possessor = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); + auto* possessor = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); if (possessor != nullptr) { auto* characterComponent = possessor->GetComponent(); diff --git a/dGame/dBehaviors/ChainBehavior.cpp b/dGame/dBehaviors/ChainBehavior.cpp index 3ef8c15b..ec0f8969 100644 --- a/dGame/dBehaviors/ChainBehavior.cpp +++ b/dGame/dBehaviors/ChainBehavior.cpp @@ -4,14 +4,19 @@ #include "dLogger.h" void ChainBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - uint32_t chain_index; + uint32_t chainIndex{}; - bitStream->Read(chain_index); + if (!bitStream->Read(chainIndex)) { + Game::logger->Log("ChainBehavior", "Unable to read chainIndex from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + } - chain_index--; + chainIndex--; - if (chain_index < this->m_behaviors.size()) { - this->m_behaviors.at(chain_index)->Handle(context, bitStream, branch); + if (chainIndex < this->m_behaviors.size()) { + this->m_behaviors.at(chainIndex)->Handle(context, bitStream, branch); + } else { + Game::logger->Log("ChainBehavior", "chainIndex out of bounds, aborting handle of chain %i bits unread %i", chainIndex, bitStream->GetNumberOfUnreadBits()); } } diff --git a/dGame/dBehaviors/ChangeIdleFlagsBehavior.cpp b/dGame/dBehaviors/ChangeIdleFlagsBehavior.cpp new file mode 100644 index 00000000..06a79fa7 --- /dev/null +++ b/dGame/dBehaviors/ChangeIdleFlagsBehavior.cpp @@ -0,0 +1,37 @@ + +#include "ChangeIdleFlagsBehavior.h" +#include "BehaviorContext.h" +#include "BehaviorBranchContext.h" + +void ChangeIdleFlagsBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + const auto target = branch.target != LWOOBJID_EMPTY ? branch.target : context->originator; + if (!target) return; + + GameMessages::SendChangeIdleFlags(target, m_FlagsOn, m_FlagsOff, UNASSIGNED_SYSTEM_ADDRESS); + + if (branch.duration > 0.0f) { + context->RegisterTimerBehavior(this, branch); + } else if (branch.start > 0) { + context->RegisterEndBehavior(this, branch); + } +} + +void ChangeIdleFlagsBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + Handle(context, bitStream, branch); +} + +void ChangeIdleFlagsBehavior::End(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { + const auto target = branch.target != LWOOBJID_EMPTY ? branch.target : context->originator; + if (!target) return; + // flip on and off to end behavior + GameMessages::SendChangeIdleFlags(target, m_FlagsOff, m_FlagsOn, UNASSIGNED_SYSTEM_ADDRESS); +} + +void ChangeIdleFlagsBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { + End(context, branch, second); +} + +void ChangeIdleFlagsBehavior::Load() { + m_FlagsOff = static_cast(GetInt("flags_off", 0)); + m_FlagsOn = static_cast(GetInt("flags_on", 0)); +} diff --git a/dGame/dBehaviors/ChangeIdleFlagsBehavior.h b/dGame/dBehaviors/ChangeIdleFlagsBehavior.h new file mode 100644 index 00000000..91f802f4 --- /dev/null +++ b/dGame/dBehaviors/ChangeIdleFlagsBehavior.h @@ -0,0 +1,23 @@ + +#pragma once +#include "Behavior.h" +#include "eAninmationFlags.h" + +class ChangeIdleFlagsBehavior final : public Behavior { +public: + + /* + * Inherited + */ + explicit ChangeIdleFlagsBehavior(const uint32_t behaviorId) : Behavior(behaviorId) {} + + void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + void Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; + void End(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; + void Load() override; + +private: + eAnimationFlags m_FlagsOff; + eAnimationFlags m_FlagsOn; +}; diff --git a/dGame/dBehaviors/ChangeOrientationBehavior.cpp b/dGame/dBehaviors/ChangeOrientationBehavior.cpp index a60be62f..445c76df 100644 --- a/dGame/dBehaviors/ChangeOrientationBehavior.cpp +++ b/dGame/dBehaviors/ChangeOrientationBehavior.cpp @@ -2,43 +2,35 @@ #include "BehaviorBranchContext.h" #include "BehaviorContext.h" #include "EntityManager.h" -#include "BaseCombatAIComponent.h" - -void ChangeOrientationBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { -} void ChangeOrientationBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - if (!m_ToTarget) return; // TODO: Add the other arguments to this behavior + Entity* sourceEntity; + if (this->m_orientCaster) sourceEntity = Game::entityManager->GetEntity(context->originator); + else sourceEntity = Game::entityManager->GetEntity(branch.target); + if (!sourceEntity) return; - auto* self = EntityManager::Instance()->GetEntity(context->originator); - auto* other = EntityManager::Instance()->GetEntity(branch.target); + if (this->m_toTarget) { + Entity* destinationEntity; + if (this->m_orientCaster) destinationEntity = Game::entityManager->GetEntity(branch.target); + else destinationEntity = Game::entityManager->GetEntity(context->originator); + if (!destinationEntity) return; - if (self == nullptr || other == nullptr) return; - - const auto source = self->GetPosition(); - const auto destination = self->GetPosition(); - - if (m_OrientCaster) { - auto* baseCombatAIComponent = self->GetComponent(); - - /*if (baseCombatAIComponent != nullptr) - { - baseCombatAIComponent->LookAt(destination); - } - else*/ - { - self->SetRotation(NiQuaternion::LookAt(source, destination)); - } - - EntityManager::Instance()->SerializeEntity(self); - } else { - other->SetRotation(NiQuaternion::LookAt(destination, source)); - - EntityManager::Instance()->SerializeEntity(other); - } + sourceEntity->SetRotation( + NiQuaternion::LookAt(sourceEntity->GetPosition(), destinationEntity->GetPosition()) + ); + } else if (this->m_toAngle){ + auto baseAngle = NiPoint3(0, 0, this->m_angle); + if (this->m_relative) baseAngle += sourceEntity->GetRotation().GetForwardVector(); + sourceEntity->SetRotation(NiQuaternion::FromEulerAngles(baseAngle)); + } else return; + Game::entityManager->SerializeEntity(sourceEntity); + return; } void ChangeOrientationBehavior::Load() { - m_OrientCaster = GetBoolean("orient_caster"); - m_ToTarget = GetBoolean("to_target"); + this->m_orientCaster = GetBoolean("orient_caster", true); + this->m_toTarget = GetBoolean("to_target", false); + this->m_toAngle = GetBoolean("to_angle", false); + this->m_angle = GetFloat("angle", 0.0f); + this->m_relative = GetBoolean("relative", false); } diff --git a/dGame/dBehaviors/ChangeOrientationBehavior.h b/dGame/dBehaviors/ChangeOrientationBehavior.h index eb038ffb..22c92bb8 100644 --- a/dGame/dBehaviors/ChangeOrientationBehavior.h +++ b/dGame/dBehaviors/ChangeOrientationBehavior.h @@ -2,24 +2,15 @@ #include "Behavior.h" -#include - -class ChangeOrientationBehavior final : public Behavior -{ +class ChangeOrientationBehavior final : public Behavior { public: - bool m_OrientCaster; - bool m_ToTarget; - - /* - * Inherited - */ - - explicit ChangeOrientationBehavior(const uint32_t behaviorId) : Behavior(behaviorId) { - } - - void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; - + explicit ChangeOrientationBehavior(const uint32_t behaviorId) : Behavior(behaviorId) {} void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; - void Load() override; +private: + bool m_orientCaster; + bool m_toTarget; + bool m_toAngle; + float m_angle; + bool m_relative; }; diff --git a/dGame/dBehaviors/ChargeUpBehavior.cpp b/dGame/dBehaviors/ChargeUpBehavior.cpp index fa9fa34b..4c7c3dac 100644 --- a/dGame/dBehaviors/ChargeUpBehavior.cpp +++ b/dGame/dBehaviors/ChargeUpBehavior.cpp @@ -1,14 +1,18 @@ #include "ChargeUpBehavior.h" #include "BehaviorBranchContext.h" #include "BehaviorContext.h" +#include "Game.h" #include "dLogger.h" void ChargeUpBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - uint32_t handle; + uint32_t handle{}; - bitStream->Read(handle); + if (!bitStream->Read(handle)) { + Game::logger->Log("ChargeUpBehavior", "Unable to read handle from bitStream, aborting Handle! variable_type"); + return; + }; - context->RegisterSyncBehavior(handle, this, branch); + context->RegisterSyncBehavior(handle, this, branch, this->m_MaxDuration); } void ChargeUpBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { @@ -20,4 +24,5 @@ void ChargeUpBehavior::Sync(BehaviorContext* context, RakNet::BitStream* bitStre void ChargeUpBehavior::Load() { this->m_action = GetAction("action"); + this->m_MaxDuration = GetFloat("max_duration"); } diff --git a/dGame/dBehaviors/ChargeUpBehavior.h b/dGame/dBehaviors/ChargeUpBehavior.h index d753895e..ceb40f6c 100644 --- a/dGame/dBehaviors/ChargeUpBehavior.h +++ b/dGame/dBehaviors/ChargeUpBehavior.h @@ -4,14 +4,7 @@ class ChargeUpBehavior final : public Behavior { public: - Behavior* m_action; - - /* - * Inherited - */ - - explicit ChargeUpBehavior(const uint32_t behaviorId) : Behavior(behaviorId) { - } + explicit ChargeUpBehavior(const uint32_t behaviorId) : Behavior(behaviorId) {} void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; @@ -20,4 +13,7 @@ public: void Sync(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; void Load() override; +private: + Behavior* m_action; + float m_MaxDuration; }; diff --git a/dGame/dBehaviors/DamageAbsorptionBehavior.cpp b/dGame/dBehaviors/DamageAbsorptionBehavior.cpp index 48dbf705..f9936767 100644 --- a/dGame/dBehaviors/DamageAbsorptionBehavior.cpp +++ b/dGame/dBehaviors/DamageAbsorptionBehavior.cpp @@ -8,7 +8,7 @@ #include "DestroyableComponent.h" void DamageAbsorptionBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) { Game::logger->Log("DamageAbsorptionBehavior", "Failed to find target (%llu)!", branch.target); @@ -34,7 +34,7 @@ void DamageAbsorptionBehavior::Calculate(BehaviorContext* context, RakNet::BitSt } void DamageAbsorptionBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, const LWOOBJID second) { - auto* target = EntityManager::Instance()->GetEntity(second); + auto* target = Game::entityManager->GetEntity(second); if (target == nullptr) { Game::logger->Log("DamageAbsorptionBehavior", "Failed to find target (%llu)!", second); diff --git a/dGame/dBehaviors/DamageReductionBehavior.cpp b/dGame/dBehaviors/DamageReductionBehavior.cpp index 2b18b7c2..7645ec7b 100644 --- a/dGame/dBehaviors/DamageReductionBehavior.cpp +++ b/dGame/dBehaviors/DamageReductionBehavior.cpp @@ -8,7 +8,7 @@ #include "DestroyableComponent.h" void DamageReductionBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) { Game::logger->Log("DamageReductionBehavior", "Failed to find target (%llu)!", branch.target); @@ -32,7 +32,7 @@ void DamageReductionBehavior::Calculate(BehaviorContext* context, RakNet::BitStr } void DamageReductionBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, const LWOOBJID second) { - auto* target = EntityManager::Instance()->GetEntity(second); + auto* target = Game::entityManager->GetEntity(second); if (target == nullptr) { Game::logger->Log("DamageReductionBehavior", "Failed to find target (%llu)!", second); diff --git a/dGame/dBehaviors/DarkInspirationBehavior.cpp b/dGame/dBehaviors/DarkInspirationBehavior.cpp new file mode 100644 index 00000000..4e9890e3 --- /dev/null +++ b/dGame/dBehaviors/DarkInspirationBehavior.cpp @@ -0,0 +1,52 @@ +#include "DarkInspirationBehavior.h" + +#include "BehaviorBranchContext.h" +#include "Entity.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "BehaviorContext.h" + +void DarkInspirationBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { + auto* target = Game::entityManager->GetEntity(branch.target); + + if (target == nullptr) { + Game::logger->LogDebug("DarkInspirationBehavior", "Failed to find target (%llu)!", branch.target); + return; + } + + auto* destroyableComponent = target->GetComponent(); + + if (destroyableComponent == nullptr) { + return; + } + + if (destroyableComponent->HasFaction(m_FactionList)) { + this->m_ActionIfFactionMatches->Handle(context, bitStream, branch); + } +} + +void DarkInspirationBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + auto* target = Game::entityManager->GetEntity(branch.target); + + if (target == nullptr) { + Game::logger->LogDebug("DarkInspirationBehavior", "Failed to find target (%llu)!", branch.target); + + return; + } + + auto* destroyableComponent = target->GetComponent(); + + if (destroyableComponent == nullptr) { + return; + } + + if (destroyableComponent->HasFaction(m_FactionList)) { + this->m_ActionIfFactionMatches->Calculate(context, bitStream, branch); + } +} + +void DarkInspirationBehavior::Load() { + this->m_ActionIfFactionMatches = GetAction("action"); + + this->m_FactionList = GetInt("faction_list"); +} diff --git a/dGame/dBehaviors/DarkInspirationBehavior.h b/dGame/dBehaviors/DarkInspirationBehavior.h new file mode 100644 index 00000000..f8298742 --- /dev/null +++ b/dGame/dBehaviors/DarkInspirationBehavior.h @@ -0,0 +1,22 @@ +#pragma once +#include "Behavior.h" + +class DarkInspirationBehavior final : public Behavior +{ +public: + /* + * Inherited + */ + + explicit DarkInspirationBehavior(const uint32_t behaviorId) : Behavior(behaviorId) { + } + + void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + + void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + + void Load() override; +private: + Behavior* m_ActionIfFactionMatches; + uint32_t m_FactionList; +}; diff --git a/dGame/dBehaviors/FallSpeedBehavior.cpp b/dGame/dBehaviors/FallSpeedBehavior.cpp new file mode 100644 index 00000000..dfbdec2a --- /dev/null +++ b/dGame/dBehaviors/FallSpeedBehavior.cpp @@ -0,0 +1,50 @@ +#include "FallSpeedBehavior.h" + +#include "ControllablePhysicsComponent.h" +#include "BehaviorContext.h" +#include "BehaviorBranchContext.h" + + +void FallSpeedBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + // make sure required parameter has non-default value + if (m_PercentSlowed == 0.0f) return; + auto* target = Game::entityManager->GetEntity(branch.target); + if (!target) return; + + auto* controllablePhysicsComponent = target->GetComponent(); + if (!controllablePhysicsComponent) return; + controllablePhysicsComponent->SetGravityScale(m_PercentSlowed); + Game::entityManager->SerializeEntity(target); + + if (branch.duration > 0.0f) { + context->RegisterTimerBehavior(this, branch); + } else if (branch.start > 0) { + context->RegisterEndBehavior(this, branch); + } +} + +void FallSpeedBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + Handle(context, bitStream, branch); +} + +void FallSpeedBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { + End(context, branch, second); +} + +void FallSpeedBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { + End(context, branch, LWOOBJID_EMPTY); +} + +void FallSpeedBehavior::End(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { + auto* target = Game::entityManager->GetEntity(branch.target); + if (!target) return; + + auto* controllablePhysicsComponent = target->GetComponent(); + if (!controllablePhysicsComponent) return; + controllablePhysicsComponent->SetGravityScale(1); + Game::entityManager->SerializeEntity(target); +} + +void FallSpeedBehavior::Load(){ + m_PercentSlowed = GetFloat("percent_slowed"); +} diff --git a/dGame/dBehaviors/FallSpeedBehavior.h b/dGame/dBehaviors/FallSpeedBehavior.h new file mode 100644 index 00000000..01f0143f --- /dev/null +++ b/dGame/dBehaviors/FallSpeedBehavior.h @@ -0,0 +1,18 @@ +#pragma once +#include "Behavior.h" + +class FallSpeedBehavior final : public Behavior +{ +public: + explicit FallSpeedBehavior(const uint32_t behaviorId) : Behavior(behaviorId) {} + + void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + void Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; + void End(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; + void UnCast(BehaviorContext* context, BehaviorBranchContext branch) override; + void Load() override; + +private: + float m_PercentSlowed; +}; diff --git a/dGame/dBehaviors/ForceMovementBehavior.cpp b/dGame/dBehaviors/ForceMovementBehavior.cpp index 55cda622..97208236 100644 --- a/dGame/dBehaviors/ForceMovementBehavior.cpp +++ b/dGame/dBehaviors/ForceMovementBehavior.cpp @@ -3,23 +3,34 @@ #include "BehaviorContext.h" #include "ControllablePhysicsComponent.h" #include "EntityManager.h" +#include "Game.h" +#include "dLogger.h" void ForceMovementBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { if (this->m_hitAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && this->m_hitEnemyAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && this->m_hitFactionAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY) { return; } - uint32_t handle; - bitStream->Read(handle); - context->RegisterSyncBehavior(handle, this, branch); + uint32_t handle{}; + if (!bitStream->Read(handle)) { + Game::logger->Log("ForceMovementBehavior", "Unable to read handle from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + } + context->RegisterSyncBehavior(handle, this, branch, this->m_Duration); } void ForceMovementBehavior::Sync(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - uint32_t next; - bitStream->Read(next); + uint32_t next{}; + if (!bitStream->Read(next)) { + Game::logger->Log("ForceMovementBehavior", "Unable to read target from bitStream, aborting Sync! %i", bitStream->GetNumberOfUnreadBits()); + return; + } - LWOOBJID target; - bitStream->Read(target); + LWOOBJID target{}; + if (!bitStream->Read(target)) { + Game::logger->Log("ForceMovementBehavior", "Unable to read target from bitStream, aborting Sync! %i", bitStream->GetNumberOfUnreadBits()); + return; + } branch.target = target; auto* behavior = CreateBehavior(next); @@ -31,7 +42,7 @@ void ForceMovementBehavior::Calculate(BehaviorContext* context, RakNet::BitStrea return; } - auto* casterEntity = EntityManager::Instance()->GetEntity(context->caster); + auto* casterEntity = Game::entityManager->GetEntity(context->caster); if (casterEntity != nullptr) { auto* controllablePhysicsComponent = casterEntity->GetComponent(); if (controllablePhysicsComponent != nullptr) { @@ -40,7 +51,7 @@ void ForceMovementBehavior::Calculate(BehaviorContext* context, RakNet::BitStrea controllablePhysicsComponent->SetVelocity(controllablePhysicsComponent->GetRotation().GetForwardVector() * 25); } - EntityManager::Instance()->SerializeEntity(casterEntity); + Game::entityManager->SerializeEntity(casterEntity); } } @@ -61,7 +72,7 @@ void ForceMovementBehavior::Load() { } void ForceMovementBehavior::SyncCalculation(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* casterEntity = EntityManager::Instance()->GetEntity(context->caster); + auto* casterEntity = Game::entityManager->GetEntity(context->caster); if (casterEntity != nullptr) { auto* controllablePhysicsComponent = casterEntity->GetComponent(); if (controllablePhysicsComponent != nullptr) { @@ -69,11 +80,11 @@ void ForceMovementBehavior::SyncCalculation(BehaviorContext* context, RakNet::Bi controllablePhysicsComponent->SetPosition(controllablePhysicsComponent->GetPosition() + controllablePhysicsComponent->GetVelocity() * m_Duration); controllablePhysicsComponent->SetVelocity({}); - EntityManager::Instance()->SerializeEntity(casterEntity); + Game::entityManager->SerializeEntity(casterEntity); } } this->m_hitAction->Calculate(context, bitStream, branch); this->m_hitEnemyAction->Calculate(context, bitStream, branch); - this->m_hitEnemyAction->Calculate(context, bitStream, branch); + this->m_hitFactionAction->Calculate(context, bitStream, branch); } diff --git a/dGame/dBehaviors/HealBehavior.cpp b/dGame/dBehaviors/HealBehavior.cpp index 2799d2e2..dae009d4 100644 --- a/dGame/dBehaviors/HealBehavior.cpp +++ b/dGame/dBehaviors/HealBehavior.cpp @@ -4,10 +4,11 @@ #include "dLogger.h" #include "EntityManager.h" #include "DestroyableComponent.h" +#include "eReplicaComponentType.h" void HealBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_stream, const BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); if (entity == nullptr) { Game::logger->Log("HealBehavior", "Failed to find entity for (%llu)!", branch.target); @@ -15,7 +16,7 @@ void HealBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_strea return; } - auto* destroyable = static_cast(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + auto* destroyable = static_cast(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (destroyable == nullptr) { Game::logger->Log("HealBehavior", "Failed to find destroyable component for %(llu)!", branch.target); diff --git a/dGame/dBehaviors/ImaginationBehavior.cpp b/dGame/dBehaviors/ImaginationBehavior.cpp index 59b192b0..50f7e046 100644 --- a/dGame/dBehaviors/ImaginationBehavior.cpp +++ b/dGame/dBehaviors/ImaginationBehavior.cpp @@ -7,7 +7,7 @@ void ImaginationBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_stream, const BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); if (entity == nullptr) { return; diff --git a/dGame/dBehaviors/ImmunityBehavior.cpp b/dGame/dBehaviors/ImmunityBehavior.cpp index 69c652f9..ba4a8b77 100644 --- a/dGame/dBehaviors/ImmunityBehavior.cpp +++ b/dGame/dBehaviors/ImmunityBehavior.cpp @@ -6,28 +6,48 @@ #include "Game.h" #include "dLogger.h" #include "DestroyableComponent.h" +#include "ControllablePhysicsComponent.h" +#include "eStateChangeType.h" void ImmunityBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); - if (target == nullptr) { + if (!target) { Game::logger->Log("DamageAbsorptionBehavior", "Failed to find target (%llu)!", branch.target); - return; } - auto* destroyable = static_cast(target->GetComponent(COMPONENT_TYPE_DESTROYABLE)); - - if (destroyable == nullptr) { - return; + auto* destroyableComponent = target->GetComponent(); + if (destroyableComponent) { + destroyableComponent->SetStatusImmunity( + eStateChangeType::PUSH, + this->m_ImmuneToBasicAttack, + this->m_ImmuneToDamageOverTime, + this->m_ImmuneToKnockback, + this->m_ImmuneToInterrupt, + this->m_ImmuneToSpeed, + this->m_ImmuneToImaginationGain, + this->m_ImmuneToImaginationLoss, + this->m_ImmuneToQuickbuildInterrupt, + this->m_ImmuneToPullToPoint + ); } - if (!this->m_immuneBasicAttack) { - return; + auto* controllablePhysicsComponent = target->GetComponent(); + if (controllablePhysicsComponent) { + controllablePhysicsComponent->SetStunImmunity( + eStateChangeType::PUSH, + context->caster, + this->m_ImmuneToStunAttack, + this->m_ImmuneToStunEquip, + this->m_ImmuneToStunInteract, + this->m_ImmuneToStunJump, + this->m_ImmuneToStunMove, + this->m_ImmuneToStunTurn, + this->m_ImmuneToStunUseItem + ); } - destroyable->PushImmunity(); - context->RegisterTimerBehavior(this, branch, target->GetObjectID()); } @@ -36,23 +56,62 @@ void ImmunityBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bi } void ImmunityBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, const LWOOBJID second) { - auto* target = EntityManager::Instance()->GetEntity(second); + auto* target = Game::entityManager->GetEntity(second); - if (target == nullptr) { + if (!target) { Game::logger->Log("DamageAbsorptionBehavior", "Failed to find target (%llu)!", second); - return; } - auto* destroyable = static_cast(target->GetComponent(COMPONENT_TYPE_DESTROYABLE)); - - if (destroyable == nullptr) { - return; + auto* destroyableComponent = target->GetComponent(); + if (destroyableComponent) { + destroyableComponent->SetStatusImmunity( + eStateChangeType::POP, + this->m_ImmuneToBasicAttack, + this->m_ImmuneToDamageOverTime, + this->m_ImmuneToKnockback, + this->m_ImmuneToInterrupt, + this->m_ImmuneToSpeed, + this->m_ImmuneToImaginationGain, + this->m_ImmuneToImaginationLoss, + this->m_ImmuneToQuickbuildInterrupt, + this->m_ImmuneToPullToPoint + ); + } + + auto* controllablePhysicsComponent = target->GetComponent(); + if (controllablePhysicsComponent) { + controllablePhysicsComponent->SetStunImmunity( + eStateChangeType::POP, + context->caster, + this->m_ImmuneToStunAttack, + this->m_ImmuneToStunEquip, + this->m_ImmuneToStunInteract, + this->m_ImmuneToStunJump, + this->m_ImmuneToStunMove, + this->m_ImmuneToStunTurn, + this->m_ImmuneToStunUseItem + ); } - destroyable->PopImmunity(); } void ImmunityBehavior::Load() { - this->m_immuneBasicAttack = GetBoolean("immune_basic_attack"); + //Stun + this->m_ImmuneToStunAttack = GetBoolean("immune_stun_attack", false); + this->m_ImmuneToStunEquip = GetBoolean("immune_stun_equip", false); + this->m_ImmuneToStunInteract = GetBoolean("immune_stun_interact", false); + this->m_ImmuneToStunMove = GetBoolean("immune_stun_move", false); + this->m_ImmuneToStunTurn = GetBoolean("immune_stun_rotate", false); + + // Status + this->m_ImmuneToBasicAttack = GetBoolean("immune_basic_attack", false); + this->m_ImmuneToDamageOverTime = GetBoolean("immune_damage_over_time", false); + this->m_ImmuneToKnockback = GetBoolean("immune_knockback", false); + this->m_ImmuneToInterrupt = GetBoolean("immune_interrupt", false); + this->m_ImmuneToSpeed = GetBoolean("immune_speed", false); + this->m_ImmuneToImaginationGain = GetBoolean("immune_imagination_gain", false); + this->m_ImmuneToImaginationLoss = GetBoolean("immune_imagination_loss", false); + this->m_ImmuneToQuickbuildInterrupt = GetBoolean("immune_quickbuild_interrupts", false); + this->m_ImmuneToPullToPoint = GetBoolean("immune_pulltopoint", false); } diff --git a/dGame/dBehaviors/ImmunityBehavior.h b/dGame/dBehaviors/ImmunityBehavior.h index f9409e4c..02cc0fae 100644 --- a/dGame/dBehaviors/ImmunityBehavior.h +++ b/dGame/dBehaviors/ImmunityBehavior.h @@ -4,8 +4,6 @@ class ImmunityBehavior final : public Behavior { public: - uint32_t m_immuneBasicAttack; - /* * Inherited */ @@ -20,4 +18,25 @@ public: void Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; void Load() override; + +private: + // stuns + bool m_ImmuneToStunAttack = false; + bool m_ImmuneToStunEquip = false; + bool m_ImmuneToStunInteract = false; + bool m_ImmuneToStunJump = false; // Unused + bool m_ImmuneToStunMove = false; + bool m_ImmuneToStunTurn = false; + bool m_ImmuneToStunUseItem = false; // Unused + + //status + bool m_ImmuneToBasicAttack = false; + bool m_ImmuneToDamageOverTime = false; + bool m_ImmuneToKnockback = false; + bool m_ImmuneToInterrupt = false; + bool m_ImmuneToSpeed = false; + bool m_ImmuneToImaginationGain = false; + bool m_ImmuneToImaginationLoss = false; + bool m_ImmuneToQuickbuildInterrupt = false; + bool m_ImmuneToPullToPoint = false; // Unused in cdclient, but used in client }; diff --git a/dGame/dBehaviors/InterruptBehavior.cpp b/dGame/dBehaviors/InterruptBehavior.cpp index b42eadb0..ffe3bb8b 100644 --- a/dGame/dBehaviors/InterruptBehavior.cpp +++ b/dGame/dBehaviors/InterruptBehavior.cpp @@ -11,7 +11,10 @@ void InterruptBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitS if (branch.target != context->originator) { bool unknown = false; - bitStream->Read(unknown); + if (!bitStream->Read(unknown)) { + Game::logger->Log("InterruptBehavior", "Unable to read unknown1 from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; if (unknown) return; } @@ -19,7 +22,10 @@ void InterruptBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitS if (!this->m_interruptBlock) { bool unknown = false; - bitStream->Read(unknown); + if (!bitStream->Read(unknown)) { + Game::logger->Log("InterruptBehavior", "Unable to read unknown2 from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; if (unknown) return; } @@ -28,12 +34,15 @@ void InterruptBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitS { bool unknown = false; - bitStream->Read(unknown); + if (!bitStream->Read(unknown)) { + Game::logger->Log("InterruptBehavior", "Unable to read unknown3 from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; } if (branch.target == context->originator) return; - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) return; @@ -58,7 +67,7 @@ void InterruptBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* b if (branch.target == context->originator) return; - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) return; diff --git a/dGame/dBehaviors/JetPackBehavior.cpp b/dGame/dBehaviors/JetPackBehavior.cpp index e7d76560..134dd0fb 100644 --- a/dGame/dBehaviors/JetPackBehavior.cpp +++ b/dGame/dBehaviors/JetPackBehavior.cpp @@ -6,7 +6,7 @@ #include "Character.h" void JetPackBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_stream, const BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); GameMessages::SendSetJetPackMode(entity, true, this->m_BypassChecks, this->m_EnableHover, this->m_effectId, this->m_Airspeed, this->m_MaxAirspeed, this->m_VerticalVelocity, this->m_WarningEffectID); @@ -20,7 +20,7 @@ void JetPackBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_st } void JetPackBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); GameMessages::SendSetJetPackMode(entity, false); diff --git a/dGame/dBehaviors/KnockbackBehavior.cpp b/dGame/dBehaviors/KnockbackBehavior.cpp index 83c82010..d67ebbc8 100644 --- a/dGame/dBehaviors/KnockbackBehavior.cpp +++ b/dGame/dBehaviors/KnockbackBehavior.cpp @@ -6,17 +6,22 @@ #include "EntityManager.h" #include "GameMessages.h" #include "DestroyableComponent.h" +#include "Game.h" +#include "dLogger.h" void KnockbackBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - bool unknown; + bool unknown{}; - bitStream->Read(unknown); + if (!bitStream->Read(unknown)) { + Game::logger->Log("KnockbackBehavior", "Unable to read unknown from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; } void KnockbackBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { bool blocked = false; - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target != nullptr) { auto* destroyableComponent = target->GetComponent(); diff --git a/dGame/dBehaviors/LootBuffBehavior.cpp b/dGame/dBehaviors/LootBuffBehavior.cpp index 6e5634fc..c7a6b36a 100644 --- a/dGame/dBehaviors/LootBuffBehavior.cpp +++ b/dGame/dBehaviors/LootBuffBehavior.cpp @@ -1,14 +1,14 @@ #include "LootBuffBehavior.h" void LootBuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto target = EntityManager::Instance()->GetEntity(context->caster); + auto target = Game::entityManager->GetEntity(context->caster); if (!target) return; auto controllablePhysicsComponent = target->GetComponent(); if (!controllablePhysicsComponent) return; controllablePhysicsComponent->AddPickupRadiusScale(m_Scale); - EntityManager::Instance()->SerializeEntity(target); + Game::entityManager->SerializeEntity(target); if (branch.duration > 0) context->RegisterTimerBehavior(this, branch); @@ -19,14 +19,14 @@ void LootBuffBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bi } void LootBuffBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { - auto target = EntityManager::Instance()->GetEntity(context->caster); + auto target = Game::entityManager->GetEntity(context->caster); if (!target) return; auto controllablePhysicsComponent = target->GetComponent(); if (!controllablePhysicsComponent) return; controllablePhysicsComponent->RemovePickupRadiusScale(m_Scale); - EntityManager::Instance()->SerializeEntity(target); + Game::entityManager->SerializeEntity(target); } void LootBuffBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { diff --git a/dGame/dBehaviors/MovementSwitchBehavior.cpp b/dGame/dBehaviors/MovementSwitchBehavior.cpp index 17ed5c40..0c11380f 100644 --- a/dGame/dBehaviors/MovementSwitchBehavior.cpp +++ b/dGame/dBehaviors/MovementSwitchBehavior.cpp @@ -4,18 +4,20 @@ #include "dLogger.h" void MovementSwitchBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { - if (this->m_groundAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && - this->m_jumpAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && - this->m_fallingAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && - this->m_doubleJumpAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && - this->m_airAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && - this->m_jetpackAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY) { + uint32_t movementType{}; + if (!bitStream->Read(movementType)) { + if (this->m_groundAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && + this->m_jumpAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && + this->m_fallingAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && + this->m_doubleJumpAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && + this->m_airAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && + this->m_jetpackAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY && + this->m_movingAction->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY) { + return; + } + Game::logger->Log("MovementSwitchBehavior", "Unable to read movementType from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); return; - } - - uint32_t movementType; - - bitStream->Read(movementType); + }; switch (movementType) { case 1: @@ -25,34 +27,40 @@ void MovementSwitchBehavior::Handle(BehaviorContext* context, RakNet::BitStream* this->m_jumpAction->Handle(context, bitStream, branch); break; case 3: - this->m_fallingAction->Handle(context, bitStream, branch); + this->m_airAction->Handle(context, bitStream, branch); break; case 4: this->m_doubleJumpAction->Handle(context, bitStream, branch); break; case 5: - this->m_airAction->Handle(context, bitStream, branch); + this->m_fallingAction->Handle(context, bitStream, branch); break; case 6: this->m_jetpackAction->Handle(context, bitStream, branch); break; default: - Game::logger->Log("MovementSwitchBehavior", "Invalid movement behavior type (%i)!", movementType); + this->m_groundAction->Handle(context, bitStream, branch); break; } } -void MovementSwitchBehavior::Load() { - this->m_airAction = GetAction("air_action"); - - this->m_doubleJumpAction = GetAction("double_jump_action"); - - this->m_fallingAction = GetAction("falling_action"); - - this->m_groundAction = GetAction("ground_action"); - - this->m_jetpackAction = GetAction("jetpack_action"); - - this->m_jumpAction = GetAction("jump_action"); +Behavior* MovementSwitchBehavior::LoadMovementType(std::string movementType) { + float actionValue = GetFloat(movementType, -1.0f); + auto loadedBehavior = GetAction(actionValue != -1.0f ? actionValue : 0.0f); + if (actionValue == -1.0f && loadedBehavior->m_templateId == BehaviorTemplates::BEHAVIOR_EMPTY) { + loadedBehavior = this->m_groundAction; + } + return loadedBehavior; } +void MovementSwitchBehavior::Load() { + float groundActionValue = GetFloat("ground_action", -1.0f); + this->m_groundAction = GetAction(groundActionValue != -1.0f ? groundActionValue : 0.0f); + + this->m_airAction = LoadMovementType("air_action"); + this->m_doubleJumpAction = LoadMovementType("double_jump_action"); + this->m_fallingAction = LoadMovementType("falling_action"); + this->m_jetpackAction = LoadMovementType("jetpack_action"); + this->m_jumpAction = LoadMovementType("jump_action"); + this->m_movingAction = LoadMovementType("moving_action"); +} diff --git a/dGame/dBehaviors/MovementSwitchBehavior.h b/dGame/dBehaviors/MovementSwitchBehavior.h index 82e1a9e9..e6ff96f6 100644 --- a/dGame/dBehaviors/MovementSwitchBehavior.h +++ b/dGame/dBehaviors/MovementSwitchBehavior.h @@ -3,7 +3,7 @@ class MovementSwitchBehavior final : public Behavior { -public: +private: /* * Members */ @@ -19,6 +19,17 @@ public: Behavior* m_jumpAction; + Behavior* m_movingAction; + + /** + * @brief Loads a movement type from the database into a behavior + * + * @param movementType The movement type to lookup in the database + * @param behaviorToLoad The Behavior where the result will be stored + */ + Behavior* LoadMovementType(std::string movementType); + +public: /* * Inherited */ diff --git a/dGame/dBehaviors/OverTimeBehavior.cpp b/dGame/dBehaviors/OverTimeBehavior.cpp index 43ac9bc2..49077f0f 100644 --- a/dGame/dBehaviors/OverTimeBehavior.cpp +++ b/dGame/dBehaviors/OverTimeBehavior.cpp @@ -6,17 +6,21 @@ #include "EntityManager.h" #include "SkillComponent.h" #include "DestroyableComponent.h" +#include "CDClientDatabase.h" +#include "CDClientManager.h" + +#include "CDSkillBehaviorTable.h" void OverTimeBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { const auto originator = context->originator; - auto* entity = EntityManager::Instance()->GetEntity(originator); + auto* entity = Game::entityManager->GetEntity(originator); if (entity == nullptr) return; for (size_t i = 0; i < m_NumIntervals; i++) { entity->AddCallbackTimer((i + 1) * m_Delay, [originator, branch, this]() { - auto* entity = EntityManager::Instance()->GetEntity(originator); + auto* entity = Game::entityManager->GetEntity(originator); if (entity == nullptr) return; @@ -37,7 +41,7 @@ void OverTimeBehavior::Load() { m_Action = GetInt("action"); // Since m_Action is a skillID and not a behavior, get is correlated behaviorID. - CDSkillBehaviorTable* skillTable = CDClientManager::Instance()->GetTable("SkillBehavior"); + CDSkillBehaviorTable* skillTable = CDClientManager::Instance().GetTable(); m_ActionBehaviorId = skillTable->GetSkillByID(m_Action).behaviorID; m_Delay = GetFloat("delay"); diff --git a/dGame/dBehaviors/ProjectileAttackBehavior.cpp b/dGame/dBehaviors/ProjectileAttackBehavior.cpp index 350234e2..eb435d7c 100644 --- a/dGame/dBehaviors/ProjectileAttackBehavior.cpp +++ b/dGame/dBehaviors/ProjectileAttackBehavior.cpp @@ -6,13 +6,17 @@ #include "dLogger.h" #include "SkillComponent.h" #include "../dWorldServer/ObjectIDManager.h" +#include "eObjectBits.h" void ProjectileAttackBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - LWOOBJID target; + LWOOBJID target{}; - bitStream->Read(target); + if (!bitStream->Read(target)) { + Game::logger->Log("ProjectileAttackBehavior", "Unable to read target from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; - auto* entity = EntityManager::Instance()->GetEntity(context->originator); + auto* entity = Game::entityManager->GetEntity(context->originator); if (entity == nullptr) { Game::logger->Log("ProjectileAttackBehavior", "Failed to find originator (%llu)!", context->originator); @@ -28,17 +32,23 @@ void ProjectileAttackBehavior::Handle(BehaviorContext* context, RakNet::BitStrea return; } - if (m_useMouseposit) { + if (m_useMouseposit && !branch.isSync) { NiPoint3 targetPosition = NiPoint3::ZERO; - bitStream->Read(targetPosition); + if (!bitStream->Read(targetPosition)) { + Game::logger->Log("ProjectileAttackBehavior", "Unable to read targetPosition from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; } - auto* targetEntity = EntityManager::Instance()->GetEntity(target); + auto* targetEntity = Game::entityManager->GetEntity(target); for (auto i = 0u; i < this->m_projectileCount; ++i) { - LWOOBJID projectileId; + LWOOBJID projectileId{}; - bitStream->Read(projectileId); + if (!bitStream->Read(projectileId)) { + Game::logger->Log("ProjectileAttackBehavior", "Unable to read projectileId from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; branch.target = target; branch.isProjectile = true; @@ -51,7 +61,7 @@ void ProjectileAttackBehavior::Handle(BehaviorContext* context, RakNet::BitStrea void ProjectileAttackBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { bitStream->Write(branch.target); - auto* entity = EntityManager::Instance()->GetEntity(context->originator); + auto* entity = Game::entityManager->GetEntity(context->originator); if (entity == nullptr) { Game::logger->Log("ProjectileAttackBehavior", "Failed to find originator (%llu)!", context->originator); @@ -68,7 +78,7 @@ void ProjectileAttackBehavior::Calculate(BehaviorContext* context, RakNet::BitSt } - auto* other = EntityManager::Instance()->GetEntity(branch.target); + auto* other = Game::entityManager->GetEntity(branch.target); if (other == nullptr) { Game::logger->Log("ProjectileAttackBehavior", "Invalid projectile target (%llu)!", branch.target); @@ -98,7 +108,7 @@ void ProjectileAttackBehavior::Calculate(BehaviorContext* context, RakNet::BitSt for (auto i = 0u; i < this->m_projectileCount; ++i) { auto id = static_cast(ObjectIDManager::Instance()->GenerateObjectID()); - id = GeneralUtils::SetBit(id, OBJECT_BIT_CLIENT); + GeneralUtils::SetBit(id, eObjectBits::SPAWNED); bitStream->Write(id); @@ -148,4 +158,6 @@ void ProjectileAttackBehavior::Load() { this->m_trackRadius = GetFloat("track_radius"); this->m_useMouseposit = GetBoolean("use_mouseposit"); + + this->m_ProjectileType = GetInt("projectile_type"); } diff --git a/dGame/dBehaviors/ProjectileAttackBehavior.h b/dGame/dBehaviors/ProjectileAttackBehavior.h index 17353a2a..b307e66c 100644 --- a/dGame/dBehaviors/ProjectileAttackBehavior.h +++ b/dGame/dBehaviors/ProjectileAttackBehavior.h @@ -23,6 +23,8 @@ public: bool m_useMouseposit; + int32_t m_ProjectileType; + /* * Inherited */ diff --git a/dGame/dBehaviors/PropertyTeleportBehavior.cpp b/dGame/dBehaviors/PropertyTeleportBehavior.cpp new file mode 100644 index 00000000..309fc929 --- /dev/null +++ b/dGame/dBehaviors/PropertyTeleportBehavior.cpp @@ -0,0 +1,61 @@ +#include "PropertyTeleportBehavior.h" + +#include "BehaviorBranchContext.h" +#include "BehaviorContext.h" +#include "Character.h" +#include "CharacterComponent.h" +#include "ChatPackets.h" +#include "WorldPackets.h" +#include "EntityManager.h" +#include "Game.h" +#include "ZoneInstanceManager.h" +#include "dZoneManager.h" + +void PropertyTeleportBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + auto* caster = Game::entityManager->GetEntity(context->caster); + if (!caster) return; + + auto* character = caster->GetCharacter(); + if (!character) return; + + LWOOBJID objId = caster->GetObjectID(); + + LWOMAPID targetMapId = m_MapId; + LWOCLONEID targetCloneId = character->GetPropertyCloneID(); + + if (Game::zoneManager->GetZoneID().GetCloneID() == character->GetPropertyCloneID()) { + targetMapId = character->GetLastNonInstanceZoneID(); + targetCloneId = 0; + } else { + character->SetLastNonInstanceZoneID(Game::zoneManager->GetZoneID().GetMapID()); + } + + ZoneInstanceManager::Instance()->RequestZoneTransfer(Game::server, targetMapId, targetCloneId, false, [objId](bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, std::string serverIP, uint16_t serverPort) { + + auto* entity = Game::entityManager->GetEntity(objId); + if (!entity) return; + + const auto sysAddr = entity->GetSystemAddress(); + + if (zoneClone != 0) ChatPackets::SendSystemMessage(sysAddr, u"Transfering to your property!"); + else ChatPackets::SendSystemMessage(sysAddr, u"Transfering back to previous world!"); + + Game::logger->Log("PropertyTeleportBehavior", "Transferring %s to Zone %i (Instance %i | Clone %i | Mythran Shift: %s) with IP %s and Port %i", sysAddr.ToString(), zoneID, zoneInstance, zoneClone, mythranShift == true ? "true" : "false", serverIP.c_str(), serverPort); + if (entity->GetCharacter()) { + entity->GetCharacter()->SetZoneID(zoneID); + entity->GetCharacter()->SetZoneInstance(zoneInstance); + entity->GetCharacter()->SetZoneClone(zoneClone); + entity->GetComponent()->SetLastRocketConfig(u""); + } + + entity->GetCharacter()->SaveXMLToDatabase(); + + WorldPackets::SendTransferToWorld(sysAddr, serverIP, serverPort, mythranShift); + return; + }); +} + +void PropertyTeleportBehavior::Load() { + this->m_CancelIfInteracting = GetBoolean("cancel_if_interacting"); // TODO unused + this->m_MapId = LWOMAPID(GetInt("mapID")); +} diff --git a/dGame/dBehaviors/PropertyTeleportBehavior.h b/dGame/dBehaviors/PropertyTeleportBehavior.h new file mode 100644 index 00000000..74eed03b --- /dev/null +++ b/dGame/dBehaviors/PropertyTeleportBehavior.h @@ -0,0 +1,21 @@ +#pragma once +#include "Behavior.h" + +class PropertyTeleportBehavior final : public Behavior +{ +public: + /* + * Inherited + */ + + explicit PropertyTeleportBehavior(const uint32_t behavior_id) : Behavior(behavior_id) { + } + + void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + + void Load() override; + +private: + LWOMAPID m_MapId; + bool m_CancelIfInteracting; +}; diff --git a/dGame/dBehaviors/PullToPointBehavior.cpp b/dGame/dBehaviors/PullToPointBehavior.cpp index 7427ccc4..e18443f7 100644 --- a/dGame/dBehaviors/PullToPointBehavior.cpp +++ b/dGame/dBehaviors/PullToPointBehavior.cpp @@ -6,9 +6,9 @@ #include "MovementAIComponent.h" void PullToPointBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(context->originator); + auto* entity = Game::entityManager->GetEntity(context->originator); - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (entity == nullptr || target == nullptr) { return; diff --git a/dGame/dBehaviors/RemoveBuffBehavior.cpp b/dGame/dBehaviors/RemoveBuffBehavior.cpp new file mode 100644 index 00000000..c6d2c9c9 --- /dev/null +++ b/dGame/dBehaviors/RemoveBuffBehavior.cpp @@ -0,0 +1,21 @@ +#include "RemoveBuffBehavior.h" + +#include "BehaviorBranchContext.h" +#include "BehaviorContext.h" +#include "EntityManager.h" +#include "BuffComponent.h" + +void RemoveBuffBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + auto* entity = Game::entityManager->GetEntity(context->caster); + if (!entity) return; + + auto* buffComponent = entity->GetComponent(); + if (!buffComponent) return; + + buffComponent->RemoveBuff(m_BuffId, false, m_RemoveImmunity); +} + +void RemoveBuffBehavior::Load() { + this->m_RemoveImmunity = GetBoolean("remove_immunity"); + this->m_BuffId = GetInt("buff_id"); +} diff --git a/dGame/dBehaviors/RemoveBuffBehavior.h b/dGame/dBehaviors/RemoveBuffBehavior.h new file mode 100644 index 00000000..f2d8547b --- /dev/null +++ b/dGame/dBehaviors/RemoveBuffBehavior.h @@ -0,0 +1,22 @@ +#pragma once +#include "Behavior.h" + +class RemoveBuffBehavior final : public Behavior +{ +public: + + /* + * Inherited + */ + + explicit RemoveBuffBehavior(const uint32_t behaviorId) : Behavior(behaviorId) { + } + + void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + + void Load() override; + +private: + bool m_RemoveImmunity; + uint32_t m_BuffId; +}; diff --git a/dGame/dBehaviors/RepairBehavior.cpp b/dGame/dBehaviors/RepairBehavior.cpp index a48141d3..d2967921 100644 --- a/dGame/dBehaviors/RepairBehavior.cpp +++ b/dGame/dBehaviors/RepairBehavior.cpp @@ -5,9 +5,10 @@ #include "EntityManager.h" #include "dLogger.h" #include "Game.h" +#include "eReplicaComponentType.h" void RepairBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_stream, const BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); if (entity == nullptr) { Game::logger->Log("RepairBehavior", "Failed to find entity for (%llu)!", branch.target); @@ -15,7 +16,7 @@ void RepairBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_str return; } - auto* destroyable = static_cast(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + auto* destroyable = static_cast(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (destroyable == nullptr) { Game::logger->Log("RepairBehavior", "Failed to find destroyable component for %(llu)!", branch.target); diff --git a/dGame/dBehaviors/SkillEventBehavior.cpp b/dGame/dBehaviors/SkillEventBehavior.cpp index 837d70c9..4205c28b 100644 --- a/dGame/dBehaviors/SkillEventBehavior.cpp +++ b/dGame/dBehaviors/SkillEventBehavior.cpp @@ -5,8 +5,8 @@ #include "CppScripts.h" void SkillEventBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); - auto* caster = EntityManager::Instance()->GetEntity(context->originator); + auto* target = Game::entityManager->GetEntity(branch.target); + auto* caster = Game::entityManager->GetEntity(context->originator); if (caster != nullptr && target != nullptr && this->m_effectHandle != nullptr && !this->m_effectHandle->empty()) { for (CppScripts::Script* script : CppScripts::GetEntityScripts(target)) { @@ -17,8 +17,8 @@ void SkillEventBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit void SkillEventBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); - auto* caster = EntityManager::Instance()->GetEntity(context->originator); + auto* target = Game::entityManager->GetEntity(branch.target); + auto* caster = Game::entityManager->GetEntity(context->originator); if (caster != nullptr && target != nullptr && this->m_effectHandle != nullptr && !this->m_effectHandle->empty()) { for (CppScripts::Script* script : CppScripts::GetEntityScripts(target)) { diff --git a/dGame/dBehaviors/SpawnBehavior.cpp b/dGame/dBehaviors/SpawnBehavior.cpp index ac7bb797..2803251a 100644 --- a/dGame/dBehaviors/SpawnBehavior.cpp +++ b/dGame/dBehaviors/SpawnBehavior.cpp @@ -7,9 +7,12 @@ #include "dLogger.h" #include "DestroyableComponent.h" #include "RebuildComponent.h" +#include "Entity.h" +#include "EntityInfo.h" +#include "eReplicaComponentType.h" void SpawnBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* origin = EntityManager::Instance()->GetEntity(context->originator); + auto* origin = Game::entityManager->GetEntity(context->originator); if (origin == nullptr) { Game::logger->Log("SpawnBehavior", "Failed to find self entity (%llu)!", context->originator); @@ -18,7 +21,7 @@ void SpawnBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStrea } if (branch.isProjectile) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target != nullptr) { origin = target; @@ -35,10 +38,10 @@ void SpawnBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStrea info.spawnerNodeID = 0; info.pos = info.pos + (info.rot.GetForwardVector() * m_Distance); - auto* entity = EntityManager::Instance()->CreateEntity( + auto* entity = Game::entityManager->CreateEntity( info, nullptr, - EntityManager::Instance()->GetEntity(context->originator) + Game::entityManager->GetEntity(context->originator) ); if (entity == nullptr) { @@ -56,7 +59,7 @@ void SpawnBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStrea rebuildComponent->SetRepositionPlayer(false); } - EntityManager::Instance()->ConstructEntity(entity); + Game::entityManager->ConstructEntity(entity); if (branch.duration > 0) { context->RegisterTimerBehavior(this, branch, entity->GetObjectID()); @@ -76,7 +79,7 @@ void SpawnBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitSt } void SpawnBehavior::Timer(BehaviorContext* context, const BehaviorBranchContext branch, const LWOOBJID second) { - auto* entity = EntityManager::Instance()->GetEntity(second); + auto* entity = Game::entityManager->GetEntity(second); if (entity == nullptr) { Game::logger->Log("SpawnBehavior", "Failed to find spawned entity (%llu)!", second); @@ -84,7 +87,7 @@ void SpawnBehavior::Timer(BehaviorContext* context, const BehaviorBranchContext return; } - auto* destroyable = static_cast(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + auto* destroyable = static_cast(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (destroyable == nullptr) { entity->Smash(context->originator); diff --git a/dGame/dBehaviors/SpeedBehavior.cpp b/dGame/dBehaviors/SpeedBehavior.cpp index c7855557..5dbad8ec 100644 --- a/dGame/dBehaviors/SpeedBehavior.cpp +++ b/dGame/dBehaviors/SpeedBehavior.cpp @@ -7,85 +7,48 @@ void SpeedBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - if (m_AffectsCaster) { - branch.target = context->caster; - } + if (m_AffectsCaster) branch.target = context->caster; - auto* target = EntityManager::Instance()->GetEntity(branch.target); - - if (target == nullptr) { - return; - } + auto* target = Game::entityManager->GetEntity(branch.target); + if (!target) return; auto* controllablePhysicsComponent = target->GetComponent(); + if (!controllablePhysicsComponent) return; - if (controllablePhysicsComponent == nullptr) { - return; - } - - const auto current = controllablePhysicsComponent->GetSpeedMultiplier(); - - controllablePhysicsComponent->SetSpeedMultiplier(current + ((m_RunSpeed - 500.0f) / 500.0f)); - - EntityManager::Instance()->SerializeEntity(target); + controllablePhysicsComponent->AddSpeedboost(m_RunSpeed); + Game::entityManager->SerializeEntity(target); if (branch.duration > 0.0f) { context->RegisterTimerBehavior(this, branch); } else if (branch.start > 0) { - controllablePhysicsComponent->SetIgnoreMultipliers(true); - context->RegisterEndBehavior(this, branch); } } +void SpeedBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + Handle(context, bitStream, branch); +} + +void SpeedBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { + End(context, branch, LWOOBJID_EMPTY); +} + void SpeedBehavior::Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); - - if (target == nullptr) { - return; - } - - auto* controllablePhysicsComponent = target->GetComponent(); - - if (controllablePhysicsComponent == nullptr) { - return; - } - - const auto current = controllablePhysicsComponent->GetSpeedMultiplier(); - - controllablePhysicsComponent->SetSpeedMultiplier(current - ((m_RunSpeed - 500.0f) / 500.0f)); - - EntityManager::Instance()->SerializeEntity(target); + End(context, branch, second); } void SpeedBehavior::End(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); - - if (target == nullptr) { - return; - } + auto* target = Game::entityManager->GetEntity(branch.target); + if (!target) return; auto* controllablePhysicsComponent = target->GetComponent(); + if (!controllablePhysicsComponent) return; - if (controllablePhysicsComponent == nullptr) { - return; - } - - const auto current = controllablePhysicsComponent->GetSpeedMultiplier(); - - controllablePhysicsComponent->SetIgnoreMultipliers(false); - - controllablePhysicsComponent->SetSpeedMultiplier(current - ((m_RunSpeed - 500.0f) / 500.0f)); - - EntityManager::Instance()->SerializeEntity(target); + controllablePhysicsComponent->RemoveSpeedboost(m_RunSpeed); + Game::entityManager->SerializeEntity(target); } void SpeedBehavior::Load() { m_RunSpeed = GetFloat("run_speed"); - - if (m_RunSpeed < 500.0f) { - m_RunSpeed = 500.0f; - } - m_AffectsCaster = GetBoolean("affects_caster"); } diff --git a/dGame/dBehaviors/SpeedBehavior.h b/dGame/dBehaviors/SpeedBehavior.h index 04c0090f..88b85820 100644 --- a/dGame/dBehaviors/SpeedBehavior.h +++ b/dGame/dBehaviors/SpeedBehavior.h @@ -13,6 +13,10 @@ public: void Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + void Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) override; + + void UnCast(BehaviorContext* context, BehaviorBranchContext branch) override; + void Timer(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; void End(BehaviorContext* context, BehaviorBranchContext branch, LWOOBJID second) override; diff --git a/dGame/dBehaviors/StunBehavior.cpp b/dGame/dBehaviors/StunBehavior.cpp index 59a81440..8e160338 100644 --- a/dGame/dBehaviors/StunBehavior.cpp +++ b/dGame/dBehaviors/StunBehavior.cpp @@ -7,6 +7,7 @@ #include "Game.h" #include "dLogger.h" #include "DestroyableComponent.h" +#include "eReplicaComponentType.h" void StunBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { @@ -14,10 +15,13 @@ void StunBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream return; } - bool blocked; - bitStream->Read(blocked); + bool blocked{}; + if (!bitStream->Read(blocked)) { + Game::logger->Log("StunBehavior", "Unable to read blocked from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) { Game::logger->Log("StunBehavior", "Failed to find target (%llu)!", branch.target); @@ -29,7 +33,7 @@ void StunBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream * If our target is an enemy we can go ahead and stun it. */ - auto* combatAiComponent = static_cast(target->GetComponent(COMPONENT_TYPE_BASE_COMBAT_AI)); + auto* combatAiComponent = static_cast(target->GetComponent(eReplicaComponentType::BASE_COMBAT_AI)); if (combatAiComponent == nullptr) { return; @@ -40,7 +44,7 @@ void StunBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream void StunBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, const BehaviorBranchContext branch) { if (this->m_stunCaster || branch.target == context->originator) { - auto* self = EntityManager::Instance()->GetEntity(context->originator); + auto* self = Game::entityManager->GetEntity(context->originator); if (self == nullptr) { Game::logger->Log("StunBehavior", "Invalid self entity (%llu)!", context->originator); @@ -52,7 +56,7 @@ void StunBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStr * See if we can stun ourselves */ - auto* combatAiComponent = static_cast(self->GetComponent(COMPONENT_TYPE_BASE_COMBAT_AI)); + auto* combatAiComponent = static_cast(self->GetComponent(eReplicaComponentType::BASE_COMBAT_AI)); if (combatAiComponent == nullptr) { return; @@ -65,7 +69,7 @@ void StunBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStr bool blocked = false; - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target != nullptr) { auto* destroyableComponent = target->GetComponent(); @@ -87,7 +91,7 @@ void StunBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStr * If our target is an enemy we can go ahead and stun it. */ - auto* combatAiComponent = static_cast(target->GetComponent(COMPONENT_TYPE_BASE_COMBAT_AI)); + auto* combatAiComponent = static_cast(target->GetComponent(eReplicaComponentType::BASE_COMBAT_AI)); if (combatAiComponent == nullptr) { return; diff --git a/dGame/dBehaviors/SwitchBehavior.cpp b/dGame/dBehaviors/SwitchBehavior.cpp index 78271b99..d65191d2 100644 --- a/dGame/dBehaviors/SwitchBehavior.cpp +++ b/dGame/dBehaviors/SwitchBehavior.cpp @@ -10,10 +10,13 @@ void SwitchBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStre auto state = true; if (this->m_imagination > 0 || !this->m_isEnemyFaction) { - bitStream->Read(state); + if (!bitStream->Read(state)) { + Game::logger->Log("SwitchBehavior", "Unable to read state from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; } - auto* entity = EntityManager::Instance()->GetEntity(context->originator); + auto* entity = Game::entityManager->GetEntity(context->originator); if (entity == nullptr) { return; @@ -25,9 +28,9 @@ void SwitchBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStre return; } - Game::logger->Log("SwitchBehavior", "[%i] State: (%d), imagination: (%i) / (%f)", entity->GetLOT(), state, destroyableComponent->GetImagination(), destroyableComponent->GetMaxImagination()); + Game::logger->LogDebug("SwitchBehavior", "[%i] State: (%d), imagination: (%i) / (%f)", entity->GetLOT(), state, destroyableComponent->GetImagination(), destroyableComponent->GetMaxImagination()); - if (state || (entity->GetLOT() == 8092 && destroyableComponent->GetImagination() >= m_imagination)) { + if (state) { this->m_actionTrue->Handle(context, bitStream, branch); } else { this->m_actionFalse->Handle(context, bitStream, branch); @@ -38,7 +41,7 @@ void SwitchBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitS auto state = true; if (this->m_imagination > 0 || !this->m_isEnemyFaction) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); state = entity != nullptr; diff --git a/dGame/dBehaviors/SwitchMultipleBehavior.cpp b/dGame/dBehaviors/SwitchMultipleBehavior.cpp index 946e5e0f..23411429 100644 --- a/dGame/dBehaviors/SwitchMultipleBehavior.cpp +++ b/dGame/dBehaviors/SwitchMultipleBehavior.cpp @@ -9,31 +9,30 @@ #include "EntityManager.h" -void SwitchMultipleBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bit_stream, BehaviorBranchContext branch) { - float value; +void SwitchMultipleBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { + float value{}; - bit_stream->Read(value); + if (!bitStream->Read(value)) { + Game::logger->Log("SwitchMultipleBehavior", "Unable to read value from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; uint32_t trigger = 0; for (unsigned int i = 0; i < this->m_behaviors.size(); i++) { const double data = this->m_behaviors.at(i).first; + trigger = i; - if (value <= data) { - - trigger = i; - - break; - } + if (value <= data) break; } auto* behavior = this->m_behaviors.at(trigger).second; - behavior->Handle(context, bit_stream, branch); + behavior->Handle(context, bitStream, branch); } -void SwitchMultipleBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bit_stream, BehaviorBranchContext branch) { +void SwitchMultipleBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { // TODO } diff --git a/dGame/dBehaviors/TacArcBehavior.cpp b/dGame/dBehaviors/TacArcBehavior.cpp index 8789f9b6..b9d871f4 100644 --- a/dGame/dBehaviors/TacArcBehavior.cpp +++ b/dGame/dBehaviors/TacArcBehavior.cpp @@ -20,12 +20,18 @@ void TacArcBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStre bool hit = false; - bitStream->Read(hit); + if (!bitStream->Read(hit)) { + Game::logger->Log("TacArcBehavior", "Unable to read hit from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; if (this->m_checkEnv) { bool blocked = false; - bitStream->Read(blocked); + if (!bitStream->Read(blocked)) { + Game::logger->Log("TacArcBehavior", "Unable to read blocked from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; if (blocked) { this->m_blockedAction->Handle(context, bitStream, branch); @@ -37,7 +43,10 @@ void TacArcBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStre if (hit) { uint32_t count = 0; - bitStream->Read(count); + if (!bitStream->Read(count)) { + Game::logger->Log("TacArcBehavior", "Unable to read count from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; if (count > m_maxTargets && m_maxTargets > 0) { count = m_maxTargets; @@ -46,9 +55,12 @@ void TacArcBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStre std::vector targets; for (auto i = 0u; i < count; ++i) { - LWOOBJID id; + LWOOBJID id{}; - bitStream->Read(id); + if (!bitStream->Read(id)) { + Game::logger->Log("TacArcBehavior", "Unable to read id from bitStream, aborting Handle! %i", bitStream->GetNumberOfUnreadBits()); + return; + }; targets.push_back(id); } @@ -64,7 +76,7 @@ void TacArcBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStre } void TacArcBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* self = EntityManager::Instance()->GetEntity(context->originator); + auto* self = Game::entityManager->GetEntity(context->originator); if (self == nullptr) { Game::logger->Log("TacArcBehavior", "Invalid self for (%llu)!", context->originator); return; @@ -73,7 +85,7 @@ void TacArcBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitS const auto* destroyableComponent = self->GetComponent(); if ((this->m_usePickedTarget || context->clientInitalized) && branch.target > 0) { - const auto* target = EntityManager::Instance()->GetEntity(branch.target); + const auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) { return; @@ -108,7 +120,7 @@ void TacArcBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitS // Find all valid targets, based on whether we target enemies or friends for (const auto& contextTarget : context->GetValidTargets()) { if (destroyableComponent != nullptr) { - const auto* targetEntity = EntityManager::Instance()->GetEntity(contextTarget); + const auto* targetEntity = Game::entityManager->GetEntity(contextTarget); if (m_targetEnemy && destroyableComponent->IsEnemy(targetEntity) || m_targetFriend && destroyableComponent->IsFriend(targetEntity)) { @@ -124,7 +136,7 @@ void TacArcBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitS break; } - auto* entity = EntityManager::Instance()->GetEntity(validTarget); + auto* entity = Game::entityManager->GetEntity(validTarget); if (entity == nullptr) { Game::logger->Log("TacArcBehavior", "Invalid target (%llu) for (%llu)!", validTarget, context->originator); diff --git a/dGame/dBehaviors/TauntBehavior.cpp b/dGame/dBehaviors/TauntBehavior.cpp index 7ed3b897..b86cd7b5 100644 --- a/dGame/dBehaviors/TauntBehavior.cpp +++ b/dGame/dBehaviors/TauntBehavior.cpp @@ -7,7 +7,7 @@ void TauntBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) { Game::logger->Log("TauntBehavior", "Failed to find target (%llu)!", branch.target); @@ -23,7 +23,7 @@ void TauntBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStrea } void TauntBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* target = EntityManager::Instance()->GetEntity(branch.target); + auto* target = Game::entityManager->GetEntity(branch.target); if (target == nullptr) { Game::logger->Log("TauntBehavior", "Failed to find target (%llu)!", branch.target); diff --git a/dGame/dBehaviors/VentureVisionBehavior.cpp b/dGame/dBehaviors/VentureVisionBehavior.cpp index 93feb8e9..397bdebf 100644 --- a/dGame/dBehaviors/VentureVisionBehavior.cpp +++ b/dGame/dBehaviors/VentureVisionBehavior.cpp @@ -5,7 +5,7 @@ void VentureVisionBehavior::Handle(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - const auto targetEntity = EntityManager::Instance()->GetEntity(branch.target); + const auto targetEntity = Game::entityManager->GetEntity(branch.target); if (targetEntity) { auto characterComponent = targetEntity->GetComponent(); @@ -21,7 +21,7 @@ void VentureVisionBehavior::Handle(BehaviorContext* context, RakNet::BitStream* } void VentureVisionBehavior::UnCast(BehaviorContext* context, BehaviorBranchContext branch) { - const auto targetEntity = EntityManager::Instance()->GetEntity(branch.target); + const auto targetEntity = Game::entityManager->GetEntity(branch.target); if (targetEntity) { auto characterComponent = targetEntity->GetComponent(); diff --git a/dGame/dBehaviors/VerifyBehavior.cpp b/dGame/dBehaviors/VerifyBehavior.cpp index 608e965b..6824a25f 100644 --- a/dGame/dBehaviors/VerifyBehavior.cpp +++ b/dGame/dBehaviors/VerifyBehavior.cpp @@ -8,14 +8,14 @@ void VerifyBehavior::Calculate(BehaviorContext* context, RakNet::BitStream* bitStream, BehaviorBranchContext branch) { - auto* entity = EntityManager::Instance()->GetEntity(branch.target); + auto* entity = Game::entityManager->GetEntity(branch.target); auto success = true; if (entity == nullptr) { success = false; } else if (this->m_rangeCheck) { - auto* self = EntityManager::Instance()->GetEntity(context->originator); + auto* self = Game::entityManager->GetEntity(context->originator); if (self == nullptr) { Game::logger->Log("VerifyBehavior", "Invalid self for (%llu)", context->originator); diff --git a/dGame/dComponents/AchievementCacheKey.h b/dGame/dComponents/AchievementCacheKey.h new file mode 100644 index 00000000..398e0231 --- /dev/null +++ b/dGame/dComponents/AchievementCacheKey.h @@ -0,0 +1,45 @@ +#include "eMissionTaskType.h" + +#ifndef __ACHIEVEMENTCACHEKEY__H__ +#define __ACHIEVEMENTCACHEKEY__H__ + +class AchievementCacheKey { +public: + AchievementCacheKey() { + targets = ""; + value = 0; + type = eMissionTaskType::UNKNOWN; + }; + + bool operator==(const AchievementCacheKey& point) const { + return this->targets == point.targets && this->value == point.value && this->type == point.type; + }; + void SetTargets(const std::string value) { this->targets = value; }; + void SetValue(uint32_t value) { this->value = value; }; + void SetType(eMissionTaskType value) { this->type = value; }; + + std::string GetTargets() const { return this->targets; }; + uint32_t GetValue() const { return this->value; }; + eMissionTaskType GetType() const { return this->type; }; +private: + std::string targets; + uint32_t value; + eMissionTaskType type; + +}; + +// Specialization of hash for the above class +namespace std { + template<> + struct hash { + size_t operator()(const AchievementCacheKey& key) const { + size_t hash = 0; + GeneralUtils::hash_combine(hash, key.GetType()); + GeneralUtils::hash_combine(hash, key.GetValue()); + GeneralUtils::hash_combine(hash, key.GetTargets()); + return hash; + }; + }; +}; + +#endif //!__ACHIEVEMENTCACHEKEY__H__ diff --git a/dGame/dComponents/BaseCombatAIComponent.cpp b/dGame/dComponents/BaseCombatAIComponent.cpp index c035a807..5df43854 100644 --- a/dGame/dComponents/BaseCombatAIComponent.cpp +++ b/dGame/dComponents/BaseCombatAIComponent.cpp @@ -22,10 +22,13 @@ #include "SkillComponent.h" #include "RebuildComponent.h" #include "DestroyableComponent.h" +#include "Metrics.hpp" +#include "CDComponentsRegistryTable.h" +#include "CDPhysicsComponentTable.h" -BaseCombatAIComponent::BaseCombatAIComponent(Entity* parent, const uint32_t id) : Component(parent) { +BaseCombatAIComponent::BaseCombatAIComponent(Entity* parent, const uint32_t id): Component(parent) { m_Target = LWOOBJID_EMPTY; - m_State = AiState::spawn; + SetAiState(AiState::spawn); m_Timer = 1.0f; m_StartPosition = parent->GetPosition(); m_MovementAI = nullptr; @@ -104,10 +107,10 @@ BaseCombatAIComponent::BaseCombatAIComponent(Entity* parent, const uint32_t id) int32_t collisionGroup = (COLLISION_GROUP_DYNAMIC | COLLISION_GROUP_ENEMY); - CDComponentsRegistryTable* componentRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - auto componentID = componentRegistryTable->GetByIDAndType(parent->GetLOT(), COMPONENT_TYPE_CONTROLLABLE_PHYSICS); + CDComponentsRegistryTable* componentRegistryTable = CDClientManager::Instance().GetTable(); + auto componentID = componentRegistryTable->GetByIDAndType(parent->GetLOT(), eReplicaComponentType::CONTROLLABLE_PHYSICS); - CDPhysicsComponentTable* physicsComponentTable = CDClientManager::Instance()->GetTable("PhysicsComponent"); + CDPhysicsComponentTable* physicsComponentTable = CDClientManager::Instance().GetTable(); if (physicsComponentTable != nullptr) { auto* info = physicsComponentTable->GetByID(componentID); @@ -170,7 +173,7 @@ void BaseCombatAIComponent::Update(const float deltaTime) { } if (m_SoftTimer <= 0.0f) { - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); m_SoftTimer = 5.0f; } else { @@ -179,7 +182,7 @@ void BaseCombatAIComponent::Update(const float deltaTime) { if (m_Disabled || m_Parent->GetIsDead()) return; - + bool stunnedThisFrame = m_Stunned; CalculateCombat(deltaTime); // Putting this here for now if (m_StartPosition == NiPoint3::ZERO) { @@ -192,7 +195,7 @@ void BaseCombatAIComponent::Update(const float deltaTime) { return; } - if (m_Stunned) { + if (stunnedThisFrame) { m_MovementAI->Stop(); return; @@ -206,7 +209,7 @@ void BaseCombatAIComponent::Update(const float deltaTime) { switch (m_State) { case AiState::spawn: Stun(2.0f); - m_State = AiState::idle; + SetAiState(AiState::idle); break; case AiState::idle: @@ -228,12 +231,24 @@ void BaseCombatAIComponent::Update(const float deltaTime) { void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { + bool hasSkillToCast = false; + for (auto& entry : m_SkillEntries) { + if (entry.cooldown > 0.0f) { + entry.cooldown -= deltaTime; + } else { + hasSkillToCast = true; + } + } + + bool hadRemainingDowntime = m_SkillTime > 0.0f; + if (m_SkillTime > 0.0f) m_SkillTime -= deltaTime; + auto* rebuild = m_Parent->GetComponent(); if (rebuild != nullptr) { const auto state = rebuild->GetState(); - if (state != REBUILD_COMPLETED) { + if (state != eRebuildState::COMPLETED) { return; } } @@ -248,19 +263,17 @@ void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { if (m_Disabled) return; - if (m_StunTime > 0.0f) { + if (m_Stunned) { m_StunTime -= deltaTime; if (m_StunTime > 0.0f) { return; } - + m_StunTime = 0.0f; m_Stunned = false; } - if (m_Stunned) { - return; - } + if (m_Stunned || hadRemainingDowntime) return; auto newTarget = FindTarget(); @@ -292,7 +305,7 @@ void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { } if (serilizationRequired) { - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } GameMessages::SendPlayFXEffect(m_Parent->GetObjectID(), 6270, u"tether", "tether"); @@ -305,7 +318,7 @@ void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { // Speed towards start position if (m_MovementAI != nullptr) { m_MovementAI->SetHaltDistance(0); - m_MovementAI->SetSpeed(m_PursuitSpeed); + m_MovementAI->SetMaxSpeed(m_PursuitSpeed); m_MovementAI->SetDestination(m_StartPosition); } @@ -320,41 +333,19 @@ void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { m_Timer = 0; } - m_State = AiState::aggro; + SetAiState(AiState::aggro); } else { - m_State = AiState::idle; + SetAiState(AiState::idle); } - for (auto i = 0; i < m_SkillEntries.size(); ++i) { - auto entry = m_SkillEntries.at(i); - - if (entry.cooldown > 0) { - entry.cooldown -= deltaTime; - - m_SkillEntries[i] = entry; - } - } - - if (m_SkillTime > 0) { - m_SkillTime -= deltaTime; - - return; - } - - if (m_Downtime > 0) { - m_Downtime -= deltaTime; - - return; - } + if (!hasSkillToCast) return; if (m_Target == LWOOBJID_EMPTY) { - m_State = AiState::idle; + SetAiState(AiState::idle); return; } - m_Downtime = 0.5f; - auto* target = GetTargetEntity(); if (target != nullptr) { @@ -375,7 +366,7 @@ void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { m_MovementAI->Stop(); } - m_State = AiState::aggro; + SetAiState(AiState::aggro); m_Timer = 0; @@ -391,8 +382,6 @@ void BaseCombatAIComponent::CalculateCombat(const float deltaTime) { } LWOOBJID BaseCombatAIComponent::FindTarget() { - //const auto reference = m_MovementAI == nullptr ? m_StartPosition : m_MovementAI->ApproximateLocation(); - NiPoint3 reference = m_StartPosition; if (m_MovementAI) reference = m_MovementAI->ApproximateLocation(); @@ -421,7 +410,7 @@ LWOOBJID BaseCombatAIComponent::FindTarget() { float biggestThreat = 0; for (const auto& entry : possibleTargets) { - auto* entity = EntityManager::Instance()->GetEntity(entry); + auto* entity = Game::entityManager->GetEntity(entry); if (entity == nullptr) { continue; @@ -467,7 +456,7 @@ LWOOBJID BaseCombatAIComponent::FindTarget() { std::vector deadThreats{}; for (const auto& threatTarget : m_ThreatEntries) { - auto* entity = EntityManager::Instance()->GetEntity(threatTarget.first); + auto* entity = Game::entityManager->GetEntity(threatTarget.first); if (entity == nullptr) { deadThreats.push_back(threatTarget.first); @@ -506,7 +495,7 @@ std::vector BaseCombatAIComponent::GetTargetWithinAggroRange() const { std::vector targets; for (auto id : m_Parent->GetTargetsInPhantom()) { - auto* other = EntityManager::Instance()->GetEntity(id); + auto* other = Game::entityManager->GetEntity(id); const auto distance = Vector3::DistanceSquared(m_Parent->GetPosition(), other->GetPosition()); @@ -532,14 +521,23 @@ bool BaseCombatAIComponent::IsMech() { void BaseCombatAIComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { - outBitStream->Write1(); - outBitStream->Write(uint32_t(m_State)); - outBitStream->Write(m_Target); + outBitStream->Write(m_DirtyStateOrTarget || bIsInitialUpdate); + if (m_DirtyStateOrTarget || bIsInitialUpdate) { + outBitStream->Write(uint32_t(m_State)); + outBitStream->Write(m_Target); + m_DirtyStateOrTarget = false; + } } +void BaseCombatAIComponent::SetAiState(AiState newState) { + if (newState == this->m_State) return; + this->m_State = newState; + m_DirtyStateOrTarget = true; + Game::entityManager->SerializeEntity(m_Parent); +} bool BaseCombatAIComponent::IsEnemy(LWOOBJID target) const { - auto* entity = EntityManager::Instance()->GetEntity(target); + auto* entity = Game::entityManager->GetEntity(target); if (entity == nullptr) { Game::logger->Log("BaseCombatAIComponent", "Invalid entity for checking validity (%llu)!", target); @@ -566,7 +564,7 @@ bool BaseCombatAIComponent::IsEnemy(LWOOBJID target) const { if (quickbuild != nullptr) { const auto state = quickbuild->GetState(); - if (state != REBUILD_COMPLETED) { + if (state != eRebuildState::COMPLETED) { return false; } } @@ -585,11 +583,14 @@ bool BaseCombatAIComponent::IsEnemy(LWOOBJID target) const { } void BaseCombatAIComponent::SetTarget(const LWOOBJID target) { + if (this->m_Target == target) return; m_Target = target; + m_DirtyStateOrTarget = true; + Game::entityManager->SerializeEntity(m_Parent); } Entity* BaseCombatAIComponent::GetTargetEntity() const { - return EntityManager::Instance()->GetEntity(m_Target); + return Game::entityManager->GetEntity(m_Target); } void BaseCombatAIComponent::Taunt(LWOOBJID offender, float threat) { @@ -657,17 +658,17 @@ void BaseCombatAIComponent::Wander() { destination.y = dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(destination); } - if (Vector3::DistanceSquared(destination, m_MovementAI->GetCurrentPosition()) < 2 * 2) { + if (Vector3::DistanceSquared(destination, m_MovementAI->GetParent()->GetPosition()) < 2 * 2) { m_MovementAI->Stop(); return; } - m_MovementAI->SetSpeed(m_TetherSpeed); + m_MovementAI->SetMaxSpeed(m_TetherSpeed); m_MovementAI->SetDestination(destination); - m_Timer += (m_MovementAI->GetCurrentPosition().x - destination.x) / m_TetherSpeed; + m_Timer += (m_MovementAI->GetParent()->GetPosition().x - destination.x) / m_TetherSpeed; } void BaseCombatAIComponent::OnAggro() { @@ -682,25 +683,25 @@ void BaseCombatAIComponent::OnAggro() { m_MovementAI->SetHaltDistance(m_AttackRadius); NiPoint3 targetPos = target->GetPosition(); - NiPoint3 currentPos = m_MovementAI->GetCurrentPosition(); + NiPoint3 currentPos = m_MovementAI->GetParent()->GetPosition(); // If the player's position is within range, attack if (Vector3::DistanceSquared(currentPos, targetPos) <= m_AttackRadius * m_AttackRadius) { m_MovementAI->Stop(); } else if (Vector3::DistanceSquared(m_StartPosition, targetPos) > m_HardTetherRadius * m_HardTetherRadius) //Return to spawn if we're too far { - m_MovementAI->SetSpeed(m_PursuitSpeed); + m_MovementAI->SetMaxSpeed(m_PursuitSpeed); m_MovementAI->SetDestination(m_StartPosition); } else //Chase the player's new position { if (IsMech() && Vector3::DistanceSquared(targetPos, currentPos) > m_AttackRadius * m_AttackRadius * 3 * 3) return; - m_MovementAI->SetSpeed(m_PursuitSpeed); + m_MovementAI->SetMaxSpeed(m_PursuitSpeed); m_MovementAI->SetDestination(targetPos); - m_State = AiState::tether; + SetAiState(AiState::tether); } m_Timer += 0.5f; @@ -722,15 +723,15 @@ void BaseCombatAIComponent::OnTether() { m_MovementAI->Stop(); } else if (Vector3::DistanceSquared(m_StartPosition, targetPos) > m_HardTetherRadius * m_HardTetherRadius) //Return to spawn if we're too far { - m_MovementAI->SetSpeed(m_PursuitSpeed); + m_MovementAI->SetMaxSpeed(m_PursuitSpeed); m_MovementAI->SetDestination(m_StartPosition); - m_State = AiState::aggro; + SetAiState(AiState::aggro); } else { if (IsMech() && Vector3::DistanceSquared(targetPos, currentPos) > m_AttackRadius * m_AttackRadius * 3 * 3) return; - m_MovementAI->SetSpeed(m_PursuitSpeed); + m_MovementAI->SetMaxSpeed(m_PursuitSpeed); m_MovementAI->SetDestination(targetPos); } diff --git a/dGame/dComponents/BaseCombatAIComponent.h b/dGame/dComponents/BaseCombatAIComponent.h index 70a88a42..8bf6140a 100644 --- a/dGame/dComponents/BaseCombatAIComponent.h +++ b/dGame/dComponents/BaseCombatAIComponent.h @@ -8,6 +8,7 @@ #include "dpWorld.h" #include "dpEntity.h" #include "Component.h" +#include "eReplicaComponentType.h" #include #include @@ -46,7 +47,7 @@ struct AiSkillEntry */ class BaseCombatAIComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_BASE_COMBAT_AI; + static const eReplicaComponentType ComponentType = eReplicaComponentType::BASE_COMBAT_AI; BaseCombatAIComponent(Entity* parentEntity, uint32_t id); ~BaseCombatAIComponent() override; @@ -243,6 +244,12 @@ private: */ std::vector GetTargetWithinAggroRange() const; + /** + * @brief Sets the AiState and prepares the entity for serialization next frame. + * + */ + void SetAiState(AiState newState); + /** * The current state of the AI */ @@ -334,11 +341,6 @@ private: */ bool m_StunImmune = false; - /** - * Time taken between actions - */ - float m_Downtime = 0; - /** * How long this entity needs to execute its skill */ @@ -374,6 +376,12 @@ private: */ bool m_DirtyThreat = false; + /** + * Whether or not the Component has dirty information and should update next frame + * + */ + bool m_DirtyStateOrTarget = false; + /** * Whether the current entity is a mech enemy, needed as mechs tether radius works differently * @return whether this entity is a mech diff --git a/dGame/dComponents/BouncerComponent.cpp b/dGame/dComponents/BouncerComponent.cpp index 1f32d6e5..bbf928dc 100644 --- a/dGame/dComponents/BouncerComponent.cpp +++ b/dGame/dComponents/BouncerComponent.cpp @@ -7,6 +7,7 @@ #include "dLogger.h" #include "GameMessages.h" #include +#include "eTriggerEventType.h" BouncerComponent::BouncerComponent(Entity* parent) : Component(parent) { m_PetEnabled = false; @@ -35,7 +36,7 @@ Entity* BouncerComponent::GetParentEntity() const { void BouncerComponent::SetPetEnabled(bool value) { m_PetEnabled = value; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void BouncerComponent::SetPetBouncerEnabled(bool value) { @@ -43,11 +44,13 @@ void BouncerComponent::SetPetBouncerEnabled(bool value) { GameMessages::SendBouncerActiveStatus(m_Parent->GetObjectID(), value, UNASSIGNED_SYSTEM_ADDRESS); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); if (value) { + m_Parent->TriggerEvent(eTriggerEventType::PET_ON_SWITCH, m_Parent); GameMessages::SendPlayFXEffect(m_Parent->GetObjectID(), 1513, u"create", "PetOnSwitch", LWOOBJID_EMPTY, 1, 1, true); } else { + m_Parent->TriggerEvent(eTriggerEventType::PET_OFF_SWITCH, m_Parent); GameMessages::SendStopFXEffect(m_Parent, true, "PetOnSwitch"); } @@ -65,7 +68,7 @@ void BouncerComponent::LookupPetSwitch() { const auto& groups = m_Parent->GetGroups(); for (const auto& group : groups) { - const auto& entities = EntityManager::Instance()->GetEntitiesInGroup(group); + const auto& entities = Game::entityManager->GetEntitiesInGroup(group); for (auto* entity : entities) { auto* switchComponent = entity->GetComponent(); @@ -76,7 +79,7 @@ void BouncerComponent::LookupPetSwitch() { m_PetSwitchLoaded = true; m_PetEnabled = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); Game::logger->Log("BouncerComponent", "Loaded pet bouncer"); } diff --git a/dGame/dComponents/BouncerComponent.h b/dGame/dComponents/BouncerComponent.h index f179998b..15665cc1 100644 --- a/dGame/dComponents/BouncerComponent.h +++ b/dGame/dComponents/BouncerComponent.h @@ -5,13 +5,14 @@ #include "RakNetTypes.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Attached to bouncer entities, allowing other entities to bounce off of it */ class BouncerComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_BOUNCER; + static const eReplicaComponentType ComponentType = eReplicaComponentType::BOUNCER; BouncerComponent(Entity* parentEntity); ~BouncerComponent() override; diff --git a/dGame/dComponents/BuffComponent.cpp b/dGame/dComponents/BuffComponent.cpp index ecb10017..68b5182c 100644 --- a/dGame/dComponents/BuffComponent.cpp +++ b/dGame/dComponents/BuffComponent.cpp @@ -9,6 +9,8 @@ #include "SkillComponent.h" #include "ControllablePhysicsComponent.h" #include "EntityManager.h" +#include "CDClientManager.h" +#include "CDSkillBehaviorTable.h" std::unordered_map> BuffComponent::m_Cache{}; @@ -100,7 +102,7 @@ void BuffComponent::ApplyBuff(const int32_t id, const float duration, const LWOO const auto& parameters = GetBuffParameters(id); for (const auto& parameter : parameters) { if (parameter.name == "overtime") { - auto* behaviorTemplateTable = CDClientManager::Instance()->GetTable("SkillBehavior"); + auto* behaviorTemplateTable = CDClientManager::Instance().GetTable(); behaviorID = behaviorTemplateTable->GetSkillByID(parameter.values[0]).behaviorID; stacks = static_cast(parameter.values[1]); @@ -123,13 +125,15 @@ void BuffComponent::ApplyBuff(const int32_t id, const float duration, const LWOO m_Buffs.emplace(id, buff); } -void BuffComponent::RemoveBuff(int32_t id) { +void BuffComponent::RemoveBuff(int32_t id, bool fromUnEquip, bool removeImmunity) { const auto& iter = m_Buffs.find(id); if (iter == m_Buffs.end()) { return; } + GameMessages::SendRemoveBuff(m_Parent, fromUnEquip, removeImmunity, id); + m_Buffs.erase(iter); RemoveBuffEffect(id); @@ -167,17 +171,10 @@ void BuffComponent::ApplyBuffEffect(int32_t id) { destroyable->SetMaxImagination(destroyable->GetMaxImagination() + maxImagination); } else if (parameter.name == "speed") { - const auto speed = parameter.value; - auto* controllablePhysicsComponent = this->GetParent()->GetComponent(); - - if (controllablePhysicsComponent == nullptr) return; - - const auto current = controllablePhysicsComponent->GetSpeedMultiplier(); - - controllablePhysicsComponent->SetSpeedMultiplier(current + ((speed - 500.0f) / 500.0f)); - - EntityManager::Instance()->SerializeEntity(this->GetParent()); + if (!controllablePhysicsComponent) return; + const auto speed = parameter.value; + controllablePhysicsComponent->AddSpeedboost(speed); } } } @@ -210,17 +207,10 @@ void BuffComponent::RemoveBuffEffect(int32_t id) { destroyable->SetMaxImagination(destroyable->GetMaxImagination() - maxImagination); } else if (parameter.name == "speed") { - const auto speed = parameter.value; - auto* controllablePhysicsComponent = this->GetParent()->GetComponent(); - - if (controllablePhysicsComponent == nullptr) return; - - const auto current = controllablePhysicsComponent->GetSpeedMultiplier(); - - controllablePhysicsComponent->SetSpeedMultiplier(current - ((speed - 500.0f) / 500.0f)); - - EntityManager::Instance()->SerializeEntity(this->GetParent()); + if (!controllablePhysicsComponent) return; + const auto speed = parameter.value; + controllablePhysicsComponent->RemoveSpeedboost(speed); } } } diff --git a/dGame/dComponents/BuffComponent.h b/dGame/dComponents/BuffComponent.h index aa669b13..d9175883 100644 --- a/dGame/dComponents/BuffComponent.h +++ b/dGame/dComponents/BuffComponent.h @@ -7,6 +7,7 @@ #include #include #include "Component.h" +#include "eReplicaComponentType.h" class Entity; @@ -41,7 +42,7 @@ struct Buff */ class BuffComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_BUFF; + static const eReplicaComponentType ComponentType = eReplicaComponentType::BUFF; explicit BuffComponent(Entity* parent); @@ -78,8 +79,9 @@ public: /** * Removes a buff from the parent entity, reversing its effects * @param id the id of the buff to remove + * @param removeImmunity whether or not to remove immunity on removing the buff */ - void RemoveBuff(int32_t id); + void RemoveBuff(int32_t id, bool fromUnEquip = false, bool removeImmunity = false); /** * Returns whether or not the entity has a buff identified by `id` diff --git a/dGame/dComponents/BuildBorderComponent.cpp b/dGame/dComponents/BuildBorderComponent.cpp index f9ead9e4..af31f939 100644 --- a/dGame/dComponents/BuildBorderComponent.cpp +++ b/dGame/dComponents/BuildBorderComponent.cpp @@ -17,7 +17,7 @@ BuildBorderComponent::~BuildBorderComponent() { void BuildBorderComponent::OnUse(Entity* originator) { if (originator->GetCharacter()) { - const auto& entities = EntityManager::Instance()->GetEntitiesInGroup("PropertyPlaque"); + const auto& entities = Game::entityManager->GetEntitiesInGroup("PropertyPlaque"); auto buildArea = m_Parent->GetObjectID(); diff --git a/dGame/dComponents/BuildBorderComponent.h b/dGame/dComponents/BuildBorderComponent.h index ba677e37..dc5afc8a 100644 --- a/dGame/dComponents/BuildBorderComponent.h +++ b/dGame/dComponents/BuildBorderComponent.h @@ -9,13 +9,14 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Component for the build border, allowing the user to start building when interacting with it */ class BuildBorderComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_BUILD_BORDER; + static const eReplicaComponentType ComponentType = eReplicaComponentType::BUILD_BORDER; BuildBorderComponent(Entity* parent); ~BuildBorderComponent() override; diff --git a/dGame/dComponents/CMakeLists.txt b/dGame/dComponents/CMakeLists.txt index 0995428b..67f4ece5 100644 --- a/dGame/dComponents/CMakeLists.txt +++ b/dGame/dComponents/CMakeLists.txt @@ -6,6 +6,7 @@ set(DGAME_DCOMPONENTS_SOURCES "BaseCombatAIComponent.cpp" "Component.cpp" "ControllablePhysicsComponent.cpp" "DestroyableComponent.cpp" + "DonationVendorComponent.cpp" "InventoryComponent.cpp" "LevelProgressionComponent.cpp" "LUPExhibitComponent.cpp" @@ -38,5 +39,6 @@ set(DGAME_DCOMPONENTS_SOURCES "BaseCombatAIComponent.cpp" "SkillComponent.cpp" "SoundTriggerComponent.cpp" "SwitchComponent.cpp" + "TriggerComponent.cpp" "VehiclePhysicsComponent.cpp" "VendorComponent.cpp" PARENT_SCOPE) diff --git a/dGame/dComponents/CharacterComponent.cpp b/dGame/dComponents/CharacterComponent.cpp index d425c066..10a4e9db 100644 --- a/dGame/dComponents/CharacterComponent.cpp +++ b/dGame/dComponents/CharacterComponent.cpp @@ -13,6 +13,9 @@ #include "VehiclePhysicsComponent.h" #include "GameMessages.h" #include "Item.h" +#include "Amf3.h" +#include "eGameMasterLevel.h" +#include "eGameActivity.h" CharacterComponent::CharacterComponent(Entity* parent, Character* character) : Component(parent) { m_Character = character; @@ -33,7 +36,7 @@ CharacterComponent::CharacterComponent(Entity* parent, Character* character) : C m_EditorLevel = m_GMLevel; m_Reputation = 0; - m_CurrentActivity = 0; + m_CurrentActivity = eGameActivity::NONE; m_CountryCode = 0; m_LastUpdateTimestamp = std::time(nullptr); } @@ -42,11 +45,11 @@ bool CharacterComponent::LandingAnimDisabled(int zoneID) { switch (zoneID) { case 0: case 556: - case 1001: case 1101: case 1202: case 1203: case 1204: + case 1261: case 1301: case 1302: case 1303: @@ -164,9 +167,9 @@ void CharacterComponent::SetPvpEnabled(const bool value) { m_PvpEnabled = value; } -void CharacterComponent::SetGMLevel(int gmlevel) { +void CharacterComponent::SetGMLevel(eGameMasterLevel gmlevel) { m_DirtyGMInfo = true; - if (gmlevel > 0) m_IsGM = true; + if (gmlevel > eGameMasterLevel::CIVILIAN) m_IsGM = true; else m_IsGM = false; m_GMLevel = gmlevel; } @@ -238,7 +241,7 @@ void CharacterComponent::LoadFromXml(tinyxml2::XMLDocument* doc) { // End custom attributes // - if (m_GMLevel > 0) { + if (m_GMLevel > eGameMasterLevel::CIVILIAN) { m_IsGM = true; m_DirtyGMInfo = true; m_EditorLevel = m_GMLevel; @@ -416,7 +419,7 @@ void CharacterComponent::TrackMissionCompletion(bool isAchievement) { // Achievements are tracked separately for the zone if (isAchievement) { - const auto mapID = dZoneManager::Instance()->GetZoneID().GetMapID(); + const auto mapID = Game::zoneManager->GetZoneID().GetMapID(); GetZoneStatisticsForMap(mapID).m_AchievementsCollected++; } } @@ -477,7 +480,7 @@ void CharacterComponent::TrackArmorDelta(int32_t armor) { void CharacterComponent::TrackRebuildComplete() { UpdatePlayerStatistic(QuickBuildsCompleted); - const auto mapID = dZoneManager::Instance()->GetZoneID().GetMapID(); + const auto mapID = Game::zoneManager->GetZoneID().GetMapID(); GetZoneStatisticsForMap(mapID).m_QuickBuildsCompleted++; } @@ -731,6 +734,6 @@ void CharacterComponent::RemoveVentureVisionEffect(std::string ventureVisionType void CharacterComponent::UpdateClientMinimap(bool showFaction, std::string ventureVisionType) const { if (!m_Parent) return; AMFArrayValue arrayToSend; - arrayToSend.InsertValue(ventureVisionType, showFaction ? static_cast(new AMFTrueValue()) : static_cast(new AMFFalseValue())); - GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent ? m_Parent->GetSystemAddress() : UNASSIGNED_SYSTEM_ADDRESS, "SetFactionVisibility", &arrayToSend); + arrayToSend.Insert(ventureVisionType, showFaction); + GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent ? m_Parent->GetSystemAddress() : UNASSIGNED_SYSTEM_ADDRESS, "SetFactionVisibility", arrayToSend); } diff --git a/dGame/dComponents/CharacterComponent.h b/dGame/dComponents/CharacterComponent.h index 196dfa01..c367aefc 100644 --- a/dGame/dComponents/CharacterComponent.h +++ b/dGame/dComponents/CharacterComponent.h @@ -9,6 +9,9 @@ #include #include "CDMissionsTable.h" #include "tinyxml2.h" +#include "eReplicaComponentType.h" + +enum class eGameActivity : uint32_t; /** * The statistics that can be achieved per zone @@ -59,7 +62,7 @@ enum StatisticID { */ class CharacterComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_CHARACTER; + static const eReplicaComponentType ComponentType = eReplicaComponentType::CHARACTER; CharacterComponent(Entity* parent, Character* character); ~CharacterComponent() override; @@ -111,13 +114,13 @@ public: * Gets the current activity that the character is partaking in, see ScriptedActivityComponent for more details * @return the current activity that the character is partaking in */ - const uint32_t GetCurrentActivity() const { return m_CurrentActivity; } + const eGameActivity GetCurrentActivity() const { return m_CurrentActivity; } /** * Set the current activity of the character, see ScriptedActivityComponent for more details * @param currentActivity the activity to set */ - void SetCurrentActivity(uint32_t currentActivity) { m_CurrentActivity = currentActivity; m_DirtyCurrentActivity = true; } + void SetCurrentActivity(eGameActivity currentActivity) { m_CurrentActivity = currentActivity; m_DirtyCurrentActivity = true; } /** * Gets if the entity is currently racing @@ -177,7 +180,7 @@ public: * Sets the GM level of the character, should be called in the entity. Here it's set for serialization * @param gmlevel the gm level to set */ - void SetGMLevel(int gmlevel); + void SetGMLevel(eGameMasterLevel gmlevel); /** * Initializes the player statistics from the string stored in the XML @@ -273,6 +276,10 @@ public: */ void UpdateClientMinimap(bool showFaction, std::string ventureVisionType) const; + void SetCurrentInteracting(LWOOBJID objectID) {m_CurrentInteracting = objectID;}; + + LWOOBJID GetCurrentInteracting() {return m_CurrentInteracting;}; + /** * Character info regarding this character, including clothing styles, etc. */ @@ -332,7 +339,7 @@ private: /** * The current GM level of this character (anything > 0 counts as a GM) */ - unsigned char m_GMLevel; + eGameMasterLevel m_GMLevel; /** * Whether the character has HF enabled @@ -342,7 +349,7 @@ private: /** * The level of the character in HF */ - unsigned char m_EditorLevel; + eGameMasterLevel m_EditorLevel; /** * Whether the currently active activity has been changed @@ -352,7 +359,7 @@ private: /** * The ID of the curently active activity */ - int m_CurrentActivity; + eGameActivity m_CurrentActivity; /** * Whether the social info has been changed @@ -557,6 +564,8 @@ private: * ID of the last rocket used */ LWOOBJID m_LastRocketItemID = LWOOBJID_EMPTY; + + LWOOBJID m_CurrentInteracting = LWOOBJID_EMPTY; }; #endif // CHARACTERCOMPONENT_H diff --git a/dGame/dComponents/ControllablePhysicsComponent.cpp b/dGame/dComponents/ControllablePhysicsComponent.cpp index d7caf6a9..a658ccd7 100644 --- a/dGame/dComponents/ControllablePhysicsComponent.cpp +++ b/dGame/dComponents/ControllablePhysicsComponent.cpp @@ -12,6 +12,8 @@ #include "EntityManager.h" #include "Character.h" #include "dZoneManager.h" +#include "LevelProgressionComponent.h" +#include "eStateChangeType.h" ControllablePhysicsComponent::ControllablePhysicsComponent(Entity* entity) : Component(entity) { m_Position = {}; @@ -30,10 +32,25 @@ ControllablePhysicsComponent::ControllablePhysicsComponent(Entity* entity) : Com m_GravityScale = 1; m_DirtyCheats = false; m_IgnoreMultipliers = false; + + m_DirtyEquippedItemInfo = true; m_PickupRadius = 0.0f; - m_DirtyPickupRadiusScale = true; + + m_DirtyBubble = false; + m_IsInBubble = false; + m_SpecialAnims = false; + m_BubbleType = eBubbleType::DEFAULT; + m_IsTeleporting = false; + m_ImmuneToStunAttackCount = 0; + m_ImmuneToStunEquipCount = 0; + m_ImmuneToStunInteractCount = 0; + m_ImmuneToStunJumpCount = 0; + m_ImmuneToStunMoveCount = 0; + m_ImmuneToStunTurnCount = 0; + m_ImmuneToStunUseItemCount = 0; + if (entity->GetLOT() != 1) // Other physics entities we care about will be added by BaseCombatAI return; @@ -70,16 +87,17 @@ void ControllablePhysicsComponent::Serialize(RakNet::BitStream* outBitStream, bo outBitStream->Write(m_JetpackBypassChecks); } - outBitStream->Write0(); //This contains info about immunities, but for now I'm leaving it out. + outBitStream->Write1(); // always write these on construction + outBitStream->Write(m_ImmuneToStunMoveCount); + outBitStream->Write(m_ImmuneToStunJumpCount); + outBitStream->Write(m_ImmuneToStunTurnCount); + outBitStream->Write(m_ImmuneToStunAttackCount); + outBitStream->Write(m_ImmuneToStunUseItemCount); + outBitStream->Write(m_ImmuneToStunEquipCount); + outBitStream->Write(m_ImmuneToStunInteractCount); } - if (m_SpeedMultiplier < 1.0f) { - m_DirtyCheats = false; - } - - if (m_IgnoreMultipliers) { - m_DirtyCheats = false; - } + if (m_IgnoreMultipliers) m_DirtyCheats = false; outBitStream->Write(m_DirtyCheats); if (m_DirtyCheats) { @@ -89,14 +107,22 @@ void ControllablePhysicsComponent::Serialize(RakNet::BitStream* outBitStream, bo m_DirtyCheats = false; } - outBitStream->Write(m_DirtyPickupRadiusScale); - if (m_DirtyPickupRadiusScale) { + outBitStream->Write(m_DirtyEquippedItemInfo); + if (m_DirtyEquippedItemInfo) { outBitStream->Write(m_PickupRadius); - outBitStream->Write0(); //No clue what this is so im leaving it false. - m_DirtyPickupRadiusScale = false; + outBitStream->Write(m_InJetpackMode); + m_DirtyEquippedItemInfo = false; } - outBitStream->Write0(); + outBitStream->Write(m_DirtyBubble); + if (m_DirtyBubble) { + outBitStream->Write(m_IsInBubble); + if (m_IsInBubble) { + outBitStream->Write(m_BubbleType); + outBitStream->Write(m_SpecialAnims); + } + m_DirtyBubble = false; + } outBitStream->Write(m_DirtyPosition || bIsInitialUpdate); if (m_DirtyPosition || bIsInitialUpdate) { @@ -168,7 +194,7 @@ void ControllablePhysicsComponent::UpdateXml(tinyxml2::XMLDocument* doc) { return; } - auto zoneInfo = dZoneManager::Instance()->GetZone()->GetZoneID(); + auto zoneInfo = Game::zoneManager->GetZone()->GetZoneID(); if (zoneInfo.GetMapID() != 0 && zoneInfo.GetCloneID() == 0) { character->SetAttribute("lzx", m_Position.x); @@ -253,7 +279,7 @@ void ControllablePhysicsComponent::AddPickupRadiusScale(float value) { m_ActivePickupRadiusScales.push_back(value); if (value > m_PickupRadius) { m_PickupRadius = value; - m_DirtyPickupRadiusScale = true; + m_DirtyEquippedItemInfo = true; } } @@ -263,16 +289,102 @@ void ControllablePhysicsComponent::RemovePickupRadiusScale(float value) { if (pos != m_ActivePickupRadiusScales.end()) { m_ActivePickupRadiusScales.erase(pos); } else { - Game::logger->Log("ControllablePhysicsComponent", "Warning: Could not find pickup radius %f in list of active radii. List has %i active radii.", value, m_ActivePickupRadiusScales.size()); + Game::logger->LogDebug("ControllablePhysicsComponent", "Warning: Could not find pickup radius %f in list of active radii. List has %i active radii.", value, m_ActivePickupRadiusScales.size()); return; } // Recalculate pickup radius since we removed one by now m_PickupRadius = 0.0f; - m_DirtyPickupRadiusScale = true; + m_DirtyEquippedItemInfo = true; for (uint32_t i = 0; i < m_ActivePickupRadiusScales.size(); i++) { auto candidateRadius = m_ActivePickupRadiusScales[i]; if (m_PickupRadius < candidateRadius) m_PickupRadius = candidateRadius; } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); +} + +void ControllablePhysicsComponent::AddSpeedboost(float value) { + m_ActiveSpeedBoosts.push_back(value); + m_SpeedBoost = value; + SetSpeedMultiplier(value / 500.0f); // 500 being the base speed +} + +void ControllablePhysicsComponent::RemoveSpeedboost(float value) { + const auto pos = std::find(m_ActiveSpeedBoosts.begin(), m_ActiveSpeedBoosts.end(), value); + if (pos != m_ActiveSpeedBoosts.end()) { + m_ActiveSpeedBoosts.erase(pos); + } else { + Game::logger->LogDebug("ControllablePhysicsComponent", "Warning: Could not find speedboost %f in list of active speedboosts. List has %i active speedboosts.", value, m_ActiveSpeedBoosts.size()); + return; + } + + // Recalculate speedboost since we removed one + m_SpeedBoost = 0.0f; + if (m_ActiveSpeedBoosts.empty()) { // no active speed boosts left, so return to base speed + auto* levelProgressionComponent = m_Parent->GetComponent(); + if (levelProgressionComponent) m_SpeedBoost = levelProgressionComponent->GetSpeedBase(); + } else { // Used the last applied speedboost + m_SpeedBoost = m_ActiveSpeedBoosts.back(); + } + SetSpeedMultiplier(m_SpeedBoost / 500.0f); // 500 being the base speed + Game::entityManager->SerializeEntity(m_Parent); +} + +void ControllablePhysicsComponent::ActivateBubbleBuff(eBubbleType bubbleType, bool specialAnims){ + if (m_IsInBubble) { + Game::logger->Log("ControllablePhysicsComponent", "Already in bubble"); + return; + } + m_BubbleType = bubbleType; + m_IsInBubble = true; + m_DirtyBubble = true; + m_SpecialAnims = specialAnims; + Game::entityManager->SerializeEntity(m_Parent); +} + +void ControllablePhysicsComponent::DeactivateBubbleBuff(){ + m_DirtyBubble = true; + m_IsInBubble = false; + Game::entityManager->SerializeEntity(m_Parent); +}; + +void ControllablePhysicsComponent::SetStunImmunity( + const eStateChangeType state, + const LWOOBJID originator, + const bool bImmuneToStunAttack, + const bool bImmuneToStunEquip, + const bool bImmuneToStunInteract, + const bool bImmuneToStunJump, + const bool bImmuneToStunMove, + const bool bImmuneToStunTurn, + const bool bImmuneToStunUseItem){ + + if (state == eStateChangeType::POP){ + if (bImmuneToStunAttack && m_ImmuneToStunAttackCount > 0) m_ImmuneToStunAttackCount -= 1; + if (bImmuneToStunEquip && m_ImmuneToStunEquipCount > 0) m_ImmuneToStunEquipCount -= 1; + if (bImmuneToStunInteract && m_ImmuneToStunInteractCount > 0) m_ImmuneToStunInteractCount -= 1; + if (bImmuneToStunJump && m_ImmuneToStunJumpCount > 0) m_ImmuneToStunJumpCount -= 1; + if (bImmuneToStunMove && m_ImmuneToStunMoveCount > 0) m_ImmuneToStunMoveCount -= 1; + if (bImmuneToStunTurn && m_ImmuneToStunTurnCount > 0) m_ImmuneToStunTurnCount -= 1; + if (bImmuneToStunUseItem && m_ImmuneToStunUseItemCount > 0) m_ImmuneToStunUseItemCount -= 1; + } else if (state == eStateChangeType::PUSH) { + if (bImmuneToStunAttack) m_ImmuneToStunAttackCount += 1; + if (bImmuneToStunEquip) m_ImmuneToStunEquipCount += 1; + if (bImmuneToStunInteract) m_ImmuneToStunInteractCount += 1; + if (bImmuneToStunJump) m_ImmuneToStunJumpCount += 1; + if (bImmuneToStunMove) m_ImmuneToStunMoveCount += 1; + if (bImmuneToStunTurn) m_ImmuneToStunTurnCount += 1; + if (bImmuneToStunUseItem) m_ImmuneToStunUseItemCount += 1; + } + + GameMessages::SendSetStunImmunity( + m_Parent->GetObjectID(), state, m_Parent->GetSystemAddress(), originator, + bImmuneToStunAttack, + bImmuneToStunEquip, + bImmuneToStunInteract, + bImmuneToStunJump, + bImmuneToStunMove, + bImmuneToStunTurn, + bImmuneToStunUseItem + ); } diff --git a/dGame/dComponents/ControllablePhysicsComponent.h b/dGame/dComponents/ControllablePhysicsComponent.h index ac481b9f..470a7af4 100644 --- a/dGame/dComponents/ControllablePhysicsComponent.h +++ b/dGame/dComponents/ControllablePhysicsComponent.h @@ -9,16 +9,19 @@ #include "Component.h" #include "dpCollisionChecks.h" #include "PhantomPhysicsComponent.h" +#include "eBubbleType.h" +#include "eReplicaComponentType.h" class Entity; class dpEntity; +enum class eStateChangeType : uint32_t; /** * Handles the movement of controllable Entities, e.g. enemies and players */ class ControllablePhysicsComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_CONTROLLABLE_PHYSICS; + static const eReplicaComponentType ComponentType = eReplicaComponentType::CONTROLLABLE_PHYSICS; ControllablePhysicsComponent(Entity* entity); ~ControllablePhysicsComponent() override; @@ -257,6 +260,63 @@ public: */ std::vector GetActivePickupRadiusScales() { return m_ActivePickupRadiusScales; }; + /** + * Add a Speed boost to the entity + * This will recalculate the speed boost based on what is being added + */ + void AddSpeedboost(float value); + + /** + * Remove speed boost from entity + * This will recalculate the speed boost based on what is the last one in te vector + */ + void RemoveSpeedboost(float value); + + /** + * The speed boosts of this component. + * @return All active Speed boosts for this component. + */ + std::vector GetActiveSpeedboosts() { return m_ActiveSpeedBoosts; }; + + /** + * Activates the Bubble Buff + */ + void ActivateBubbleBuff(eBubbleType bubbleType = eBubbleType::DEFAULT, bool specialAnims = true); + + /** + * Deactivates the Bubble Buff + */ + void DeactivateBubbleBuff(); + + /** + * Gets if the Entity is in a bubble + */ + bool GetIsInBubble(){ return m_IsInBubble; }; + + /** + * Push or Pop a layer of stun immunity to this entity + */ + void SetStunImmunity( + const eStateChangeType state, + const LWOOBJID originator = LWOOBJID_EMPTY, + const bool bImmuneToStunAttack = false, + const bool bImmuneToStunEquip = false, + const bool bImmuneToStunInteract = false, + const bool bImmuneToStunJump = false, + const bool bImmuneToStunMove = false, + const bool bImmuneToStunTurn = false, + const bool bImmuneToStunUseItem = false + ); + + // getters for stun immunities + const bool GetImmuneToStunAttack() { return m_ImmuneToStunAttackCount > 0;}; + const bool GetImmuneToStunEquip() { return m_ImmuneToStunEquipCount > 0;}; + const bool GetImmuneToStunInteract() { return m_ImmuneToStunInteractCount > 0;}; + const bool GetImmuneToStunJump() { return m_ImmuneToStunJumpCount > 0;}; + const bool GetImmuneToStunMove() { return m_ImmuneToStunMoveCount > 0;}; + const bool GetImmuneToStunTurn() { return m_ImmuneToStunTurnCount > 0;}; + const bool GetImmuneToStunUseItem() { return m_ImmuneToStunUseItemCount > 0;}; + private: /** * The entity that owns this component @@ -356,7 +416,7 @@ private: /** * Whether the pickup scale is dirty. */ - bool m_DirtyPickupRadiusScale; + bool m_DirtyEquippedItemInfo; /** * The list of pickup radius scales for this entity @@ -372,6 +432,47 @@ private: * If the entity is teleporting */ bool m_IsTeleporting; + + /** + * The list of speed boosts for this entity + */ + std::vector m_ActiveSpeedBoosts; + + /** + * The active speed boost for this entity + */ + float m_SpeedBoost; + + /* + * If Bubble info is dirty + */ + bool m_DirtyBubble; + + /* + * If the entity is in a bubble + */ + bool m_IsInBubble; + + /* + * The type of bubble the entity has + */ + eBubbleType m_BubbleType; + + /* + * If the entity should be using the special animations + */ + bool m_SpecialAnims; + + /** + * stun immunity counters + */ + int32_t m_ImmuneToStunAttackCount; + int32_t m_ImmuneToStunEquipCount; + int32_t m_ImmuneToStunInteractCount; + int32_t m_ImmuneToStunJumpCount; + int32_t m_ImmuneToStunMoveCount; + int32_t m_ImmuneToStunTurnCount; + int32_t m_ImmuneToStunUseItemCount; }; #endif // CONTROLLABLEPHYSICSCOMPONENT_H diff --git a/dGame/dComponents/DestroyableComponent.cpp b/dGame/dComponents/DestroyableComponent.cpp index 8b9ae42d..4c726bad 100644 --- a/dGame/dComponents/DestroyableComponent.cpp +++ b/dGame/dComponents/DestroyableComponent.cpp @@ -2,9 +2,10 @@ #include #include "dLogger.h" #include "Game.h" +#include "dConfig.h" -#include "AMFFormat.h" -#include "AMFFormat_BitStream.h" +#include "Amf3.h" +#include "AmfSerialize.h" #include "GameMessages.h" #include "User.h" #include "CDClientManager.h" @@ -28,7 +29,14 @@ #include "CharacterComponent.h" #include "PossessableComponent.h" #include "PossessorComponent.h" +#include "InventoryComponent.h" #include "dZoneManager.h" +#include "WorldConfig.h" +#include "eMissionTaskType.h" +#include "eStateChangeType.h" +#include "eGameActivity.h" + +#include "CDComponentsRegistryTable.h" DestroyableComponent::DestroyableComponent(Entity* parent) : Component(parent) { m_iArmor = 0; @@ -43,7 +51,7 @@ DestroyableComponent::DestroyableComponent(Entity* parent) : Component(parent) { m_IsGMImmune = false; m_IsShielded = false; m_DamageToAbsorb = 0; - m_HasBricks = false; + m_IsModuleAssembly = m_Parent->HasComponent(eReplicaComponentType::MODULE_ASSEMBLY); m_DirtyThreatList = false; m_HasThreats = false; m_ExplodeFactor = 1.0f; @@ -53,26 +61,35 @@ DestroyableComponent::DestroyableComponent(Entity* parent) : Component(parent) { m_LootMatrixID = 0; m_MinCoins = 0; m_MaxCoins = 0; - m_ImmuneStacks = 0; m_DamageReduction = 0; + + m_ImmuneToBasicAttackCount = 0; + m_ImmuneToDamageOverTimeCount = 0; + m_ImmuneToKnockbackCount = 0; + m_ImmuneToInterruptCount = 0; + m_ImmuneToSpeedCount = 0; + m_ImmuneToImaginationGainCount = 0; + m_ImmuneToImaginationLossCount = 0; + m_ImmuneToQuickbuildInterruptCount = 0; + m_ImmuneToPullToPointCount = 0; } DestroyableComponent::~DestroyableComponent() { } void DestroyableComponent::Reinitialize(LOT templateID) { - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); - int32_t buffComponentID = compRegistryTable->GetByIDAndType(templateID, COMPONENT_TYPE_BUFF); - int32_t collectibleComponentID = compRegistryTable->GetByIDAndType(templateID, COMPONENT_TYPE_COLLECTIBLE); - int32_t rebuildComponentID = compRegistryTable->GetByIDAndType(templateID, COMPONENT_TYPE_REBUILD); + int32_t buffComponentID = compRegistryTable->GetByIDAndType(templateID, eReplicaComponentType::BUFF); + int32_t collectibleComponentID = compRegistryTable->GetByIDAndType(templateID, eReplicaComponentType::COLLECTIBLE); + int32_t rebuildComponentID = compRegistryTable->GetByIDAndType(templateID, eReplicaComponentType::QUICK_BUILD); int32_t componentID = 0; if (collectibleComponentID > 0) componentID = collectibleComponentID; if (rebuildComponentID > 0) componentID = rebuildComponentID; if (buffComponentID > 0) componentID = buffComponentID; - CDDestructibleComponentTable* destCompTable = CDClientManager::Instance()->GetTable("DestructibleComponent"); + CDDestructibleComponentTable* destCompTable = CDClientManager::Instance().GetTable(); std::vector destCompData = destCompTable->Query([=](CDDestructibleComponent entry) { return (entry.id == componentID); }); if (componentID > 0) { @@ -104,7 +121,16 @@ void DestroyableComponent::Reinitialize(LOT templateID) { void DestroyableComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, uint32_t& flags) { if (bIsInitialUpdate) { - outBitStream->Write0(); //Contains info about immunities this object has, but it's left out for now. + outBitStream->Write1(); // always write these on construction + outBitStream->Write(m_ImmuneToBasicAttackCount); + outBitStream->Write(m_ImmuneToDamageOverTimeCount); + outBitStream->Write(m_ImmuneToKnockbackCount); + outBitStream->Write(m_ImmuneToInterruptCount); + outBitStream->Write(m_ImmuneToSpeedCount); + outBitStream->Write(m_ImmuneToImaginationGainCount); + outBitStream->Write(m_ImmuneToImaginationLossCount); + outBitStream->Write(m_ImmuneToQuickbuildInterruptCount); + outBitStream->Write(m_ImmuneToPullToPointCount); } outBitStream->Write(m_DirtyHealth || bIsInitialUpdate); @@ -137,26 +163,18 @@ void DestroyableComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsIn outBitStream->Write(m_IsSmashed); if (m_IsSmashable) { - outBitStream->Write(m_HasBricks); - - if (m_ExplodeFactor != 1.0f) { - outBitStream->Write1(); - outBitStream->Write(m_ExplodeFactor); - } else { - outBitStream->Write0(); - } + outBitStream->Write(m_IsModuleAssembly); + outBitStream->Write(m_ExplodeFactor != 1.0f); + if (m_ExplodeFactor != 1.0f) outBitStream->Write(m_ExplodeFactor); } } - m_DirtyHealth = false; } + outBitStream->Write(m_DirtyThreatList || bIsInitialUpdate); if (m_DirtyThreatList || bIsInitialUpdate) { - outBitStream->Write1(); outBitStream->Write(m_HasThreats); m_DirtyThreatList = false; - } else { - outBitStream->Write0(); } } @@ -227,19 +245,15 @@ void DestroyableComponent::SetMaxHealth(float value, bool playAnim) { if (playAnim) { // Now update the player bar if (!m_Parent->GetParentUser()) return; - AMFStringValue* amount = new AMFStringValue(); - amount->SetStringValue(std::to_string(difference)); - AMFStringValue* type = new AMFStringValue(); - type->SetStringValue("health"); AMFArrayValue args; - args.InsertValue("amount", amount); - args.InsertValue("type", type); + args.Insert("amount", std::to_string(difference)); + args.Insert("type", "health"); - GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent->GetParentUser()->GetSystemAddress(), "MaxPlayerBarUpdate", &args); + GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent->GetParentUser()->GetSystemAddress(), "MaxPlayerBarUpdate", args); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void DestroyableComponent::SetArmor(int32_t value) { @@ -272,19 +286,15 @@ void DestroyableComponent::SetMaxArmor(float value, bool playAnim) { if (playAnim) { // Now update the player bar if (!m_Parent->GetParentUser()) return; - AMFStringValue* amount = new AMFStringValue(); - amount->SetStringValue(std::to_string(value)); - AMFStringValue* type = new AMFStringValue(); - type->SetStringValue("armor"); AMFArrayValue args; - args.InsertValue("amount", amount); - args.InsertValue("type", type); + args.Insert("amount", std::to_string(value)); + args.Insert("type", "armor"); - GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent->GetParentUser()->GetSystemAddress(), "MaxPlayerBarUpdate", &args); + GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent->GetParentUser()->GetSystemAddress(), "MaxPlayerBarUpdate", args); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void DestroyableComponent::SetImagination(int32_t value) { @@ -316,18 +326,14 @@ void DestroyableComponent::SetMaxImagination(float value, bool playAnim) { if (playAnim) { // Now update the player bar if (!m_Parent->GetParentUser()) return; - AMFStringValue* amount = new AMFStringValue(); - amount->SetStringValue(std::to_string(difference)); - AMFStringValue* type = new AMFStringValue(); - type->SetStringValue("imagination"); AMFArrayValue args; - args.InsertValue("amount", amount); - args.InsertValue("type", type); + args.Insert("amount", std::to_string(difference)); + args.Insert("type", "imagination"); - GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent->GetParentUser()->GetSystemAddress(), "MaxPlayerBarUpdate", &args); + GameMessages::SendUIMessageServerToSingleClient(m_Parent, m_Parent->GetParentUser()->GetSystemAddress(), "MaxPlayerBarUpdate", args); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void DestroyableComponent::SetDamageToAbsorb(int32_t value) { @@ -342,7 +348,7 @@ void DestroyableComponent::SetDamageReduction(int32_t value) { void DestroyableComponent::SetIsImmune(bool value) { m_DirtyHealth = true; - m_ImmuneStacks = value ? 1 : 0; + m_ImmuneToBasicAttackCount = value ? 1 : 0; } void DestroyableComponent::SetIsGMImmune(bool value) { @@ -438,7 +444,6 @@ void DestroyableComponent::AddEnemyFaction(int32_t factionID) { void DestroyableComponent::SetIsSmashable(bool value) { m_DirtyHealth = true; m_IsSmashable = value; - //m_HasBricks = value; } void DestroyableComponent::SetAttacksToBlock(const uint32_t value) { @@ -446,18 +451,18 @@ void DestroyableComponent::SetAttacksToBlock(const uint32_t value) { } bool DestroyableComponent::IsImmune() const { - return m_ImmuneStacks > 0 || m_IsGMImmune; + return m_IsGMImmune || m_ImmuneToBasicAttackCount > 0; } bool DestroyableComponent::IsKnockbackImmune() const { auto* characterComponent = m_Parent->GetComponent(); auto* inventoryComponent = m_Parent->GetComponent(); - if (characterComponent != nullptr && inventoryComponent != nullptr && characterComponent->GetCurrentActivity() == eGameActivities::ACTIVITY_QUICKBUILDING) { + if (characterComponent != nullptr && inventoryComponent != nullptr && characterComponent->GetCurrentActivity() == eGameActivity::QUICKBUILDING) { const auto hasPassive = inventoryComponent->HasAnyPassive({ - ItemSetPassiveAbilityID::EngineerRank2, ItemSetPassiveAbilityID::EngineerRank3, - ItemSetPassiveAbilityID::SummonerRank2, ItemSetPassiveAbilityID::SummonerRank3, - ItemSetPassiveAbilityID::InventorRank2, ItemSetPassiveAbilityID::InventorRank3, + eItemSetPassiveAbilityID::EngineerRank2, eItemSetPassiveAbilityID::EngineerRank3, + eItemSetPassiveAbilityID::SummonerRank2, eItemSetPassiveAbilityID::SummonerRank3, + eItemSetPassiveAbilityID::InventorRank2, eItemSetPassiveAbilityID::InventorRank3, }, 5); if (hasPassive) { @@ -477,11 +482,11 @@ LWOOBJID DestroyableComponent::GetKillerID() const { } Entity* DestroyableComponent::GetKiller() const { - return EntityManager::Instance()->GetEntity(m_KillerID); + return Game::entityManager->GetEntity(m_KillerID); } bool DestroyableComponent::CheckValidity(const LWOOBJID target, const bool ignoreFactions, const bool targetEnemy, const bool targetFriend) const { - auto* targetEntity = EntityManager::Instance()->GetEntity(target); + auto* targetEntity = Game::entityManager->GetEntity(target); if (targetEntity == nullptr) { Game::logger->Log("DestroyableComponent", "Invalid entity for checking validity (%llu)!", target); @@ -499,7 +504,7 @@ bool DestroyableComponent::CheckValidity(const LWOOBJID target, const bool ignor if (targetQuickbuild != nullptr) { const auto state = targetQuickbuild->GetState(); - if (state != REBUILD_COMPLETED) { + if (state != eRebuildState::COMPLETED) { return false; } } @@ -527,7 +532,7 @@ void DestroyableComponent::Heal(const uint32_t health) { SetHealth(current); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } @@ -545,7 +550,7 @@ void DestroyableComponent::Imagine(const int32_t deltaImagination) { SetImagination(current); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } @@ -559,7 +564,7 @@ void DestroyableComponent::Repair(const uint32_t armor) { SetArmor(current); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } @@ -621,7 +626,7 @@ void DestroyableComponent::Damage(uint32_t damage, const LWOOBJID source, uint32 if (possessor) { auto possessableId = possessor->GetPossessable(); if (possessableId != LWOOBJID_EMPTY) { - auto possessable = EntityManager::Instance()->GetEntity(possessableId); + auto possessable = Game::entityManager->GetEntity(possessableId); if (possessable) { possessor->Dismount(possessable); } @@ -633,12 +638,13 @@ void DestroyableComponent::Damage(uint32_t damage, const LWOOBJID source, uint32 } if (echo) { - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } - auto* attacker = EntityManager::Instance()->GetEntity(source); + auto* attacker = Game::entityManager->GetEntity(source); m_Parent->OnHit(attacker); m_Parent->OnHitOrHealResult(attacker, sourceDamage); + NotifySubscribers(attacker, sourceDamage); for (const auto& cb : m_OnHitCallbacks) { cb(attacker); @@ -653,20 +659,49 @@ void DestroyableComponent::Damage(uint32_t damage, const LWOOBJID source, uint32 return; } + + //check if hardcore mode is enabled + if (Game::entityManager->GetHardcoreMode()) { + DoHardcoreModeDrops(source); + } + Smash(source, eKillType::VIOLENT, u"", skillID); } +void DestroyableComponent::Subscribe(LWOOBJID scriptObjId, CppScripts::Script* scriptToAdd) { + m_SubscribedScripts.insert(std::make_pair(scriptObjId, scriptToAdd)); + Game::logger->LogDebug("DestroyableComponent", "Added script %llu to entity %llu", scriptObjId, m_Parent->GetObjectID()); + Game::logger->LogDebug("DestroyableComponent", "Number of subscribed scripts %i", m_SubscribedScripts.size()); +} + +void DestroyableComponent::Unsubscribe(LWOOBJID scriptObjId) { + auto foundScript = m_SubscribedScripts.find(scriptObjId); + if (foundScript != m_SubscribedScripts.end()) { + m_SubscribedScripts.erase(foundScript); + Game::logger->LogDebug("DestroyableComponent", "Removed script %llu from entity %llu", scriptObjId, m_Parent->GetObjectID()); + } else { + Game::logger->LogDebug("DestroyableComponent", "Tried to remove a script for Entity %llu but script %llu didnt exist", m_Parent->GetObjectID(), scriptObjId); + } + Game::logger->LogDebug("DestroyableComponent", "Number of subscribed scripts %i", m_SubscribedScripts.size()); +} + +void DestroyableComponent::NotifySubscribers(Entity* attacker, uint32_t damage) { + for (auto script : m_SubscribedScripts) { + script.second->NotifyHitOrHealResult(m_Parent, attacker, damage); + } +} + void DestroyableComponent::Smash(const LWOOBJID source, const eKillType killType, const std::u16string& deathType, uint32_t skillID) { if (m_iHealth > 0) { SetArmor(0); SetHealth(0); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } m_KillerID = source; - auto* owner = EntityManager::Instance()->GetEntity(source); + auto* owner = Game::entityManager->GetEntity(source); if (owner != nullptr) { owner = owner->GetOwner(); // If the owner is overwritten, we collect that here @@ -678,15 +713,15 @@ void DestroyableComponent::Smash(const LWOOBJID source, const eKillType killType auto* inventoryComponent = owner->GetComponent(); if (inventoryComponent != nullptr && isEnemy) { - inventoryComponent->TriggerPassiveAbility(PassiveAbilityTrigger::EnemySmashed); + inventoryComponent->TriggerPassiveAbility(PassiveAbilityTrigger::EnemySmashed, m_Parent); } auto* missions = owner->GetComponent(); if (missions != nullptr) { - if (team != nullptr && isEnemy) { + if (team != nullptr) { for (const auto memberId : team->members) { - auto* member = EntityManager::Instance()->GetEntity(memberId); + auto* member = Game::entityManager->GetEntity(memberId); if (member == nullptr) continue; @@ -694,12 +729,12 @@ void DestroyableComponent::Smash(const LWOOBJID source, const eKillType killType if (memberMissions == nullptr) continue; - memberMissions->Progress(MissionTaskType::MISSION_TASK_TYPE_SMASH, m_Parent->GetLOT()); - memberMissions->Progress(MissionTaskType::MISSION_TASK_TYPE_SKILL, m_Parent->GetLOT(), skillID); + memberMissions->Progress(eMissionTaskType::SMASH, m_Parent->GetLOT()); + memberMissions->Progress(eMissionTaskType::USE_SKILL, m_Parent->GetLOT(), skillID); } } else { - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_SMASH, m_Parent->GetLOT()); - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_SKILL, m_Parent->GetLOT(), skillID); + missions->Progress(eMissionTaskType::SMASH, m_Parent->GetLOT()); + missions->Progress(eMissionTaskType::USE_SKILL, m_Parent->GetLOT(), skillID); } } } @@ -726,12 +761,12 @@ void DestroyableComponent::Smash(const LWOOBJID source, const eKillType killType if (team->lootOption == 0) { // Round robin specificOwner = TeamManager::Instance()->GetNextLootOwner(team); - auto* member = EntityManager::Instance()->GetEntity(specificOwner); + auto* member = Game::entityManager->GetEntity(specificOwner); if (member) LootGenerator::Instance().DropLoot(member, m_Parent, lootMatrixId, GetMinCoins(), GetMaxCoins()); } else { for (const auto memberId : team->members) { // Free for all - auto* member = EntityManager::Instance()->GetEntity(memberId); + auto* member = Game::entityManager->GetEntity(memberId); if (member == nullptr) continue; @@ -744,35 +779,30 @@ void DestroyableComponent::Smash(const LWOOBJID source, const eKillType killType } } else { //Check if this zone allows coin drops - if (dZoneManager::Instance()->GetPlayerLoseCoinOnDeath()) { + if (Game::zoneManager->GetPlayerLoseCoinOnDeath()) { auto* character = m_Parent->GetCharacter(); uint64_t coinsTotal = character->GetCoins(); + const uint64_t minCoinsToLose = Game::zoneManager->GetWorldConfig()->coinsLostOnDeathMin; + if (coinsTotal >= minCoinsToLose) { + const uint64_t maxCoinsToLose = Game::zoneManager->GetWorldConfig()->coinsLostOnDeathMax; + const float coinPercentageToLose = Game::zoneManager->GetWorldConfig()->coinsLostOnDeathPercent; - if (coinsTotal > 0) { - uint64_t coinsToLoose = 1; + uint64_t coinsToLose = std::max(static_cast(coinsTotal * coinPercentageToLose), minCoinsToLose); + coinsToLose = std::min(maxCoinsToLose, coinsToLose); - if (coinsTotal >= 200) { - float hundreth = (coinsTotal / 100.0f); - coinsToLoose = static_cast(hundreth); - } + coinsTotal -= coinsToLose; - if (coinsToLoose > 10000) { - coinsToLoose = 10000; - } - - coinsTotal -= coinsToLoose; - - LootGenerator::Instance().DropLoot(m_Parent, m_Parent, -1, coinsToLoose, coinsToLoose); - character->SetCoins(coinsTotal, eLootSourceType::LOOT_SOURCE_PICKUP); + LootGenerator::Instance().DropLoot(m_Parent, m_Parent, -1, coinsToLose, coinsToLose); + character->SetCoins(coinsTotal, eLootSourceType::PICKUP); } } - Entity* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); + Entity* zoneControl = Game::entityManager->GetZoneControlEntity(); for (CppScripts::Script* script : CppScripts::GetEntityScripts(zoneControl)) { script->OnPlayerDied(zoneControl, m_Parent); } - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPTED_ACTIVITY); for (Entity* scriptEntity : scriptedActs) { if (scriptEntity->GetObjectID() != zoneControl->GetObjectID()) { // Don't want to trigger twice on instance worlds for (CppScripts::Script* script : CppScripts::GetEntityScripts(scriptEntity)) { @@ -792,12 +822,53 @@ void DestroyableComponent::SetFaction(int32_t factionID, bool ignoreChecks) { AddFaction(factionID, ignoreChecks); } -void DestroyableComponent::PushImmunity(int32_t stacks) { - m_ImmuneStacks += stacks; -} +void DestroyableComponent::SetStatusImmunity( + const eStateChangeType state, + const bool bImmuneToBasicAttack, + const bool bImmuneToDamageOverTime, + const bool bImmuneToKnockback, + const bool bImmuneToInterrupt, + const bool bImmuneToSpeed, + const bool bImmuneToImaginationGain, + const bool bImmuneToImaginationLoss, + const bool bImmuneToQuickbuildInterrupt, + const bool bImmuneToPullToPoint) { -void DestroyableComponent::PopImmunity(int32_t stacks) { - m_ImmuneStacks -= stacks; + if (state == eStateChangeType::POP) { + if (bImmuneToBasicAttack && m_ImmuneToBasicAttackCount > 0) m_ImmuneToBasicAttackCount -= 1; + if (bImmuneToDamageOverTime && m_ImmuneToDamageOverTimeCount > 0) m_ImmuneToDamageOverTimeCount -= 1; + if (bImmuneToKnockback && m_ImmuneToKnockbackCount > 0) m_ImmuneToKnockbackCount -= 1; + if (bImmuneToInterrupt && m_ImmuneToInterruptCount > 0) m_ImmuneToInterruptCount -= 1; + if (bImmuneToSpeed && m_ImmuneToSpeedCount > 0) m_ImmuneToSpeedCount -= 1; + if (bImmuneToImaginationGain && m_ImmuneToImaginationGainCount > 0) m_ImmuneToImaginationGainCount -= 1; + if (bImmuneToImaginationLoss && m_ImmuneToImaginationLossCount > 0) m_ImmuneToImaginationLossCount -= 1; + if (bImmuneToQuickbuildInterrupt && m_ImmuneToQuickbuildInterruptCount > 0) m_ImmuneToQuickbuildInterruptCount -= 1; + if (bImmuneToPullToPoint && m_ImmuneToPullToPointCount > 0) m_ImmuneToPullToPointCount -= 1; + + } else if (state == eStateChangeType::PUSH){ + if (bImmuneToBasicAttack) m_ImmuneToBasicAttackCount += 1; + if (bImmuneToDamageOverTime) m_ImmuneToDamageOverTimeCount += 1; + if (bImmuneToKnockback) m_ImmuneToKnockbackCount += 1; + if (bImmuneToInterrupt) m_ImmuneToInterruptCount += 1; + if (bImmuneToSpeed) m_ImmuneToSpeedCount += 1; + if (bImmuneToImaginationGain) m_ImmuneToImaginationGainCount += 1; + if (bImmuneToImaginationLoss) m_ImmuneToImaginationLossCount += 1; + if (bImmuneToQuickbuildInterrupt) m_ImmuneToQuickbuildInterruptCount += 1; + if (bImmuneToPullToPoint) m_ImmuneToPullToPointCount += 1; + } + + GameMessages::SendSetStatusImmunity( + m_Parent->GetObjectID(), state, m_Parent->GetSystemAddress(), + bImmuneToBasicAttack, + bImmuneToDamageOverTime, + bImmuneToKnockback, + bImmuneToInterrupt, + bImmuneToSpeed, + bImmuneToImaginationGain, + bImmuneToImaginationLoss, + bImmuneToQuickbuildInterrupt, + bImmuneToPullToPoint + ); } void DestroyableComponent::FixStats() { @@ -894,9 +965,81 @@ void DestroyableComponent::FixStats() { destroyableComponent->SetImagination(currentImagination); // Serialize the entity - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } void DestroyableComponent::AddOnHitCallback(const std::function& callback) { m_OnHitCallbacks.push_back(callback); } + +void DestroyableComponent::DoHardcoreModeDrops(const LWOOBJID source){ + //check if this is a player: + if (m_Parent->IsPlayer()) { + //remove hardcore_lose_uscore_on_death_percent from the player's uscore: + auto* character = m_Parent->GetComponent(); + auto uscore = character->GetUScore(); + + auto uscoreToLose = uscore * (Game::entityManager->GetHardcoreLoseUscoreOnDeathPercent() / 100); + character->SetUScore(uscore - uscoreToLose); + + GameMessages::SendModifyLEGOScore(m_Parent, m_Parent->GetSystemAddress(), -uscoreToLose, eLootSourceType::MISSION); + + if (Game::entityManager->GetHardcoreDropinventoryOnDeath()) { + //drop all items from inventory: + auto* inventory = m_Parent->GetComponent(); + if (inventory) { + //get the items inventory: + auto items = inventory->GetInventory(eInventoryType::ITEMS); + if (items){ + auto itemMap = items->GetItems(); + if (!itemMap.empty()){ + for (const auto& item : itemMap) { + //drop the item: + if (!item.second) continue; + // don't drop the thinkng cap + if (item.second->GetLot() == 6086) continue; + GameMessages::SendDropClientLoot(m_Parent, source, item.second->GetLot(), 0, m_Parent->GetPosition(), item.second->GetCount()); + item.second->SetCount(0, false, false); + } + Game::entityManager->SerializeEntity(m_Parent); + } + } + } + } + + //get character: + auto* chars = m_Parent->GetCharacter(); + if (chars) { + auto coins = chars->GetCoins(); + + //lose all coins: + chars->SetCoins(0, eLootSourceType::NONE); + + //drop all coins: + GameMessages::SendDropClientLoot(m_Parent, source, LOT_NULL, coins, m_Parent->GetPosition()); + } + + // Reload the player since we can't normally reduce uscore from the server and we want the UI to update + // do this last so we don't get killed.... again + Game::entityManager->DestructEntity(m_Parent); + Game::entityManager->ConstructEntity(m_Parent); + return; + } + + //award the player some u-score: + auto* player = Game::entityManager->GetEntity(source); + if (player && player->IsPlayer()) { + auto* playerStats = player->GetComponent(); + if (playerStats) { + //get the maximum health from this enemy: + auto maxHealth = GetMaxHealth(); + + int uscore = maxHealth * Game::entityManager->GetHardcoreUscoreEnemiesMultiplier(); + + playerStats->SetUScore(playerStats->GetUScore() + uscore); + GameMessages::SendModifyLEGOScore(player, player->GetSystemAddress(), uscore, eLootSourceType::MISSION); + + Game::entityManager->SerializeEntity(m_Parent); + } + } +} diff --git a/dGame/dComponents/DestroyableComponent.h b/dGame/dComponents/DestroyableComponent.h index a403717b..5e5133b7 100644 --- a/dGame/dComponents/DestroyableComponent.h +++ b/dGame/dComponents/DestroyableComponent.h @@ -6,6 +6,12 @@ #include "tinyxml2.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" + +namespace CppScripts { + class Script; +}; //! namespace CppScripts +enum class eStateChangeType : uint32_t; /** * Represents the stats of an entity, for example its health, imagination and armor. Also handles factions, which @@ -13,7 +19,7 @@ */ class DestroyableComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_DESTROYABLE; + static const eReplicaComponentType ComponentType = eReplicaComponentType::DESTROYABLE; DestroyableComponent(Entity* parentEntity); ~DestroyableComponent() override; @@ -233,13 +239,13 @@ public: * Returns whether or not this entity has bricks flying out when smashed * @return whether or not this entity has bricks flying out when smashed */ - bool GetHasBricks() const { return m_HasBricks; } + bool GetHasBricks() const { return m_IsModuleAssembly; } /** * Sets the multiplier for the explosion that's visible when the bricks fly out when this entity is smashed * @param value the multiplier for the explosion that's visible when the bricks fly out when this entity is smashed */ - void SetExplodeFactor(float value); + void SetExplodeFactor(float value) { m_ExplodeFactor = value; }; /** * Returns the current multiplier for explosions @@ -392,16 +398,31 @@ public: void Smash(LWOOBJID source, eKillType killType = eKillType::VIOLENT, const std::u16string& deathType = u"", uint32_t skillID = 0); /** - * Pushes a layer of immunity to this entity, making it immune for longer - * @param stacks the amount of immunity to add + * Push or Pop a layer of status immunity to this entity */ - void PushImmunity(int32_t stacks = 1); + void SetStatusImmunity( + const eStateChangeType state, + const bool bImmuneToBasicAttack = false, + const bool bImmuneToDamageOverTime = false, + const bool bImmuneToKnockback = false, + const bool bImmuneToInterrupt = false, + const bool bImmuneToSpeed = false, + const bool bImmuneToImaginationGain = false, + const bool bImmuneToImaginationLoss = false, + const bool bImmuneToQuickbuildInterrupt = false, + const bool bImmuneToPullToPoint = false + ); - /** - * Pops layers of immunity, making it immune for less longer - * @param stacks the number of layers of immunity to remove - */ - void PopImmunity(int32_t stacks = 1); + // Getters for status immunities + const bool GetImmuneToBasicAttack() {return m_ImmuneToBasicAttackCount > 0;}; + const bool GetImmuneToDamageOverTime() {return m_ImmuneToDamageOverTimeCount > 0;}; + const bool GetImmuneToKnockback() {return m_ImmuneToKnockbackCount > 0;}; + const bool GetImmuneToInterrupt() {return m_ImmuneToInterruptCount > 0;}; + const bool GetImmuneToSpeed() {return m_ImmuneToSpeedCount > 0;}; + const bool GetImmuneToImaginationGain() {return m_ImmuneToImaginationGainCount > 0;}; + const bool GetImmuneToImaginationLoss() {return m_ImmuneToImaginationLossCount > 0;}; + const bool GetImmuneToQuickbuildInterrupt() {return m_ImmuneToQuickbuildInterruptCount > 0;}; + const bool GetImmuneToPullToPoint() {return m_ImmuneToPullToPointCount > 0;}; /** * Utility to reset all stats to the default stats based on items and completed missions @@ -414,6 +435,28 @@ public: */ void AddOnHitCallback(const std::function& callback); + /** + * Pushes a faction back to the list of factions. + * @param value Faction to add to list. + * + * This method should only be used for testing. Use AddFaction(int32_t, bool) for adding a faction properly. + */ + void AddFactionNoLookup(int32_t faction) { m_FactionIDs.push_back(faction); }; + + /** + * Notify subscribed scripts of Damage actions. + * + * @param attacker The attacking Entity + * @param damage The amount of damage that was done + */ + void NotifySubscribers(Entity* attacker, uint32_t damage); + + void Subscribe(LWOOBJID scriptObjId, CppScripts::Script* scriptToAdd); + void Unsubscribe(LWOOBJID scriptObjId); + + // handle hardcode mode drops + void DoHardcoreModeDrops(const LWOOBJID source); + private: /** * Whether or not the health should be serialized @@ -470,11 +513,6 @@ private: */ uint32_t m_AttacksToBlock; - /** - * The layers of immunity this entity has left - */ - int32_t m_ImmuneStacks; - /** * The amount of damage that should be reduced from every attack */ @@ -508,7 +546,7 @@ private: /** * Whether this entity has bricks flying out when smashed (causes the client to look up the files) */ - bool m_HasBricks; + bool m_IsModuleAssembly; /** * The rate at which bricks fly out when smashed @@ -549,6 +587,24 @@ private: * The list of callbacks that will be called when this entity gets hit */ std::vector> m_OnHitCallbacks; + + /** + * The list of scripts subscribed to this components actions + */ + std::map m_SubscribedScripts; + + /** + * status immunity counters + */ + uint32_t m_ImmuneToBasicAttackCount; + uint32_t m_ImmuneToDamageOverTimeCount; + uint32_t m_ImmuneToKnockbackCount; + uint32_t m_ImmuneToInterruptCount; + uint32_t m_ImmuneToSpeedCount; + uint32_t m_ImmuneToImaginationGainCount; + uint32_t m_ImmuneToImaginationLossCount; + uint32_t m_ImmuneToQuickbuildInterruptCount; + uint32_t m_ImmuneToPullToPointCount; }; #endif // DESTROYABLECOMPONENT_H diff --git a/dGame/dComponents/DonationVendorComponent.cpp b/dGame/dComponents/DonationVendorComponent.cpp new file mode 100644 index 00000000..f19ba9b7 --- /dev/null +++ b/dGame/dComponents/DonationVendorComponent.cpp @@ -0,0 +1,50 @@ +#include "DonationVendorComponent.h" +#include "Database.h" + +DonationVendorComponent::DonationVendorComponent(Entity* parent) : VendorComponent(parent) { + //LoadConfigData + m_PercentComplete = 0.0; + m_TotalDonated = 0; + m_TotalRemaining = 0; + + // custom attribute to calculate other values + m_Goal = m_Parent->GetVar(u"donationGoal"); + if (m_Goal == 0) m_Goal = INT32_MAX; + + // Default to the nexus tower jawbox activity and setup settings + m_ActivityId = m_Parent->GetVar(u"activityID"); + if ((m_ActivityId == 0) || (m_ActivityId == 117)) { + m_ActivityId = 117; + m_PercentComplete = 1.0; + m_TotalDonated = INT32_MAX; + m_TotalRemaining = 0; + m_Goal = INT32_MAX; + return; + } + + std::unique_ptr query(Database::CreatePreppedStmt("SELECT SUM(primaryScore) as donation_total FROM leaderboard WHERE game_id = ?;")); + query->setInt(1, m_ActivityId); + std::unique_ptr donation_total(query->executeQuery()); + if (donation_total->next()) m_TotalDonated = donation_total->getInt("donation_total"); + m_TotalRemaining = m_Goal - m_TotalDonated; + m_PercentComplete = m_TotalDonated/static_cast(m_Goal); +} + +void DonationVendorComponent::SubmitDonation(uint32_t count) { + if (count <= 0 && ((m_TotalDonated + count) > 0)) return; + m_TotalDonated += count; + m_TotalRemaining = m_Goal - m_TotalDonated; + m_PercentComplete = m_TotalDonated/static_cast(m_Goal); + m_DirtyDonationVendor = true; +} + +void DonationVendorComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { + VendorComponent::Serialize(outBitStream, bIsInitialUpdate, flags); + outBitStream->Write(bIsInitialUpdate || m_DirtyDonationVendor); + if (bIsInitialUpdate || m_DirtyDonationVendor) { + outBitStream->Write(m_PercentComplete); + outBitStream->Write(m_TotalDonated); + outBitStream->Write(m_TotalRemaining); + if (!bIsInitialUpdate) m_DirtyDonationVendor = false; + } +} diff --git a/dGame/dComponents/DonationVendorComponent.h b/dGame/dComponents/DonationVendorComponent.h new file mode 100644 index 00000000..6c706bf9 --- /dev/null +++ b/dGame/dComponents/DonationVendorComponent.h @@ -0,0 +1,27 @@ +#ifndef __DONATIONVENDORCOMPONENT__H__ +#define __DONATIONVENDORCOMPONENT__H__ + +#include "VendorComponent.h" +#include "eReplicaComponentType.h" + +class Entity; + +class DonationVendorComponent final : public VendorComponent { +public: + inline static const eReplicaComponentType ComponentType = eReplicaComponentType::DONATION_VENDOR; + DonationVendorComponent(Entity* parent); + void Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags); + uint32_t GetActivityID() {return m_ActivityId;}; + void SubmitDonation(uint32_t count); + +private: + bool m_DirtyDonationVendor = false; + float m_PercentComplete = 0.0; + int32_t m_TotalDonated = 0; + int32_t m_TotalRemaining = 0; + uint32_t m_ActivityId = 0; + int32_t m_Goal = 0; +}; + + +#endif //!__DONATIONVENDORCOMPONENT__H__ diff --git a/dGame/dComponents/InventoryComponent.cpp b/dGame/dComponents/InventoryComponent.cpp index 930ace8d..8689c0bc 100644 --- a/dGame/dComponents/InventoryComponent.cpp +++ b/dGame/dComponents/InventoryComponent.cpp @@ -27,6 +27,16 @@ #include "dConfig.h" #include "eItemType.h" #include "eUnequippableActiveType.h" +#include "CppScripts.h" +#include "eMissionTaskType.h" +#include "eStateChangeType.h" +#include "eUseItemResponse.h" + +#include "CDComponentsRegistryTable.h" +#include "CDInventoryComponentTable.h" +#include "CDScriptComponentTable.h" +#include "CDObjectSkillsTable.h" +#include "CDSkillBehaviorTable.h" InventoryComponent::InventoryComponent(Entity* parent, tinyxml2::XMLDocument* document) : Component(parent) { this->m_Dirty = true; @@ -45,10 +55,10 @@ InventoryComponent::InventoryComponent(Entity* parent, tinyxml2::XMLDocument* do return; } - auto* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - const auto componentId = compRegistryTable->GetByIDAndType(lot, COMPONENT_TYPE_INVENTORY); + auto* compRegistryTable = CDClientManager::Instance().GetTable(); + const auto componentId = compRegistryTable->GetByIDAndType(lot, eReplicaComponentType::INVENTORY); - auto* inventoryComponentTable = CDClientManager::Instance()->GetTable("InventoryComponent"); + auto* inventoryComponentTable = CDClientManager::Instance().GetTable(); auto items = inventoryComponentTable->Query([=](const CDInventoryComponent entry) { return entry.id == componentId; }); auto slot = 0u; @@ -106,6 +116,9 @@ Inventory* InventoryComponent::GetInventory(const eInventoryType type) { case eInventoryType::VENDOR_BUYBACK: size = 27u; break; + case eInventoryType::DONATION: + size = 24u; + break; default: break; } @@ -179,7 +192,7 @@ void InventoryComponent::AddItem( inventoryType = Inventory::FindInventoryTypeForLot(lot); } - auto* missions = static_cast(this->m_Parent->GetComponent(COMPONENT_TYPE_MISSION)); + auto* missions = static_cast(this->m_Parent->GetComponent(eReplicaComponentType::MISSION)); auto* inventory = GetInventory(inventoryType); @@ -195,7 +208,7 @@ void InventoryComponent::AddItem( auto* item = new Item(lot, inventory, slot, count, config, parent, showFlyingLoot, isModMoveAndEquip, subKey, bound, lootSourceType); if (missions != nullptr && !IsTransferInventory(inventoryType)) { - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, lot, LWOOBJID_EMPTY, "", count, IsTransferInventory(inventorySourceType)); + missions->Progress(eMissionTaskType::GATHER, lot, LWOOBJID_EMPTY, "", count, IsTransferInventory(inventorySourceType)); } return; @@ -209,9 +222,11 @@ void InventoryComponent::AddItem( auto stack = static_cast(info.stackSize); + bool isBrick = inventoryType == eInventoryType::BRICKS || (stack == 0 && info.itemType == 1); + // info.itemType of 1 is item type brick - if (inventoryType == eInventoryType::BRICKS || (stack == 0 && info.itemType == 1)) { - stack = 999; + if (isBrick) { + stack = UINT32_MAX; } else if (stack == 0) { stack = 1; } @@ -232,7 +247,8 @@ void InventoryComponent::AddItem( } } - while (left > 0) { + // If we have some leftover and we aren't bricks, make a new stack + while (left > 0 && (!isBrick || (isBrick && !existing))) { const auto size = std::min(left, stack); left -= size; @@ -280,7 +296,7 @@ void InventoryComponent::AddItem( } if (missions != nullptr && !IsTransferInventory(inventoryType)) { - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, lot, LWOOBJID_EMPTY, "", count - outOfSpace, IsTransferInventory(inventorySourceType)); + missions->Progress(eMissionTaskType::GATHER, lot, LWOOBJID_EMPTY, "", count - outOfSpace, IsTransferInventory(inventorySourceType)); } } @@ -327,7 +343,9 @@ void InventoryComponent::MoveItemToInventory(Item* item, const eInventoryType in const auto lot = item->GetLot(); - if (item->GetConfig().empty() && !item->GetBound() || (item->GetBound() && item->GetInfo().isBOP)) { + const auto subkey = item->GetSubKey(); + + if (subkey == LWOOBJID_EMPTY && item->GetConfig().empty() && (!item->GetBound() || (item->GetBound() && item->GetInfo().isBOP))) { auto left = std::min(count, origin->GetLotCount(lot)); while (left > 0) { @@ -343,7 +361,7 @@ void InventoryComponent::MoveItemToInventory(Item* item, const eInventoryType in left -= delta; - AddItem(lot, delta, eLootSourceType::LOOT_SOURCE_NONE, inventory, {}, LWOOBJID_EMPTY, showFlyingLot, isModMoveAndEquip, LWOOBJID_EMPTY, origin->GetType(), 0, false, preferredSlot); + AddItem(lot, delta, eLootSourceType::NONE, inventory, {}, LWOOBJID_EMPTY, showFlyingLot, isModMoveAndEquip, LWOOBJID_EMPTY, origin->GetType(), 0, false, preferredSlot); item->SetCount(item->GetCount() - delta, false, false); @@ -358,7 +376,7 @@ void InventoryComponent::MoveItemToInventory(Item* item, const eInventoryType in const auto delta = std::min(item->GetCount(), count); - AddItem(lot, delta, eLootSourceType::LOOT_SOURCE_NONE, inventory, config, LWOOBJID_EMPTY, showFlyingLot, isModMoveAndEquip, LWOOBJID_EMPTY, origin->GetType(), 0, item->GetBound(), preferredSlot); + AddItem(lot, delta, eLootSourceType::NONE, inventory, config, LWOOBJID_EMPTY, showFlyingLot, isModMoveAndEquip, subkey, origin->GetType(), 0, item->GetBound(), preferredSlot); item->SetCount(item->GetCount() - delta, false, false); } @@ -367,7 +385,7 @@ void InventoryComponent::MoveItemToInventory(Item* item, const eInventoryType in if (missionComponent != nullptr) { if (IsTransferInventory(inventory)) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, lot, LWOOBJID_EMPTY, "", -static_cast(count)); + missionComponent->Progress(eMissionTaskType::GATHER, lot, LWOOBJID_EMPTY, "", -static_cast(count)); } } } @@ -605,16 +623,17 @@ void InventoryComponent::UpdateXml(tinyxml2::XMLDocument* document) { return; } - std::vector inventories; + std::vector inventoriesToSave; + // Need to prevent some transfer inventories from being saved for (const auto& pair : this->m_Inventories) { auto* inventory = pair.second; - if (inventory->GetType() == VENDOR_BUYBACK) { + if (inventory->GetType() == VENDOR_BUYBACK || inventory->GetType() == eInventoryType::MODELS_IN_BBB) { continue; } - inventories.push_back(inventory); + inventoriesToSave.push_back(inventory); } inventoryElement->SetAttribute("csl", m_Consumable); @@ -629,7 +648,7 @@ void InventoryComponent::UpdateXml(tinyxml2::XMLDocument* document) { bags->DeleteChildren(); - for (const auto* inventory : inventories) { + for (const auto* inventory : inventoriesToSave) { auto* bag = document->NewElement("b"); bag->SetAttribute("t", inventory->GetType()); @@ -648,7 +667,7 @@ void InventoryComponent::UpdateXml(tinyxml2::XMLDocument* document) { items->DeleteChildren(); - for (auto* inventory : inventories) { + for (auto* inventory : inventoriesToSave) { if (inventory->GetSize() == 0) { continue; } @@ -810,18 +829,26 @@ void InventoryComponent::EquipItem(Item* item, const bool skipChecks) { if (character != nullptr && !skipChecks) { // Hacky proximity rocket if (item->GetLot() == 6416) { - const auto rocketLauchPads = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_ROCKET_LAUNCH); + const auto rocketLauchPads = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::ROCKET_LAUNCH); const auto position = m_Parent->GetPosition(); - for (auto* lauchPad : rocketLauchPads) { - if (Vector3::DistanceSquared(lauchPad->GetPosition(), position) > 13 * 13) continue; + for (auto* launchPad : rocketLauchPads) { + if (!launchPad) continue; + + auto prereq = launchPad->GetVarAsString(u"rocketLaunchPreCondition"); + if (!prereq.empty()) { + PreconditionExpression expression(prereq); + if (!expression.Check(m_Parent)) continue; + } + + if (Vector3::DistanceSquared(launchPad->GetPosition(), position) > 13 * 13) continue; auto* characterComponent = m_Parent->GetComponent(); if (characterComponent != nullptr) characterComponent->SetLastRocketItemID(item->GetId()); - lauchPad->OnUse(m_Parent); + launchPad->OnUse(m_Parent); break; } @@ -834,9 +861,9 @@ void InventoryComponent::EquipItem(Item* item, const bool skipChecks) { const auto type = static_cast(item->GetInfo().itemType); - if (!building && (item->GetLot() == 6086 || type == eItemType::ITEM_TYPE_LOOT_MODEL || type == eItemType::ITEM_TYPE_VEHICLE)) return; + if (!building && (item->GetLot() == 6086 || type == eItemType::LOOT_MODEL || type == eItemType::VEHICLE)) return; - if (type != eItemType::ITEM_TYPE_LOOT_MODEL && type != eItemType::ITEM_TYPE_MODEL) { + if (type != eItemType::LOOT_MODEL && type != eItemType::MODEL) { if (!item->GetBound() && !item->GetPreconditionExpression()->Check(m_Parent)) { return; } @@ -861,7 +888,9 @@ void InventoryComponent::EquipItem(Item* item, const bool skipChecks) { AddItemSkills(item->GetLot()); - EntityManager::Instance()->SerializeEntity(m_Parent); + EquipScripts(item); + + Game::entityManager->SerializeEntity(m_Parent); } void InventoryComponent::UnEquipItem(Item* item) { @@ -889,12 +918,45 @@ void InventoryComponent::UnEquipItem(Item* item) { PurgeProxies(item); - EntityManager::Instance()->SerializeEntity(m_Parent); + UnequipScripts(item); + + Game::entityManager->SerializeEntity(m_Parent); // Trigger property event if (PropertyManagementComponent::Instance() != nullptr && item->GetCount() > 0 && Inventory::FindInventoryTypeForLot(item->GetLot()) == MODELS) { PropertyManagementComponent::Instance()->GetParent()->OnZonePropertyModelRemovedWhileEquipped(m_Parent); - dZoneManager::Instance()->GetZoneControlObject()->OnZonePropertyModelRemovedWhileEquipped(m_Parent); + Game::zoneManager->GetZoneControlObject()->OnZonePropertyModelRemovedWhileEquipped(m_Parent); + } +} + + +void InventoryComponent::EquipScripts(Item* equippedItem) { + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + if (!compRegistryTable) return; + int32_t scriptComponentID = compRegistryTable->GetByIDAndType(equippedItem->GetLot(), eReplicaComponentType::SCRIPT, -1); + if (scriptComponentID > -1) { + CDScriptComponentTable* scriptCompTable = CDClientManager::Instance().GetTable(); + CDScriptComponent scriptCompData = scriptCompTable->GetByID(scriptComponentID); + auto* itemScript = CppScripts::GetScript(m_Parent, scriptCompData.script_name); + if (!itemScript) { + Game::logger->Log("InventoryComponent", "null script?"); + } + itemScript->OnFactionTriggerItemEquipped(m_Parent, equippedItem->GetId()); + } +} + +void InventoryComponent::UnequipScripts(Item* unequippedItem) { + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + if (!compRegistryTable) return; + int32_t scriptComponentID = compRegistryTable->GetByIDAndType(unequippedItem->GetLot(), eReplicaComponentType::SCRIPT, -1); + if (scriptComponentID > -1) { + CDScriptComponentTable* scriptCompTable = CDClientManager::Instance().GetTable(); + CDScriptComponent scriptCompData = scriptCompTable->GetByID(scriptComponentID); + auto* itemScript = CppScripts::GetScript(m_Parent, scriptCompData.script_name); + if (!itemScript) { + Game::logger->Log("InventoryComponent", "null script?"); + } + itemScript->OnFactionTriggerItemUnequipped(m_Parent, unequippedItem->GetId()); } } @@ -909,7 +971,7 @@ void InventoryComponent::HandlePossession(Item* item) { if (possessorComponent->GetIsDismounting()) return; // Check to see if we are already mounting something - auto* currentlyPossessedEntity = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); + auto* currentlyPossessedEntity = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); auto currentlyPossessedItem = possessorComponent->GetMountItemID(); if (currentlyPossessedItem) { @@ -917,7 +979,7 @@ void InventoryComponent::HandlePossession(Item* item) { return; } - GameMessages::SendSetStunned(m_Parent->GetObjectID(), eStunState::PUSH, m_Parent->GetSystemAddress(), LWOOBJID_EMPTY, true, false, true, false, false, false, false, true, true, true, true, true, true, true, true, true); + GameMessages::SendSetStunned(m_Parent->GetObjectID(), eStateChangeType::PUSH, m_Parent->GetSystemAddress(), LWOOBJID_EMPTY, true, false, true, false, false, false, false, true, true, true, true, true, true, true, true, true); // Set the mount Item ID so that we know what were handling possessorComponent->SetMountItemID(item->GetId()); @@ -932,26 +994,15 @@ void InventoryComponent::HandlePossession(Item* item) { info.rot = startRotation; info.spawnerID = m_Parent->GetObjectID(); - auto* mount = EntityManager::Instance()->CreateEntity(info, nullptr, m_Parent); + auto* mount = Game::entityManager->CreateEntity(info, nullptr, m_Parent); // Check to see if the mount is a vehicle, if so, flip it auto* vehicleComponent = mount->GetComponent(); - if (vehicleComponent) { - auto angles = startRotation.GetEulerAngles(); - // Make it right side up - angles.x -= PI; - // Make it going in the direction of the player - angles.y -= PI; - startRotation = NiQuaternion::FromEulerAngles(angles); - mount->SetRotation(startRotation); - // We're pod racing now - characterComponent->SetIsRacing(true); - } + if (vehicleComponent) characterComponent->SetIsRacing(true); // Setup the destroyable stats auto* destroyableComponent = mount->GetComponent(); if (destroyableComponent) { - destroyableComponent->SetIsSmashable(false); destroyableComponent->SetIsImmune(true); } @@ -968,9 +1019,9 @@ void InventoryComponent::HandlePossession(Item* item) { GameMessages::SendSetJetPackMode(m_Parent, false); // Make it go to the client - EntityManager::Instance()->ConstructEntity(mount); + Game::entityManager->ConstructEntity(mount); // Update the possessor - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); // have to unlock the input so it vehicle can be driven if (vehicleComponent) GameMessages::SendVehicleUnlockInput(mount->GetObjectID(), false, m_Parent->GetSystemAddress()); @@ -985,6 +1036,7 @@ void InventoryComponent::ApplyBuff(Item* item) const { } } +// TODO Something needs to send the remove buff GameMessage as well when it is unequipping items that would remove buffs. void InventoryComponent::RemoveBuff(Item* item) const { const auto buffs = FindBuffs(item, false); @@ -1031,7 +1083,7 @@ void InventoryComponent::PopEquippedItems() { destroyableComponent->SetHealth(static_cast(destroyableComponent->GetMaxHealth())); destroyableComponent->SetArmor(static_cast(destroyableComponent->GetMaxArmor())); destroyableComponent->SetImagination(static_cast(destroyableComponent->GetMaxImagination())); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } m_Dirty = true; @@ -1153,20 +1205,20 @@ void InventoryComponent::RemoveItemSkills(const LOT lot) { } } -void InventoryComponent::TriggerPassiveAbility(PassiveAbilityTrigger trigger) { +void InventoryComponent::TriggerPassiveAbility(PassiveAbilityTrigger trigger, Entity* target) { for (auto* set : m_Itemsets) { - set->TriggerPassiveAbility(trigger); + set->TriggerPassiveAbility(trigger, target); } } -bool InventoryComponent::HasAnyPassive(const std::vector& passiveIDs, int32_t equipmentRequirement) const { +bool InventoryComponent::HasAnyPassive(const std::vector& passiveIDs, int32_t equipmentRequirement) const { for (auto* set : m_Itemsets) { if (set->GetEquippedCount() < equipmentRequirement) { continue; } // Check if the set has any of the passive abilities - if (std::find(passiveIDs.begin(), passiveIDs.end(), static_cast(set->GetID())) != passiveIDs.end()) { + if (std::find(passiveIDs.begin(), passiveIDs.end(), static_cast(set->GetID())) != passiveIDs.end()) { return true; } } @@ -1197,7 +1249,7 @@ void InventoryComponent::SpawnPet(Item* item) { auto destroyableComponent = m_Parent->GetComponent(); if (Game::config->GetValue("pets_take_imagination") == "1" && destroyableComponent && destroyableComponent->GetImagination() <= 0) { - GameMessages::SendUseItemRequirementsResponse(m_Parent->GetObjectID(), m_Parent->GetSystemAddress(), UseItemResponse::NoImaginationForPet); + GameMessages::SendUseItemRequirementsResponse(m_Parent->GetObjectID(), m_Parent->GetSystemAddress(), eUseItemResponse::NoImaginationForPet); return; } @@ -1207,7 +1259,7 @@ void InventoryComponent::SpawnPet(Item* item) { info.rot = NiQuaternion::IDENTITY; info.spawnerID = m_Parent->GetObjectID(); - auto* pet = EntityManager::Instance()->CreateEntity(info); + auto* pet = Game::entityManager->CreateEntity(info); auto* petComponent = pet->GetComponent(); @@ -1215,7 +1267,7 @@ void InventoryComponent::SpawnPet(Item* item) { petComponent->Activate(item); } - EntityManager::Instance()->ConstructEntity(pet); + Game::entityManager->ConstructEntity(pet); } void InventoryComponent::SetDatabasePet(LWOOBJID id, const DatabasePet& data) { @@ -1242,15 +1294,15 @@ void InventoryComponent::RemoveDatabasePet(LWOOBJID id) { BehaviorSlot InventoryComponent::FindBehaviorSlot(const eItemType type) { switch (type) { - case eItemType::ITEM_TYPE_HAT: + case eItemType::HAT: return BehaviorSlot::Head; - case eItemType::ITEM_TYPE_NECK: + case eItemType::NECK: return BehaviorSlot::Neck; - case eItemType::ITEM_TYPE_LEFT_HAND: + case eItemType::LEFT_HAND: return BehaviorSlot::Offhand; - case eItemType::ITEM_TYPE_RIGHT_HAND: + case eItemType::RIGHT_HAND: return BehaviorSlot::Primary; - case eItemType::ITEM_TYPE_CONSUMABLE: + case eItemType::CONSUMABLE: return BehaviorSlot::Consumable; default: return BehaviorSlot::Invalid; @@ -1258,11 +1310,11 @@ BehaviorSlot InventoryComponent::FindBehaviorSlot(const eItemType type) { } bool InventoryComponent::IsTransferInventory(eInventoryType type) { - return type == VENDOR_BUYBACK || type == VAULT_ITEMS || type == VAULT_MODELS || type == TEMP_ITEMS || type == TEMP_MODELS; + return type == VENDOR_BUYBACK || type == VAULT_ITEMS || type == VAULT_MODELS || type == TEMP_ITEMS || type == TEMP_MODELS || type == MODELS_IN_BBB; } uint32_t InventoryComponent::FindSkill(const LOT lot) { - auto* table = CDClientManager::Instance()->GetTable("ObjectSkills"); + auto* table = CDClientManager::Instance().GetTable(); const auto results = table->Query([=](const CDObjectSkills& entry) { return entry.objectTemplate == static_cast(lot); @@ -1280,14 +1332,14 @@ uint32_t InventoryComponent::FindSkill(const LOT lot) { std::vector InventoryComponent::FindBuffs(Item* item, bool castOnEquip) const { std::vector buffs; if (item == nullptr) return buffs; - auto* table = CDClientManager::Instance()->GetTable("ObjectSkills"); - auto* behaviors = CDClientManager::Instance()->GetTable("SkillBehavior"); + auto* table = CDClientManager::Instance().GetTable(); + auto* behaviors = CDClientManager::Instance().GetTable(); const auto results = table->Query([=](const CDObjectSkills& entry) { return entry.objectTemplate == static_cast(item->GetLot()); }); - auto* missions = static_cast(m_Parent->GetComponent(COMPONENT_TYPE_MISSION)); + auto* missions = static_cast(m_Parent->GetComponent(eReplicaComponentType::MISSION)); for (const auto& result : results) { if (result.castOnType == 1) { @@ -1300,7 +1352,7 @@ std::vector InventoryComponent::FindBuffs(Item* item, bool castOnEquip } if (missions != nullptr && castOnEquip) { - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_SKILL, result.skillID); + missions->Progress(eMissionTaskType::USE_SKILL, result.skillID); } // If item is not a proxy, add its buff to the added buffs. @@ -1324,7 +1376,7 @@ void InventoryComponent::SetNPCItems(const std::vector& items) { UpdateSlot(info.equipLocation, { id, static_cast(item), 1, slot++ }, true); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } InventoryComponent::~InventoryComponent() { diff --git a/dGame/dComponents/InventoryComponent.h b/dGame/dComponents/InventoryComponent.h index 394cb801..801f9f51 100644 --- a/dGame/dComponents/InventoryComponent.h +++ b/dGame/dComponents/InventoryComponent.h @@ -17,8 +17,11 @@ #include "DatabasePet.h" #include "Component.h" #include "ItemSetPassiveAbility.h" -#include "ItemSetPassiveAbilityID.h" +#include "eItemSetPassiveAbilityID.h" #include "PossessorComponent.h" +#include "eInventoryType.h" +#include "eReplicaComponentType.h" +#include "eLootSourceType.h" class Entity; class ItemSet; @@ -35,7 +38,7 @@ enum class eItemType : int32_t; class InventoryComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_INVENTORY; + static const eReplicaComponentType ComponentType = eReplicaComponentType::INVENTORY; explicit InventoryComponent(Entity* parent, tinyxml2::XMLDocument* document = nullptr); void Update(float deltaTime) override; @@ -97,7 +100,7 @@ public: void AddItem( LOT lot, uint32_t count, - eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE, + eLootSourceType lootSourceType = eLootSourceType::NONE, eInventoryType inventoryType = INVALID, const std::vector& config = {}, LWOOBJID parent = LWOOBJID_EMPTY, @@ -282,7 +285,7 @@ public: * Triggers one of the passive abilities from the equipped item set * @param trigger the trigger to fire */ - void TriggerPassiveAbility(PassiveAbilityTrigger trigger); + void TriggerPassiveAbility(PassiveAbilityTrigger trigger, Entity* target = nullptr); /** * Returns if the entity has any of the passed passive abilities equipped @@ -290,7 +293,7 @@ public: * @param equipmentRequirement the number of equipment required to be allowed to have the ability * @return if the entity has any of the passed passive abilities equipped */ - bool HasAnyPassive(const std::vector& passiveIDs, int32_t equipmentRequirement) const; + bool HasAnyPassive(const std::vector& passiveIDs, int32_t equipmentRequirement) const; /** * Despawns the currently active pet, if any @@ -351,6 +354,20 @@ public: */ static uint32_t FindSkill(LOT lot); + /** + * Call this when you equip an item. This calls OnFactionTriggerItemEquipped for any scripts found on the items. + * + * @param equippedItem The item script to lookup and call equip on + */ + void EquipScripts(Item* equippedItem); + + /** + * Call this when you unequip an item. This calls OnFactionTriggerItemUnequipped for any scripts found on the items. + * + * @param unequippedItem The item script to lookup and call unequip on + */ + void UnequipScripts(Item* unequippedItem); + ~InventoryComponent() override; private: diff --git a/dGame/dComponents/LUPExhibitComponent.cpp b/dGame/dComponents/LUPExhibitComponent.cpp index 7b8c85ba..deb3cc8c 100644 --- a/dGame/dComponents/LUPExhibitComponent.cpp +++ b/dGame/dComponents/LUPExhibitComponent.cpp @@ -35,7 +35,7 @@ void LUPExhibitComponent::NextExhibit() { m_Exhibit = m_Exhibits[m_ExhibitIndex]; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void LUPExhibitComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, uint32_t& flags) { diff --git a/dGame/dComponents/LUPExhibitComponent.h b/dGame/dComponents/LUPExhibitComponent.h index 2d128f6c..587d1b2f 100644 --- a/dGame/dComponents/LUPExhibitComponent.h +++ b/dGame/dComponents/LUPExhibitComponent.h @@ -2,6 +2,7 @@ #include "Component.h" #include "Entity.h" +#include "eReplicaComponentType.h" /** * Component that handles the LOT that is shown in the LUP exhibit in the LUP world. Works by setting a timer and @@ -10,7 +11,7 @@ class LUPExhibitComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_EXHIBIT; + static const eReplicaComponentType ComponentType = eReplicaComponentType::EXHIBIT; LUPExhibitComponent(Entity* parent); ~LUPExhibitComponent(); diff --git a/dGame/dComponents/LevelProgressionComponent.cpp b/dGame/dComponents/LevelProgressionComponent.cpp index ee3cfc6b..8163e736 100644 --- a/dGame/dComponents/LevelProgressionComponent.cpp +++ b/dGame/dComponents/LevelProgressionComponent.cpp @@ -4,9 +4,13 @@ #include "CharacterComponent.h" #include "tinyxml2.h" +#include "CDRewardsTable.h" + LevelProgressionComponent::LevelProgressionComponent(Entity* parent) : Component(parent) { m_Parent = parent; m_Level = 1; + m_SpeedBase = 500.0f; + m_CharacterVersion = eCharacterVersion::LIVE; } void LevelProgressionComponent::UpdateXml(tinyxml2::XMLDocument* doc) { @@ -16,7 +20,8 @@ void LevelProgressionComponent::UpdateXml(tinyxml2::XMLDocument* doc) { return; } level->SetAttribute("l", m_Level); - + level->SetAttribute("sb", m_SpeedBase); + level->SetAttribute("cv", static_cast(m_CharacterVersion)); } void LevelProgressionComponent::LoadFromXml(tinyxml2::XMLDocument* doc) { @@ -26,7 +31,10 @@ void LevelProgressionComponent::LoadFromXml(tinyxml2::XMLDocument* doc) { return; } level->QueryAttribute("l", &m_Level); - + level->QueryAttribute("sb", &m_SpeedBase); + uint32_t characterVersion; + level->QueryAttribute("cv", &characterVersion); + m_CharacterVersion = static_cast(characterVersion); } void LevelProgressionComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { @@ -36,7 +44,7 @@ void LevelProgressionComponent::Serialize(RakNet::BitStream* outBitStream, bool } void LevelProgressionComponent::HandleLevelUp() { - auto* rewardsTable = CDClientManager::Instance()->GetTable("Rewards"); + auto* rewardsTable = CDClientManager::Instance().GetTable(); const auto& rewards = rewardsTable->GetByLevelID(m_Level); bool rewardingItem = rewards.size() > 0; @@ -51,7 +59,7 @@ void LevelProgressionComponent::HandleLevelUp() { for (auto* reward : rewards) { switch (reward->rewardType) { case 0: - inventoryComponent->AddItem(reward->value, reward->count, eLootSourceType::LOOT_SOURCE_LEVEL_REWARD); + inventoryComponent->AddItem(reward->value, reward->count, eLootSourceType::LEVEL_REWARD); break; case 4: { @@ -60,7 +68,8 @@ void LevelProgressionComponent::HandleLevelUp() { } break; case 9: - controllablePhysicsComponent->SetSpeedMultiplier(static_cast(reward->value) / 500.0f); + SetSpeedBase(static_cast(reward->value) ); + controllablePhysicsComponent->SetSpeedMultiplier(GetSpeedBase() / 500.0f); break; case 11: case 12: @@ -72,3 +81,9 @@ void LevelProgressionComponent::HandleLevelUp() { // Tell the client we have finished sending level rewards. if (rewardingItem) GameMessages::NotifyLevelRewards(m_Parent->GetObjectID(), m_Parent->GetSystemAddress(), m_Level, !rewardingItem); } + +void LevelProgressionComponent::SetRetroactiveBaseSpeed(){ + if (m_Level >= 20) m_SpeedBase = 525.0f; + auto* controllablePhysicsComponent = m_Parent->GetComponent(); + if (controllablePhysicsComponent) controllablePhysicsComponent->SetSpeedMultiplier(m_SpeedBase / 500.0f); +} diff --git a/dGame/dComponents/LevelProgressionComponent.h b/dGame/dComponents/LevelProgressionComponent.h index 53f6693e..06908a81 100644 --- a/dGame/dComponents/LevelProgressionComponent.h +++ b/dGame/dComponents/LevelProgressionComponent.h @@ -3,14 +3,17 @@ #include "Entity.h" #include "GameMessages.h" #include "Component.h" +#include "eCharacterVersion.h" +#include "eReplicaComponentType.h" /** * Component that handles level progression and serilization. * */ + class LevelProgressionComponent : public Component { public: - static const uint32_t ComponentType = eReplicaComponentType::COMPONENT_TYPE_LEVEL_PROGRESSION; + static const eReplicaComponentType ComponentType = eReplicaComponentType::LEVEL_PROGRESSION; /** * Constructor for this component @@ -44,11 +47,40 @@ public: */ void SetLevel(uint32_t level) { m_Level = level; m_DirtyLevelInfo = true; } + /** + * Gets the current Speed Base of the entity + * @return the current Speed Base of the entity + */ + const uint32_t GetSpeedBase() const { return m_SpeedBase; } + + /** + * Sets the Speed Base of the entity + * @param SpeedBase the Speed Base to set + */ + void SetSpeedBase(uint32_t SpeedBase) { m_SpeedBase = SpeedBase; } + /** * Gives the player rewards for the last level that they leveled up from */ void HandleLevelUp(); + /** + * Gets the current Character Version of the entity + * @return the current Character Version of the entity + */ + const eCharacterVersion GetCharacterVersion() const { return m_CharacterVersion; } + + /** + * Sets the Character Version of the entity + * @param CharacterVersion the Character Version to set + */ + void SetCharacterVersion(eCharacterVersion CharacterVersion) { m_CharacterVersion = CharacterVersion; } + + /** + * Set the Base Speed retroactively of the entity + */ + void SetRetroactiveBaseSpeed(); + private: /** * whether the level is dirty @@ -59,4 +91,15 @@ private: * Level of the entity */ uint32_t m_Level; + + /** + * The base speed of the entity + */ + float m_SpeedBase; + + /** + * The Character format version + */ + eCharacterVersion m_CharacterVersion; + }; diff --git a/dGame/dComponents/MissionComponent.cpp b/dGame/dComponents/MissionComponent.cpp index 96f213e5..40411902 100644 --- a/dGame/dComponents/MissionComponent.cpp +++ b/dGame/dComponents/MissionComponent.cpp @@ -13,18 +13,20 @@ #include "InventoryComponent.h" #include "GameMessages.h" #include "Game.h" -#include "AMFFormat.h" +#include "Amf3.h" #include "dZoneManager.h" #include "Mail.h" #include "MissionPrerequisites.h" +#include "AchievementCacheKey.h" +#include "eMissionState.h" // MARK: Mission Component -std::unordered_map> MissionComponent::m_AchievementCache = {}; +std::unordered_map> MissionComponent::m_AchievementCache = {}; //! Initializer MissionComponent::MissionComponent(Entity* parent) : Component(parent) { - m_LastUsedMissionOrderUID = dZoneManager::Instance()->GetUniqueMissionIdStartingValue(); + m_LastUsedMissionOrderUID = Game::zoneManager->GetUniqueMissionIdStartingValue(); } //! Destructor @@ -52,11 +54,11 @@ Mission* MissionComponent::GetMission(const uint32_t missionId) const { } -MissionState MissionComponent::GetMissionState(const uint32_t missionId) const { +eMissionState MissionComponent::GetMissionState(const uint32_t missionId) const { auto* mission = GetMission(missionId); if (mission == nullptr) { - return CanAccept(missionId) ? MissionState::MISSION_STATE_AVAILABLE : MissionState::MISSION_STATE_UNKNOWN; + return CanAccept(missionId) ? eMissionState::AVAILABLE : eMissionState::UNKNOWN; } return mission->GetMissionState(); @@ -142,7 +144,7 @@ void MissionComponent::RemoveMission(uint32_t missionId) { m_Missions.erase(missionId); } -void MissionComponent::Progress(MissionTaskType type, int32_t value, LWOOBJID associate, const std::string& targets, int32_t count, bool ignoreAchievements) { +void MissionComponent::Progress(eMissionTaskType type, int32_t value, LWOOBJID associate, const std::string& targets, int32_t count, bool ignoreAchievements) { for (const auto& pair : m_Missions) { auto* mission = pair.second; @@ -214,7 +216,7 @@ void MissionComponent::ForceProgressTaskType(const uint32_t missionId, const uin } for (auto* element : mission->GetTasks()) { - if (element->GetType() != static_cast(taskType)) continue; + if (element->GetType() != static_cast(taskType)) continue; element->AddProgress(value); } @@ -252,7 +254,7 @@ void MissionComponent::ForceProgressValue(uint32_t missionId, uint32_t taskType, } for (auto* element : mission->GetTasks()) { - if (element->GetType() != static_cast(taskType) || !element->InAllTargets(value)) continue; + if (element->GetType() != static_cast(taskType) || !element->InAllTargets(value)) continue; element->AddProgress(1); } @@ -263,7 +265,7 @@ void MissionComponent::ForceProgressValue(uint32_t missionId, uint32_t taskType, } bool MissionComponent::GetMissionInfo(uint32_t missionId, CDMissions& result) { - auto* missionsTable = CDClientManager::Instance()->GetTable("Missions"); + auto* missionsTable = CDClientManager::Instance().GetTable(); const auto missions = missionsTable->Query([=](const CDMissions& entry) { return entry.id == static_cast(missionId); @@ -280,7 +282,7 @@ bool MissionComponent::GetMissionInfo(uint32_t missionId, CDMissions& result) { #define MISSION_NEW_METHOD -bool MissionComponent::LookForAchievements(MissionTaskType type, int32_t value, bool progress, LWOOBJID associate, const std::string& targets, int32_t count) { +bool MissionComponent::LookForAchievements(eMissionTaskType type, int32_t value, bool progress, LWOOBJID associate, const std::string& targets, int32_t count) { #ifdef MISSION_NEW_METHOD // Query for achievments, using the cache const auto& result = QueryAchievements(type, value, targets); @@ -317,8 +319,8 @@ bool MissionComponent::LookForAchievements(MissionTaskType type, int32_t value, return any; #else - auto* missionTasksTable = CDClientManager::Instance()->GetTable("MissionTasks"); - auto* missionsTable = CDClientManager::Instance()->GetTable("Missions"); + auto* missionTasksTable = CDClientManager::Instance().GetTable(); + auto* missionsTable = CDClientManager::Instance().GetTable(); auto tasks = missionTasksTable->Query([=](const CDMissionTasks& entry) { return entry.taskType == static_cast(type); @@ -389,14 +391,14 @@ bool MissionComponent::LookForAchievements(MissionTaskType type, int32_t value, #endif } -const std::vector& MissionComponent::QueryAchievements(MissionTaskType type, int32_t value, const std::string targets) { +const std::vector& MissionComponent::QueryAchievements(eMissionTaskType type, int32_t value, const std::string targets) { // Create a hash which represent this query for achievements - size_t hash = 0; - GeneralUtils::hash_combine(hash, type); - GeneralUtils::hash_combine(hash, value); - GeneralUtils::hash_combine(hash, targets); + AchievementCacheKey toFind; + toFind.SetType(type); + toFind.SetValue(value); + toFind.SetTargets(targets); - const std::unordered_map>::iterator& iter = m_AchievementCache.find(hash); + const auto& iter = m_AchievementCache.find(toFind); // Check if this query is cached if (iter != m_AchievementCache.end()) { @@ -404,8 +406,8 @@ const std::vector& MissionComponent::QueryAchievements(MissionTaskType } // Find relevent tables - auto* missionTasksTable = CDClientManager::Instance()->GetTable("MissionTasks"); - auto* missionsTable = CDClientManager::Instance()->GetTable("Missions"); + auto* missionTasksTable = CDClientManager::Instance().GetTable(); + auto* missionsTable = CDClientManager::Instance().GetTable(); std::vector result; @@ -447,11 +449,9 @@ const std::vector& MissionComponent::QueryAchievements(MissionTaskType } } } - // Insert into cache - m_AchievementCache.insert_or_assign(hash, result); - - return m_AchievementCache.find(hash)->second; + m_AchievementCache.insert_or_assign(toFind, result); + return m_AchievementCache.find(toFind)->second; } bool MissionComponent::RequiresItem(const LOT lot) { @@ -485,7 +485,7 @@ bool MissionComponent::RequiresItem(const LOT lot) { } for (auto* task : mission->GetTasks()) { - if (task->IsComplete() || task->GetType() != MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION) { + if (task->IsComplete() || task->GetType() != eMissionTaskType::GATHER) { continue; } @@ -497,7 +497,7 @@ bool MissionComponent::RequiresItem(const LOT lot) { } } - const auto required = LookForAchievements(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, lot, false); + const auto required = LookForAchievements(eMissionTaskType::GATHER, lot, false); return required; } diff --git a/dGame/dComponents/MissionComponent.h b/dGame/dComponents/MissionComponent.h index 58185a68..eeaaa726 100644 --- a/dGame/dComponents/MissionComponent.h +++ b/dGame/dComponents/MissionComponent.h @@ -16,15 +16,18 @@ #include "CDClientManager.h" #include "CDMissionsTable.h" #include "Component.h" +#include "eReplicaComponentType.h" + +class AchievementCacheKey; /** * The mission inventory of an entity. Tracks mission state for each mission that can be accepted and allows for - * progression of each of the mission task types (see MissionTaskType). + * progression of each of the mission task types (see eMissionTaskType). */ class MissionComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_MISSION; + static const eReplicaComponentType ComponentType = eReplicaComponentType::MISSION; explicit MissionComponent(Entity* parent); ~MissionComponent() override; @@ -50,7 +53,7 @@ public: * @param missionId the ID of the mission to get the mission state for * @return the mission state of the mission specified by the ID */ - MissionState GetMissionState(uint32_t missionId) const; + eMissionState GetMissionState(uint32_t missionId) const; /** * Checks if the entity has all the requirements for accepting the mission specified by the ID. @@ -91,7 +94,7 @@ public: * @param count the number to progress by, for example the number of items * @param ignoreAchievements do not progress achievements */ - void Progress(MissionTaskType type, int32_t value, LWOOBJID associate = 0, const std::string& targets = "", int32_t count = 1, bool ignoreAchievements = false); + void Progress(eMissionTaskType type, int32_t value, LWOOBJID associate = 0, const std::string& targets = "", int32_t count = 1, bool ignoreAchievements = false); /** * Forces progression for a mission and task, ignoring checks @@ -138,7 +141,7 @@ public: * @param count the number of values to progress by (differs by task type) * @return true if a achievement was accepted, false otherwise */ - bool LookForAchievements(MissionTaskType type, int32_t value, bool progress = true, LWOOBJID associate = LWOOBJID_EMPTY, const std::string& targets = "", int32_t count = 1); + bool LookForAchievements(eMissionTaskType type, int32_t value, bool progress = true, LWOOBJID associate = LWOOBJID_EMPTY, const std::string& targets = "", int32_t count = 1); /** * Checks if there's a mission active that requires the collection of the specified LOT @@ -186,13 +189,13 @@ private: * @param targets optional targets to progress with * @return list of mission IDs (achievements) that can be progressed for the given parameters */ - static const std::vector& QueryAchievements(MissionTaskType type, int32_t value, const std::string targets); + static const std::vector& QueryAchievements(eMissionTaskType type, int32_t value, const std::string targets); /** * As achievements can be hard to query, we here store a list of all the mission IDs that can be unlocked for a * combination of tasks and values, so that they can be easily re-queried later */ - static std::unordered_map> m_AchievementCache; + static std::unordered_map> m_AchievementCache; /** * Order of missions in the UI. This value is incremented by 1 diff --git a/dGame/dComponents/MissionOfferComponent.cpp b/dGame/dComponents/MissionOfferComponent.cpp index 2f2ed0f0..e4c94ebd 100644 --- a/dGame/dComponents/MissionOfferComponent.cpp +++ b/dGame/dComponents/MissionOfferComponent.cpp @@ -14,6 +14,9 @@ #include "dLogger.h" #include "Game.h" #include "MissionPrerequisites.h" +#include "eMissionState.h" + +#include "CDComponentsRegistryTable.h" OfferedMission::OfferedMission(const uint32_t missionId, const bool offersMission, const bool acceptsMission) { this->missionId = missionId; @@ -37,15 +40,15 @@ bool OfferedMission::GetAcceptMission() const { //------------------------ MissionOfferComponent below ------------------------ MissionOfferComponent::MissionOfferComponent(Entity* parent, const LOT parentLot) : Component(parent) { - auto* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); + auto* compRegistryTable = CDClientManager::Instance().GetTable(); - auto value = compRegistryTable->GetByIDAndType(parentLot, COMPONENT_TYPE_MISSION_OFFER, -1); + auto value = compRegistryTable->GetByIDAndType(parentLot, eReplicaComponentType::MISSION_OFFER, -1); if (value != -1) { const uint32_t componentId = value; // Now lookup the missions in the MissionNPCComponent table - auto* missionNpcComponentTable = CDClientManager::Instance()->GetTable("MissionNPCComponent"); + auto* missionNpcComponentTable = CDClientManager::Instance().GetTable(); auto missions = missionNpcComponentTable->Query([=](const CDMissionNPCComponent& entry) { return entry.id == static_cast(componentId); @@ -76,7 +79,7 @@ void MissionOfferComponent::OnUse(Entity* originator) { void MissionOfferComponent::OfferMissions(Entity* entity, const uint32_t specifiedMissionId) { // First, get the entity's MissionComponent. If there is not one, then we cannot offer missions to this entity. - auto* missionComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + auto* missionComponent = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (!missionComponent) { Game::logger->Log("MissionOfferComponent", "Unable to get mission component for Entity %llu", entity->GetObjectID()); @@ -170,10 +173,10 @@ void MissionOfferComponent::OfferMissions(Entity* entity, const uint32_t specifi for (const auto sample : randomMissionPool) { const auto state = missionComponent->GetMissionState(sample); - if (state == MissionState::MISSION_STATE_ACTIVE || - state == MissionState::MISSION_STATE_COMPLETE_ACTIVE || - state == MissionState::MISSION_STATE_READY_TO_COMPLETE || - state == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE || + if (state == eMissionState::ACTIVE || + state == eMissionState::COMPLETE_ACTIVE || + state == eMissionState::READY_TO_COMPLETE || + state == eMissionState::COMPLETE_READY_TO_COMPLETE || sample == specifiedMissionId) { mission = missionComponent->GetMission(sample); diff --git a/dGame/dComponents/MissionOfferComponent.h b/dGame/dComponents/MissionOfferComponent.h index 42aad3c4..6e22ca05 100644 --- a/dGame/dComponents/MissionOfferComponent.h +++ b/dGame/dComponents/MissionOfferComponent.h @@ -10,6 +10,7 @@ #include "Component.h" #include #include +#include "eReplicaComponentType.h" class Entity; @@ -60,7 +61,7 @@ private: */ class MissionOfferComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_MISSION_OFFER; + static const eReplicaComponentType ComponentType = eReplicaComponentType::MISSION_OFFER; MissionOfferComponent(Entity* parent, LOT parentLot); ~MissionOfferComponent() override; diff --git a/dGame/dComponents/ModelComponent.cpp b/dGame/dComponents/ModelComponent.cpp index 8fa085f2..74f614d1 100644 --- a/dGame/dComponents/ModelComponent.cpp +++ b/dGame/dComponents/ModelComponent.cpp @@ -10,7 +10,7 @@ ModelComponent::ModelComponent(Entity* parent) : Component(parent) { void ModelComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { // ItemComponent Serialization. Pets do not get this serialization. - if (!m_Parent->HasComponent(COMPONENT_TYPE_PET)) { + if (!m_Parent->HasComponent(eReplicaComponentType::PET)) { outBitStream->Write1(); outBitStream->Write(m_userModelID != LWOOBJID_EMPTY ? m_userModelID : m_Parent->GetObjectID()); outBitStream->Write(0); diff --git a/dGame/dComponents/ModelComponent.h b/dGame/dComponents/ModelComponent.h index 81342059..b5224869 100644 --- a/dGame/dComponents/ModelComponent.h +++ b/dGame/dComponents/ModelComponent.h @@ -4,6 +4,7 @@ #include "NiPoint3.h" #include "NiQuaternion.h" #include "Component.h" +#include "eReplicaComponentType.h" class Entity; @@ -12,7 +13,7 @@ class Entity; */ class ModelComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_MODEL; + static const eReplicaComponentType ComponentType = eReplicaComponentType::MODEL; ModelComponent(Entity* parent); diff --git a/dGame/dComponents/ModuleAssemblyComponent.h b/dGame/dComponents/ModuleAssemblyComponent.h index 24e1c1ee..39670c9a 100644 --- a/dGame/dComponents/ModuleAssemblyComponent.h +++ b/dGame/dComponents/ModuleAssemblyComponent.h @@ -3,6 +3,7 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Component that belongs to an object that may be modularly built, like cars and rockets. Note that this is not the @@ -11,9 +12,9 @@ */ class ModuleAssemblyComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_MODULE_ASSEMBLY; + static const eReplicaComponentType ComponentType = eReplicaComponentType::MODULE_ASSEMBLY; - ModuleAssemblyComponent(Entity* MSG_CHAT_INTERNAL_PLAYER_REMOVED_NOTIFICATION); + ModuleAssemblyComponent(Entity* parent); ~ModuleAssemblyComponent() override; void Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags); diff --git a/dGame/dComponents/MovementAIComponent.cpp b/dGame/dComponents/MovementAIComponent.cpp index ed6ed483..1966638d 100644 --- a/dGame/dComponents/MovementAIComponent.cpp +++ b/dGame/dComponents/MovementAIComponent.cpp @@ -9,82 +9,78 @@ #include "dpWorld.h" #include "EntityManager.h" #include "SimplePhysicsComponent.h" +#include "CDClientManager.h" +#include "Game.h" +#include "dZoneManager.h" -std::map MovementAIComponent::m_PhysicsSpeedCache = {}; +#include "CDComponentsRegistryTable.h" +#include "CDPhysicsComponentTable.h" + +namespace { + /** + * Cache of all lots and their respective speeds + */ + std::map m_PhysicsSpeedCache; +} MovementAIComponent::MovementAIComponent(Entity* parent, MovementAIInfo info) : Component(parent) { - m_Info = std::move(info); - m_Done = true; + m_Info = info; + m_AtFinalWaypoint = true; m_BaseCombatAI = nullptr; - m_BaseCombatAI = reinterpret_cast(m_Parent->GetComponent(COMPONENT_TYPE_BASE_COMBAT_AI)); + m_BaseCombatAI = m_Parent->GetComponent(); //Try and fix the insane values: - if (m_Info.wanderRadius > 5.0f) m_Info.wanderRadius = m_Info.wanderRadius * 0.5f; + if (m_Info.wanderRadius > 5.0f) m_Info.wanderRadius *= 0.5f; if (m_Info.wanderRadius > 8.0f) m_Info.wanderRadius = 8.0f; - if (m_Info.wanderSpeed > 0.5f) m_Info.wanderSpeed = m_Info.wanderSpeed * 0.5f; + if (m_Info.wanderSpeed > 0.5f) m_Info.wanderSpeed *= 0.5f; m_BaseSpeed = GetBaseSpeed(m_Parent->GetLOT()); - m_NextWaypoint = GetCurrentPosition(); + m_NextWaypoint = m_Parent->GetPosition(); m_Acceleration = 0.4f; - m_Interrupted = false; - m_PullPoint = {}; + m_PullingToPoint = false; + m_PullPoint = NiPoint3::ZERO; m_HaltDistance = 0; - m_Timer = 0; + m_TimeToTravel = 0; + m_TimeTravelled = 0; m_CurrentSpeed = 0; - m_Speed = 0; - m_TotalTime = 0; + m_MaxSpeed = 0; m_LockRotation = false; } -MovementAIComponent::~MovementAIComponent() = default; - void MovementAIComponent::Update(const float deltaTime) { - if (m_Interrupted) { + if (m_PullingToPoint) { const auto source = GetCurrentWaypoint(); const auto speed = deltaTime * 2.5f; - NiPoint3 velocity; - - velocity.x = (m_PullPoint.x - source.x) * speed; - velocity.y = (m_PullPoint.y - source.y) * speed; - velocity.z = (m_PullPoint.z - source.z) * speed; + NiPoint3 velocity = (m_PullPoint - source) * speed; SetPosition(source + velocity); - if (Vector3::DistanceSquared(GetCurrentPosition(), m_PullPoint) < 2 * 2) { - m_Interrupted = false; + if (Vector3::DistanceSquared(m_Parent->GetPosition(), m_PullPoint) < std::pow(2, 2)) { + m_PullingToPoint = false; } return; } - if (AtFinalWaypoint()) // Are we done? - { - return; - } + // Are we done? + if (AtFinalWaypoint()) return; if (m_HaltDistance > 0) { - if (Vector3::DistanceSquared(ApproximateLocation(), GetDestination()) < m_HaltDistance * m_HaltDistance) // Prevent us from hugging the target - { + // Prevent us from hugging the target + if (Vector3::DistanceSquared(ApproximateLocation(), GetDestination()) < std::pow(m_HaltDistance, 2)) { Stop(); - return; } } - if (m_Timer > 0) { - m_Timer -= deltaTime; - - if (m_Timer > 0) { - return; - } - - m_Timer = 0; - } + m_TimeTravelled += deltaTime; + if (m_TimeTravelled < m_TimeToTravel) return; + m_TimeTravelled = 0.0f; const auto source = GetCurrentWaypoint(); @@ -92,60 +88,56 @@ void MovementAIComponent::Update(const float deltaTime) { NiPoint3 velocity = NiPoint3::ZERO; - if (AdvanceWaypointIndex()) // Do we have another waypoint to seek? + if (m_Acceleration > 0 && m_BaseSpeed > 0 && AdvanceWaypointIndex()) // Do we have another waypoint to seek? { m_NextWaypoint = GetCurrentWaypoint(); if (m_NextWaypoint == source) { - m_Timer = 0; + m_TimeToTravel = 0.0f; goto nextAction; } - if (m_CurrentSpeed < m_Speed) { + if (m_CurrentSpeed < m_MaxSpeed) { m_CurrentSpeed += m_Acceleration; } - if (m_CurrentSpeed > m_Speed) { - m_CurrentSpeed = m_Speed; + if (m_CurrentSpeed > m_MaxSpeed) { + m_CurrentSpeed = m_MaxSpeed; } - const auto speed = m_CurrentSpeed * m_BaseSpeed; + const auto speed = m_CurrentSpeed * m_BaseSpeed; // scale speed based on base speed const auto delta = m_NextWaypoint - source; // Normalize the vector - const auto length = sqrtf(delta.x * delta.x + delta.y * delta.y + delta.z * delta.z); - + const auto length = delta.Length(); if (length > 0) { - velocity.x = (delta.x / length) * speed; - velocity.y = (delta.y / length) * speed; - velocity.z = (delta.z / length) * speed; + velocity = (delta / length) * speed; } // Calclute the time it will take to reach the next waypoint with the current speed - m_TotalTime = m_Timer = length / speed; + m_TimeTravelled = 0.0f; + m_TimeToTravel = length / speed; SetRotation(NiQuaternion::LookAt(source, m_NextWaypoint)); } else { // Check if there are more waypoints in the queue, if so set our next destination to the next waypoint - if (!m_Queue.empty()) { - SetDestination(m_Queue.top()); - - m_Queue.pop(); - } else { - // We have reached our final waypoint + if (m_CurrentPath.empty()) { Stop(); return; } + SetDestination(m_CurrentPath.top()); + + m_CurrentPath.pop(); } nextAction: SetVelocity(velocity); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } const MovementAIInfo& MovementAIComponent::GetInfo() const { @@ -153,7 +145,7 @@ const MovementAIInfo& MovementAIComponent::GetInfo() const { } bool MovementAIComponent::AdvanceWaypointIndex() { - if (m_PathIndex >= m_CurrentPath.size()) { + if (m_PathIndex >= m_InterpolatedWaypoints.size()) { return false; } @@ -163,37 +155,19 @@ bool MovementAIComponent::AdvanceWaypointIndex() { } NiPoint3 MovementAIComponent::GetCurrentWaypoint() const { - if (m_PathIndex >= m_CurrentPath.size()) { - return GetCurrentPosition(); - } - - return m_CurrentPath[m_PathIndex]; -} - -NiPoint3 MovementAIComponent::GetNextWaypoint() const { - return m_NextWaypoint; -} - -NiPoint3 MovementAIComponent::GetCurrentPosition() const { - return m_Parent->GetPosition(); + return m_PathIndex >= m_InterpolatedWaypoints.size() ? m_Parent->GetPosition() : m_InterpolatedWaypoints[m_PathIndex]; } NiPoint3 MovementAIComponent::ApproximateLocation() const { - auto source = GetCurrentPosition(); + auto source = m_Parent->GetPosition(); - if (m_Done) { - return source; - } + if (AtFinalWaypoint()) return source; auto destination = m_NextWaypoint; - auto factor = m_TotalTime > 0 ? (m_TotalTime - m_Timer) / m_TotalTime : 0; + auto percentageToWaypoint = m_TimeToTravel > 0 ? m_TimeTravelled / m_TimeToTravel : 0; - auto x = source.x + factor * (destination.x - source.x); - auto y = source.y + factor * (destination.y - source.y); - auto z = source.z + factor * (destination.z - source.z); - - NiPoint3 approximation = NiPoint3(x, y, z); + auto approximation = source + ((destination - source) * percentageToWaypoint); if (dpWorld::Instance().IsLoaded()) { approximation.y = dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(approximation); @@ -217,58 +191,47 @@ bool MovementAIComponent::Warp(const NiPoint3& point) { SetPosition(destination); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); return true; } -float MovementAIComponent::GetTimer() const { - return m_Timer; -} - -bool MovementAIComponent::AtFinalWaypoint() const { - return m_Done; -} - void MovementAIComponent::Stop() { - if (m_Done) { - return; - } + if (AtFinalWaypoint()) return; SetPosition(ApproximateLocation()); SetVelocity(NiPoint3::ZERO); - m_TotalTime = m_Timer = 0; + m_TimeToTravel = 0; + m_TimeTravelled = 0; - m_Done = true; + m_AtFinalWaypoint = true; - m_CurrentPath = {}; + m_InterpolatedWaypoints.clear(); + while (!m_CurrentPath.empty()) m_CurrentPath.pop(); m_PathIndex = 0; m_CurrentSpeed = 0; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void MovementAIComponent::PullToPoint(const NiPoint3& point) { Stop(); - m_Interrupted = true; + m_PullingToPoint = true; m_PullPoint = point; } void MovementAIComponent::SetPath(std::vector path) { - std::reverse(path.begin(), path.end()); + if (path.empty()) return; + std::for_each(path.rbegin(), path.rend() - 1, [this](const NiPoint3& point) { + this->m_CurrentPath.push(point); + }); - for (const auto& point : path) { - m_Queue.push(point); - } - - SetDestination(m_Queue.top()); - - m_Queue.pop(); + SetDestination(path.front()); } float MovementAIComponent::GetBaseSpeed(LOT lot) { @@ -279,37 +242,26 @@ float MovementAIComponent::GetBaseSpeed(LOT lot) { return it->second; } - CDComponentsRegistryTable* componentRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - CDPhysicsComponentTable* physicsComponentTable = CDClientManager::Instance()->GetTable("PhysicsComponent"); + CDComponentsRegistryTable* componentRegistryTable = CDClientManager::Instance().GetTable(); + CDPhysicsComponentTable* physicsComponentTable = CDClientManager::Instance().GetTable(); int32_t componentID; CDPhysicsComponent* physicsComponent = nullptr; - componentID = componentRegistryTable->GetByIDAndType(lot, COMPONENT_TYPE_CONTROLLABLE_PHYSICS, -1); + componentID = componentRegistryTable->GetByIDAndType(lot, eReplicaComponentType::CONTROLLABLE_PHYSICS, -1); - if (componentID != -1) { - physicsComponent = physicsComponentTable->GetByID(componentID); - - goto foundComponent; + if (componentID == -1) { + componentID = componentRegistryTable->GetByIDAndType(lot, eReplicaComponentType::SIMPLE_PHYSICS, -1); } - componentID = componentRegistryTable->GetByIDAndType(lot, COMPONENT_TYPE_SIMPLE_PHYSICS, -1); + physicsComponent = physicsComponentTable->GetByID(componentID); - if (componentID != -1) { - physicsComponent = physicsComponentTable->GetByID(componentID); + // Client defaults speed to 10 and if the speed is also null in the table, it defaults to 10. + float speed = physicsComponent != nullptr ? physicsComponent->speed : 10.0f; - goto foundComponent; - } + float delta = fabs(speed) - 1.0f; -foundComponent: - - float speed; - - if (physicsComponent == nullptr) { - speed = 8; - } else { - speed = physicsComponent->speed; - } + if (delta <= std::numeric_limits::epsilon()) speed = 10.0f; m_PhysicsSpeedCache[lot] = speed; @@ -317,39 +269,11 @@ foundComponent: } void MovementAIComponent::SetPosition(const NiPoint3& value) { - auto* controllablePhysicsComponent = m_Parent->GetComponent(); - - if (controllablePhysicsComponent != nullptr) { - controllablePhysicsComponent->SetPosition(value); - - return; - } - - auto* simplePhysicsComponent = m_Parent->GetComponent(); - - if (simplePhysicsComponent != nullptr) { - simplePhysicsComponent->SetPosition(value); - } + m_Parent->SetPosition(value); } void MovementAIComponent::SetRotation(const NiQuaternion& value) { - if (m_LockRotation) { - return; - } - - auto* controllablePhysicsComponent = m_Parent->GetComponent(); - - if (controllablePhysicsComponent != nullptr) { - controllablePhysicsComponent->SetRotation(value); - - return; - } - - auto* simplePhysicsComponent = m_Parent->GetComponent(); - - if (simplePhysicsComponent != nullptr) { - simplePhysicsComponent->SetRotation(value); - } + if (!m_LockRotation) m_Parent->SetRotation(value); } void MovementAIComponent::SetVelocity(const NiPoint3& value) { @@ -368,15 +292,8 @@ void MovementAIComponent::SetVelocity(const NiPoint3& value) { } } -void MovementAIComponent::SetDestination(const NiPoint3& value) { - if (m_Interrupted) { - return; - } - - /*if (Vector3::DistanceSquared(value, GetDestination()) < 2 * 2) - { - return; - }*/ +void MovementAIComponent::SetDestination(const NiPoint3& destination) { + if (m_PullingToPoint) return; const auto location = ApproximateLocation(); @@ -385,97 +302,53 @@ void MovementAIComponent::SetDestination(const NiPoint3& value) { } std::vector computedPath; - if (dpWorld::Instance().IsLoaded()) { - computedPath = dpWorld::Instance().GetNavMesh()->GetPath(GetCurrentPosition(), value, m_Info.wanderSpeed); - } else { + computedPath = dpWorld::Instance().GetNavMesh()->GetPath(m_Parent->GetPosition(), destination, m_Info.wanderSpeed); + } + + // Somehow failed + if (computedPath.empty()) { // Than take 10 points between the current position and the destination and make that the path - auto point = location; + auto start = location; - auto delta = value - point; + auto delta = destination - start; - auto step = delta / 10; + auto step = delta / 10.0f; for (int i = 0; i < 10; i++) { - point = point + step; + // TODO: Replace this with += when the NiPoint3::operator+= is fixed + start = start + step; - computedPath.push_back(point); + computedPath.push_back(start); } } - if (computedPath.empty()) // Somehow failed - { - return; - } - - m_CurrentPath.clear(); - - m_CurrentPath.push_back(location); + m_InterpolatedWaypoints.clear(); // Simply path - for (auto point : computedPath) { + for (auto& point : computedPath) { if (dpWorld::Instance().IsLoaded()) { point.y = dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(point); } - m_CurrentPath.push_back(point); + m_InterpolatedWaypoints.push_back(point); } - m_CurrentPath.push_back(computedPath[computedPath.size() - 1]); - m_PathIndex = 0; - m_TotalTime = m_Timer = 0; + m_TimeTravelled = 0; + m_TimeToTravel = 0; - m_Done = false; + m_AtFinalWaypoint = false; } NiPoint3 MovementAIComponent::GetDestination() const { - if (m_CurrentPath.empty()) { - return GetCurrentPosition(); - } - - return m_CurrentPath[m_CurrentPath.size() - 1]; + return m_InterpolatedWaypoints.empty() ? m_Parent->GetPosition() : m_InterpolatedWaypoints.back(); } -void MovementAIComponent::SetSpeed(const float value) { - m_Speed = value; +void MovementAIComponent::SetMaxSpeed(const float value) { + if (value == m_MaxSpeed) return; + m_MaxSpeed = value; m_Acceleration = value / 5; } - -float MovementAIComponent::GetSpeed() const { - return m_Speed; -} - -void MovementAIComponent::SetAcceleration(const float value) { - m_Acceleration = value; -} - -float MovementAIComponent::GetAcceleration() const { - return m_Acceleration; -} - -void MovementAIComponent::SetHaltDistance(const float value) { - m_HaltDistance = value; -} - -float MovementAIComponent::GetHaltDistance() const { - return m_HaltDistance; -} - -void MovementAIComponent::SetCurrentSpeed(float value) { - m_CurrentSpeed = value; -} - -float MovementAIComponent::GetCurrentSpeed() const { - return m_CurrentSpeed; -} - -void MovementAIComponent::SetLockRotation(bool value) { - m_LockRotation = value; -} - -bool MovementAIComponent::GetLockRotation() const { - return m_LockRotation; -} diff --git a/dGame/dComponents/MovementAIComponent.h b/dGame/dComponents/MovementAIComponent.h index 82cd48a0..4a4e4c0a 100644 --- a/dGame/dComponents/MovementAIComponent.h +++ b/dGame/dComponents/MovementAIComponent.h @@ -1,6 +1,6 @@ /* * Darkflame Universe - * Copyright 2018 + * Copyright 2023 */ #ifndef MOVEMENTAICOMPONENT_H @@ -13,6 +13,7 @@ #include "Game.h" #include "dLogger.h" #include "Component.h" +#include "eReplicaComponentType.h" #include class ControllablePhysicsComponent; @@ -56,10 +57,9 @@ struct MovementAIInfo { */ class MovementAIComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_MOVEMENT_AI; + static const eReplicaComponentType ComponentType = eReplicaComponentType::MOVEMENT_AI; MovementAIComponent(Entity* parentEntity, MovementAIInfo info); - ~MovementAIComponent() override; void Update(float deltaTime) override; @@ -85,61 +85,55 @@ public: * Sets the max speed at which this entity may run * @param value the speed value to set */ - void SetSpeed(float value); - - /** - * Returns the max speed at which this entity may run - * @return the max speed at which this entity may run - */ - float GetSpeed() const; + void SetMaxSpeed(float value); /** * Sets how fast the entity will accelerate when not running at full speed * @param value the acceleration to set */ - void SetAcceleration(float value); + void SetAcceleration(float value) { m_Acceleration = value; }; /** * Returns the current speed at which this entity accelerates when not running at full speed * @return the current speed at which this entity accelerates when not running at full speed */ - float GetAcceleration() const; + float GetAcceleration() const { return m_Acceleration; }; /** * Sets the halting distance (the distance at which we consider the target to be reached) * @param value the halting distance to set */ - void SetHaltDistance(float value); + void SetHaltDistance(float value) { m_HaltDistance = value; } /** * Returns the current halting distance (the distance at which we consider the target to be reached) * @return the current halting distance */ - float GetHaltDistance() const; + float GetHaltDistance() const { return m_HaltDistance; } /** * Sets the speed the entity is currently running at * @param value the speed value to set */ - void SetCurrentSpeed(float value); + void SetCurrentSpeed(float value) { m_CurrentSpeed = value; } /** * Returns the speed the entity is currently running at * @return the speed the entity is currently running at */ - float GetCurrentSpeed() const; + float GetCurrentSpeed() const { return m_CurrentSpeed; } /** * Locks the rotation of this entity in place, depending on the argument * @param value if true, the entity will be rotationally locked */ - void SetLockRotation(bool value); + void SetLockRotation(bool value) { m_LockRotation = value; } /** * Returns whether this entity is currently rotationally locked * @return true if the entity is rotationally locked, false otherwise */ - bool GetLockRotation() const; + bool GetLockRotation() const { return m_LockRotation; }; /** * Attempts to update the waypoint index, making the entity move to the next waypoint @@ -157,13 +151,7 @@ public: * Returns the waypoint this entity is supposed to move towards next * @return the waypoint this entity is supposed to move towards next */ - NiPoint3 GetNextWaypoint() const; - - /** - * Returns the current position of this entity - * @return the current position of this entity - */ - NiPoint3 GetCurrentPosition() const; + NiPoint3 GetNextWaypoint() const { return m_NextWaypoint; } /** * Returns the approximate current location of the entity, including y coordinates @@ -179,17 +167,11 @@ public: */ bool Warp(const NiPoint3& point); - /** - * Returns the time it will take to reach the final waypoint according to the current speed - * @return the time it will take to reach the final waypoint according to the current speed - */ - float GetTimer() const; - /** * Returns if the entity is at its final waypoint * @return if the entity is at its final waypoint */ - bool AtFinalWaypoint() const; + bool AtFinalWaypoint() const { return m_AtFinalWaypoint; } /** * Renders the entity stationary @@ -249,17 +231,12 @@ private: /** * The max speed this entity may move at */ - float m_Speed; + float m_MaxSpeed; /** * The time it will take to reach the next waypoint using the current speed */ - float m_Timer; - - /** - * The total time it will take to reach the waypoint form its starting point - */ - float m_TotalTime; + float m_TimeTravelled; /** * The path this entity is currently traversing @@ -269,7 +246,7 @@ private: /** * If the entity has reached it last waypoint */ - bool m_Done; + bool m_AtFinalWaypoint; /** * The speed the entity is currently moving at @@ -286,6 +263,11 @@ private: */ float m_HaltDistance; + /** + * The total time it will take to reach the waypoint form its starting point + */ + float m_TimeToTravel; + /** * The base speed this entity has */ @@ -294,7 +276,7 @@ private: /** * If the AI is currently turned of (e.g. when teleporting to some location) */ - bool m_Interrupted; + bool m_PullingToPoint; /** * A position that the entity is currently moving towards while being interrupted @@ -314,17 +296,12 @@ private: /** * The path the entity is currently following */ - std::vector m_CurrentPath; + std::vector m_InterpolatedWaypoints; /** - * Queue of positions to traverse + * The path from the current position to the destination. */ - std::stack m_Queue; - - /** - * Cache of all lots and their respective speeds - */ - static std::map m_PhysicsSpeedCache; + std::stack m_CurrentPath; }; #endif // MOVEMENTAICOMPONENT_H diff --git a/dGame/dComponents/MovingPlatformComponent.cpp b/dGame/dComponents/MovingPlatformComponent.cpp index 42699672..f4dcdbe9 100644 --- a/dGame/dComponents/MovingPlatformComponent.cpp +++ b/dGame/dComponents/MovingPlatformComponent.cpp @@ -12,11 +12,12 @@ #include "GameMessages.h" #include "CppScripts.h" #include "SimplePhysicsComponent.h" +#include "Zone.h" MoverSubComponent::MoverSubComponent(const NiPoint3& startPos) { mPosition = {}; - mState = MovementPlatformState::Stopped; + mState = eMovementPlatformState::Stopped; mDesiredWaypointIndex = 0; // -1; mInReverse = false; mShouldStopAtDesiredWaypoint = false; @@ -58,7 +59,7 @@ MovingPlatformComponent::MovingPlatformComponent(Entity* parent, const std::stri m_MoverSubComponentType = eMoverSubComponentType::mover; m_MoverSubComponent = new MoverSubComponent(m_Parent->GetDefaultPosition()); m_PathName = GeneralUtils::ASCIIToUTF16(pathName); - m_Path = dZoneManager::Instance()->GetZone()->GetPath(pathName); + m_Path = Game::zoneManager->GetZone()->GetPath(pathName); m_NoAutoStart = false; if (m_Path == nullptr) { @@ -127,12 +128,12 @@ void MovingPlatformComponent::OnCompleteRebuild() { StartPathing(); } -void MovingPlatformComponent::SetMovementState(MovementPlatformState value) { +void MovingPlatformComponent::SetMovementState(eMovementPlatformState value) { auto* subComponent = static_cast(m_MoverSubComponent); subComponent->mState = value; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void MovingPlatformComponent::GotoWaypoint(uint32_t index, bool stopAtWaypoint) { @@ -152,7 +153,7 @@ void MovingPlatformComponent::StartPathing() { auto* subComponent = static_cast(m_MoverSubComponent); subComponent->mShouldStopAtDesiredWaypoint = true; - subComponent->mState = MovementPlatformState::Stationary; + subComponent->mState = eMovementPlatformState::Stationary; NiPoint3 targetPosition; @@ -174,7 +175,7 @@ void MovingPlatformComponent::StartPathing() { } m_Parent->AddCallbackTimer(subComponent->mWaitTime, [this] { - SetMovementState(MovementPlatformState::Moving); + SetMovementState(eMovementPlatformState::Moving); }); const auto travelTime = Vector3::Distance(targetPosition, subComponent->mPosition) / subComponent->mSpeed + 1.5f; @@ -193,13 +194,13 @@ void MovingPlatformComponent::StartPathing() { //GameMessages::SendPlatformResync(m_Parent, UNASSIGNED_SYSTEM_ADDRESS); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void MovingPlatformComponent::ContinuePathing() { auto* subComponent = static_cast(m_MoverSubComponent); - subComponent->mState = MovementPlatformState::Stationary; + subComponent->mState = eMovementPlatformState::Stationary; subComponent->mCurrentWaypointIndex = subComponent->mNextWaypointIndex; @@ -241,7 +242,7 @@ void MovingPlatformComponent::ContinuePathing() { subComponent->mCurrentWaypointIndex = pathSize; switch (behavior) { case PathBehavior::Once: - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); return; case PathBehavior::Bounce: @@ -282,7 +283,7 @@ void MovingPlatformComponent::ContinuePathing() { m_Parent->CancelCallbackTimers(); m_Parent->AddCallbackTimer(subComponent->mWaitTime, [this] { - SetMovementState(MovementPlatformState::Moving); + SetMovementState(eMovementPlatformState::Moving); }); auto travelTime = Vector3::Distance(targetPosition, subComponent->mPosition) / subComponent->mSpeed + 1.5; @@ -303,7 +304,7 @@ void MovingPlatformComponent::ContinuePathing() { ContinuePathing(); }); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void MovingPlatformComponent::StopPathing() { @@ -313,11 +314,11 @@ void MovingPlatformComponent::StopPathing() { m_PathingStopped = true; - subComponent->mState = MovementPlatformState::Stopped; + subComponent->mState = eMovementPlatformState::Stopped; subComponent->mDesiredWaypointIndex = -1; subComponent->mShouldStopAtDesiredWaypoint = false; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); //GameMessages::SendPlatformResync(m_Parent, UNASSIGNED_SYSTEM_ADDRESS); } @@ -340,7 +341,7 @@ void MovingPlatformComponent::WarpToWaypoint(size_t index) { m_Parent->SetPosition(waypoint.position); m_Parent->SetRotation(waypoint.rotation); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } size_t MovingPlatformComponent::GetLastWaypointIndex() const { diff --git a/dGame/dComponents/MovingPlatformComponent.h b/dGame/dComponents/MovingPlatformComponent.h index efa3a4cf..9e4c1ecf 100644 --- a/dGame/dComponents/MovingPlatformComponent.h +++ b/dGame/dComponents/MovingPlatformComponent.h @@ -13,6 +13,10 @@ #include "dCommonVars.h" #include "EntityManager.h" #include "Component.h" +#include "eMovementPlatformState.h" +#include "eReplicaComponentType.h" + +class Path; /** * Different types of available platforms @@ -26,16 +30,6 @@ enum class eMoverSubComponentType : uint32_t { simpleMover = 5, }; -/** - * The different types of platform movement state, supposedly a bitmap - */ -enum class MovementPlatformState : uint32_t -{ - Moving = 0b00010, - Stationary = 0b11001, - Stopped = 0b01100 -}; - /** * Sub component for moving platforms that determine the actual current movement state */ @@ -49,7 +43,7 @@ public: /** * The state the platform is currently in */ - MovementPlatformState mState = MovementPlatformState::Stationary; + eMovementPlatformState mState = eMovementPlatformState::Stationary; /** * The waypoint this platform currently wants to traverse to @@ -112,7 +106,7 @@ public: */ class MovingPlatformComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_MOVING_PLATFORM; + static const eReplicaComponentType ComponentType = eReplicaComponentType::MOVING_PLATFORM; MovingPlatformComponent(Entity* parent, const std::string& pathName); ~MovingPlatformComponent() override; @@ -133,7 +127,7 @@ public: * Updates the movement state for the moving platform * @param value the movement state to set */ - void SetMovementState(MovementPlatformState value); + void SetMovementState(eMovementPlatformState value); /** * Instructs the moving platform to go to some waypoint diff --git a/dGame/dComponents/PetComponent.cpp b/dGame/dComponents/PetComponent.cpp index 8863f9be..7df08c60 100644 --- a/dGame/dComponents/PetComponent.cpp +++ b/dGame/dComponents/PetComponent.cpp @@ -15,11 +15,20 @@ #include "PetDigServer.h" #include "../dWorldServer/ObjectIDManager.h" #include "eUnequippableActiveType.h" +#include "eTerminateType.h" +#include "ePetTamingNotifyType.h" +#include "eUseItemResponse.h" +#include "ePlayerFlag.h" #include "Game.h" #include "dConfig.h" #include "dChatFilter.h" #include "Database.h" +#include "EntityInfo.h" +#include "eMissionTaskType.h" +#include "RenderComponent.h" +#include "eObjectBits.h" +#include "eGameMasterLevel.h" std::unordered_map PetComponent::buildCache{}; std::unordered_map PetComponent::currentActivities{}; @@ -29,7 +38,7 @@ std::unordered_map PetComponent::activePets{}; * Maps all the pet lots to a flag indicating that the player has caught it. All basic pets have been guessed by ObjID * while the faction ones could be checked using their respective missions. */ -std::map PetComponent::petFlags = { +std::map PetComponent::petFlags = { { 3050, 801 }, // Elephant { 3054, 803 }, // Cat { 3195, 806 }, // Triceratops @@ -59,7 +68,7 @@ std::map PetComponent::petFlags = { { 13067, 838 }, // Skeleton dragon }; -PetComponent::PetComponent(Entity* parent, uint32_t componentId) : Component(parent) { +PetComponent::PetComponent(Entity* parent, uint32_t componentId): Component(parent) { m_ComponentId = componentId; m_Interaction = LWOOBJID_EMPTY; @@ -118,21 +127,23 @@ void PetComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpd outBitStream->Write(m_Owner); } - outBitStream->Write(tamed); - if (tamed) { - outBitStream->Write(m_ModerationStatus); + if (bIsInitialUpdate) { + outBitStream->Write(tamed); + if (tamed) { + outBitStream->Write(m_ModerationStatus); - const auto nameData = GeneralUtils::UTF8ToUTF16(m_Name); - const auto ownerNameData = GeneralUtils::UTF8ToUTF16(m_OwnerName); + const auto nameData = GeneralUtils::UTF8ToUTF16(m_Name); + const auto ownerNameData = GeneralUtils::UTF8ToUTF16(m_OwnerName); - outBitStream->Write(static_cast(nameData.size())); - for (const auto c : nameData) { - outBitStream->Write(c); - } + outBitStream->Write(static_cast(nameData.size())); + for (const auto c : nameData) { + outBitStream->Write(c); + } - outBitStream->Write(static_cast(ownerNameData.size())); - for (const auto c : ownerNameData) { - outBitStream->Write(c); + outBitStream->Write(static_cast(ownerNameData.size())); + for (const auto c : ownerNameData) { + outBitStream->Write(c); + } } } } @@ -143,7 +154,7 @@ void PetComponent::OnUse(Entity* originator) { } if (m_Tamer != LWOOBJID_EMPTY) { - auto* tamer = EntityManager::Instance()->GetEntity(m_Tamer); + auto* tamer = Game::entityManager->GetEntity(m_Tamer); if (tamer != nullptr) { return; @@ -194,21 +205,7 @@ void PetComponent::OnUse(Entity* originator) { return; } - auto lxfAsset = std::string(result.getStringField(0)); - - std::vector lxfAssetSplit = GeneralUtils::SplitString(lxfAsset, '\\'); - - lxfAssetSplit.erase(lxfAssetSplit.begin()); - - buildFile = "res/BrickModels"; - - for (auto part : lxfAssetSplit) { - std::transform(part.begin(), part.end(), part.begin(), [](unsigned char c) { - return std::tolower(c); - }); - - buildFile += "/" + part; - } + buildFile = std::string(result.getStringField(0)); PetPuzzleData data; data.buildFile = buildFile; @@ -239,7 +236,7 @@ void PetComponent::OnUse(Entity* originator) { return; } - auto& bricks = BrickDatabase::Instance()->GetBricks(buildFile); + const auto& bricks = BrickDatabase::GetBricks(buildFile); if (bricks.empty()) { ChatPackets::SendSystemMessage(originator->GetSystemAddress(), u"Failed to load the puzzle minigame for this pet."); @@ -293,7 +290,7 @@ void PetComponent::OnUse(Entity* originator) { m_Parent->GetObjectID(), LWOOBJID_EMPTY, true, - NOTIFY_TYPE_BEGIN, + ePetTamingNotifyType::BEGIN, petPosition, position, rotation, @@ -305,7 +302,7 @@ void PetComponent::OnUse(Entity* originator) { LWOOBJID_EMPTY, originator->GetObjectID(), true, - NOTIFY_TYPE_BEGIN, + ePetTamingNotifyType::BEGIN, petPosition, position, rotation, @@ -321,7 +318,7 @@ void PetComponent::OnUse(Entity* originator) { // Notify the start of a pet taming minigame for (CppScripts::Script* script : CppScripts::GetEntityScripts(m_Parent)) { - script->OnNotifyPetTamingMinigame(m_Parent, originator, NOTIFY_TYPE_BEGIN); + script->OnNotifyPetTamingMinigame(m_Parent, originator, ePetTamingNotifyType::BEGIN); } } @@ -347,7 +344,7 @@ void PetComponent::Update(float deltaTime) { if (m_Timer <= 0) { Wander(); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } else { m_Timer = 5; @@ -372,7 +369,7 @@ void PetComponent::Update(float deltaTime) { } if (m_TresureTime > 0) { - auto* tresure = EntityManager::Instance()->GetEntity(m_Interaction); + auto* tresure = Game::entityManager->GetEntity(m_Interaction); if (tresure == nullptr) { m_TresureTime = 0; @@ -398,7 +395,7 @@ void PetComponent::Update(float deltaTime) { } auto destination = owner->GetPosition(); - NiPoint3 position = m_MovementAI->GetCurrentPosition(); + NiPoint3 position = m_MovementAI->GetParent()->GetPosition(); float distanceToOwner = Vector3::DistanceSquared(position, destination); @@ -452,7 +449,7 @@ void PetComponent::Update(float deltaTime) { NiPoint3 tresurePosition = closestTresure->GetPosition(); float distance = Vector3::DistanceSquared(position, tresurePosition); - if (distance < 3 * 3) { + if (distance < 5 * 5) { m_Interaction = closestTresure->GetObjectID(); Command(NiPoint3::ZERO, LWOOBJID_EMPTY, 1, 202, true); @@ -469,7 +466,7 @@ skipTresure: m_MovementAI->SetHaltDistance(haltDistance); - m_MovementAI->SetSpeed(2.5f); + m_MovementAI->SetMaxSpeed(2.5f); m_MovementAI->SetDestination(destination); @@ -479,7 +476,7 @@ skipTresure: void PetComponent::TryBuild(uint32_t numBricks, bool clientFailed) { if (m_Tamer == LWOOBJID_EMPTY) return; - auto* tamer = EntityManager::Instance()->GetEntity(m_Tamer); + auto* tamer = Game::entityManager->GetEntity(m_Tamer); if (tamer == nullptr) { m_Tamer = LWOOBJID_EMPTY; @@ -501,7 +498,7 @@ void PetComponent::TryBuild(uint32_t numBricks, bool clientFailed) { destroyableComponent->SetImagination(imagination); - EntityManager::Instance()->SerializeEntity(tamer); + Game::entityManager->SerializeEntity(tamer); if (clientFailed) { if (imagination < cached->second.imaginationCost) { @@ -519,7 +516,7 @@ void PetComponent::TryBuild(uint32_t numBricks, bool clientFailed) { void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { if (m_Tamer == LWOOBJID_EMPTY) return; - auto* tamer = EntityManager::Instance()->GetEntity(m_Tamer); + auto* tamer = Game::entityManager->GetEntity(m_Tamer); if (tamer == nullptr) { m_Tamer = LWOOBJID_EMPTY; @@ -534,7 +531,7 @@ void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { } GameMessages::SendPlayFXEffect(tamer, -1, u"petceleb", "", LWOOBJID_EMPTY, 1, 1, true); - GameMessages::SendPlayAnimation(tamer, u"rebuild-celebrate"); + RenderComponent::PlayAnimation(tamer, u"rebuild-celebrate"); EntityInfo info{}; info.lot = cached->second.puzzleModelLot; @@ -542,11 +539,11 @@ void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { info.rot = NiQuaternion::IDENTITY; info.spawnerID = tamer->GetObjectID(); - auto* modelEntity = EntityManager::Instance()->CreateEntity(info); + auto* modelEntity = Game::entityManager->CreateEntity(info); m_ModelId = modelEntity->GetObjectID(); - EntityManager::Instance()->ConstructEntity(modelEntity); + Game::entityManager->ConstructEntity(modelEntity); GameMessages::SendNotifyTamingModelLoadedOnServer(m_Tamer, tamer->GetSystemAddress()); @@ -560,8 +557,8 @@ void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { LWOOBJID petSubKey = ObjectIDManager::Instance()->GenerateRandomObjectID(); - petSubKey = GeneralUtils::SetBit(petSubKey, OBJECT_BIT_CHARACTER); - petSubKey = GeneralUtils::SetBit(petSubKey, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(petSubKey, eObjectBits::CHARACTER); + GeneralUtils::SetBit(petSubKey, eObjectBits::PERSISTENT); m_DatabaseId = petSubKey; @@ -574,7 +571,7 @@ void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { GameMessages::SendRegisterPetDBID(m_Tamer, petSubKey, tamer->GetSystemAddress()); - inventoryComponent->AddItem(m_Parent->GetLOT(), 1, eLootSourceType::LOOT_SOURCE_ACTIVITY, eInventoryType::MODELS, {}, LWOOBJID_EMPTY, true, false, petSubKey); + inventoryComponent->AddItem(m_Parent->GetLOT(), 1, eLootSourceType::ACTIVITY, eInventoryType::MODELS, {}, LWOOBJID_EMPTY, true, false, petSubKey); auto* item = inventoryComponent->FindItemBySubKey(petSubKey, MODELS); if (item == nullptr) { @@ -598,7 +595,7 @@ void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { LWOOBJID_EMPTY, LWOOBJID_EMPTY, false, - NOTIFY_TYPE_NAMINGPET, + ePetTamingNotifyType::NAMINGPET, NiPoint3::ZERO, NiPoint3::ZERO, NiQuaternion::IDENTITY, @@ -613,7 +610,7 @@ void PetComponent::NotifyTamingBuildSuccess(NiPoint3 position) { auto* missionComponent = tamer->GetComponent(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_PET_TAMING, m_Parent->GetLOT()); + missionComponent->Progress(eMissionTaskType::PET_TAMING, m_Parent->GetLOT()); } SetStatus(1); @@ -642,7 +639,7 @@ void PetComponent::RequestSetPetName(std::u16string name) { return; } - auto* tamer = EntityManager::Instance()->GetEntity(m_Tamer); + auto* tamer = Game::entityManager->GetEntity(m_Tamer); if (tamer == nullptr) { m_Tamer = LWOOBJID_EMPTY; @@ -664,7 +661,7 @@ void PetComponent::RequestSetPetName(std::u16string name) { //Save our pet's new name to the db: SetPetNameForModeration(GeneralUtils::UTF16ToWTF8(name)); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); std::u16string u16name = GeneralUtils::UTF8ToUTF16(m_Name); std::u16string u16ownerName = GeneralUtils::UTF8ToUTF16(m_OwnerName); @@ -678,7 +675,7 @@ void PetComponent::RequestSetPetName(std::u16string name) { m_Parent->GetObjectID(), m_Tamer, false, - NOTIFY_TYPE_SUCCESS, + ePetTamingNotifyType::SUCCESS, NiPoint3::ZERO, NiPoint3::ZERO, NiQuaternion::IDENTITY, @@ -687,7 +684,7 @@ void PetComponent::RequestSetPetName(std::u16string name) { GameMessages::SendTerminateInteraction(m_Tamer, eTerminateType::FROM_INTERACTION, m_Parent->GetObjectID()); - auto* modelEntity = EntityManager::Instance()->GetEntity(m_ModelId); + auto* modelEntity = Game::entityManager->GetEntity(m_ModelId); if (modelEntity != nullptr) { modelEntity->Smash(m_Tamer); @@ -699,14 +696,14 @@ void PetComponent::RequestSetPetName(std::u16string name) { // Notify the end of a pet taming minigame for (CppScripts::Script* script : CppScripts::GetEntityScripts(m_Parent)) { - script->OnNotifyPetTamingMinigame(m_Parent, tamer, NOTIFY_TYPE_SUCCESS); + script->OnNotifyPetTamingMinigame(m_Parent, tamer, ePetTamingNotifyType::SUCCESS); } } void PetComponent::ClientExitTamingMinigame(bool voluntaryExit) { if (m_Tamer == LWOOBJID_EMPTY) return; - auto* tamer = EntityManager::Instance()->GetEntity(m_Tamer); + auto* tamer = Game::entityManager->GetEntity(m_Tamer); if (tamer == nullptr) { m_Tamer = LWOOBJID_EMPTY; @@ -719,7 +716,7 @@ void PetComponent::ClientExitTamingMinigame(bool voluntaryExit) { m_Parent->GetObjectID(), m_Tamer, false, - NOTIFY_TYPE_QUIT, + ePetTamingNotifyType::QUIT, NiPoint3::ZERO, NiPoint3::ZERO, NiQuaternion::IDENTITY, @@ -736,11 +733,11 @@ void PetComponent::ClientExitTamingMinigame(bool voluntaryExit) { m_Tamer = LWOOBJID_EMPTY; m_Timer = 0; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); // Notify the end of a pet taming minigame for (CppScripts::Script* script : CppScripts::GetEntityScripts(m_Parent)) { - script->OnNotifyPetTamingMinigame(m_Parent, tamer, NOTIFY_TYPE_QUIT); + script->OnNotifyPetTamingMinigame(m_Parent, tamer, ePetTamingNotifyType::QUIT); } } @@ -757,7 +754,7 @@ void PetComponent::StartTimer() { void PetComponent::ClientFailTamingMinigame() { if (m_Tamer == LWOOBJID_EMPTY) return; - auto* tamer = EntityManager::Instance()->GetEntity(m_Tamer); + auto* tamer = Game::entityManager->GetEntity(m_Tamer); if (tamer == nullptr) { m_Tamer = LWOOBJID_EMPTY; @@ -770,7 +767,7 @@ void PetComponent::ClientFailTamingMinigame() { m_Parent->GetObjectID(), m_Tamer, false, - NOTIFY_TYPE_FAILED, + ePetTamingNotifyType::FAILED, NiPoint3::ZERO, NiPoint3::ZERO, NiQuaternion::IDENTITY, @@ -787,11 +784,11 @@ void PetComponent::ClientFailTamingMinigame() { m_Tamer = LWOOBJID_EMPTY; m_Timer = 0; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); // Notify the end of a pet taming minigame for (CppScripts::Script* script : CppScripts::GetEntityScripts(m_Parent)) { - script->OnNotifyPetTamingMinigame(m_Parent, tamer, NOTIFY_TYPE_FAILED); + script->OnNotifyPetTamingMinigame(m_Parent, tamer, ePetTamingNotifyType::FAILED); } } @@ -825,17 +822,17 @@ void PetComponent::Wander() { destination.y = dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(destination); } - if (Vector3::DistanceSquared(destination, m_MovementAI->GetCurrentPosition()) < 2 * 2) { + if (Vector3::DistanceSquared(destination, m_MovementAI->GetParent()->GetPosition()) < 2 * 2) { m_MovementAI->Stop(); return; } - m_MovementAI->SetSpeed(info.wanderSpeed); + m_MovementAI->SetMaxSpeed(info.wanderSpeed); m_MovementAI->SetDestination(destination); - m_Timer += (m_MovementAI->GetCurrentPosition().x - destination.x) / info.wanderSpeed; + m_Timer += (m_MovementAI->GetParent()->GetPosition().x - destination.x) / info.wanderSpeed; } void PetComponent::Activate(Item* item, bool registerPet, bool fromTaming) { @@ -890,9 +887,9 @@ void PetComponent::Activate(Item* item, bool registerPet, bool fromTaming) { m_Timer = 3; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); - owner->GetCharacter()->SetPlayerFlag(69, true); + owner->GetCharacter()->SetPlayerFlag(ePlayerFlag::FIRST_MANUAL_PET_HIBERNATE, true); if (registerPet) { GameMessages::SendAddPetToPlayer(m_Owner, 0, GeneralUtils::UTF8ToUTF16(m_Name), m_DatabaseId, m_Parent->GetLOT(), owner->GetSystemAddress()); @@ -930,16 +927,16 @@ void PetComponent::AddDrainImaginationTimer(Item* item, bool fromTaming) { return; } - // If we are out of imagination despawn the pet. - if (playerDestroyableComponent->GetImagination() == 0) { - this->Deactivate(); - auto playerEntity = playerDestroyableComponent->GetParent(); - if (!playerEntity) return; + // If we are out of imagination despawn the pet. + if (playerDestroyableComponent->GetImagination() == 0) { + this->Deactivate(); + auto playerEntity = playerDestroyableComponent->GetParent(); + if (!playerEntity) return; - GameMessages::SendUseItemRequirementsResponse(playerEntity->GetObjectID(), playerEntity->GetSystemAddress(), UseItemResponse::NoImaginationForPet); - } + GameMessages::SendUseItemRequirementsResponse(playerEntity->GetObjectID(), playerEntity->GetSystemAddress(), eUseItemResponse::NoImaginationForPet); + } - this->AddDrainImaginationTimer(item); + this->AddDrainImaginationTimer(item); }); } @@ -997,7 +994,7 @@ void PetComponent::Command(NiPoint3 position, LWOOBJID source, int32_t commandTy // TODO: Go to player } - if (owner->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (owner->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { ChatPackets::SendSystemMessage(owner->GetSystemAddress(), u"Commmand Type: " + (GeneralUtils::to_u16string(commandType)) + u" - Type Id: " + (GeneralUtils::to_u16string(typeId))); } } @@ -1007,7 +1004,7 @@ LWOOBJID PetComponent::GetOwnerId() const { } Entity* PetComponent::GetOwner() const { - return EntityManager::Instance()->GetEntity(m_Owner); + return Game::entityManager->GetEntity(m_Owner); } LWOOBJID PetComponent::GetDatabaseId() const { @@ -1049,7 +1046,7 @@ PetComponent* PetComponent::GetTamingPet(LWOOBJID tamer) { return nullptr; } - auto* entity = EntityManager::Instance()->GetEntity(pair->second); + auto* entity = Game::entityManager->GetEntity(pair->second); if (entity == nullptr) { currentActivities.erase(tamer); @@ -1067,7 +1064,7 @@ PetComponent* PetComponent::GetActivePet(LWOOBJID owner) { return nullptr; } - auto* entity = EntityManager::Instance()->GetEntity(pair->second); + auto* entity = Game::entityManager->GetEntity(pair->second); if (entity == nullptr) { activePets.erase(owner); @@ -1089,7 +1086,7 @@ void PetComponent::SetPetNameForModeration(const std::string& petName) { int approved = 1; //default, in mod //Make sure that the name isn't already auto-approved: - if (Game::chatFilter->IsSentenceOkay(petName, 0).empty()) { + if (Game::chatFilter->IsSentenceOkay(petName, eGameMasterLevel::CIVILIAN).empty()) { approved = 2; //approved } diff --git a/dGame/dComponents/PetComponent.h b/dGame/dComponents/PetComponent.h index efde0e8a..b3d089a9 100644 --- a/dGame/dComponents/PetComponent.h +++ b/dGame/dComponents/PetComponent.h @@ -4,6 +4,7 @@ #include "MovementAIComponent.h" #include "Component.h" #include "Preconditions.h" +#include "eReplicaComponentType.h" enum class PetAbilityType { @@ -20,7 +21,7 @@ enum class PetAbilityType class PetComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PET; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PET; explicit PetComponent(Entity* parentEntity, uint32_t componentId); ~PetComponent() override; @@ -262,7 +263,7 @@ private: /** * Flags that indicate that a player has tamed a pet, indexed by the LOT of the pet */ - static std::map petFlags; + static std::map petFlags; /** * The ID of the component in the pet component table diff --git a/dGame/dComponents/PhantomPhysicsComponent.cpp b/dGame/dComponents/PhantomPhysicsComponent.cpp index cef9cc36..640281f8 100644 --- a/dGame/dComponents/PhantomPhysicsComponent.cpp +++ b/dGame/dComponents/PhantomPhysicsComponent.cpp @@ -14,11 +14,13 @@ #include "EntityManager.h" #include "ControllablePhysicsComponent.h" #include "GameMessages.h" +#include "ePhysicsEffectType.h" #include "CDClientManager.h" #include "CDComponentsRegistryTable.h" #include "CDPhysicsComponentTable.h" #include "dServer.h" +#include "EntityInfo.h" #include "dpWorld.h" #include "dpEntity.h" @@ -35,7 +37,7 @@ PhantomPhysicsComponent::PhantomPhysicsComponent(Entity* parent) : Component(par m_PositionInfoDirty = false; m_IsPhysicsEffectActive = false; - m_EffectType = 0; + m_EffectType = ePhysicsEffectType::PUSH; m_DirectionalMultiplier = 0.0f; m_MinMax = false; @@ -142,10 +144,10 @@ PhantomPhysicsComponent::PhantomPhysicsComponent(Entity* parent) : Component(par */ if (!m_HasCreatedPhysics) { - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - auto componentID = compRegistryTable->GetByIDAndType(m_Parent->GetLOT(), COMPONENT_TYPE_PHANTOM_PHYSICS); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + auto componentID = compRegistryTable->GetByIDAndType(m_Parent->GetLOT(), eReplicaComponentType::PHANTOM_PHYSICS); - CDPhysicsComponentTable* physComp = CDClientManager::Instance()->GetTable("PhysicsComponent"); + CDPhysicsComponentTable* physComp = CDClientManager::Instance().GetTable(); if (physComp == nullptr) return; @@ -214,6 +216,13 @@ PhantomPhysicsComponent::PhantomPhysicsComponent(Entity* parent) : Component(par m_dpEntity->SetRotation(m_Rotation); m_dpEntity->SetPosition(m_Position); dpWorld::Instance().AddEntity(m_dpEntity); + } else if (info->physicsAsset == "env\\env_won_fv_gas-blocking-volume.hkx"){ + m_dpEntity = new dpEntity(m_Parent->GetObjectID(), 390.496826f, 111.467964f, 600.821534f, true); + m_dpEntity->SetScale(m_Scale); + m_dpEntity->SetRotation(m_Rotation); + m_Position.y -= (111.467964f * m_Scale) / 2; + m_dpEntity->SetPosition(m_Position); + dpWorld::Instance().AddEntity(m_dpEntity); } else { //Game::logger->Log("PhantomPhysicsComponent", "This one is supposed to have %s", info->physicsAsset.c_str()); @@ -252,10 +261,10 @@ void PhantomPhysicsComponent::CreatePhysics() { y = m_Parent->GetVar(u"primitiveModelValueY"); z = m_Parent->GetVar(u"primitiveModelValueZ"); } else { - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - auto componentID = compRegistryTable->GetByIDAndType(m_Parent->GetLOT(), COMPONENT_TYPE_PHANTOM_PHYSICS); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + auto componentID = compRegistryTable->GetByIDAndType(m_Parent->GetLOT(), eReplicaComponentType::PHANTOM_PHYSICS); - CDPhysicsComponentTable* physComp = CDClientManager::Instance()->GetTable("PhysicsComponent"); + CDPhysicsComponentTable* physComp = CDClientManager::Instance().GetTable(); if (physComp == nullptr) return; @@ -353,7 +362,7 @@ void PhantomPhysicsComponent::Update(float deltaTime) { //If we are a respawn volume, inform the client: if (m_IsRespawnVolume) { - auto entity = EntityManager::Instance()->GetEntity(en->GetObjectID()); + auto entity = Game::entityManager->GetEntity(en->GetObjectID()); if (entity) { GameMessages::SendPlayerReachedRespawnCheckpoint(entity, m_RespawnPos, m_RespawnRot); @@ -394,8 +403,8 @@ void PhantomPhysicsComponent::SpawnVertices() { info.spawnerID = m_Parent->GetObjectID(); info.spawnerNodeID = 0; - Entity* newEntity = EntityManager::Instance()->CreateEntity(info, nullptr); - EntityManager::Instance()->ConstructEntity(newEntity); + Entity* newEntity = Game::entityManager->CreateEntity(info, nullptr); + Game::entityManager->ConstructEntity(newEntity); } } @@ -404,7 +413,7 @@ void PhantomPhysicsComponent::SetDirectionalMultiplier(float mul) { m_EffectInfoDirty = true; } -void PhantomPhysicsComponent::SetEffectType(uint32_t type) { +void PhantomPhysicsComponent::SetEffectType(ePhysicsEffectType type) { m_EffectType = type; m_EffectInfoDirty = true; } diff --git a/dGame/dComponents/PhantomPhysicsComponent.h b/dGame/dComponents/PhantomPhysicsComponent.h index 1e9a7b60..cc0d1844 100644 --- a/dGame/dComponents/PhantomPhysicsComponent.h +++ b/dGame/dComponents/PhantomPhysicsComponent.h @@ -12,10 +12,12 @@ #include "CppScripts.h" #include "InvalidScript.h" #include "Component.h" +#include "eReplicaComponentType.h" class LDFBaseData; class Entity; class dpEntity; +enum class ePhysicsEffectType : uint32_t ; /** * Allows the creation of phantom physics for an entity: a physics object that is generally invisible but can be @@ -25,7 +27,7 @@ class dpEntity; */ class PhantomPhysicsComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PHANTOM_PHYSICS; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PHANTOM_PHYSICS; PhantomPhysicsComponent(Entity* parent); ~PhantomPhysicsComponent() override; @@ -102,13 +104,13 @@ public: * Returns the effect that's currently active, defaults to 0 * @return the effect that's currently active */ - uint32_t GetEffectType() const { return m_EffectType; } + ePhysicsEffectType GetEffectType() const { return m_EffectType; } /** * Sets the effect that's currently active * @param type the effect to set */ - void SetEffectType(uint32_t type); + void SetEffectType(ePhysicsEffectType type); /** * Returns the Physics entity for the component @@ -167,7 +169,7 @@ private: /** * The physics effect that's currently active, defaults to 0 */ - uint32_t m_EffectType; + ePhysicsEffectType m_EffectType; /** * A scaling multiplier to add to the directional vector diff --git a/dGame/dComponents/PlayerForcedMovementComponent.cpp b/dGame/dComponents/PlayerForcedMovementComponent.cpp index 3bcad677..76993507 100644 --- a/dGame/dComponents/PlayerForcedMovementComponent.cpp +++ b/dGame/dComponents/PlayerForcedMovementComponent.cpp @@ -7,8 +7,8 @@ PlayerForcedMovementComponent::PlayerForcedMovementComponent(Entity* parent) : C PlayerForcedMovementComponent::~PlayerForcedMovementComponent() {} void PlayerForcedMovementComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { - outBitStream->Write(m_DirtyInfo); - if (m_DirtyInfo) { + outBitStream->Write(m_DirtyInfo || bIsInitialUpdate); + if (m_DirtyInfo || bIsInitialUpdate) { outBitStream->Write(m_PlayerOnRail); outBitStream->Write(m_ShowBillboard); } diff --git a/dGame/dComponents/PlayerForcedMovementComponent.h b/dGame/dComponents/PlayerForcedMovementComponent.h index 43b99997..90708c9a 100644 --- a/dGame/dComponents/PlayerForcedMovementComponent.h +++ b/dGame/dComponents/PlayerForcedMovementComponent.h @@ -2,6 +2,7 @@ #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Component that handles player forced movement @@ -9,7 +10,7 @@ */ class PlayerForcedMovementComponent : public Component { public: - static const uint32_t ComponentType = eReplicaComponentType::COMPONENT_TYPE_PLAYER_FORCED_MOVEMENT; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PLAYER_FORCED_MOVEMENT; /** * Constructor for this component diff --git a/dGame/dComponents/PossessableComponent.cpp b/dGame/dComponents/PossessableComponent.cpp index 37591532..5c45a6c1 100644 --- a/dGame/dComponents/PossessableComponent.cpp +++ b/dGame/dComponents/PossessableComponent.cpp @@ -34,8 +34,8 @@ void PossessableComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsIn outBitStream->Write(m_Possessor != LWOOBJID_EMPTY); if (m_Possessor != LWOOBJID_EMPTY) outBitStream->Write(m_Possessor); - outBitStream->Write(m_AnimationFlag != eAnimationFlags::IDLE_INVALID); - if (m_AnimationFlag != eAnimationFlags::IDLE_INVALID) outBitStream->Write(m_AnimationFlag); + outBitStream->Write(m_AnimationFlag != eAnimationFlags::IDLE_NONE); + if (m_AnimationFlag != eAnimationFlags::IDLE_NONE) outBitStream->Write(m_AnimationFlag); outBitStream->Write(m_ImmediatelyDepossess); m_ImmediatelyDepossess = false; // reset flag diff --git a/dGame/dComponents/PossessableComponent.h b/dGame/dComponents/PossessableComponent.h index 43ce8610..2026c11e 100644 --- a/dGame/dComponents/PossessableComponent.h +++ b/dGame/dComponents/PossessableComponent.h @@ -6,6 +6,7 @@ #include "Item.h" #include "PossessorComponent.h" #include "eAninmationFlags.h" +#include "eReplicaComponentType.h" /** * Represents an entity that can be controlled by some other entity, generally used by cars to indicate that some @@ -13,7 +14,7 @@ */ class PossessableComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_POSSESSABLE; + static const eReplicaComponentType ComponentType = eReplicaComponentType::POSSESSABLE; PossessableComponent(Entity* parentEntity, uint32_t componentId); @@ -109,7 +110,7 @@ private: * @brief What animaiton flag to use * */ - eAnimationFlags m_AnimationFlag = eAnimationFlags::IDLE_INVALID; + eAnimationFlags m_AnimationFlag = eAnimationFlags::IDLE_NONE; /** * @brief Should this be immediately depossessed diff --git a/dGame/dComponents/PossessorComponent.cpp b/dGame/dComponents/PossessorComponent.cpp index f0cad5ca..8019f91c 100644 --- a/dGame/dComponents/PossessorComponent.cpp +++ b/dGame/dComponents/PossessorComponent.cpp @@ -4,6 +4,8 @@ #include "EntityManager.h" #include "GameMessages.h" #include "eUnequippableActiveType.h" +#include "eControlScheme.h" +#include "eStateChangeType.h" PossessorComponent::PossessorComponent(Entity* parent) : Component(parent) { m_Possessable = LWOOBJID_EMPTY; @@ -11,7 +13,7 @@ PossessorComponent::PossessorComponent(Entity* parent) : Component(parent) { PossessorComponent::~PossessorComponent() { if (m_Possessable != LWOOBJID_EMPTY) { - auto* mount = EntityManager::Instance()->GetEntity(m_Possessable); + auto* mount = Game::entityManager->GetEntity(m_Possessable); if (mount) { auto* possessable = mount->GetComponent(); if (possessable) { @@ -54,10 +56,10 @@ void PossessorComponent::Mount(Entity* mount) { // GM's to send GameMessages::SendSetJetPackMode(m_Parent, false); GameMessages::SendVehicleUnlockInput(mount->GetObjectID(), false, m_Parent->GetSystemAddress()); - GameMessages::SendSetStunned(m_Parent->GetObjectID(), eStunState::PUSH, m_Parent->GetSystemAddress(), LWOOBJID_EMPTY, true, false, true, false, false, false, false, true, true, true, true, true, true, true, true, true); + GameMessages::SendSetStunned(m_Parent->GetObjectID(), eStateChangeType::PUSH, m_Parent->GetSystemAddress(), LWOOBJID_EMPTY, true, false, true, false, false, false, false, true, true, true, true, true, true, true, true, true); - EntityManager::Instance()->SerializeEntity(m_Parent); - EntityManager::Instance()->SerializeEntity(mount); + Game::entityManager->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(mount); } void PossessorComponent::Dismount(Entity* mount, bool forceDismount) { @@ -71,12 +73,12 @@ void PossessorComponent::Dismount(Entity* mount, bool forceDismount) { possessableComponent->SetPossessor(LWOOBJID_EMPTY); if (forceDismount) possessableComponent->ForceDepossess(); } - EntityManager::Instance()->SerializeEntity(m_Parent); - EntityManager::Instance()->SerializeEntity(mount); + Game::entityManager->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(mount); auto characterComponent = m_Parent->GetComponent(); if (characterComponent) characterComponent->SetIsRacing(false); } // Make sure we don't have wacky controls - GameMessages::SendSetPlayerControlScheme(m_Parent, eControlSceme::SCHEME_A); + GameMessages::SendSetPlayerControlScheme(m_Parent, eControlScheme::SCHEME_A); } diff --git a/dGame/dComponents/PossessorComponent.h b/dGame/dComponents/PossessorComponent.h index 00b24445..4456af27 100644 --- a/dGame/dComponents/PossessorComponent.h +++ b/dGame/dComponents/PossessorComponent.h @@ -3,6 +3,7 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" // possession types enum class ePossessionType : uint8_t { @@ -17,7 +18,7 @@ enum class ePossessionType : uint8_t { */ class PossessorComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_POSSESSOR; + static const eReplicaComponentType ComponentType = eReplicaComponentType::POSSESSOR; PossessorComponent(Entity* parent); ~PossessorComponent() override; diff --git a/dGame/dComponents/PropertyComponent.h b/dGame/dComponents/PropertyComponent.h index 2096a475..41f93677 100644 --- a/dGame/dComponents/PropertyComponent.h +++ b/dGame/dComponents/PropertyComponent.h @@ -9,6 +9,7 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" struct PropertyState { LWOOBJID ownerID; @@ -21,7 +22,7 @@ struct PropertyState { */ class PropertyComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PROPERTY; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PROPERTY; explicit PropertyComponent(Entity* parentEntity); ~PropertyComponent() override; [[nodiscard]] PropertyState* GetPropertyState() const { return m_PropertyState; }; diff --git a/dGame/dComponents/PropertyEntranceComponent.cpp b/dGame/dComponents/PropertyEntranceComponent.cpp index e6540417..bff917d8 100644 --- a/dGame/dComponents/PropertyEntranceComponent.cpp +++ b/dGame/dComponents/PropertyEntranceComponent.cpp @@ -11,11 +11,14 @@ #include "CharacterComponent.h" #include "UserManager.h" #include "dLogger.h" +#include "Amf3.h" +#include "eObjectBits.h" +#include "eGameMasterLevel.h" PropertyEntranceComponent::PropertyEntranceComponent(uint32_t componentID, Entity* parent) : Component(parent) { this->propertyQueries = {}; - auto table = CDClientManager::Instance()->GetTable("PropertyEntranceComponent"); + auto table = CDClientManager::Instance().GetTable(); const auto& entry = table->GetByID(componentID); this->m_MapID = entry.mapID; @@ -33,12 +36,9 @@ void PropertyEntranceComponent::OnUse(Entity* entity) { AMFArrayValue args; - auto* state = new AMFStringValue(); - state->SetStringValue("property_menu"); + args.Insert("state", "property_menu"); - args.InsertValue("state", state); - - GameMessages::SendUIMessageServerToSingleClient(entity, entity->GetSystemAddress(), "pushGameState", &args); + GameMessages::SendUIMessageServerToSingleClient(entity, entity->GetSystemAddress(), "pushGameState", args); } void PropertyEntranceComponent::OnEnterProperty(Entity* entity, uint32_t index, bool returnToZone, const SystemAddress& sysAddr) { @@ -241,8 +241,8 @@ void PropertyEntranceComponent::OnPropertyEntranceSync(Entity* entity, bool incl // Convert owner char id to LWOOBJID LWOOBJID ownerObjId = owner; - ownerObjId = GeneralUtils::SetBit(ownerObjId, OBJECT_BIT_CHARACTER); - ownerObjId = GeneralUtils::SetBit(ownerObjId, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(ownerObjId, eObjectBits::CHARACTER); + GeneralUtils::SetBit(ownerObjId, eObjectBits::PERSISTENT); // Query to get friend and best friend fields auto friendCheck = Database::CreatePreppedStmt("SELECT best_friend FROM friends WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?)"); @@ -270,7 +270,7 @@ void PropertyEntranceComponent::OnPropertyEntranceSync(Entity* entity, bool incl bool isModeratorApproved = propertyEntry->getBoolean(10); - if (!isModeratorApproved && entity->GetGMLevel() >= GAME_MASTER_LEVEL_LEAD_MODERATOR) { + if (!isModeratorApproved && entity->GetGMLevel() >= eGameMasterLevel::LEAD_MODERATOR) { propertyName = "[AWAITING APPROVAL]"; propertyDescription = "[AWAITING APPROVAL]"; isModeratorApproved = true; diff --git a/dGame/dComponents/PropertyEntranceComponent.h b/dGame/dComponents/PropertyEntranceComponent.h index 4110e7d9..e37d1daa 100644 --- a/dGame/dComponents/PropertyEntranceComponent.h +++ b/dGame/dComponents/PropertyEntranceComponent.h @@ -6,13 +6,14 @@ #include "Entity.h" #include "EntityManager.h" #include "GameMessages.h" +#include "eReplicaComponentType.h" /** * Represents the launch pad that's used to select and browse properties */ class PropertyEntranceComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PROPERTY_ENTRANCE; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PROPERTY_ENTRANCE; explicit PropertyEntranceComponent(uint32_t componentID, Entity* parent); /** diff --git a/dGame/dComponents/PropertyManagementComponent.cpp b/dGame/dComponents/PropertyManagementComponent.cpp index f6954e4d..c8937c35 100644 --- a/dGame/dComponents/PropertyManagementComponent.cpp +++ b/dGame/dComponents/PropertyManagementComponent.cpp @@ -17,6 +17,9 @@ #include "Player.h" #include "RocketLaunchpadControlComponent.h" #include "PropertyEntranceComponent.h" +#include "InventoryComponent.h" +#include "eMissionTaskType.h" +#include "eObjectBits.h" #include #include "CppScripts.h" @@ -35,7 +38,7 @@ PropertyManagementComponent::PropertyManagementComponent(Entity* parent) : Compo instance = this; - const auto& worldId = dZoneManager::Instance()->GetZone()->GetZoneID(); + const auto& worldId = Game::zoneManager->GetZone()->GetZoneID(); const auto zoneId = worldId.GetMapID(); const auto cloneId = worldId.GetCloneID(); @@ -64,8 +67,8 @@ PropertyManagementComponent::PropertyManagementComponent(Entity* parent) : Compo if (propertyEntry->next()) { this->propertyId = propertyEntry->getUInt64(1); this->owner = propertyEntry->getUInt64(2); - this->owner = GeneralUtils::SetBit(this->owner, OBJECT_BIT_CHARACTER); - this->owner = GeneralUtils::SetBit(this->owner, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(this->owner, eObjectBits::CHARACTER); + GeneralUtils::SetBit(this->owner, eObjectBits::PERSISTENT); this->clone_Id = propertyEntry->getInt(2); this->propertyName = propertyEntry->getString(5).c_str(); this->propertyDescription = propertyEntry->getString(6).c_str(); @@ -87,7 +90,7 @@ LWOOBJID PropertyManagementComponent::GetOwnerId() const { } Entity* PropertyManagementComponent::GetOwner() const { - return EntityManager::Instance()->GetEntity(owner); + return Game::entityManager->GetEntity(owner); } void PropertyManagementComponent::SetOwner(Entity* value) { @@ -95,7 +98,7 @@ void PropertyManagementComponent::SetOwner(Entity* value) { } std::vector PropertyManagementComponent::GetPaths() const { - const auto zoneId = dZoneManager::Instance()->GetZone()->GetWorldID(); + const auto zoneId = Game::zoneManager->GetZone()->GetWorldID(); auto query = CDClientDatabase::CreatePreppedStmt( "SELECT path FROM PropertyTemplate WHERE mapID = ?;"); @@ -182,14 +185,14 @@ bool PropertyManagementComponent::Claim(const LWOOBJID playerId) { return false; } - auto* entity = EntityManager::Instance()->GetEntity(playerId); + auto* entity = Game::entityManager->GetEntity(playerId); auto* user = entity->GetParentUser(); auto character = entity->GetCharacter(); if (!character) return false; - auto* zone = dZoneManager::Instance()->GetZone(); + auto* zone = Game::zoneManager->GetZone(); const auto& worldId = zone->GetZoneID(); const auto propertyZoneId = worldId.GetMapID(); @@ -200,6 +203,16 @@ bool PropertyManagementComponent::Claim(const LWOOBJID playerId) { // If we are not on our clone do not allow us to claim the property if (propertyCloneId != playerCloneId) return false; + std::string name = zone->GetZoneName(); + std::string description = ""; + + auto prop_path = zone->GetPath(m_Parent->GetVarAsString(u"propertyName")); + + if (prop_path){ + if (!prop_path->property.displayName.empty()) name = prop_path->property.displayName; + description = prop_path->property.displayDesc; + } + SetOwnerId(playerId); propertyId = ObjectIDManager::GenerateRandomObjectID(); @@ -207,14 +220,15 @@ bool PropertyManagementComponent::Claim(const LWOOBJID playerId) { auto* insertion = Database::CreatePreppedStmt( "INSERT INTO properties" "(id, owner_id, template_id, clone_id, name, description, rent_amount, rent_due, privacy_option, last_updated, time_claimed, rejection_reason, reputation, zone_id, performance_cost)" - "VALUES (?, ?, ?, ?, ?, '', 0, 0, 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), '', 0, ?, 0.0)" + "VALUES (?, ?, ?, ?, ?, ?, 0, 0, 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), '', 0, ?, 0.0)" ); insertion->setUInt64(1, propertyId); insertion->setUInt64(2, (uint32_t)playerId); insertion->setUInt(3, templateId); insertion->setUInt64(4, playerCloneId); - insertion->setString(5, zone->GetZoneName().c_str()); - insertion->setInt(6, propertyZoneId); + insertion->setString(5, name.c_str()); + insertion->setString(6, description.c_str()); + insertion->setInt(7, propertyZoneId); // Try and execute the query, print an error if it fails. try { @@ -226,7 +240,7 @@ bool PropertyManagementComponent::Claim(const LWOOBJID playerId) { return false; } - auto* zoneControlObject = dZoneManager::Instance()->GetZoneControlObject(); + auto* zoneControlObject = Game::zoneManager->GetZoneControlObject(); for (CppScripts::Script* script : CppScripts::GetEntityScripts(zoneControlObject)) { script->OnZonePropertyRented(zoneControlObject, entity); } @@ -242,7 +256,7 @@ void PropertyManagementComponent::OnStartBuilding() { LWOMAPID zoneId = 1100; - const auto entrance = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_PROPERTY_ENTRANCE); + const auto entrance = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::PROPERTY_ENTRANCE); originalPrivacyOption = privacyOption; @@ -325,9 +339,9 @@ void PropertyManagementComponent::UpdateModelPosition(const LWOOBJID id, const N info.settings.push_back(setting->Copy()); } - Entity* newEntity = EntityManager::Instance()->CreateEntity(info); + Entity* newEntity = Game::entityManager->CreateEntity(info); if (newEntity != nullptr) { - EntityManager::Instance()->ConstructEntity(newEntity); + Game::entityManager->ConstructEntity(newEntity); // Make sure the propMgmt doesn't delete our model after the server dies // Trying to do this after the entity is constructed. Shouldn't really change anything but @@ -357,18 +371,17 @@ void PropertyManagementComponent::UpdateModelPosition(const LWOOBJID id, const N info.respawnTime = 10; info.emulated = true; - info.emulator = EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(); + info.emulator = Game::entityManager->GetZoneControlEntity()->GetObjectID(); - LWOOBJID id = static_cast(persistentId) | 1ull << OBJECT_BIT_CLIENT; + info.spawnerID = persistentId; + GeneralUtils::SetBit(info.spawnerID, eObjectBits::CLIENT); - info.spawnerID = id; + const auto spawnerId = Game::zoneManager->MakeSpawner(info); - const auto spawnerId = dZoneManager::Instance()->MakeSpawner(info); - - auto* spawner = dZoneManager::Instance()->GetSpawner(spawnerId); + auto* spawner = Game::zoneManager->GetSpawner(spawnerId); auto ldfModelBehavior = new LDFData(u"modelBehaviors", 0); - auto userModelID = new LDFData(u"userModelID", id); + auto userModelID = new LDFData(u"userModelID", info.spawnerID); auto modelType = new LDFData(u"modelType", 2); auto propertyObjectID = new LDFData(u"propertyObjectID", true); auto componentWhitelist = new LDFData(u"componentWhitelist", 1); @@ -388,11 +401,11 @@ void PropertyManagementComponent::UpdateModelPosition(const LWOOBJID id, const N GameMessages::SendGetModelsOnProperty(entity->GetObjectID(), GetModels(), UNASSIGNED_SYSTEM_ADDRESS); - EntityManager::Instance()->GetZoneControlEntity()->OnZonePropertyModelPlaced(entity); + Game::entityManager->GetZoneControlEntity()->OnZonePropertyModelPlaced(entity); }); // Progress place model missions auto missionComponent = entity->GetComponent(); - if (missionComponent != nullptr) missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_PLACE_MODEL, 0); + if (missionComponent != nullptr) missionComponent->Progress(eMissionTaskType::PLACE_MODEL, 0); } void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int deleteReason) { @@ -420,7 +433,7 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet const auto spawnerId = index->second; - auto* spawner = dZoneManager::Instance()->GetSpawner(spawnerId); + auto* spawner = Game::zoneManager->GetSpawner(spawnerId); models.erase(id); @@ -428,7 +441,7 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet Game::logger->Log("PropertyManagementComponent", "Failed to find spawner"); } - auto* model = EntityManager::Instance()->GetEntity(id); + auto* model = Game::entityManager->GetEntity(id); if (model == nullptr) { Game::logger->Log("PropertyManagementComponent", "Failed to find model entity"); @@ -436,7 +449,7 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet return; } - EntityManager::Instance()->DestructEntity(model); + Game::entityManager->DestructEntity(model); Game::logger->Log("PropertyManagementComponent", "Deleting model LOT %i", model->GetLOT()); @@ -463,7 +476,7 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet settings.push_back(propertyObjectID); settings.push_back(modelType); - inventoryComponent->AddItem(6662, 1, eLootSourceType::LOOT_SOURCE_DELETION, eInventoryType::HIDDEN, settings, LWOOBJID_EMPTY, false, false, spawnerId); + inventoryComponent->AddItem(6662, 1, eLootSourceType::DELETION, eInventoryType::MODELS_IN_BBB, settings, LWOOBJID_EMPTY, false, false, spawnerId); auto* item = inventoryComponent->FindItemBySubKey(spawnerId); if (item == nullptr) { @@ -483,9 +496,9 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet GameMessages::SendPlaceModelResponse(entity->GetObjectID(), entity->GetSystemAddress(), NiPoint3::ZERO, LWOOBJID_EMPTY, 16, NiQuaternion::IDENTITY); if (spawner != nullptr) { - dZoneManager::Instance()->RemoveSpawner(spawner->m_Info.spawnerID); + Game::zoneManager->RemoveSpawner(spawner->m_Info.spawnerID); } else { - model->Smash(SILENT); + model->Smash(LWOOBJID_EMPTY, eKillType::SILENT); } item->SetCount(0, true, false, false); @@ -493,7 +506,7 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet return; } - inventoryComponent->AddItem(model->GetLOT(), 1, eLootSourceType::LOOT_SOURCE_DELETION, INVALID, {}, LWOOBJID_EMPTY, false); + inventoryComponent->AddItem(model->GetLOT(), 1, eLootSourceType::DELETION, INVALID, {}, LWOOBJID_EMPTY, false); auto* item = inventoryComponent->FindItemByLot(model->GetLOT()); @@ -507,13 +520,13 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet item->Equip(); GameMessages::SendUGCEquipPostDeleteBasedOnEditMode(entity->GetObjectID(), entity->GetSystemAddress(), item->GetId(), item->GetCount()); - EntityManager::Instance()->GetZoneControlEntity()->OnZonePropertyModelPickedUp(entity); + Game::entityManager->GetZoneControlEntity()->OnZonePropertyModelPickedUp(entity); break; } case 1: // Return to inv { - EntityManager::Instance()->GetZoneControlEntity()->OnZonePropertyModelRemoved(entity); + Game::entityManager->GetZoneControlEntity()->OnZonePropertyModelRemoved(entity); break; } @@ -536,9 +549,9 @@ void PropertyManagementComponent::DeleteModel(const LWOOBJID id, const int delet GameMessages::SendPlaceModelResponse(entity->GetObjectID(), entity->GetSystemAddress(), NiPoint3::ZERO, LWOOBJID_EMPTY, 16, NiQuaternion::IDENTITY); if (spawner != nullptr) { - dZoneManager::Instance()->RemoveSpawner(spawner->m_Info.spawnerID); + Game::zoneManager->RemoveSpawner(spawner->m_Info.spawnerID); } else { - model->Smash(SILENT); + model->Smash(LWOOBJID_EMPTY, eKillType::SILENT); } } @@ -600,7 +613,7 @@ void PropertyManagementComponent::Load() { info.respawnTime = 10; //info.emulated = true; - //info.emulator = EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(); + //info.emulator = Game::entityManager->GetZoneControlEntity()->GetObjectID(); info.spawnerID = id; @@ -609,8 +622,8 @@ void PropertyManagementComponent::Load() { //BBB property models need to have extra stuff set for them: if (lot == 14) { LWOOBJID blueprintID = lookupResult->getUInt(10); - blueprintID = GeneralUtils::SetBit(blueprintID, OBJECT_BIT_CHARACTER); - blueprintID = GeneralUtils::SetBit(blueprintID, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(blueprintID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(blueprintID, eObjectBits::PERSISTENT); LDFBaseData* ldfBlueprintID = new LDFData(u"blueprintid", blueprintID); LDFBaseData* componentWhitelist = new LDFData(u"componentWhitelist", 1); @@ -639,9 +652,9 @@ void PropertyManagementComponent::Load() { node->config = settings; - const auto spawnerId = dZoneManager::Instance()->MakeSpawner(info); + const auto spawnerId = Game::zoneManager->MakeSpawner(info); - auto* spawner = dZoneManager::Instance()->GetSpawner(spawnerId); + auto* spawner = Game::zoneManager->GetSpawner(spawnerId); auto* model = spawner->Spawn(); @@ -656,7 +669,7 @@ void PropertyManagementComponent::Save() { return; } - auto* insertion = Database::CreatePreppedStmt("INSERT INTO properties_contents VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"); + auto* insertion = Database::CreatePreppedStmt("INSERT INTO properties_contents VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"); auto* update = Database::CreatePreppedStmt("UPDATE properties_contents SET x = ?, y = ?, z = ?, rx = ?, ry = ?, rz = ?, rw = ? WHERE id = ?;"); auto* lookup = Database::CreatePreppedStmt("SELECT id FROM properties_contents WHERE property_id = ?;"); auto* remove = Database::CreatePreppedStmt("DELETE FROM properties_contents WHERE id = ?;"); @@ -685,7 +698,7 @@ void PropertyManagementComponent::Save() { modelIds.push_back(id); - auto* entity = EntityManager::Instance()->GetEntity(pair.first); + auto* entity = Game::entityManager->GetEntity(pair.first); if (entity == nullptr) { continue; @@ -706,6 +719,13 @@ void PropertyManagementComponent::Save() { insertion->setDouble(9, rotation.y); insertion->setDouble(10, rotation.z); insertion->setDouble(11, rotation.w); + insertion->setString(12, ("Objects_" + std::to_string(entity->GetLOT()) + "_name").c_str()); // Model name. TODO make this customizable + insertion->setString(13, ""); // Model description. TODO implement this. + insertion->setDouble(14, 0); // behavior 1. TODO implement this. + insertion->setDouble(15, 0); // behavior 2. TODO implement this. + insertion->setDouble(16, 0); // behavior 3. TODO implement this. + insertion->setDouble(17, 0); // behavior 4. TODO implement this. + insertion->setDouble(18, 0); // behavior 5. TODO implement this. try { insertion->execute(); } catch (sql::SQLException& ex) { @@ -766,7 +786,7 @@ void PropertyManagementComponent::OnQueryPropertyData(Entity* originator, const author = m_Parent->GetObjectID(); } - const auto& worldId = dZoneManager::Instance()->GetZone()->GetZoneID(); + const auto& worldId = Game::zoneManager->GetZone()->GetZoneID(); const auto zoneId = worldId.GetMapID(); Game::logger->Log("Properties", "Getting property info for %d", zoneId); diff --git a/dGame/dComponents/PropertyManagementComponent.h b/dGame/dComponents/PropertyManagementComponent.h index d9526015..2ee010a8 100644 --- a/dGame/dComponents/PropertyManagementComponent.h +++ b/dGame/dComponents/PropertyManagementComponent.h @@ -3,6 +3,7 @@ #include #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Information regarding which players may visit this property @@ -31,7 +32,7 @@ enum class PropertyPrivacyOption class PropertyManagementComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PROPERTY_MANAGEMENT; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PROPERTY_MANAGEMENT; PropertyManagementComponent(Entity* parent); static PropertyManagementComponent* Instance(); diff --git a/dGame/dComponents/PropertyVendorComponent.cpp b/dGame/dComponents/PropertyVendorComponent.cpp index ed89bfc7..cff4995c 100644 --- a/dGame/dComponents/PropertyVendorComponent.cpp +++ b/dGame/dComponents/PropertyVendorComponent.cpp @@ -43,7 +43,7 @@ void PropertyVendorComponent::OnBuyFromVendor(Entity* originator, const bool con GameMessages::SendPropertyRentalResponse(m_Parent->GetObjectID(), 0, 0, 0, 0, originator->GetSystemAddress()); - auto* controller = dZoneManager::Instance()->GetZoneControlObject(); + auto* controller = Game::zoneManager->GetZoneControlObject(); controller->OnFireEventServerSide(m_Parent, "propertyRented"); diff --git a/dGame/dComponents/PropertyVendorComponent.h b/dGame/dComponents/PropertyVendorComponent.h index f947d745..5055b445 100644 --- a/dGame/dComponents/PropertyVendorComponent.h +++ b/dGame/dComponents/PropertyVendorComponent.h @@ -2,6 +2,7 @@ #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * The property guard that stands on a property before it's claimed, allows entities to attempt claiming this property. @@ -9,7 +10,7 @@ class PropertyVendorComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PROPERTY_VENDOR; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PROPERTY_VENDOR; explicit PropertyVendorComponent(Entity* parent); /** diff --git a/dGame/dComponents/ProximityMonitorComponent.h b/dGame/dComponents/ProximityMonitorComponent.h index a98397a2..2f51917d 100644 --- a/dGame/dComponents/ProximityMonitorComponent.h +++ b/dGame/dComponents/ProximityMonitorComponent.h @@ -11,6 +11,7 @@ #include "dpWorld.h" #include "dpEntity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Utility component for detecting how close entities are to named proximities for this entity. Allows you to store @@ -18,7 +19,7 @@ */ class ProximityMonitorComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PROXIMITY_MONITOR; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PROXIMITY_MONITOR; ProximityMonitorComponent(Entity* parentEntity, int smallRadius = -1, int largeRadius = -1); ~ProximityMonitorComponent() override; diff --git a/dGame/dComponents/RacingControlComponent.cpp b/dGame/dComponents/RacingControlComponent.cpp index 75465669..5de24445 100644 --- a/dGame/dComponents/RacingControlComponent.cpp +++ b/dGame/dComponents/RacingControlComponent.cpp @@ -15,12 +15,17 @@ #include "Player.h" #include "PossessableComponent.h" #include "PossessorComponent.h" -#include "RacingTaskParam.h" +#include "eRacingTaskParam.h" #include "Spawner.h" #include "VehiclePhysicsComponent.h" #include "dServer.h" #include "dZoneManager.h" #include "dConfig.h" +#include "Loot.h" +#include "eMissionTaskType.h" +#include "LeaderboardManager.h" +#include "dZoneManager.h" +#include "CDActivitiesTable.h" #ifndef M_PI #define M_PI 3.14159265358979323846264338327950288 @@ -43,36 +48,14 @@ RacingControlComponent::RacingControlComponent(Entity* parent) m_EmptyTimer = 0; m_SoloRacing = Game::config->GetValue("solo_racing") == "1"; - // Select the main world ID as fallback when a player fails to load. - + m_MainWorld = 1200; const auto worldID = Game::server->GetZoneID(); + if (Game::zoneManager->CheckIfAccessibleZone((worldID/10)*10)) m_MainWorld = (worldID/10)*10; - switch (worldID) { - case 1203: - m_ActivityID = 42; - m_MainWorld = 1200; - break; - - case 1261: - m_ActivityID = 60; - m_MainWorld = 1260; - break; - - case 1303: - m_ActivityID = 39; - m_MainWorld = 1300; - break; - - case 1403: - m_ActivityID = 54; - m_MainWorld = 1400; - break; - - default: - m_ActivityID = 42; - m_MainWorld = 1200; - break; - } + m_ActivityID = 42; + CDActivitiesTable* activitiesTable = CDClientManager::Instance().GetTable(); + std::vector activities = activitiesTable->Query([=](CDActivities entry) {return (entry.instanceMapID == worldID); }); + for (CDActivities activity : activities) m_ActivityID = activity.ActivityID; } RacingControlComponent::~RacingControlComponent() {} @@ -98,7 +81,7 @@ void RacingControlComponent::OnPlayerLoaded(Entity* player) { } void RacingControlComponent::LoadPlayerVehicle(Entity* player, - bool initialLoad) { + uint32_t positionNumber, bool initialLoad) { // Load the player's vehicle. if (player == nullptr) { @@ -123,42 +106,26 @@ void RacingControlComponent::LoadPlayerVehicle(Entity* player, // Calculate the vehicle's starting position. - auto* path = dZoneManager::Instance()->GetZone()->GetPath( + auto* path = Game::zoneManager->GetZone()->GetPath( GeneralUtils::UTF16ToWTF8(m_PathName)); - auto startPosition = path->pathWaypoints[0].position + NiPoint3::UNIT_Y * 3; - - const auto spacing = 15; - - // This sometimes spawns the vehicle out of the map if there are lots of - // players loaded. - - const auto range = m_LoadedPlayers * spacing; - - startPosition = - startPosition + NiPoint3::UNIT_Z * ((m_LeadingPlayer / 2) + - m_RacingPlayers.size() * spacing); - - auto startRotation = - NiQuaternion::LookAt(startPosition, startPosition + NiPoint3::UNIT_X); - - auto angles = startRotation.GetEulerAngles(); - - angles.y -= M_PI; - - startRotation = NiQuaternion::FromEulerAngles(angles); - - Game::logger->Log("RacingControlComponent", - "Start position <%f, %f, %f>, <%f, %f, %f>", - startPosition.x, startPosition.y, startPosition.z, - angles.x * (180.0f / M_PI), angles.y * (180.0f / M_PI), - angles.z * (180.0f / M_PI)); + auto spawnPointEntities = Game::entityManager->GetEntitiesByLOT(4843); + auto startPosition = NiPoint3::ZERO; + auto startRotation = NiQuaternion::IDENTITY; + const std::string placementAsString = std::to_string(positionNumber); + for (auto entity : spawnPointEntities) { + if (!entity) continue; + if (entity->GetVarAsString(u"placement") == placementAsString) { + startPosition = entity->GetPosition(); + startRotation = entity->GetRotation(); + break; + } + } // Make sure the player is at the correct position. GameMessages::SendTeleport(player->GetObjectID(), startPosition, - startRotation, player->GetSystemAddress(), true, - true); + startRotation, player->GetSystemAddress(), true); // Spawn the vehicle entity. @@ -169,7 +136,7 @@ void RacingControlComponent::LoadPlayerVehicle(Entity* player, info.spawnerID = m_Parent->GetObjectID(); auto* carEntity = - EntityManager::Instance()->CreateEntity(info, nullptr, m_Parent); + Game::entityManager->CreateEntity(info, nullptr, m_Parent); // Make the vehicle a child of the racing controller. m_Parent->AddChild(carEntity); @@ -240,9 +207,9 @@ void RacingControlComponent::LoadPlayerVehicle(Entity* player, // Construct and serialize everything when done. - EntityManager::Instance()->ConstructEntity(carEntity); - EntityManager::Instance()->SerializeEntity(player); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->ConstructEntity(carEntity); + Game::entityManager->SerializeEntity(player); + Game::entityManager->SerializeEntity(m_Parent); GameMessages::SendRacingSetPlayerResetInfo( m_Parent->GetObjectID(), 0, 0, player->GetObjectID(), startPosition, 1, @@ -253,7 +220,7 @@ void RacingControlComponent::LoadPlayerVehicle(Entity* player, // Reset the player to the start position during downtime, in case something // went wrong. m_Parent->AddCallbackTimer(1, [this, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) { return; @@ -276,11 +243,9 @@ void RacingControlComponent::LoadPlayerVehicle(Entity* player, // Make sure everything has the correct position. GameMessages::SendTeleport(player->GetObjectID(), startPosition, - startRotation, player->GetSystemAddress(), true, - true); + startRotation, player->GetSystemAddress(), true); GameMessages::SendTeleport(carEntity->GetObjectID(), startPosition, - startRotation, player->GetSystemAddress(), true, - true); + startRotation, player->GetSystemAddress(), true); } void RacingControlComponent::OnRacingClientReady(Entity* player) { @@ -304,7 +269,7 @@ void RacingControlComponent::OnRacingClientReady(Entity* player) { racingPlayer.vehicleID, UNASSIGNED_SYSTEM_ADDRESS); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void RacingControlComponent::OnRequestDie(Entity* player) { @@ -317,32 +282,60 @@ void RacingControlComponent::OnRequestDie(Entity* player) { } auto* vehicle = - EntityManager::Instance()->GetEntity(racingPlayer.vehicleID); + Game::entityManager->GetEntity(racingPlayer.vehicleID); - if (vehicle == nullptr) { - return; - } + if (!vehicle) return; if (!racingPlayer.noSmashOnReload) { racingPlayer.smashedTimes++; + GameMessages::SendDie(vehicle, vehicle->GetObjectID(), LWOOBJID_EMPTY, true, + eKillType::VIOLENT, u"", 0, 0, 90.0f, false, true, 0); + + auto* destroyableComponent = vehicle->GetComponent(); + uint32_t respawnImagination = 0; + // Reset imagination to half its current value, rounded up to the nearest value divisible by 10, as it was done in live. + // Do not actually change the value yet. Do that on respawn. + if (destroyableComponent) { + respawnImagination = static_cast(ceil(destroyableComponent->GetImagination() / 2.0f / 10.0f)) * 10.0f; + GameMessages::SendSetResurrectRestoreValues(vehicle, -1, -1, respawnImagination); + } + + // Respawn the player in 2 seconds, as was done in live. Not sure if this value is in a setting somewhere else... + vehicle->AddCallbackTimer(2.0f, [=]() { + if (!vehicle || !this->m_Parent) return; + GameMessages::SendRacingResetPlayerToLastReset( + m_Parent->GetObjectID(), racingPlayer.playerID, + UNASSIGNED_SYSTEM_ADDRESS); + + GameMessages::SendVehicleStopBoost(vehicle, player->GetSystemAddress(), true); + + GameMessages::SendRacingSetPlayerResetInfo( + m_Parent->GetObjectID(), racingPlayer.lap, + racingPlayer.respawnIndex, player->GetObjectID(), + racingPlayer.respawnPosition, racingPlayer.respawnIndex + 1, + UNASSIGNED_SYSTEM_ADDRESS); + + GameMessages::SendResurrect(vehicle); + auto* destroyableComponent = vehicle->GetComponent(); + // Reset imagination to half its current value, rounded up to the nearest value divisible by 10, as it was done in live. + if (destroyableComponent) destroyableComponent->SetImagination(respawnImagination); + Game::entityManager->SerializeEntity(vehicle); + }); + + auto* characterComponent = player->GetComponent(); + if (characterComponent != nullptr) { + characterComponent->UpdatePlayerStatistic(RacingTimesWrecked); + } + } else { + GameMessages::SendRacingSetPlayerResetInfo( + m_Parent->GetObjectID(), racingPlayer.lap, + racingPlayer.respawnIndex, player->GetObjectID(), + racingPlayer.respawnPosition, racingPlayer.respawnIndex + 1, + UNASSIGNED_SYSTEM_ADDRESS); + GameMessages::SendRacingResetPlayerToLastReset( + m_Parent->GetObjectID(), racingPlayer.playerID, + UNASSIGNED_SYSTEM_ADDRESS); } - - // Reset player to last checkpoint - GameMessages::SendRacingSetPlayerResetInfo( - m_Parent->GetObjectID(), racingPlayer.lap, - racingPlayer.respawnIndex, player->GetObjectID(), - racingPlayer.respawnPosition, racingPlayer.respawnIndex + 1, - UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendRacingResetPlayerToLastReset( - m_Parent->GetObjectID(), racingPlayer.playerID, - UNASSIGNED_SYSTEM_ADDRESS); - - auto* characterComponent = player->GetComponent(); - if (characterComponent != nullptr) { - characterComponent->UpdatePlayerStatistic(RacingTimesWrecked); - } - - return; } } @@ -355,33 +348,19 @@ void RacingControlComponent::OnRacingPlayerInfoResetFinished(Entity* player) { } auto* vehicle = - EntityManager::Instance()->GetEntity(racingPlayer.vehicleID); + Game::entityManager->GetEntity(racingPlayer.vehicleID); if (vehicle == nullptr) { return; } - if (!racingPlayer.noSmashOnReload) { - GameMessages::SendDie(vehicle, LWOOBJID_EMPTY, LWOOBJID_EMPTY, true, - VIOLENT, u"", 0, 0, 0, true, false, 0); - - GameMessages::SendVehicleUnlockInput(racingPlayer.vehicleID, false, - UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendVehicleSetWheelLockState( - racingPlayer.vehicleID, false, false, - UNASSIGNED_SYSTEM_ADDRESS); - - GameMessages::SendResurrect(vehicle); - } - racingPlayer.noSmashOnReload = false; return; } } -void RacingControlComponent::HandleMessageBoxResponse(Entity* player, - const std::string& id) { +void RacingControlComponent::HandleMessageBoxResponse(Entity* player, int32_t button, const std::string& id) { auto* data = GetPlayerData(player->GetObjectID()); if (data == nullptr) { @@ -389,15 +368,17 @@ void RacingControlComponent::HandleMessageBoxResponse(Entity* player, } if (id == "rewardButton") { - if (data->collectedRewards) { - return; - } + if (data->collectedRewards) return; data->collectedRewards = true; // Calculate the score, different loot depending on player count - const auto score = m_LoadedPlayers * 10 + data->finished; + auto playersRating = m_LoadedPlayers; + if(m_LoadedPlayers == 1 && m_SoloRacing) { + playersRating *= 2; + } + const auto score = playersRating * 10 + data->finished; LootGenerator::Instance().GiveActivityLoot(player, m_Parent, m_ActivityID, score); // Giving rewards @@ -409,22 +390,22 @@ void RacingControlComponent::HandleMessageBoxResponse(Entity* player, if (missionComponent == nullptr) return; - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, 0, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_COMPETED_IN_RACE); // Progress task for competing in a race - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, data->smashedTimes, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_SAFE_DRIVER); // Finish a race without being smashed. + missionComponent->Progress(eMissionTaskType::RACING, 0, (LWOOBJID)eRacingTaskParam::COMPETED_IN_RACE); // Progress task for competing in a race + missionComponent->Progress(eMissionTaskType::RACING, data->smashedTimes, (LWOOBJID)eRacingTaskParam::SAFE_DRIVER); // Finish a race without being smashed. // If solo racing is enabled OR if there are 3 players in the race, progress placement tasks. if (m_SoloRacing || m_LoadedPlayers > 2) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, data->finished, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_FINISH_WITH_PLACEMENT); // Finish in 1st place on a race + missionComponent->Progress(eMissionTaskType::RACING, data->finished, (LWOOBJID)eRacingTaskParam::FINISH_WITH_PLACEMENT); // Finish in 1st place on a race if (data->finished == 1) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, dZoneManager::Instance()->GetZone()->GetWorldID(), (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_FIRST_PLACE_MULTIPLE_TRACKS); // Finish in 1st place on multiple tracks. - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, dZoneManager::Instance()->GetZone()->GetWorldID(), (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_WIN_RACE_IN_WORLD); // Finished first place in specific world. + missionComponent->Progress(eMissionTaskType::RACING, Game::zoneManager->GetZone()->GetWorldID(), (LWOOBJID)eRacingTaskParam::FIRST_PLACE_MULTIPLE_TRACKS); // Finish in 1st place on multiple tracks. + missionComponent->Progress(eMissionTaskType::RACING, Game::zoneManager->GetZone()->GetWorldID(), (LWOOBJID)eRacingTaskParam::WIN_RACE_IN_WORLD); // Finished first place in specific world. } if (data->finished == m_LoadedPlayers) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, dZoneManager::Instance()->GetZone()->GetWorldID(), (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_LAST_PLACE_FINISH); // Finished first place in specific world. + missionComponent->Progress(eMissionTaskType::RACING, Game::zoneManager->GetZone()->GetWorldID(), (LWOOBJID)eRacingTaskParam::LAST_PLACE_FINISH); // Finished first place in specific world. } } - } else if (id == "ACT_RACE_EXIT_THE_RACE?" || id == "Exit") { - auto* vehicle = EntityManager::Instance()->GetEntity(data->vehicleID); + } else if ((id == "ACT_RACE_EXIT_THE_RACE?" || id == "Exit") && button == m_ActivityExitConfirm) { + auto* vehicle = Game::entityManager->GetEntity(data->vehicleID); if (vehicle == nullptr) { return; @@ -525,7 +506,7 @@ void RacingControlComponent::Update(float deltaTime) { // Check if any players has disconnected before loading in for (size_t i = 0; i < m_LobbyPlayers.size(); i++) { auto* playerEntity = - EntityManager::Instance()->GetEntity(m_LobbyPlayers[i]); + Game::entityManager->GetEntity(m_LobbyPlayers[i]); if (playerEntity == nullptr) { --m_LoadedPlayers; @@ -547,7 +528,7 @@ void RacingControlComponent::Update(float deltaTime) { if (m_EmptyTimer >= 30) { for (const auto player : m_LobbyPlayers) { auto* playerEntity = - EntityManager::Instance()->GetEntity(player); + Game::entityManager->GetEntity(player); if (playerEntity == nullptr) { continue; @@ -567,12 +548,12 @@ void RacingControlComponent::Update(float deltaTime) { Game::logger->Log("RacingControlComponent", "Loading all players..."); - for (size_t i = 0; i < m_LobbyPlayers.size(); i++) { + for (size_t positionNumber = 0; positionNumber < m_LobbyPlayers.size(); positionNumber++) { Game::logger->Log("RacingControlComponent", "Loading player now!"); auto* player = - EntityManager::Instance()->GetEntity(m_LobbyPlayers[i]); + Game::entityManager->GetEntity(m_LobbyPlayers[positionNumber]); if (player == nullptr) { return; @@ -581,7 +562,7 @@ void RacingControlComponent::Update(float deltaTime) { Game::logger->Log("RacingControlComponent", "Loading player now NOW!"); - LoadPlayerVehicle(player, true); + LoadPlayerVehicle(player, positionNumber + 1, true); m_Loaded = true; } @@ -596,7 +577,7 @@ void RacingControlComponent::Update(float deltaTime) { if (!m_Started) { // Check if anyone has disconnected during this period for (size_t i = 0; i < m_RacingPlayers.size(); i++) { - auto* playerEntity = EntityManager::Instance()->GetEntity( + auto* playerEntity = Game::entityManager->GetEntity( m_RacingPlayers[i].playerID); if (playerEntity == nullptr) { @@ -612,7 +593,7 @@ void RacingControlComponent::Update(float deltaTime) { if (m_LoadedPlayers < 2 && !(m_LoadedPlayers == 1 && m_SoloRacing)) { for (const auto player : m_LobbyPlayers) { auto* playerEntity = - EntityManager::Instance()->GetEntity(player); + Game::entityManager->GetEntity(player); if (playerEntity == nullptr) { continue; @@ -645,15 +626,15 @@ void RacingControlComponent::Update(float deltaTime) { for (const auto& player : m_RacingPlayers) { auto* vehicle = - EntityManager::Instance()->GetEntity(player.vehicleID); + Game::entityManager->GetEntity(player.vehicleID); auto* playerEntity = - EntityManager::Instance()->GetEntity(player.playerID); + Game::entityManager->GetEntity(player.playerID); if (vehicle != nullptr && playerEntity != nullptr) { GameMessages::SendTeleport( player.playerID, player.respawnPosition, player.respawnRotation, - playerEntity->GetSystemAddress(), true, true); + playerEntity->GetSystemAddress(), true); vehicle->SetPosition(player.respawnPosition); vehicle->SetRotation(player.respawnRotation); @@ -665,18 +646,18 @@ void RacingControlComponent::Update(float deltaTime) { destroyableComponent->SetImagination(0); } - EntityManager::Instance()->SerializeEntity(vehicle); - EntityManager::Instance()->SerializeEntity( + Game::entityManager->SerializeEntity(vehicle); + Game::entityManager->SerializeEntity( playerEntity); } } // Spawn imagination pickups - auto* minSpawner = dZoneManager::Instance()->GetSpawnersByName( + auto* minSpawner = Game::zoneManager->GetSpawnersByName( "ImaginationSpawn_Min")[0]; - auto* medSpawner = dZoneManager::Instance()->GetSpawnersByName( + auto* medSpawner = Game::zoneManager->GetSpawnersByName( "ImaginationSpawn_Med")[0]; - auto* maxSpawner = dZoneManager::Instance()->GetSpawnersByName( + auto* maxSpawner = Game::zoneManager->GetSpawnersByName( "ImaginationSpawn_Max")[0]; minSpawner->Activate(); @@ -692,9 +673,9 @@ void RacingControlComponent::Update(float deltaTime) { // Reset players to their start location, without smashing them for (auto& player : m_RacingPlayers) { auto* vehicleEntity = - EntityManager::Instance()->GetEntity(player.vehicleID); + Game::entityManager->GetEntity(player.vehicleID); auto* playerEntity = - EntityManager::Instance()->GetEntity(player.playerID); + Game::entityManager->GetEntity(player.playerID); if (vehicleEntity == nullptr || playerEntity == nullptr) { continue; @@ -711,9 +692,9 @@ void RacingControlComponent::Update(float deltaTime) { // Activate the players movement for (auto& player : m_RacingPlayers) { auto* vehicleEntity = - EntityManager::Instance()->GetEntity(player.vehicleID); + Game::entityManager->GetEntity(player.vehicleID); auto* playerEntity = - EntityManager::Instance()->GetEntity(player.playerID); + Game::entityManager->GetEntity(player.playerID); if (vehicleEntity == nullptr || playerEntity == nullptr) { continue; @@ -731,7 +712,7 @@ void RacingControlComponent::Update(float deltaTime) { Game::logger->Log("RacingControlComponent", "Starting race"); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); m_StartTime = std::time(nullptr); } @@ -745,13 +726,13 @@ void RacingControlComponent::Update(float deltaTime) { } // Race routines - auto* path = dZoneManager::Instance()->GetZone()->GetPath( + auto* path = Game::zoneManager->GetZone()->GetPath( GeneralUtils::UTF16ToWTF8(m_PathName)); for (auto& player : m_RacingPlayers) { - auto* vehicle = EntityManager::Instance()->GetEntity(player.vehicleID); + auto* vehicle = Game::entityManager->GetEntity(player.vehicleID); auto* playerEntity = - EntityManager::Instance()->GetEntity(player.playerID); + Game::entityManager->GetEntity(player.playerID); if (vehicle == nullptr || playerEntity == nullptr) { continue; @@ -763,7 +744,7 @@ void RacingControlComponent::Update(float deltaTime) { // be smashed by death plane if (vehiclePosition.y < -500) { GameMessages::SendDie(vehicle, m_Parent->GetObjectID(), - LWOOBJID_EMPTY, true, VIOLENT, u"", 0, 0, 0, + LWOOBJID_EMPTY, true, eKillType::VIOLENT, u"", 0, 0, 0, true, false, 0); OnRequestDie(playerEntity); @@ -842,7 +823,7 @@ void RacingControlComponent::Update(float deltaTime) { if (missionComponent != nullptr) { // Progress lap time tasks - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, (lapTime) * 1000, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_LAP_TIME); + missionComponent->Progress(eMissionTaskType::RACING, (lapTime) * 1000, (LWOOBJID)eRacingTaskParam::LAP_TIME); if (player.lap == 3) { m_Finished++; @@ -857,8 +838,9 @@ void RacingControlComponent::Update(float deltaTime) { "Completed time %llu, %llu", raceTime, raceTime * 1000); + LeaderboardManager::SaveScore(playerEntity->GetObjectID(), m_ActivityID, static_cast(player.raceTime), static_cast(player.bestLapTime), static_cast(player.finished == 1)); // Entire race time - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, (raceTime) * 1000, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_TOTAL_TRACK_TIME); + missionComponent->Progress(eMissionTaskType::RACING, (raceTime) * 1000, (LWOOBJID)eRacingTaskParam::TOTAL_TRACK_TIME); auto* characterComponent = playerEntity->GetComponent(); if (characterComponent != nullptr) { diff --git a/dGame/dComponents/RacingControlComponent.h b/dGame/dComponents/RacingControlComponent.h index dac60962..a81121e1 100644 --- a/dGame/dComponents/RacingControlComponent.h +++ b/dGame/dComponents/RacingControlComponent.h @@ -7,6 +7,7 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Information for each player in the race @@ -104,7 +105,7 @@ struct RacingPlayerInfo { */ class RacingControlComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_RACING_CONTROL; + static const eReplicaComponentType ComponentType = eReplicaComponentType::RACING_CONTROL; RacingControlComponent(Entity* parentEntity); ~RacingControlComponent(); @@ -123,7 +124,7 @@ public: * @param player The player who's vehicle to initialize. * @param initialLoad Is this the first time the player is loading in this race? */ - void LoadPlayerVehicle(Entity* player, bool initialLoad = false); + void LoadPlayerVehicle(Entity* player, uint32_t positionNumber, bool initialLoad = false); /** * Invoked when the client says it has loaded in. @@ -143,7 +144,7 @@ public: /** * Invoked when the player responds to the GUI. */ - void HandleMessageBoxResponse(Entity* player, const std::string& id); + void HandleMessageBoxResponse(Entity* player, int32_t button, const std::string& id); /** * Get the racing data from a player's LWOOBJID. @@ -245,4 +246,9 @@ private: float m_EmptyTimer; bool m_SoloRacing; + + /** + * Value for message box response to know if we are exiting the race via the activity dialogue + */ + const int32_t m_ActivityExitConfirm = 1; }; diff --git a/dGame/dComponents/RailActivatorComponent.cpp b/dGame/dComponents/RailActivatorComponent.cpp index 1b94fc4a..c7d58999 100644 --- a/dGame/dComponents/RailActivatorComponent.cpp +++ b/dGame/dComponents/RailActivatorComponent.cpp @@ -7,11 +7,13 @@ #include "RebuildComponent.h" #include "Game.h" #include "dLogger.h" +#include "RenderComponent.h" +#include "EntityManager.h" +#include "eStateChangeType.h" RailActivatorComponent::RailActivatorComponent(Entity* parent, int32_t componentID) : Component(parent) { m_ComponentID = componentID; - const auto tableData = CDClientManager::Instance() - ->GetTable("RailActivatorComponent")->GetEntryByID(componentID); + const auto tableData = CDClientManager::Instance().GetTable()->GetEntryByID(componentID);; m_Path = parent->GetVar(u"rail_path"); m_PathDirection = parent->GetVar(u"rail_path_direction"); @@ -42,7 +44,7 @@ RailActivatorComponent::~RailActivatorComponent() = default; void RailActivatorComponent::OnUse(Entity* originator) { auto* rebuildComponent = m_Parent->GetComponent(); - if (rebuildComponent != nullptr && rebuildComponent->GetState() != REBUILD_COMPLETED) + if (rebuildComponent != nullptr && rebuildComponent->GetState() != eRebuildState::COMPLETED) return; if (rebuildComponent != nullptr) { @@ -57,29 +59,16 @@ void RailActivatorComponent::OnUse(Entity* originator) { GameMessages::SendPlayFXEffect(originator->GetObjectID(), m_StartEffect.first, m_StartEffect.second, std::to_string(m_StartEffect.first)); } - + + float animationLength = 0.5f; if (!m_StartAnimation.empty()) { - GameMessages::SendPlayAnimation(originator, m_StartAnimation); - } - - float animationLength; - - if (m_StartAnimation == u"whirlwind-rail-up-earth") { - animationLength = 1.5f; - } else if (m_StartAnimation == u"whirlwind-rail-up-lightning") { - animationLength = 0.5f; - } else if (m_StartAnimation == u"whirlwind-rail-up-ice") { - animationLength = 0.5f; - } else if (m_StartAnimation == u"whirlwind-rail-up-fire") { - animationLength = 0.5f; - } else { - animationLength = 0.5f; + animationLength = RenderComponent::PlayAnimation(originator, m_StartAnimation); } const auto originatorID = originator->GetObjectID(); m_Parent->AddCallbackTimer(animationLength, [originatorID, this]() { - auto* originator = EntityManager::Instance()->GetEntity(originatorID); + auto* originator = Game::entityManager->GetEntity(originatorID); if (originator == nullptr) { return; @@ -95,7 +84,7 @@ void RailActivatorComponent::OnUse(Entity* originator) { void RailActivatorComponent::OnRailMovementReady(Entity* originator) const { // Stun the originator - GameMessages::SendSetStunned(originator->GetObjectID(), PUSH, originator->GetSystemAddress(), LWOOBJID_EMPTY, + GameMessages::SendSetStunned(originator->GetObjectID(), eStateChangeType::PUSH, originator->GetSystemAddress(), LWOOBJID_EMPTY, true, true, true, true, true, true, true ); @@ -112,7 +101,7 @@ void RailActivatorComponent::OnRailMovementReady(Entity* originator) const { } if (!m_LoopAnimation.empty()) { - GameMessages::SendPlayAnimation(originator, m_LoopAnimation); + RenderComponent::PlayAnimation(originator, m_LoopAnimation); } GameMessages::SendSetRailMovement(originator->GetObjectID(), m_PathDirection, m_Path, m_PathStart, @@ -123,7 +112,7 @@ void RailActivatorComponent::OnRailMovementReady(Entity* originator) const { void RailActivatorComponent::OnCancelRailMovement(Entity* originator) { // Remove the stun from the originator - GameMessages::SendSetStunned(originator->GetObjectID(), POP, originator->GetSystemAddress(), LWOOBJID_EMPTY, + GameMessages::SendSetStunned(originator->GetObjectID(), eStateChangeType::POP, originator->GetSystemAddress(), LWOOBJID_EMPTY, true, true, true, true, true, true, true ); @@ -147,7 +136,7 @@ void RailActivatorComponent::OnCancelRailMovement(Entity* originator) { } if (!m_StopAnimation.empty()) { - GameMessages::SendPlayAnimation(originator, m_StopAnimation); + RenderComponent::PlayAnimation(originator, m_StopAnimation); } // Remove the player after they've signalled they're done railing diff --git a/dGame/dComponents/RailActivatorComponent.h b/dGame/dComponents/RailActivatorComponent.h index 8f67e7e3..5d625d2a 100644 --- a/dGame/dComponents/RailActivatorComponent.h +++ b/dGame/dComponents/RailActivatorComponent.h @@ -4,6 +4,7 @@ #include #include "dCommonVars.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Component that handles the traveling using rails, e.g. the ninjago posts that can be used to travel using Spinjitzu. @@ -14,7 +15,7 @@ public: explicit RailActivatorComponent(Entity* parent, int32_t componentID); ~RailActivatorComponent() override; - static const uint32_t ComponentType = COMPONENT_TYPE_RAIL_ACTIVATOR; + static const eReplicaComponentType ComponentType = eReplicaComponentType::RAIL_ACTIVATOR; /** * Handles the OnUse event from some entity, initiates the rail movement diff --git a/dGame/dComponents/RebuildComponent.cpp b/dGame/dComponents/RebuildComponent.cpp index 6ec9b964..e669489d 100644 --- a/dGame/dComponents/RebuildComponent.cpp +++ b/dGame/dComponents/RebuildComponent.cpp @@ -7,13 +7,20 @@ #include "dLogger.h" #include "CharacterComponent.h" #include "MissionComponent.h" -#include "MissionTaskType.h" +#include "eMissionTaskType.h" +#include "eTriggerEventType.h" +#include "eQuickBuildFailReason.h" +#include "eTerminateType.h" +#include "eGameActivity.h" #include "dServer.h" #include "PacketUtils.h" #include "Spawner.h" #include "MovingPlatformComponent.h" #include "Preconditions.h" +#include "Loot.h" +#include "TeamManager.h" +#include "RenderComponent.h" #include "CppScripts.h" @@ -44,14 +51,14 @@ RebuildComponent::~RebuildComponent() { Entity* builder = GetBuilder(); if (builder) { - CancelRebuild(builder, eFailReason::REASON_BUILD_ENDED, true); + CancelRebuild(builder, eQuickBuildFailReason::BUILD_ENDED, true); } DespawnActivator(); } void RebuildComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { - if (m_Parent->GetComponent(COMPONENT_TYPE_DESTROYABLE) == nullptr) { + if (m_Parent->GetComponent(eReplicaComponentType::DESTROYABLE) == nullptr) { if (bIsInitialUpdate) { outBitStream->Write(false); } @@ -63,7 +70,7 @@ void RebuildComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitia // If build state is completed and we've already serialized once in the completed state, // don't serializing this component anymore as this will cause the build to jump again. // If state changes, serialization will begin again. - if (!m_StateDirty && m_State == REBUILD_COMPLETED) { + if (!m_StateDirty && m_State == eRebuildState::COMPLETED) { outBitStream->Write0(); outBitStream->Write0(); return; @@ -87,7 +94,7 @@ void RebuildComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitia outBitStream->Write1(); - outBitStream->Write(m_State); + outBitStream->Write(m_State); outBitStream->Write(m_ShowResetEffect); outBitStream->Write(m_Activator != nullptr); @@ -113,11 +120,11 @@ void RebuildComponent::Update(float deltaTime) { else { m_SoftTimer = 5.0f; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); }*/ switch (m_State) { - case REBUILD_OPEN: { + case eRebuildState::OPEN: { SpawnActivator(); m_TimeBeforeDrain = 0; @@ -132,7 +139,7 @@ void RebuildComponent::Update(float deltaTime) { if (m_TimerIncomplete >= m_TimeBeforeSmash - 4.0f) { m_ShowResetEffect = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } if (m_TimerIncomplete >= m_TimeBeforeSmash) { @@ -147,7 +154,7 @@ void RebuildComponent::Update(float deltaTime) { break; } - case REBUILD_COMPLETED: { + case eRebuildState::COMPLETED: { m_Timer += deltaTime; // For reset times < 0 this has to be handled manually @@ -156,7 +163,7 @@ void RebuildComponent::Update(float deltaTime) { if (!m_ShowResetEffect) { m_ShowResetEffect = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } @@ -169,11 +176,11 @@ void RebuildComponent::Update(float deltaTime) { } break; } - case REBUILD_BUILDING: + case eRebuildState::BUILDING: { Entity* builder = GetBuilder(); - if (builder == nullptr) { + if (!builder) { ResetRebuild(false); return; @@ -190,18 +197,18 @@ void RebuildComponent::Update(float deltaTime) { DestroyableComponent* destComp = builder->GetComponent(); if (!destComp) break; - int newImagination = destComp->GetImagination() - 1; - - destComp->SetImagination(newImagination); - EntityManager::Instance()->SerializeEntity(builder); + int newImagination = destComp->GetImagination(); ++m_DrainedImagination; + --newImagination; + destComp->SetImagination(newImagination); + Game::entityManager->SerializeEntity(builder); - if (newImagination == 0 && m_DrainedImagination < m_TakeImagination) { - CancelRebuild(builder, eFailReason::REASON_OUT_OF_IMAGINATION, true); - + if (newImagination <= 0) { + CancelRebuild(builder, eQuickBuildFailReason::OUT_OF_IMAGINATION, true); break; } + } if (m_Timer >= m_CompleteTime && m_DrainedImagination >= m_TakeImagination) { @@ -210,7 +217,7 @@ void RebuildComponent::Update(float deltaTime) { break; } - case REBUILD_INCOMPLETE: { + case eRebuildState::INCOMPLETE: { m_TimerIncomplete += deltaTime; // For reset times < 0 this has to be handled manually @@ -218,7 +225,7 @@ void RebuildComponent::Update(float deltaTime) { if (m_TimerIncomplete >= m_TimeBeforeSmash - 4.0f) { m_ShowResetEffect = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } if (m_TimerIncomplete >= m_TimeBeforeSmash) { @@ -231,11 +238,12 @@ void RebuildComponent::Update(float deltaTime) { } break; } + case eRebuildState::RESETTING: break; } } void RebuildComponent::OnUse(Entity* originator) { - if (GetBuilder() != nullptr || m_State == REBUILD_COMPLETED) { + if (GetBuilder() != nullptr || m_State == eRebuildState::COMPLETED) { return; } @@ -255,20 +263,20 @@ void RebuildComponent::SpawnActivator() { info.spawnerID = m_Parent->GetObjectID(); info.pos = m_ActivatorPosition == NiPoint3::ZERO ? m_Parent->GetPosition() : m_ActivatorPosition; - m_Activator = EntityManager::Instance()->CreateEntity(info, nullptr, m_Parent); + m_Activator = Game::entityManager->CreateEntity(info, nullptr, m_Parent); if (m_Activator) { m_ActivatorId = m_Activator->GetObjectID(); - EntityManager::Instance()->ConstructEntity(m_Activator); + Game::entityManager->ConstructEntity(m_Activator); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } } void RebuildComponent::DespawnActivator() { if (m_Activator) { - EntityManager::Instance()->DestructEntity(m_Activator); + Game::entityManager->DestructEntity(m_Activator); m_Activator->ScheduleKillAfterUpdate(); @@ -279,7 +287,7 @@ void RebuildComponent::DespawnActivator() { } Entity* RebuildComponent::GetActivator() { - return EntityManager::Instance()->GetEntity(m_ActivatorId); + return Game::entityManager->GetEntity(m_ActivatorId); } NiPoint3 RebuildComponent::GetActivatorPosition() { @@ -327,7 +335,7 @@ eRebuildState RebuildComponent::GetState() { } Entity* RebuildComponent::GetBuilder() const { - auto* builder = EntityManager::Instance()->GetEntity(m_Builder); + auto* builder = Game::entityManager->GetEntity(m_Builder); return builder; } @@ -389,20 +397,20 @@ void RebuildComponent::SetRepositionPlayer(bool value) { } void RebuildComponent::StartRebuild(Entity* user) { - if (m_State == eRebuildState::REBUILD_OPEN || m_State == eRebuildState::REBUILD_COMPLETED || m_State == eRebuildState::REBUILD_INCOMPLETE) { + if (m_State == eRebuildState::OPEN || m_State == eRebuildState::COMPLETED || m_State == eRebuildState::INCOMPLETE) { m_Builder = user->GetObjectID(); auto* character = user->GetComponent(); - character->SetCurrentActivity(eGameActivities::ACTIVITY_QUICKBUILDING); + character->SetCurrentActivity(eGameActivity::QUICKBUILDING); - EntityManager::Instance()->SerializeEntity(user); + Game::entityManager->SerializeEntity(user); - GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::REBUILD_BUILDING, user->GetObjectID()); - GameMessages::SendEnableRebuild(m_Parent, true, false, false, eFailReason::REASON_NOT_GIVEN, 0.0f, user->GetObjectID()); + GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::BUILDING, user->GetObjectID()); + GameMessages::SendEnableRebuild(m_Parent, true, false, false, eQuickBuildFailReason::NOT_GIVEN, 0.0f, user->GetObjectID()); - m_State = eRebuildState::REBUILD_BUILDING; + m_State = eRebuildState::BUILDING; m_StateDirty = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); auto* movingPlatform = m_Parent->GetComponent(); if (movingPlatform != nullptr) { @@ -428,27 +436,27 @@ void RebuildComponent::CompleteRebuild(Entity* user) { auto* characterComponent = user->GetComponent(); if (characterComponent != nullptr) { - characterComponent->SetCurrentActivity(eGameActivities::ACTIVITY_NONE); + characterComponent->SetCurrentActivity(eGameActivity::NONE); characterComponent->TrackRebuildComplete(); } else { Game::logger->Log("RebuildComponent", "Some user tried to finish the rebuild but they didn't have a character somehow."); return; } - EntityManager::Instance()->SerializeEntity(user); + Game::entityManager->SerializeEntity(user); - GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::REBUILD_COMPLETED, user->GetObjectID()); + GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::COMPLETED, user->GetObjectID()); GameMessages::SendPlayFXEffect(m_Parent, 507, u"create", "BrickFadeUpVisCompleteEffect", LWOOBJID_EMPTY, 0.4f, 1.0f, true); - GameMessages::SendEnableRebuild(m_Parent, false, false, true, eFailReason::REASON_NOT_GIVEN, m_ResetTime, user->GetObjectID()); + GameMessages::SendEnableRebuild(m_Parent, false, false, true, eQuickBuildFailReason::NOT_GIVEN, m_ResetTime, user->GetObjectID()); GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, m_Parent->GetObjectID()); - m_State = eRebuildState::REBUILD_COMPLETED; + m_State = eRebuildState::COMPLETED; m_StateDirty = true; m_Timer = 0.0f; m_DrainedImagination = 0; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); // Removes extra item requirements, isn't live accurate. // In live, all items were removed at the start of the quickbuild, then returned if it was cancelled. @@ -464,12 +472,20 @@ void RebuildComponent::CompleteRebuild(Entity* user) { auto* builder = GetBuilder(); - if (builder != nullptr) { - auto* missionComponent = builder->GetComponent(); - if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ACTIVITY, m_ActivityId); + if (builder) { + auto* team = TeamManager::Instance()->GetTeam(builder->GetObjectID()); + if (team) { + for (const auto memberId : team->members) { // progress missions for all team members + auto* member = Game::entityManager->GetEntity(memberId); + if (member) { + auto* missionComponent = member->GetComponent(); + if (missionComponent) missionComponent->Progress(eMissionTaskType::ACTIVITY, m_ActivityId); + } + } + } else { + auto* missionComponent = builder->GetComponent(); + if (missionComponent) missionComponent->Progress(eMissionTaskType::ACTIVITY, m_ActivityId); } - LootGenerator::Instance().DropActivityLoot(builder, m_Parent, m_ActivityId, 1); } @@ -485,6 +501,8 @@ void RebuildComponent::CompleteRebuild(Entity* user) { for (const auto& callback : m_RebuildCompleteCallbacks) callback(user); + m_Parent->TriggerEvent(eTriggerEventType::REBUILD_COMPLETE, user); + auto* movingPlatform = m_Parent->GetComponent(); if (movingPlatform != nullptr) { movingPlatform->OnCompleteRebuild(); @@ -500,30 +518,30 @@ void RebuildComponent::CompleteRebuild(Entity* user) { character->SetPlayerFlag(flagNumber, true); } } - GameMessages::SendPlayAnimation(user, u"rebuild-celebrate", 1.09f); + RenderComponent::PlayAnimation(user, u"rebuild-celebrate", 1.09f); } void RebuildComponent::ResetRebuild(bool failed) { Entity* builder = GetBuilder(); - if (m_State == eRebuildState::REBUILD_BUILDING && builder) { - GameMessages::SendEnableRebuild(m_Parent, false, false, failed, eFailReason::REASON_NOT_GIVEN, m_ResetTime, builder->GetObjectID()); + if (m_State == eRebuildState::BUILDING && builder) { + GameMessages::SendEnableRebuild(m_Parent, false, false, failed, eQuickBuildFailReason::NOT_GIVEN, m_ResetTime, builder->GetObjectID()); if (failed) { - GameMessages::SendPlayAnimation(builder, u"rebuild-fail"); + RenderComponent::PlayAnimation(builder, u"rebuild-fail"); } } - GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::REBUILD_RESETTING, LWOOBJID_EMPTY); + GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::RESETTING, LWOOBJID_EMPTY); - m_State = eRebuildState::REBUILD_RESETTING; + m_State = eRebuildState::RESETTING; m_StateDirty = true; m_Timer = 0.0f; m_TimerIncomplete = 0.0f; m_ShowResetEffect = false; m_DrainedImagination = 0; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); // Notify scripts and possible subscribers for (auto* script : CppScripts::GetEntityScripts(m_Parent)) @@ -538,15 +556,15 @@ void RebuildComponent::ResetRebuild(bool failed) { } } -void RebuildComponent::CancelRebuild(Entity* entity, eFailReason failReason, bool skipChecks) { - if (m_State != eRebuildState::REBUILD_COMPLETED || skipChecks) { +void RebuildComponent::CancelRebuild(Entity* entity, eQuickBuildFailReason failReason, bool skipChecks) { + if (m_State != eRebuildState::COMPLETED || skipChecks) { m_Builder = LWOOBJID_EMPTY; const auto entityID = entity != nullptr ? entity->GetObjectID() : LWOOBJID_EMPTY; // Notify the client that a state has changed - GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::REBUILD_INCOMPLETE, entityID); + GameMessages::SendRebuildNotifyState(m_Parent, m_State, eRebuildState::INCOMPLETE, entityID); GameMessages::SendEnableRebuild(m_Parent, false, true, false, failReason, m_Timer, entityID); // Now terminate any interaction with the rebuild @@ -554,7 +572,7 @@ void RebuildComponent::CancelRebuild(Entity* entity, eFailReason failReason, boo GameMessages::SendTerminateInteraction(m_Parent->GetObjectID(), eTerminateType::FROM_INTERACTION, m_Parent->GetObjectID()); // Now update the component itself - m_State = eRebuildState::REBUILD_INCOMPLETE; + m_State = eRebuildState::INCOMPLETE; m_StateDirty = true; // Notify scripts and possible subscribers @@ -563,7 +581,7 @@ void RebuildComponent::CancelRebuild(Entity* entity, eFailReason failReason, boo for (const auto& cb : m_RebuildStateCallbacks) cb(m_State); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } if (entity == nullptr) { @@ -572,8 +590,8 @@ void RebuildComponent::CancelRebuild(Entity* entity, eFailReason failReason, boo CharacterComponent* characterComponent = entity->GetComponent(); if (characterComponent) { - characterComponent->SetCurrentActivity(eGameActivities::ACTIVITY_NONE); - EntityManager::Instance()->SerializeEntity(entity); + characterComponent->SetCurrentActivity(eGameActivity::NONE); + Game::entityManager->SerializeEntity(entity); } } diff --git a/dGame/dComponents/RebuildComponent.h b/dGame/dComponents/RebuildComponent.h index bd3edd7d..09dd0465 100644 --- a/dGame/dComponents/RebuildComponent.h +++ b/dGame/dComponents/RebuildComponent.h @@ -9,8 +9,11 @@ #include "ScriptedActivityComponent.h" #include "Preconditions.h" #include "Component.h" +#include "eReplicaComponentType.h" +#include "eRebuildState.h" class Entity; +enum class eQuickBuildFailReason : uint32_t; /** * Component that handles entities that can be built into other entities using the quick build mechanic. Generally @@ -19,7 +22,7 @@ class Entity; */ class RebuildComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_REBUILD; + static const eReplicaComponentType ComponentType = eReplicaComponentType::QUICK_BUILD; RebuildComponent(Entity* entity); ~RebuildComponent() override; @@ -214,7 +217,7 @@ public: * @param failReason the reason the rebuild was cancelled * @param skipChecks whether or not to skip the check for the rebuild not being completed */ - void CancelRebuild(Entity* builder, eFailReason failReason, bool skipChecks = false); + void CancelRebuild(Entity* builder, eQuickBuildFailReason failReason, bool skipChecks = false); private: /** * Whether or not the quickbuild state has been changed since we last serialized it. @@ -224,7 +227,7 @@ private: /** * The state the rebuild is currently in */ - eRebuildState m_State = eRebuildState::REBUILD_OPEN; + eRebuildState m_State = eRebuildState::OPEN; /** * The time that has passed since initiating the rebuild diff --git a/dGame/dComponents/RenderComponent.cpp b/dGame/dComponents/RenderComponent.cpp index 3002ae33..94f5fb5d 100644 --- a/dGame/dComponents/RenderComponent.cpp +++ b/dGame/dComponents/RenderComponent.cpp @@ -11,72 +11,36 @@ #include "GameMessages.h" #include "Game.h" #include "dLogger.h" +#include "CDAnimationsTable.h" std::unordered_map RenderComponent::m_DurationCache{}; -RenderComponent::RenderComponent(Entity* parent) : Component(parent) { +RenderComponent::RenderComponent(Entity* parent, int32_t componentId): Component(parent) { m_Effects = std::vector(); + m_LastAnimationName = ""; + if (componentId == -1) return; - return; + auto query = CDClientDatabase::CreatePreppedStmt("SELECT * FROM RenderComponent WHERE id = ?;"); + query.bind(1, componentId); + auto result = query.execQuery(); - /* - auto* table = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - - const auto entry = table->GetByIDAndType(parent->GetLOT(), COMPONENT_TYPE_RENDER); - - std::stringstream query; - - query << "SELECT effect1, effect2, effect3, effect4, effect5, effect6 FROM RenderComponent WHERE id = " << std::to_string(entry) << ";"; - - auto result = CDClientDatabase::ExecuteQuery(query.str()); - - if (result.eof()) - { - return; - } - - for (auto i = 0; i < 6; ++i) - { - if (result.fieldIsNull(i)) - { - continue; - } - - const auto id = result.getIntField(i); - - if (id <= 0) - { - continue; - } - - query.clear(); - - query << "SELECT effectType, effectName FROM BehaviorEffect WHERE effectID = " << std::to_string(id) << ";"; - - auto effectResult = CDClientDatabase::ExecuteQuery(query.str()); - - while (!effectResult.eof()) - { - const auto type = effectResult.fieldIsNull(0) ? "" : std::string(effectResult.getStringField(0)); - - const auto name = effectResult.fieldIsNull(1) ? "" : std::string(effectResult.getStringField(1)); - - auto* effect = new Effect(); - - effect->name = name; - effect->type = GeneralUtils::ASCIIToUTF16(type); - effect->scale = 1; - effect->effectID = id; - effect->secondary = LWOOBJID_EMPTY; - - m_Effects.push_back(effect); - - effectResult.nextRow(); + if (!result.eof()) { + auto animationGroupIDs = std::string(result.getStringField("animationGroupIDs", "")); + if (!animationGroupIDs.empty()) { + auto* animationsTable = CDClientManager::Instance().GetTable(); + auto groupIdsSplit = GeneralUtils::SplitString(animationGroupIDs, ','); + for (auto& groupId : groupIdsSplit) { + int32_t groupIdInt; + if (!GeneralUtils::TryParse(groupId, groupIdInt)) { + Game::logger->Log("RenderComponent", "bad animation group Id %s", groupId.c_str()); + continue; + } + m_animationGroupIds.push_back(groupIdInt); + animationsTable->CacheAnimationGroup(groupIdInt); + } } } - result.finalize(); - */ } RenderComponent::~RenderComponent() { @@ -224,3 +188,45 @@ void RenderComponent::StopEffect(const std::string& name, const bool killImmedia std::vector& RenderComponent::GetEffects() { return m_Effects; } + + +float RenderComponent::PlayAnimation(Entity* self, const std::u16string& animation, float priority, float scale) { + if (!self) return 0.0f; + return RenderComponent::PlayAnimation(self, GeneralUtils::UTF16ToWTF8(animation), priority, scale); +} + +float RenderComponent::PlayAnimation(Entity* self, const std::string& animation, float priority, float scale) { + if (!self) return 0.0f; + return RenderComponent::DoAnimation(self, animation, true, priority, scale); +} + +float RenderComponent::GetAnimationTime(Entity* self, const std::u16string& animation) { + if (!self) return 0.0f; + return RenderComponent::GetAnimationTime(self, GeneralUtils::UTF16ToWTF8(animation)); +} + +float RenderComponent::GetAnimationTime(Entity* self, const std::string& animation) { + if (!self) return 0.0f; + return RenderComponent::DoAnimation(self, animation, false); +} + + +float RenderComponent::DoAnimation(Entity* self, const std::string& animation, bool sendAnimation, float priority, float scale) { + float returnlength = 0.0f; + if (!self) return returnlength; + auto* renderComponent = self->GetComponent(); + if (!renderComponent) return returnlength; + + auto* animationsTable = CDClientManager::Instance().GetTable(); + for (auto& groupId : renderComponent->m_animationGroupIds) { + auto animationGroup = animationsTable->GetAnimation(animation, renderComponent->GetLastAnimationName(), groupId); + if (animationGroup.FoundData()) { + auto data = animationGroup.Data(); + renderComponent->SetLastAnimationName(data.animation_name); + returnlength = data.animation_length; + } + } + if (sendAnimation) GameMessages::SendPlayAnimation(self, GeneralUtils::ASCIIToUTF16(animation), priority, scale); + if (returnlength == 0.0f) Game::logger->Log("RenderComponent", "WARNING: Unable to find animation %s for lot %i in any group.", animation.c_str(), self->GetLOT()); + return returnlength; +} diff --git a/dGame/dComponents/RenderComponent.h b/dGame/dComponents/RenderComponent.h index e6184564..cdf32160 100644 --- a/dGame/dComponents/RenderComponent.h +++ b/dGame/dComponents/RenderComponent.h @@ -6,8 +6,9 @@ #include #include -#include "AMFFormat.h" +#include "Amf3.h" #include "Component.h" +#include "eReplicaComponentType.h" class Entity; @@ -55,9 +56,9 @@ struct Effect { */ class RenderComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_RENDER; + static const eReplicaComponentType ComponentType = eReplicaComponentType::RENDER; - RenderComponent(Entity* entity); + RenderComponent(Entity* entity, int32_t componentId = -1); ~RenderComponent() override; void Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags); @@ -103,6 +104,32 @@ public: */ std::vector& GetEffects(); + /** + * Verifies that an animation can be played on this entity by checking + * if it has the animation assigned to its group. If it does, the animation is echo'd + * down to all clients to be played and the duration of the played animation is returned. + * If the animation did not exist or the function was called in an invalid state, 0 is returned. + * + * The logic here matches the exact client logic. + * + * @param self The entity that wants to play an animation + * @param animation The animation_type (animationID in the client) to be played. + * @param sendAnimation Whether or not to echo the animation down to all clients. + * @param priority The priority of the animation. Only used if sendAnimation is true. + * @param scale The scale of the animation. Only used if sendAnimation is true. + * + * @return The duration of the animation that was played. + */ + static float DoAnimation(Entity* self, const std::string& animation, bool sendAnimation, float priority = 0.0f, float scale = 1.0f); + + static float PlayAnimation(Entity* self, const std::u16string& animation, float priority = 0.0f, float scale = 1.0f); + static float PlayAnimation(Entity* self, const std::string& animation, float priority = 0.0f, float scale = 1.0f); + static float GetAnimationTime(Entity* self, const std::string& animation); + static float GetAnimationTime(Entity* self, const std::u16string& animation); + + const std::string& GetLastAnimationName() const { return m_LastAnimationName; }; + void SetLastAnimationName(const std::string& name) { m_LastAnimationName = name; }; + private: /** @@ -110,6 +137,11 @@ private: */ std::vector m_Effects; + std::vector m_animationGroupIds; + + // The last animationName that was played + std::string m_LastAnimationName; + /** * Cache of queries that look for the length of each effect, indexed by effect ID */ diff --git a/dGame/dComponents/RigidbodyPhantomPhysicsComponent.h b/dGame/dComponents/RigidbodyPhantomPhysicsComponent.h index c8faa930..480f9b81 100644 --- a/dGame/dComponents/RigidbodyPhantomPhysicsComponent.h +++ b/dGame/dComponents/RigidbodyPhantomPhysicsComponent.h @@ -11,6 +11,7 @@ #include "NiPoint3.h" #include "NiQuaternion.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Component that handles rigid bodies that can be interacted with, mostly client-side rendered. An example is the @@ -18,7 +19,7 @@ */ class RigidbodyPhantomPhysicsComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_PHANTOM_PHYSICS; + static const eReplicaComponentType ComponentType = eReplicaComponentType::PHANTOM_PHYSICS; RigidbodyPhantomPhysicsComponent(Entity* parent); ~RigidbodyPhantomPhysicsComponent() override; diff --git a/dGame/dComponents/RocketLaunchLupComponent.h b/dGame/dComponents/RocketLaunchLupComponent.h index 3fc0b444..226fa1b2 100644 --- a/dGame/dComponents/RocketLaunchLupComponent.h +++ b/dGame/dComponents/RocketLaunchLupComponent.h @@ -3,6 +3,7 @@ #include "Entity.h" #include "GameMessages.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Component that handles the LUP/WBL rocket launchpad that can be interacted with to travel to WBL worlds. @@ -10,7 +11,7 @@ */ class RocketLaunchLupComponent : public Component { public: - static const uint32_t ComponentType = eReplicaComponentType::COMPONENT_TYPE_ROCKET_LAUNCH_LUP; + static const eReplicaComponentType ComponentType = eReplicaComponentType::ROCKET_LAUNCH_LUP; /** * Constructor for this component, builds the m_LUPWorlds vector diff --git a/dGame/dComponents/RocketLaunchpadControlComponent.cpp b/dGame/dComponents/RocketLaunchpadControlComponent.cpp index 4f248a40..10908d9e 100644 --- a/dGame/dComponents/RocketLaunchpadControlComponent.cpp +++ b/dGame/dComponents/RocketLaunchpadControlComponent.cpp @@ -15,8 +15,10 @@ #include "PropertyEntranceComponent.h" #include "RocketLaunchLupComponent.h" #include "dServer.h" -#include "dMessageIdentifiers.h" #include "PacketUtils.h" +#include "eObjectWorldState.h" +#include "eConnectionType.h" +#include "eMasterMessageType.h" RocketLaunchpadControlComponent::RocketLaunchpadControlComponent(Entity* parent, int rocketId) : Component(parent) { auto query = CDClientDatabase::CreatePreppedStmt( @@ -77,9 +79,9 @@ void RocketLaunchpadControlComponent::Launch(Entity* originator, LWOMAPID mapId, GameMessages::SendFireEventClientSide(m_Parent->GetObjectID(), originator->GetSystemAddress(), u"RocketEquipped", rocket->GetId(), cloneId, -1, originator->GetObjectID()); - GameMessages::SendChangeObjectWorldState(rocket->GetId(), WORLDSTATE_ATTACHED, UNASSIGNED_SYSTEM_ADDRESS); + GameMessages::SendChangeObjectWorldState(rocket->GetId(), eObjectWorldState::ATTACHED, UNASSIGNED_SYSTEM_ADDRESS); - EntityManager::Instance()->SerializeEntity(originator); + Game::entityManager->SerializeEntity(originator); } void RocketLaunchpadControlComponent::OnUse(Entity* originator) { @@ -135,7 +137,7 @@ LWOCLONEID RocketLaunchpadControlComponent::GetSelectedCloneId(LWOOBJID player) void RocketLaunchpadControlComponent::TellMasterToPrepZone(int zoneID) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_PREP_ZONE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::PREP_ZONE); bitStream.Write(zoneID); Game::server->SendToMaster(&bitStream); } diff --git a/dGame/dComponents/RocketLaunchpadControlComponent.h b/dGame/dComponents/RocketLaunchpadControlComponent.h index 8a10f0f7..84cff22d 100644 --- a/dGame/dComponents/RocketLaunchpadControlComponent.h +++ b/dGame/dComponents/RocketLaunchpadControlComponent.h @@ -9,6 +9,7 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" class PreconditionExpression; @@ -17,7 +18,7 @@ class PreconditionExpression; */ class RocketLaunchpadControlComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_ROCKET_LAUNCH; + static const eReplicaComponentType ComponentType = eReplicaComponentType::ROCKET_LAUNCH; RocketLaunchpadControlComponent(Entity* parent, int rocketId); ~RocketLaunchpadControlComponent() override; diff --git a/dGame/dComponents/ScriptedActivityComponent.cpp b/dGame/dComponents/ScriptedActivityComponent.cpp index 4b5ec41d..81b3e9a9 100644 --- a/dGame/dComponents/ScriptedActivityComponent.cpp +++ b/dGame/dComponents/ScriptedActivityComponent.cpp @@ -16,18 +16,27 @@ #include "GeneralUtils.h" #include "dZoneManager.h" #include "dConfig.h" +#include "InventoryComponent.h" #include "DestroyableComponent.h" +#include "Loot.h" +#include "eMissionTaskType.h" +#include "eMatchUpdate.h" +#include "eConnectionType.h" +#include "eChatInternalMessageType.h" + +#include "CDCurrencyTableTable.h" +#include "CDActivityRewardsTable.h" +#include "CDActivitiesTable.h" +#include "LeaderboardManager.h" ScriptedActivityComponent::ScriptedActivityComponent(Entity* parent, int activityID) : Component(parent) { - CDActivitiesTable* activitiesTable = CDClientManager::Instance()->GetTable("Activities"); - std::vector activities = activitiesTable->Query([=](CDActivities entry) {return (entry.ActivityID == activityID); }); + m_ActivityID = activityID; + CDActivitiesTable* activitiesTable = CDClientManager::Instance().GetTable(); + std::vector activities = activitiesTable->Query([=](CDActivities entry) {return (entry.ActivityID == m_ActivityID); }); for (CDActivities activity : activities) { m_ActivityInfo = activity; - - const auto mapID = m_ActivityInfo.instanceMapID; - - if ((mapID == 1203 || mapID == 1261 || mapID == 1303 || mapID == 1403) && Game::config->GetValue("solo_racing") == "1") { + if (static_cast(activity.leaderboardType) == Leaderboard::Type::Racing && Game::config->GetValue("solo_racing") == "1") { m_ActivityInfo.minTeamSize = 1; m_ActivityInfo.minTeams = 1; } @@ -48,7 +57,7 @@ ScriptedActivityComponent::ScriptedActivityComponent(Entity* parent, int activit if (destroyableComponent) { // check for LMIs and set the loot LMIs - CDActivityRewardsTable* activityRewardsTable = CDClientManager::Instance()->GetTable("ActivityRewards"); + CDActivityRewardsTable* activityRewardsTable = CDClientManager::Instance().GetTable(); std::vector activityRewards = activityRewardsTable->Query([=](CDActivityRewards entry) {return (entry.LootMatrixIndex == destroyableComponent->GetLootMatrixID()); }); uint32_t startingLMI = 0; @@ -88,6 +97,21 @@ void ScriptedActivityComponent::Serialize(RakNet::BitStream* outBitStream, bool } } +void ScriptedActivityComponent::ReloadConfig() { + CDActivitiesTable* activitiesTable = CDClientManager::Instance().GetTable(); + std::vector activities = activitiesTable->Query([=](CDActivities entry) {return (entry.ActivityID == m_ActivityID); }); + for (auto activity : activities) { + auto mapID = m_ActivityInfo.instanceMapID; + if ((mapID == 1203 || mapID == 1261 || mapID == 1303 || mapID == 1403) && Game::config->GetValue("solo_racing") == "1") { + m_ActivityInfo.minTeamSize = 1; + m_ActivityInfo.minTeams = 1; + } else { + m_ActivityInfo.minTeamSize = activity.minTeamSize; + m_ActivityInfo.minTeams = activity.minTeams; + } + } +} + void ScriptedActivityComponent::HandleMessageBoxResponse(Entity* player, const std::string& id) { if (m_ActivityInfo.ActivityID == 103) { return; @@ -113,11 +137,11 @@ void ScriptedActivityComponent::PlayerJoin(Entity* player) { instance->AddParticipant(player); } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void ScriptedActivityComponent::PlayerJoinLobby(Entity* player) { - if (!m_Parent->HasComponent(COMPONENT_TYPE_REBUILD)) + if (!m_Parent->HasComponent(eReplicaComponentType::QUICK_BUILD)) GameMessages::SendMatchResponse(player, player->GetSystemAddress(), 0); // tell the client they joined a lobby LobbyPlayer* newLobbyPlayer = new LobbyPlayer(); newLobbyPlayer->entityID = player->GetObjectID(); @@ -125,7 +149,7 @@ void ScriptedActivityComponent::PlayerJoinLobby(Entity* player) { auto* character = player->GetCharacter(); if (character != nullptr) - character->SetLastNonInstanceZoneID(dZoneManager::Instance()->GetZone()->GetWorldID()); + character->SetLastNonInstanceZoneID(Game::zoneManager->GetZone()->GetWorldID()); for (Lobby* lobby : m_Queue) { if (lobby->players.size() < m_ActivityInfo.maxTeamSize || m_ActivityInfo.maxTeamSize == 1 && lobby->players.size() < m_ActivityInfo.maxTeams) { @@ -143,9 +167,9 @@ void ScriptedActivityComponent::PlayerJoinLobby(Entity* player) { } std::string matchUpdate = "player=9:" + std::to_string(entity->GetObjectID()) + "\nplayerName=0:" + entity->GetCharacter()->GetName(); - GameMessages::SendMatchUpdate(player, player->GetSystemAddress(), matchUpdate, eMatchUpdate::MATCH_UPDATE_PLAYER_JOINED); + GameMessages::SendMatchUpdate(player, player->GetSystemAddress(), matchUpdate, eMatchUpdate::PLAYER_ADDED); PlayerReady(entity, joinedPlayer->ready); - GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchUpdateJoined, eMatchUpdate::MATCH_UPDATE_PLAYER_JOINED); + GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchUpdateJoined, eMatchUpdate::PLAYER_ADDED); } } } @@ -161,7 +185,7 @@ void ScriptedActivityComponent::PlayerJoinLobby(Entity* player) { if (m_ActivityInfo.maxTeamSize != 1 && playerLobby->players.size() >= m_ActivityInfo.minTeamSize || m_ActivityInfo.maxTeamSize == 1 && playerLobby->players.size() >= m_ActivityInfo.minTeams) { // Update the joining player on the match timer std::string matchTimerUpdate = "time=3:" + std::to_string(playerLobby->timer); - GameMessages::SendMatchUpdate(player, player->GetSystemAddress(), matchTimerUpdate, eMatchUpdate::MATCH_UPDATE_TIME); + GameMessages::SendMatchUpdate(player, player->GetSystemAddress(), matchTimerUpdate, eMatchUpdate::PHASE_WAIT_READY); } } @@ -177,7 +201,7 @@ void ScriptedActivityComponent::PlayerLeave(LWOOBJID playerID) { if (entity == nullptr) continue; - GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchUpdateLeft, eMatchUpdate::MATCH_UPDATE_PLAYER_LEFT); + GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchUpdateLeft, eMatchUpdate::PLAYER_REMOVED); } delete lobby->players[i]; @@ -191,7 +215,7 @@ void ScriptedActivityComponent::PlayerLeave(LWOOBJID playerID) { } void ScriptedActivityComponent::Update(float deltaTime) { - + std::vector lobbiesToRemove{}; // Ticks all the lobbies, not applicable for non-instance activities for (Lobby* lobby : m_Queue) { for (LobbyPlayer* player : lobby->players) { @@ -202,6 +226,11 @@ void ScriptedActivityComponent::Update(float deltaTime) { } } + if (lobby->players.empty()) { + lobbiesToRemove.push_back(lobby); + continue; + } + // Update the match time for all players if (m_ActivityInfo.maxTeamSize != 1 && lobby->players.size() >= m_ActivityInfo.minTeamSize || m_ActivityInfo.maxTeamSize == 1 && lobby->players.size() >= m_ActivityInfo.minTeams) { @@ -213,7 +242,7 @@ void ScriptedActivityComponent::Update(float deltaTime) { continue; std::string matchTimerUpdate = "time=3:" + std::to_string(lobby->timer); - GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchTimerUpdate, eMatchUpdate::MATCH_UPDATE_TIME); + GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchTimerUpdate, eMatchUpdate::PHASE_WAIT_READY); } } @@ -238,20 +267,24 @@ void ScriptedActivityComponent::Update(float deltaTime) { if (entity == nullptr) continue; - GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchTimerUpdate, eMatchUpdate::MATCH_UPDATE_TIME_START_DELAY); + GameMessages::SendMatchUpdate(entity, entity->GetSystemAddress(), matchTimerUpdate, eMatchUpdate::PHASE_WAIT_START); } } // The timer has elapsed, start the instance if (lobby->timer <= 0.0f) { Game::logger->Log("ScriptedActivityComponent", "Setting up instance."); - ActivityInstance* instance = NewInstance(); LoadPlayersIntoInstance(instance, lobby->players); - RemoveLobby(lobby); instance->StartZone(); + lobbiesToRemove.push_back(lobby); } } + + while (!lobbiesToRemove.empty()) { + RemoveLobby(lobbiesToRemove.front()); + lobbiesToRemove.erase(lobbiesToRemove.begin()); + } } void ScriptedActivityComponent::RemoveLobby(Lobby* lobby) { @@ -270,14 +303,14 @@ bool ScriptedActivityComponent::HasLobby() const { bool ScriptedActivityComponent::IsValidActivity(Entity* player) { // Makes it so that scripted activities with an unimplemented map cannot be joined - /*if (player->GetGMLevel() < GAME_MASTER_LEVEL_DEVELOPER && (m_ActivityInfo.instanceMapID == 1302 || m_ActivityInfo.instanceMapID == 1301)) { + /*if (player->GetGMLevel() < eGameMasterLevel::DEVELOPER && (m_ActivityInfo.instanceMapID == 1302 || m_ActivityInfo.instanceMapID == 1301)) { if (m_Parent->GetLOT() == 4860) { auto* missionComponent = player->GetComponent(); missionComponent->CompleteMission(229); } ChatPackets::SendSystemMessage(player->GetSystemAddress(), u"Sorry, this activity is not ready."); - static_cast(player)->SendToZone(dZoneManager::Instance()->GetZone()->GetWorldID()); // Gets them out of this stuck state + static_cast(player)->SendToZone(Game::zoneManager->GetZone()->GetWorldID()); // Gets them out of this stuck state return false; }*/ @@ -342,8 +375,8 @@ void ScriptedActivityComponent::PlayerReady(Entity* player, bool bReady) { // Update players in lobby on player being ready std::string matchReadyUpdate = "player=9:" + std::to_string(player->GetObjectID()); - eMatchUpdate readyStatus = eMatchUpdate::MATCH_UPDATE_PLAYER_READY; - if (!bReady) readyStatus = eMatchUpdate::MATCH_UPDATE_PLAYER_UNREADY; + eMatchUpdate readyStatus = eMatchUpdate::PLAYER_READY; + if (!bReady) readyStatus = eMatchUpdate::PLAYER_NOT_READY; for (LobbyPlayer* otherPlayer : lobby->players) { auto* entity = otherPlayer->GetEntity(); if (entity == nullptr) @@ -412,7 +445,7 @@ void ScriptedActivityComponent::RemoveActivityPlayerData(LWOOBJID playerID) { m_ActivityPlayers[i] = nullptr; m_ActivityPlayers.erase(m_ActivityPlayers.begin() + i); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); return; } @@ -425,7 +458,7 @@ ActivityPlayer* ScriptedActivityComponent::AddActivityPlayerData(LWOOBJID player return data; m_ActivityPlayers.push_back(new ActivityPlayer{ playerID, {} }); - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); return GetActivityPlayerData(playerID); } @@ -447,7 +480,7 @@ void ScriptedActivityComponent::SetActivityValue(LWOOBJID playerID, uint32_t ind data->values[std::min(index, (uint32_t)9)] = value; } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void ScriptedActivityComponent::PlayerRemove(LWOOBJID playerID) { @@ -483,7 +516,7 @@ void ActivityInstance::StartZone() { // only make a team if we have more than one participant if (participants.size() > 1) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_CREATE_TEAM); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::CREATE_TEAM); bitStream.Write(leader->GetObjectID()); bitStream.Write(m_Participants.size()); @@ -502,7 +535,7 @@ void ActivityInstance::StartZone() { const auto objid = player->GetObjectID(); ZoneInstanceManager::Instance()->RequestZoneTransfer(Game::server, m_ActivityInfo.instanceMapID, cloneId, false, [objid](bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, std::string serverIP, uint16_t serverPort) { - auto* player = EntityManager::Instance()->GetEntity(objid); + auto* player = Game::entityManager->GetEntity(objid); if (player == nullptr) return; @@ -524,18 +557,18 @@ void ActivityInstance::StartZone() { void ActivityInstance::RewardParticipant(Entity* participant) { auto* missionComponent = participant->GetComponent(); if (missionComponent) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ACTIVITY, m_ActivityInfo.ActivityID); + missionComponent->Progress(eMissionTaskType::ACTIVITY, m_ActivityInfo.ActivityID); } // First, get the activity data - auto* activityRewardsTable = CDClientManager::Instance()->GetTable("ActivityRewards"); + auto* activityRewardsTable = CDClientManager::Instance().GetTable(); std::vector activityRewards = activityRewardsTable->Query([=](CDActivityRewards entry) { return (entry.objectTemplate == m_ActivityInfo.ActivityID); }); if (!activityRewards.empty()) { uint32_t minCoins = 0; uint32_t maxCoins = 0; - auto* currencyTableTable = CDClientManager::Instance()->GetTable("CurrencyTable"); + auto* currencyTableTable = CDClientManager::Instance().GetTable(); std::vector currencyTable = currencyTableTable->Query([=](CDCurrencyTable entry) { return (entry.currencyIndex == activityRewards[0].CurrencyIndex && entry.npcminlevel == 1); }); if (!currencyTable.empty()) { @@ -552,7 +585,7 @@ std::vector ActivityInstance::GetParticipants() const { entities.reserve(m_Participants.size()); for (const auto& id : m_Participants) { - auto* entity = EntityManager::Instance()->GetEntity(id); + auto* entity = Game::entityManager->GetEntity(id); if (entity != nullptr) entities.push_back(entity); } @@ -584,5 +617,5 @@ void ActivityInstance::SetScore(uint32_t score) { } Entity* LobbyPlayer::GetEntity() const { - return EntityManager::Instance()->GetEntity(entityID); + return Game::entityManager->GetEntity(entityID); } diff --git a/dGame/dComponents/ScriptedActivityComponent.h b/dGame/dComponents/ScriptedActivityComponent.h index 66ca799f..1d49a62d 100644 --- a/dGame/dComponents/ScriptedActivityComponent.h +++ b/dGame/dComponents/ScriptedActivityComponent.h @@ -11,6 +11,9 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" + +#include "CDActivitiesTable.h" /** * Represents an instance of an activity, having participants and score @@ -153,7 +156,7 @@ struct ActivityPlayer { */ class ScriptedActivityComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SCRIPTED_ACTIVITY; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SCRIPTED_ACTIVITY; ScriptedActivityComponent(Entity* parent, int activityID); ~ScriptedActivityComponent() override; @@ -276,6 +279,12 @@ public: */ ActivityInstance* GetInstance(const LWOOBJID playerID); + /** + * @brief Reloads the config settings for this component + * + */ + void ReloadConfig(); + /** * Removes all the instances */ @@ -361,6 +370,12 @@ private: * LMIs for team sizes */ std::unordered_map m_ActivityLootMatrices; + + /** + * The activity id + * + */ + int32_t m_ActivityID; }; #endif // SCRIPTEDACTIVITYCOMPONENT_H diff --git a/dGame/dComponents/ShootingGalleryComponent.cpp b/dGame/dComponents/ShootingGalleryComponent.cpp index d5e12b28..ed91ac96 100644 --- a/dGame/dComponents/ShootingGalleryComponent.cpp +++ b/dGame/dComponents/ShootingGalleryComponent.cpp @@ -14,7 +14,7 @@ void ShootingGalleryComponent::SetStaticParams(const StaticShootingGalleryParams void ShootingGalleryComponent::SetDynamicParams(const DynamicShootingGalleryParams& params) { m_DynamicParams = params; m_Dirty = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } void ShootingGalleryComponent::Serialize(RakNet::BitStream* outBitStream, bool isInitialUpdate, uint32_t& flags) const { diff --git a/dGame/dComponents/ShootingGalleryComponent.h b/dGame/dComponents/ShootingGalleryComponent.h index c43f20c2..c31575f1 100644 --- a/dGame/dComponents/ShootingGalleryComponent.h +++ b/dGame/dComponents/ShootingGalleryComponent.h @@ -3,6 +3,7 @@ #include "NiPoint3.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Parameters for the shooting gallery that change during playtime @@ -72,7 +73,7 @@ struct StaticShootingGalleryParams { */ class ShootingGalleryComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SHOOTING_GALLERY; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SHOOTING_GALLERY; explicit ShootingGalleryComponent(Entity* parent); ~ShootingGalleryComponent(); diff --git a/dGame/dComponents/SimplePhysicsComponent.h b/dGame/dComponents/SimplePhysicsComponent.h index ebb8b124..51356710 100644 --- a/dGame/dComponents/SimplePhysicsComponent.h +++ b/dGame/dComponents/SimplePhysicsComponent.h @@ -11,6 +11,7 @@ #include "NiPoint3.h" #include "NiQuaternion.h" #include "Component.h" +#include "eReplicaComponentType.h" class Entity; @@ -27,7 +28,7 @@ enum class eClimbableType : int32_t { */ class SimplePhysicsComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SIMPLE_PHYSICS; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SIMPLE_PHYSICS; SimplePhysicsComponent(uint32_t componentID, Entity* parent); ~SimplePhysicsComponent() override; diff --git a/dGame/dComponents/SkillComponent.cpp b/dGame/dComponents/SkillComponent.cpp index f7e8e7d9..5c1d221a 100644 --- a/dGame/dComponents/SkillComponent.cpp +++ b/dGame/dComponents/SkillComponent.cpp @@ -19,11 +19,18 @@ #include "BaseCombatAIComponent.h" #include "ScriptComponent.h" #include "BuffComponent.h" - +#include "EchoStartSkill.h" +#include "DoClientProjectileImpact.h" +#include "CDClientManager.h" +#include "CDSkillBehaviorTable.h" +#include "eConnectionType.h" +#include "eClientMessageType.h" ProjectileSyncEntry::ProjectileSyncEntry() { } +std::unordered_map SkillComponent::m_skillBehaviorCache = {}; + bool SkillComponent::CastPlayerSkill(const uint32_t behaviorId, const uint32_t skillUid, RakNet::BitStream* bitStream, const LWOOBJID target, uint32_t skillID) { auto* context = new BehaviorContext(this->m_Parent->GetObjectID()); @@ -123,10 +130,14 @@ void SkillComponent::RegisterPlayerProjectile(const LWOOBJID projectileId, Behav } void SkillComponent::Update(const float deltaTime) { - if (!m_Parent->HasComponent(COMPONENT_TYPE_BASE_COMBAT_AI) && m_Parent->GetLOT() != 1) { + if (!m_Parent->HasComponent(eReplicaComponentType::BASE_COMBAT_AI) && m_Parent->GetLOT() != 1) { CalculateUpdate(deltaTime); } + if (m_Parent->IsPlayer()) { + for (const auto& pair : this->m_managedBehaviors) pair.second->UpdatePlayerSyncs(deltaTime); + } + std::map keep{}; for (const auto& pair : this->m_managedBehaviors) { @@ -181,17 +192,19 @@ void SkillComponent::Reset() { } void SkillComponent::Interrupt() { - if (m_Parent->IsPlayer()) return; - + // TODO: need to check immunities on the destroyable component, but they aren't implemented auto* combat = m_Parent->GetComponent(); - - if (combat != nullptr && combat->GetStunImmune()) { - return; - } + if (combat != nullptr && combat->GetStunImmune()) return; for (const auto& behavior : this->m_managedBehaviors) { + for (const auto& behaviorEndEntry : behavior.second->endEntries) { + behaviorEndEntry.behavior->End(behavior.second, behaviorEndEntry.branchContext, behaviorEndEntry.second); + } + behavior.second->endEntries.clear(); + if (m_Parent->IsPlayer()) continue; behavior.second->Interrupt(); } + } void SkillComponent::RegisterCalculatedProjectile(const LWOOBJID projectileId, BehaviorContext* context, const BehaviorBranchContext& branch, const LOT lot, const float maxTime, @@ -214,6 +227,29 @@ void SkillComponent::RegisterCalculatedProjectile(const LWOOBJID projectileId, B this->m_managedProjectiles.push_back(entry); } +bool SkillComponent::CastSkill(const uint32_t skillId, LWOOBJID target, const LWOOBJID optionalOriginatorID) { + uint32_t behaviorId = -1; + // try to find it via the cache + const auto& pair = m_skillBehaviorCache.find(skillId); + + // if it's not in the cache look it up and cache it + if (pair == m_skillBehaviorCache.end()) { + auto skillTable = CDClientManager::Instance().GetTable(); + behaviorId = skillTable->GetSkillByID(skillId).behaviorID; + m_skillBehaviorCache.insert_or_assign(skillId, behaviorId); + } else { + behaviorId = pair->second; + } + + // check to see if we got back a valid behavior + if (behaviorId == -1) { + Game::logger->LogDebug("SkillComponent", "Tried to cast skill %i but found no behavior", skillId); + return false; + } + + return CalculateBehavior(skillId, behaviorId, target, false, false, optionalOriginatorID).success; +} + SkillExecutionResult SkillComponent::CalculateBehavior(const uint32_t skillId, const uint32_t behaviorId, const LWOOBJID target, const bool ignoreTarget, const bool clientInitalized, const LWOOBJID originatorOverride) { auto* bitStream = new RakNet::BitStream(); @@ -224,6 +260,8 @@ SkillExecutionResult SkillComponent::CalculateBehavior(const uint32_t skillId, c context->caster = m_Parent->GetObjectID(); + context->skillID = skillId; + context->clientInitalized = clientInitalized; context->foundTarget = target != LWOOBJID_EMPTY || ignoreTarget || clientInitalized; @@ -246,14 +284,15 @@ SkillExecutionResult SkillComponent::CalculateBehavior(const uint32_t skillId, c if (!clientInitalized) { // Echo start skill - GameMessages::EchoStartSkill start; + EchoStartSkill start; start.iCastType = 0; start.skillID = skillId; start.uiSkillHandle = context->skillUId; start.optionalOriginatorID = context->originator; + start.optionalTargetID = target; - auto* originator = EntityManager::Instance()->GetEntity(context->originator); + auto* originator = Game::entityManager->GetEntity(context->originator); if (originator != nullptr) { start.originatorRot = originator->GetRotation(); @@ -265,7 +304,7 @@ SkillExecutionResult SkillComponent::CalculateBehavior(const uint32_t skillId, c // Write message RakNet::BitStream message; - PacketUtils::WriteHeader(message, CLIENT, MSG_CLIENT_GAME_MSG); + PacketUtils::WriteHeader(message, eConnectionType::CLIENT, eClientMessageType::GAME_MSG); message.Write(this->m_Parent->GetObjectID()); start.Serialize(&message); @@ -299,7 +338,7 @@ void SkillComponent::CalculateUpdate(const float deltaTime) { entry.time += deltaTime; - auto* origin = EntityManager::Instance()->GetEntity(entry.context->originator); + auto* origin = Game::entityManager->GetEntity(entry.context->originator); if (origin == nullptr) { continue; @@ -310,7 +349,7 @@ void SkillComponent::CalculateUpdate(const float deltaTime) { const auto position = entry.startPosition + (entry.velocity * entry.time); for (const auto& targetId : targets) { - auto* target = EntityManager::Instance()->GetEntity(targetId); + auto* target = Game::entityManager->GetEntity(targetId); const auto targetPosition = target->GetPosition(); @@ -319,34 +358,7 @@ void SkillComponent::CalculateUpdate(const float deltaTime) { const auto distance = Vector3::DistanceSquared(targetPosition, closestPoint); if (distance > 3 * 3) { - /* - if (entry.TrackTarget && distance <= entry.TrackRadius) - { - const auto rotation = NiQuaternion::LookAtUnlocked(position, targetPosition); - - const auto speed = entry.Velocity.Length(); - - const auto homingTarget = rotation.GetForwardVector() * speed; - - Vector3 homing; - - // Move towards - - const auto difference = homingTarget - entry.Velocity; - const auto mag = difference.Length(); - if (mag <= speed || mag == 0) - { - homing = homingTarget; - } - else - { - entry.Velocity + homingTarget / mag * speed; - } - - entry.Velocity = homing; - } - */ - + // TODO There is supposed to be an implementation for homing projectiles here continue; } @@ -385,7 +397,7 @@ void SkillComponent::CalculateUpdate(const float deltaTime) { void SkillComponent::SyncProjectileCalculation(const ProjectileSyncEntry& entry) const { - auto* other = EntityManager::Instance()->GetEntity(entry.branchContext.target); + auto* other = Game::entityManager->GetEntity(entry.branchContext.target); if (other == nullptr) { if (entry.branchContext.target != LWOOBJID_EMPTY) { @@ -416,7 +428,7 @@ void SkillComponent::SyncProjectileCalculation(const ProjectileSyncEntry& entry) behavior->Calculate(entry.context, bitStream, entry.branchContext); - GameMessages::DoClientProjectileImpact projectileImpact; + DoClientProjectileImpact projectileImpact; projectileImpact.sBitStream.assign((char*)bitStream->GetData(), bitStream->GetNumberOfBytesUsed()); projectileImpact.i64OwnerID = this->m_Parent->GetObjectID(); @@ -425,7 +437,7 @@ void SkillComponent::SyncProjectileCalculation(const ProjectileSyncEntry& entry) RakNet::BitStream message; - PacketUtils::WriteHeader(message, CLIENT, MSG_CLIENT_GAME_MSG); + PacketUtils::WriteHeader(message, eConnectionType::CLIENT, eClientMessageType::GAME_MSG); message.Write(this->m_Parent->GetObjectID()); projectileImpact.Serialize(&message); @@ -465,7 +477,7 @@ void SkillComponent::HandleUnCast(const uint32_t behaviorId, const LWOOBJID targ delete context; } -SkillComponent::SkillComponent(Entity* parent) : Component(parent) { +SkillComponent::SkillComponent(Entity* parent): Component(parent) { this->m_skillUid = 0; } diff --git a/dGame/dComponents/SkillComponent.h b/dGame/dComponents/SkillComponent.h index f43276f1..034e65ce 100644 --- a/dGame/dComponents/SkillComponent.h +++ b/dGame/dComponents/SkillComponent.h @@ -13,6 +13,7 @@ #include "Component.h" #include "Entity.h" #include "dLogger.h" +#include "eReplicaComponentType.h" struct ProjectileSyncEntry { LWOOBJID id = LWOOBJID_EMPTY; @@ -58,7 +59,7 @@ struct SkillExecutionResult { */ class SkillComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SKILL; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SKILL; explicit SkillComponent(Entity* parent); ~SkillComponent() override; @@ -119,6 +120,15 @@ public: */ void RegisterPlayerProjectile(LWOOBJID projectileId, BehaviorContext* context, const BehaviorBranchContext& branch, LOT lot); + /** + * Wrapper for CalculateBehavior that mimics the call structure in scripts and helps reduce magic numbers + * @param skillId the skill to cast + * @param target the target of the skill + * @param optionalOriginatorID change the originator of the skill + * @return if the case succeeded + */ + bool CastSkill(const uint32_t skillId, LWOOBJID target = LWOOBJID_EMPTY, const LWOOBJID optionalOriginatorID = LWOOBJID_EMPTY); + /** * Initializes a server-side skill calculation. * @param skillId the skill ID @@ -190,6 +200,11 @@ private: */ uint32_t m_skillUid; + /** + * Cache for looking up a behavior id via a skill ID + */ + static std::unordered_map m_skillBehaviorCache; + /** * Sync a server-side projectile calculation. * @param entry the projectile information diff --git a/dGame/dComponents/SoundTriggerComponent.cpp b/dGame/dComponents/SoundTriggerComponent.cpp index be62beee..21d30948 100644 --- a/dGame/dComponents/SoundTriggerComponent.cpp +++ b/dGame/dComponents/SoundTriggerComponent.cpp @@ -76,7 +76,7 @@ void SoundTriggerComponent::ActivateMusicCue(const std::string& name) { -1.0f }); dirty = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } @@ -88,6 +88,6 @@ void SoundTriggerComponent::DeactivateMusicCue(const std::string& name) { if (musicCue != this->musicCues.end()) { this->musicCues.erase(musicCue); dirty = true; - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } diff --git a/dGame/dComponents/SoundTriggerComponent.h b/dGame/dComponents/SoundTriggerComponent.h index e1178150..954d8495 100644 --- a/dGame/dComponents/SoundTriggerComponent.h +++ b/dGame/dComponents/SoundTriggerComponent.h @@ -3,6 +3,7 @@ #include "Entity.h" #include "GUID.h" #include "Component.h" +#include "eReplicaComponentType.h" /** * Music that should be played by the client @@ -19,7 +20,7 @@ struct MusicCue { */ class SoundTriggerComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SOUND_TRIGGER; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SOUND_TRIGGER; explicit SoundTriggerComponent(Entity* parent); ~SoundTriggerComponent() override; diff --git a/dGame/dComponents/SwitchComponent.cpp b/dGame/dComponents/SwitchComponent.cpp index 2263a866..eee54342 100644 --- a/dGame/dComponents/SwitchComponent.cpp +++ b/dGame/dComponents/SwitchComponent.cpp @@ -1,5 +1,7 @@ #include "SwitchComponent.h" #include "EntityManager.h" +#include "eTriggerEventType.h" +#include "RenderComponent.h" std::vector SwitchComponent::petSwitches; @@ -38,16 +40,16 @@ bool SwitchComponent::GetActive() const { void SwitchComponent::EntityEnter(Entity* entity) { if (!m_Active) { if (m_Rebuild) { - if (m_Rebuild->GetState() != eRebuildState::REBUILD_COMPLETED) return; + if (m_Rebuild->GetState() != eRebuildState::COMPLETED) return; } m_Active = true; if (!m_Parent) return; - m_Parent->TriggerEvent("OnActivated"); + m_Parent->TriggerEvent(eTriggerEventType::ACTIVATED, entity); const auto grpName = m_Parent->GetVarAsString(u"grp_name"); if (!grpName.empty()) { - const auto entities = EntityManager::Instance()->GetEntitiesInGroup(grpName); + const auto entities = Game::entityManager->GetEntitiesInGroup(grpName); for (auto* entity : entities) { entity->OnFireEventServerSide(entity, "OnActivated"); @@ -58,10 +60,10 @@ void SwitchComponent::EntityEnter(Entity* entity) { if (m_PetBouncer != nullptr) { GameMessages::SendPlayFXEffect(m_Parent->GetObjectID(), 2602, u"pettriggeractive", "BounceEffect", LWOOBJID_EMPTY, 1, 1, true); - GameMessages::SendPlayAnimation(m_Parent, u"engaged", 0, 1); + RenderComponent::PlayAnimation(m_Parent, u"engaged"); m_PetBouncer->SetPetBouncerEnabled(true); } else { - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } @@ -78,12 +80,12 @@ void SwitchComponent::Update(float deltaTime) { if (m_Timer <= 0.0f) { m_Active = false; if (!m_Parent) return; - m_Parent->TriggerEvent("OnDectivated"); + m_Parent->TriggerEvent(eTriggerEventType::DEACTIVATED, m_Parent); const auto grpName = m_Parent->GetVarAsString(u"grp_name"); if (!grpName.empty()) { - const auto entities = EntityManager::Instance()->GetEntitiesInGroup(grpName); + const auto entities = Game::entityManager->GetEntitiesInGroup(grpName); for (auto* entity : entities) { entity->OnFireEventServerSide(entity, "OnDectivated"); @@ -93,7 +95,7 @@ void SwitchComponent::Update(float deltaTime) { if (m_PetBouncer != nullptr) { m_PetBouncer->SetPetBouncerEnabled(false); } else { - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); } } } diff --git a/dGame/dComponents/SwitchComponent.h b/dGame/dComponents/SwitchComponent.h index ea5955d8..fde3cfc0 100644 --- a/dGame/dComponents/SwitchComponent.h +++ b/dGame/dComponents/SwitchComponent.h @@ -9,13 +9,14 @@ #include "BouncerComponent.h" #include #include "Component.h" +#include "eReplicaComponentType.h" /** * A component for switches in game, including pet triggered switches. */ class SwitchComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SWITCH; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SWITCH; SwitchComponent(Entity* parent); ~SwitchComponent() override; diff --git a/dGame/dComponents/TriggerComponent.cpp b/dGame/dComponents/TriggerComponent.cpp new file mode 100644 index 00000000..ab52c4e5 --- /dev/null +++ b/dGame/dComponents/TriggerComponent.cpp @@ -0,0 +1,443 @@ +#include "TriggerComponent.h" +#include "dZoneManager.h" +#include "TeamManager.h" +#include "eTriggerCommandType.h" +#include "eMissionTaskType.h" +#include "ePhysicsEffectType.h" + +#include "CharacterComponent.h" +#include "ControllablePhysicsComponent.h" +#include "MissionComponent.h" +#include "PhantomPhysicsComponent.h" +#include "Player.h" +#include "RebuildComponent.h" +#include "SkillComponent.h" +#include "eEndBehavior.h" + + +TriggerComponent::TriggerComponent(Entity* parent, const std::string triggerInfo): Component(parent) { + m_Parent = parent; + m_Trigger = nullptr; + + std::vector tokens = GeneralUtils::SplitString(triggerInfo, ':'); + + uint32_t sceneID; + GeneralUtils::TryParse(tokens.at(0), sceneID); + uint32_t triggerID; + GeneralUtils::TryParse(tokens.at(1), triggerID); + + m_Trigger = Game::zoneManager->GetZone()->GetTrigger(sceneID, triggerID); + + if (!m_Trigger) m_Trigger = new LUTriggers::Trigger(); +} + +void TriggerComponent::TriggerEvent(eTriggerEventType event, Entity* optionalTarget) { + if (m_Trigger && m_Trigger->enabled) { + for (LUTriggers::Event* triggerEvent : m_Trigger->events) { + if (triggerEvent->id == event) { + for (LUTriggers::Command* command : triggerEvent->commands) { + HandleTriggerCommand(command, optionalTarget); + } + } + } + } +} + +void TriggerComponent::HandleTriggerCommand(LUTriggers::Command* command, Entity* optionalTarget) { + auto argArray = GeneralUtils::SplitString(command->args, ','); + + // determine targets + std::vector targetEntities = GatherTargets(command, optionalTarget); + + // if we have no targets, then we are done + if (targetEntities.empty()) return; + + for (Entity* targetEntity : targetEntities) { + if (!targetEntity) continue; + + switch (command->id) { + case eTriggerCommandType::ZONE_PLAYER: break; + case eTriggerCommandType::FIRE_EVENT: + HandleFireEvent(targetEntity, command->args); + break; + case eTriggerCommandType::DESTROY_OBJ: + HandleDestroyObject(targetEntity, command->args); + break; + case eTriggerCommandType::TOGGLE_TRIGGER: + HandleToggleTrigger(targetEntity, command->args); + break; + case eTriggerCommandType::RESET_REBUILD: + HandleResetRebuild(targetEntity, command->args); + break; + case eTriggerCommandType::SET_PATH: break; + case eTriggerCommandType::SET_PICK_TYPE: break; + case eTriggerCommandType::MOVE_OBJECT: + HandleMoveObject(targetEntity, argArray); + break; + case eTriggerCommandType::ROTATE_OBJECT: + HandleRotateObject(targetEntity, argArray); + break; + case eTriggerCommandType::PUSH_OBJECT: + HandlePushObject(targetEntity, argArray); + break; + case eTriggerCommandType::REPEL_OBJECT: + HandleRepelObject(targetEntity, command->args); + break; + case eTriggerCommandType::SET_TIMER: + HandleSetTimer(targetEntity, argArray); + break; + case eTriggerCommandType::CANCEL_TIMER: + HandleCancelTimer(targetEntity, command->args); + break; + case eTriggerCommandType::PLAY_CINEMATIC: + HandlePlayCinematic(targetEntity, argArray); + break; + case eTriggerCommandType::TOGGLE_BBB: + HandleToggleBBB(targetEntity, command->args); + break; + case eTriggerCommandType::UPDATE_MISSION: + HandleUpdateMission(targetEntity, argArray); + break; + case eTriggerCommandType::SET_BOUNCER_STATE: break; + case eTriggerCommandType::BOUNCE_ALL_ON_BOUNCER: break; + case eTriggerCommandType::TURN_AROUND_ON_PATH: break; + case eTriggerCommandType::GO_FORWARD_ON_PATH: break; + case eTriggerCommandType::GO_BACKWARD_ON_PATH: break; + case eTriggerCommandType::STOP_PATHING: break; + case eTriggerCommandType::START_PATHING: break; + case eTriggerCommandType::LOCK_OR_UNLOCK_CONTROLS: break; + case eTriggerCommandType::PLAY_EFFECT: + HandlePlayEffect(targetEntity, argArray); + break; + case eTriggerCommandType::STOP_EFFECT: + GameMessages::SendStopFXEffect(targetEntity, true, command->args); + break; + case eTriggerCommandType::CAST_SKILL: + HandleCastSkill(targetEntity, command->args); + break; + case eTriggerCommandType::DISPLAY_ZONE_SUMMARY: + GameMessages::SendDisplayZoneSummary(targetEntity->GetObjectID(), targetEntity->GetSystemAddress(), false, command->args == "1", m_Parent->GetObjectID()); + break; + case eTriggerCommandType::SET_PHYSICS_VOLUME_EFFECT: + HandleSetPhysicsVolumeEffect(targetEntity, argArray); + break; + case eTriggerCommandType::SET_PHYSICS_VOLUME_STATUS: + HandleSetPhysicsVolumeStatus(targetEntity, command->args); + break; + case eTriggerCommandType::SET_MODEL_TO_BUILD: break; + case eTriggerCommandType::SPAWN_MODEL_BRICKS: break; + case eTriggerCommandType::ACTIVATE_SPAWNER_NETWORK: + HandleActivateSpawnerNetwork(command->args); + break; + case eTriggerCommandType::DEACTIVATE_SPAWNER_NETWORK: + HandleDeactivateSpawnerNetwork(command->args); + break; + case eTriggerCommandType::RESET_SPAWNER_NETWORK: + HandleResetSpawnerNetwork(command->args); + break; + case eTriggerCommandType::DESTROY_SPAWNER_NETWORK_OBJECTS: + HandleDestroySpawnerNetworkObjects(command->args); + break; + case eTriggerCommandType::GO_TO_WAYPOINT: break; + case eTriggerCommandType::ACTIVATE_PHYSICS: + HandleActivatePhysics(targetEntity, command->args); + break; + // DEPRECATED BLOCK START + case eTriggerCommandType::ACTIVATE_MUSIC_CUE: break; + case eTriggerCommandType::DEACTIVATE_MUSIC_CUE: break; + case eTriggerCommandType::FLASH_MUSIC_CUE: break; + case eTriggerCommandType::SET_MUSIC_PARAMETER: break; + case eTriggerCommandType::PLAY_2D_AMBIENT_SOUND: break; + case eTriggerCommandType::STOP_2D_AMBIENT_SOUND: break; + case eTriggerCommandType::PLAY_3D_AMBIENT_SOUND: break; + case eTriggerCommandType::STOP_3D_AMBIENT_SOUND: break; + case eTriggerCommandType::ACTIVATE_MIXER_PROGRAM: break; + case eTriggerCommandType::DEACTIVATE_MIXER_PROGRAM: break; + // DEPRECATED BLOCK END + default: + Game::logger->LogDebug("TriggerComponent", "Event %i was not handled!", command->id); + break; + } + } +} + +std::vector TriggerComponent::GatherTargets(LUTriggers::Command* command, Entity* optionalTarget) { + std::vector entities = {}; + + if (command->target == "self") entities.push_back(m_Parent); + else if (command->target == "zone") { /*TODO*/ } + else if (command->target == "target" && optionalTarget) entities.push_back(optionalTarget); + else if (command->target == "targetTeam" && optionalTarget) { + auto* team = TeamManager::Instance()->GetTeam(optionalTarget->GetObjectID()); + for (const auto memberId : team->members) { + auto* member = Game::entityManager->GetEntity(memberId); + if (member) entities.push_back(member); + } + } else if (command->target == "objGroup") entities = Game::entityManager->GetEntitiesInGroup(command->targetName); + else if (command->target == "allPlayers") { + for (auto* player : Player::GetAllPlayers()) { + entities.push_back(player); + } + } else if (command->target == "allNPCs") { /*UNUSED*/ } + + return entities; +} + +void TriggerComponent::HandleFireEvent(Entity* targetEntity, std::string args) { + for (CppScripts::Script* script : CppScripts::GetEntityScripts(targetEntity)) { + script->OnFireEventServerSide(targetEntity, m_Parent, args, 0, 0, 0); + } +} + +void TriggerComponent::HandleDestroyObject(Entity* targetEntity, std::string args){ + uint32_t killType; + GeneralUtils::TryParse(args, killType); + targetEntity->Smash(m_Parent->GetObjectID(), static_cast(killType)); +} + +void TriggerComponent::HandleToggleTrigger(Entity* targetEntity, std::string args){ + auto* triggerComponent = targetEntity->GetComponent(); + if (!triggerComponent) { + Game::logger->LogDebug("TriggerComponent::HandleToggleTrigger", "Trigger component not found!"); + return; + } + triggerComponent->SetTriggerEnabled(args == "1"); +} + +void TriggerComponent::HandleResetRebuild(Entity* targetEntity, std::string args){ + auto* rebuildComponent = targetEntity->GetComponent(); + if (!rebuildComponent) { + Game::logger->LogDebug("TriggerComponent::HandleResetRebuild", "Rebuild component not found!"); + return; + } + rebuildComponent->ResetRebuild(args == "1"); +} + +void TriggerComponent::HandleMoveObject(Entity* targetEntity, std::vector argArray){ + if (argArray.size() <= 2) return; + + auto position = targetEntity->GetPosition(); + NiPoint3 offset = NiPoint3::ZERO; + GeneralUtils::TryParse(argArray.at(0), argArray.at(1), argArray.at(2), offset); + + position += offset; + targetEntity->SetPosition(position); +} + +void TriggerComponent::HandleRotateObject(Entity* targetEntity, std::vector argArray){ + if (argArray.size() <= 2) return; + + NiPoint3 vector = NiPoint3::ZERO; + GeneralUtils::TryParse(argArray.at(0), argArray.at(1), argArray.at(2), vector); + + NiQuaternion rotation = NiQuaternion::FromEulerAngles(vector); + targetEntity->SetRotation(rotation); +} + +void TriggerComponent::HandlePushObject(Entity* targetEntity, std::vector argArray){ + if (argArray.size() < 3) return; + + auto* phantomPhysicsComponent = m_Parent->GetComponent(); + if (!phantomPhysicsComponent) { + Game::logger->LogDebug("TriggerComponent::HandlePushObject", "Phantom Physics component not found!"); + return; + } + phantomPhysicsComponent->SetPhysicsEffectActive(true); + phantomPhysicsComponent->SetEffectType(ePhysicsEffectType::PUSH); + phantomPhysicsComponent->SetDirectionalMultiplier(1); + NiPoint3 direction = NiPoint3::ZERO; + GeneralUtils::TryParse(argArray.at(0), argArray.at(1), argArray.at(2), direction); + phantomPhysicsComponent->SetDirection(direction); + + Game::entityManager->SerializeEntity(m_Parent); +} + + +void TriggerComponent::HandleRepelObject(Entity* targetEntity, std::string args){ + auto* phantomPhysicsComponent = m_Parent->GetComponent(); + if (!phantomPhysicsComponent) { + Game::logger->LogDebug("TriggerComponent::HandleRepelObject", "Phantom Physics component not found!"); + return; + } + float forceMultiplier; + GeneralUtils::TryParse(args, forceMultiplier); + phantomPhysicsComponent->SetPhysicsEffectActive(true); + phantomPhysicsComponent->SetEffectType(ePhysicsEffectType::REPULSE); + phantomPhysicsComponent->SetDirectionalMultiplier(forceMultiplier); + + auto triggerPos = m_Parent->GetPosition(); + auto targetPos = targetEntity->GetPosition(); + + // normalize the vectors to get the direction + auto delta = targetPos - triggerPos; + auto length = delta.Length(); + NiPoint3 direction = delta / length; + phantomPhysicsComponent->SetDirection(direction); + + Game::entityManager->SerializeEntity(m_Parent); +} + +void TriggerComponent::HandleSetTimer(Entity* targetEntity, std::vector argArray){ + if (argArray.size() != 2) { + Game::logger->LogDebug("TriggerComponent::HandleSetTimer", "Not ehought variables!"); + return; + } + float time = 0.0; + GeneralUtils::TryParse(argArray.at(1), time); + m_Parent->AddTimer(argArray.at(0), time); +} + +void TriggerComponent::HandleCancelTimer(Entity* targetEntity, std::string args){ + m_Parent->CancelTimer(args); +} + +void TriggerComponent::HandlePlayCinematic(Entity* targetEntity, std::vector argArray) { + float leadIn = -1.0; + auto wait = eEndBehavior::RETURN; + bool unlock = true; + bool leaveLocked = false; + bool hidePlayer = false; + + if (argArray.size() >= 2) { + GeneralUtils::TryParse(argArray.at(1), leadIn); + if (argArray.size() >= 3 && argArray.at(2) == "wait") { + wait = eEndBehavior::WAIT; + if (argArray.size() >= 4 && argArray.at(3) == "unlock") { + unlock = false; + if (argArray.size() >= 5 && argArray.at(4) == "leavelocked") { + leaveLocked = true; + if (argArray.size() >= 6 && argArray.at(5) == "hideplayer") { + hidePlayer = true; + } + } + } + } + } + + GameMessages::SendPlayCinematic(targetEntity->GetObjectID(), GeneralUtils::UTF8ToUTF16(argArray.at(0)), targetEntity->GetSystemAddress(), true, true, false, false, wait, hidePlayer, leadIn, leaveLocked, unlock); +} + +void TriggerComponent::HandleToggleBBB(Entity* targetEntity, std::string args) { + auto* character = targetEntity->GetCharacter(); + if (!character) { + Game::logger->LogDebug("TriggerComponent::HandleToggleBBB", "Character was not found!"); + return; + } + bool buildMode = !(character->GetBuildMode()); + if (args == "enter") buildMode = true; + else if (args == "exit") buildMode = false; + character->SetBuildMode(buildMode); +} + +void TriggerComponent::HandleUpdateMission(Entity* targetEntity, std::vector argArray) { + // there are only explore tasks used + // If others need to be implemented for modding + // then we need a good way to convert this from a string to that enum + if (argArray.at(0) != "exploretask") return; + MissionComponent* missionComponent = targetEntity->GetComponent(); + if (!missionComponent){ + Game::logger->LogDebug("TriggerComponent::HandleUpdateMission", "Mission component not found!"); + return; + } + missionComponent->Progress(eMissionTaskType::EXPLORE, 0, 0, argArray.at(4)); +} + +void TriggerComponent::HandlePlayEffect(Entity* targetEntity, std::vector argArray) { + if (argArray.size() < 3) return; + int32_t effectID = 0; + if (!GeneralUtils::TryParse(argArray.at(1), effectID)) return; + std::u16string effectType = GeneralUtils::UTF8ToUTF16(argArray.at(2)); + float priority = 1; + if (argArray.size() == 4) GeneralUtils::TryParse(argArray.at(3), priority); + GameMessages::SendPlayFXEffect(targetEntity, effectID, effectType, argArray.at(0), LWOOBJID_EMPTY, priority); +} + +void TriggerComponent::HandleCastSkill(Entity* targetEntity, std::string args){ + auto* skillComponent = targetEntity->GetComponent(); + if (!skillComponent) { + Game::logger->LogDebug("TriggerComponent::HandleCastSkill", "Skill component not found!"); + return; + } + uint32_t skillId; + GeneralUtils::TryParse(args, skillId); + skillComponent->CastSkill(skillId, targetEntity->GetObjectID()); +} + +void TriggerComponent::HandleSetPhysicsVolumeEffect(Entity* targetEntity, std::vector argArray) { + auto* phantomPhysicsComponent = targetEntity->GetComponent(); + if (!phantomPhysicsComponent) { + Game::logger->LogDebug("TriggerComponent::HandleSetPhysicsVolumeEffect", "Phantom Physics component not found!"); + return; + } + phantomPhysicsComponent->SetPhysicsEffectActive(true); + ePhysicsEffectType effectType = ePhysicsEffectType::PUSH; + std::transform(argArray.at(0).begin(), argArray.at(0).end(), argArray.at(0).begin(), ::tolower); //Transform to lowercase + if (argArray.at(0) == "push") effectType = ePhysicsEffectType::PUSH; + else if (argArray.at(0) == "attract") effectType = ePhysicsEffectType::ATTRACT; + else if (argArray.at(0) == "repulse") effectType = ePhysicsEffectType::REPULSE; + else if (argArray.at(0) == "gravity") effectType = ePhysicsEffectType::GRAVITY_SCALE; + else if (argArray.at(0) == "friction") effectType = ePhysicsEffectType::FRICTION; + + phantomPhysicsComponent->SetEffectType(effectType); + phantomPhysicsComponent->SetDirectionalMultiplier(std::stof(argArray.at(1))); + if (argArray.size() > 4) { + NiPoint3 direction = NiPoint3::ZERO; + GeneralUtils::TryParse(argArray.at(2), argArray.at(3), argArray.at(4), direction); + phantomPhysicsComponent->SetDirection(direction); + } + if (argArray.size() > 5) { + uint32_t min; + GeneralUtils::TryParse(argArray.at(6), min); + phantomPhysicsComponent->SetMin(min); + + uint32_t max; + GeneralUtils::TryParse(argArray.at(7), max); + phantomPhysicsComponent->SetMax(max); + } + + Game::entityManager->SerializeEntity(targetEntity); +} + +void TriggerComponent::HandleSetPhysicsVolumeStatus(Entity* targetEntity, std::string args) { + auto* phantomPhysicsComponent = targetEntity->GetComponent(); + if (!phantomPhysicsComponent) { + Game::logger->LogDebug("TriggerComponent::HandleSetPhysicsVolumeEffect", "Phantom Physics component not found!"); + return; + } + phantomPhysicsComponent->SetPhysicsEffectActive(args == "On"); + Game::entityManager->SerializeEntity(targetEntity); +} + +void TriggerComponent::HandleActivateSpawnerNetwork(std::string args){ + for (auto* spawner : Game::zoneManager->GetSpawnersByName(args)) { + if (spawner) spawner->Activate(); + } +} + +void TriggerComponent::HandleDeactivateSpawnerNetwork(std::string args){ + for (auto* spawner : Game::zoneManager->GetSpawnersByName(args)) { + if (spawner) spawner->Deactivate(); + } +} + +void TriggerComponent::HandleResetSpawnerNetwork(std::string args){ + for (auto* spawner : Game::zoneManager->GetSpawnersByName(args)) { + if (spawner) spawner->Reset(); + } +} + +void TriggerComponent::HandleDestroySpawnerNetworkObjects(std::string args){ + for (auto* spawner : Game::zoneManager->GetSpawnersByName(args)) { + if (spawner) spawner->DestroyAllEntities(); + } +} + +void TriggerComponent::HandleActivatePhysics(Entity* targetEntity, std::string args) { + if (args == "true") { + // TODO add physics entity if there isn't one + } else if (args == "false"){ + // TODO remove Phsyics entity if there is one + } else { + Game::logger->LogDebug("TriggerComponent", "Invalid argument for ActivatePhysics Trigger: %s", args.c_str()); + } +} diff --git a/dGame/dComponents/TriggerComponent.h b/dGame/dComponents/TriggerComponent.h new file mode 100644 index 00000000..df65707f --- /dev/null +++ b/dGame/dComponents/TriggerComponent.h @@ -0,0 +1,50 @@ +#ifndef __TRIGGERCOMPONENT__H__ +#define __TRIGGERCOMPONENT__H__ + +#include "Component.h" +#include "LUTriggers.h" +#include "eReplicaComponentType.h" + +class TriggerComponent : public Component { +public: + static const eReplicaComponentType ComponentType = eReplicaComponentType::TRIGGER; + + explicit TriggerComponent(Entity* parent, const std::string triggerInfo); + + void TriggerEvent(eTriggerEventType event, Entity* optionalTarget = nullptr); + LUTriggers::Trigger* GetTrigger() const { return m_Trigger; } + void SetTriggerEnabled(bool enabled){ m_Trigger->enabled = enabled; }; + + +private: + + void HandleTriggerCommand(LUTriggers::Command* command, Entity* optionalTarget); + std::vector GatherTargets(LUTriggers::Command* command, Entity* optionalTarget); + + // Trigger Event Handlers + void HandleFireEvent(Entity* targetEntity, std::string args); + void HandleDestroyObject(Entity* targetEntity, std::string args); + void HandleToggleTrigger(Entity* targetEntity, std::string args); + void HandleResetRebuild(Entity* targetEntity, std::string args); + void HandleMoveObject(Entity* targetEntity, std::vector argArray); + void HandleRotateObject(Entity* targetEntity, std::vector argArray); + void HandlePushObject(Entity* targetEntity, std::vector argArray); + void HandleRepelObject(Entity* targetEntity, std::string args); + void HandleSetTimer(Entity* targetEntity, std::vector argArray); + void HandleCancelTimer(Entity* targetEntity, std::string args); + void HandlePlayCinematic(Entity* targetEntity, std::vector argArray); + void HandleToggleBBB(Entity* targetEntity, std::string args); + void HandleUpdateMission(Entity* targetEntity, std::vector argArray); + void HandlePlayEffect(Entity* targetEntity, std::vector argArray); + void HandleCastSkill(Entity* targetEntity, std::string args); + void HandleSetPhysicsVolumeEffect(Entity* targetEntity, std::vector argArray); + void HandleSetPhysicsVolumeStatus(Entity* targetEntity, std::string args); + void HandleActivateSpawnerNetwork(std::string args); + void HandleDeactivateSpawnerNetwork(std::string args); + void HandleResetSpawnerNetwork(std::string args); + void HandleDestroySpawnerNetworkObjects(std::string args); + void HandleActivatePhysics(Entity* targetEntity, std::string args); + + LUTriggers::Trigger* m_Trigger; +}; +#endif //!__TRIGGERCOMPONENT__H__ diff --git a/dGame/dComponents/VehiclePhysicsComponent.cpp b/dGame/dComponents/VehiclePhysicsComponent.cpp index bf6a3bb7..684b135b 100644 --- a/dGame/dComponents/VehiclePhysicsComponent.cpp +++ b/dGame/dComponents/VehiclePhysicsComponent.cpp @@ -11,6 +11,7 @@ VehiclePhysicsComponent::VehiclePhysicsComponent(Entity* parent) : Component(par m_DirtyPosition = true; m_DirtyVelocity = true; m_DirtyAngularVelocity = true; + m_EndBehavior = GeneralUtils::GenerateRandomNumber(0, 7); } VehiclePhysicsComponent::~VehiclePhysicsComponent() { @@ -18,34 +19,47 @@ VehiclePhysicsComponent::~VehiclePhysicsComponent() { } void VehiclePhysicsComponent::SetPosition(const NiPoint3& pos) { + if (pos == m_Position) return; + m_DirtyPosition = true; m_Position = pos; } void VehiclePhysicsComponent::SetRotation(const NiQuaternion& rot) { + if (rot == m_Rotation) return; m_DirtyPosition = true; m_Rotation = rot; } void VehiclePhysicsComponent::SetVelocity(const NiPoint3& vel) { + if (vel == m_Velocity) return; m_DirtyPosition = true; m_Velocity = vel; } void VehiclePhysicsComponent::SetAngularVelocity(const NiPoint3& vel) { + if (vel == m_AngularVelocity) return; m_DirtyPosition = true; m_AngularVelocity = vel; } void VehiclePhysicsComponent::SetIsOnGround(bool val) { + if (val == m_IsOnGround) return; m_DirtyPosition = true; m_IsOnGround = val; } void VehiclePhysicsComponent::SetIsOnRail(bool val) { + if (val == m_IsOnRail) return; m_DirtyPosition = true; m_IsOnRail = val; } +void VehiclePhysicsComponent::SetRemoteInputInfo(const RemoteInputInfo& remoteInputInfo) { + if (m_RemoteInputInfo == remoteInputInfo) return; + this->m_RemoteInputInfo = remoteInputInfo; + m_DirtyRemoteInput = true; +} + void VehiclePhysicsComponent::SetDirtyPosition(bool val) { m_DirtyPosition = val; } @@ -62,9 +76,15 @@ void VehiclePhysicsComponent::Serialize(RakNet::BitStream* outBitStream, bool bI outBitStream->Write(bIsInitialUpdate || m_DirtyPosition); if (bIsInitialUpdate || m_DirtyPosition) { - outBitStream->Write(m_Position); + m_DirtyPosition = false; + outBitStream->Write(m_Position.x); + outBitStream->Write(m_Position.y); + outBitStream->Write(m_Position.z); - outBitStream->Write(m_Rotation); + outBitStream->Write(m_Rotation.x); + outBitStream->Write(m_Rotation.y); + outBitStream->Write(m_Rotation.z); + outBitStream->Write(m_Rotation.w); outBitStream->Write(m_IsOnGround); outBitStream->Write(m_IsOnRail); @@ -72,20 +92,33 @@ void VehiclePhysicsComponent::Serialize(RakNet::BitStream* outBitStream, bool bI outBitStream->Write(bIsInitialUpdate || m_DirtyVelocity); if (bIsInitialUpdate || m_DirtyVelocity) { - outBitStream->Write(m_Velocity); + outBitStream->Write(m_Velocity.x); + outBitStream->Write(m_Velocity.y); + outBitStream->Write(m_Velocity.z); + m_DirtyVelocity = false; } outBitStream->Write(bIsInitialUpdate || m_DirtyAngularVelocity); if (bIsInitialUpdate || m_DirtyAngularVelocity) { - outBitStream->Write(m_AngularVelocity); + outBitStream->Write(m_AngularVelocity.x); + outBitStream->Write(m_AngularVelocity.y); + outBitStream->Write(m_AngularVelocity.z); + m_DirtyAngularVelocity = false; } - outBitStream->Write0(); + outBitStream->Write0(); // local_space_info. TODO: Implement this - outBitStream->Write0(); + outBitStream->Write(m_DirtyRemoteInput || bIsInitialUpdate); // remote_input_info + if (m_DirtyRemoteInput || bIsInitialUpdate) { + outBitStream->Write(m_RemoteInputInfo.m_RemoteInputX); + outBitStream->Write(m_RemoteInputInfo.m_RemoteInputY); + outBitStream->Write(m_RemoteInputInfo.m_IsPowersliding); + outBitStream->Write(m_RemoteInputInfo.m_IsModified); + m_DirtyRemoteInput = false; + } - outBitStream->Write(0.0f); + outBitStream->Write(125.0f); // remote_input_ping TODO: Figure out how this should be calculated as it seems to be constant through the whole race. if (!bIsInitialUpdate) { outBitStream->Write0(); @@ -93,8 +126,8 @@ void VehiclePhysicsComponent::Serialize(RakNet::BitStream* outBitStream, bool bI } if (bIsInitialUpdate) { - outBitStream->Write(5); - outBitStream->Write1(); + outBitStream->Write(m_EndBehavior); + outBitStream->Write1(); // is input locked? } outBitStream->Write0(); @@ -102,8 +135,7 @@ void VehiclePhysicsComponent::Serialize(RakNet::BitStream* outBitStream, bool bI void VehiclePhysicsComponent::Update(float deltaTime) { if (m_SoftUpdate > 5) { - EntityManager::Instance()->SerializeEntity(m_Parent); - + Game::entityManager->SerializeEntity(m_Parent); m_SoftUpdate = 0; } else { m_SoftUpdate += deltaTime; diff --git a/dGame/dComponents/VehiclePhysicsComponent.h b/dGame/dComponents/VehiclePhysicsComponent.h index f5ab1917..e314bef1 100644 --- a/dGame/dComponents/VehiclePhysicsComponent.h +++ b/dGame/dComponents/VehiclePhysicsComponent.h @@ -3,13 +3,32 @@ #include "BitStream.h" #include "Entity.h" #include "Component.h" +#include "eReplicaComponentType.h" + +struct RemoteInputInfo { + void operator=(const RemoteInputInfo& other) { + m_RemoteInputX = other.m_RemoteInputX; + m_RemoteInputY = other.m_RemoteInputY; + m_IsPowersliding = other.m_IsPowersliding; + m_IsModified = other.m_IsModified; + } + + bool operator==(const RemoteInputInfo& other) { + return m_RemoteInputX == other.m_RemoteInputX && m_RemoteInputY == other.m_RemoteInputY && m_IsPowersliding == other.m_IsPowersliding && m_IsModified == other.m_IsModified; + } + + float m_RemoteInputX; + float m_RemoteInputY; + bool m_IsPowersliding; + bool m_IsModified; +}; /** * Physics component for vehicles. */ class VehiclePhysicsComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_VEHICLE_PHYSICS; + static const eReplicaComponentType ComponentType = eReplicaComponentType::VEHICLE_PHYSICS; VehiclePhysicsComponent(Entity* parentEntity); ~VehiclePhysicsComponent() override; @@ -93,6 +112,7 @@ public: void SetDirtyPosition(bool val); void SetDirtyVelocity(bool val); void SetDirtyAngularVelocity(bool val); + void SetRemoteInputInfo(const RemoteInputInfo&); private: bool m_DirtyPosition; @@ -108,4 +128,7 @@ private: bool m_IsOnRail; float m_SoftUpdate = 0; + uint32_t m_EndBehavior; + RemoteInputInfo m_RemoteInputInfo; + bool m_DirtyRemoteInput; }; diff --git a/dGame/dComponents/VendorComponent.cpp b/dGame/dComponents/VendorComponent.cpp index 96dfb2c0..245ea9b1 100644 --- a/dGame/dComponents/VendorComponent.cpp +++ b/dGame/dComponents/VendorComponent.cpp @@ -1,21 +1,28 @@ #include "VendorComponent.h" - -#include - -#include "Game.h" +#include "BitStream.h" #include "dServer.h" +#include "dZoneManager.h" +#include "WorldConfig.h" +#include "CDComponentsRegistryTable.h" +#include "CDVendorComponentTable.h" +#include "CDLootMatrixTable.h" +#include "CDLootTableTable.h" +#include "CDItemComponentTable.h" VendorComponent::VendorComponent(Entity* parent) : Component(parent) { + m_HasStandardCostItems = false; + m_HasMultiCostItems = false; SetupConstants(); RefreshInventory(true); } -VendorComponent::~VendorComponent() = default; - void VendorComponent::Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags) { - outBitStream->Write1(); - outBitStream->Write1(); // Has standard items (Required for vendors with missions.) - outBitStream->Write(HasCraftingStation()); // Has multi use items + outBitStream->Write(bIsInitialUpdate || m_DirtyVendor); + if (bIsInitialUpdate || m_DirtyVendor) { + outBitStream->Write(m_HasStandardCostItems); + outBitStream->Write(m_HasMultiCostItems); + if (!bIsInitialUpdate) m_DirtyVendor = false; + } } void VendorComponent::OnUse(Entity* originator) { @@ -23,109 +30,125 @@ void VendorComponent::OnUse(Entity* originator) { GameMessages::SendVendorStatusUpdate(m_Parent, originator->GetSystemAddress()); } -float VendorComponent::GetBuyScalar() const { - return m_BuyScalar; -} - -float VendorComponent::GetSellScalar() const { - return m_SellScalar; -} - -void VendorComponent::SetBuyScalar(float value) { - m_BuyScalar = value; -} - -void VendorComponent::SetSellScalar(float value) { - m_SellScalar = value; -} - -std::map& VendorComponent::GetInventory() { - return m_Inventory; -} - -bool VendorComponent::HasCraftingStation() { - // As far as we know, only Umami has a crafting station - return m_Parent->GetLOT() == 13800; -} - void VendorComponent::RefreshInventory(bool isCreation) { - //Custom code for Max vanity NPC - if (m_Parent->GetLOT() == 9749 && Game::server->GetZoneID() == 1201) { - if (!isCreation) return; - m_Inventory.insert({ 11909, 0 }); //Top hat w frog - m_Inventory.insert({ 7785, 0 }); //Flash bulb - m_Inventory.insert({ 12764, 0 }); //Big fountain soda - m_Inventory.insert({ 12241, 0 }); //Hot cocoa (from fb) + SetHasStandardCostItems(false); + SetHasMultiCostItems(false); + m_Inventory.clear(); + + // Custom code for Max vanity NPC and Mr.Ree cameras + if(isCreation && m_Parent->GetLOT() == 9749 && Game::server->GetZoneID() == 1201) { + SetupMaxCustomVendor(); return; } - m_Inventory.clear(); - auto* lootMatrixTable = CDClientManager::Instance()->GetTable("LootMatrix"); - std::vector lootMatrices = lootMatrixTable->Query([=](CDLootMatrix entry) { return (entry.LootMatrixIndex == m_LootMatrixID); }); + + auto* lootMatrixTable = CDClientManager::Instance().GetTable(); + const auto lootMatrices = lootMatrixTable->Query([=](CDLootMatrix entry) { return (entry.LootMatrixIndex == m_LootMatrixID); }); if (lootMatrices.empty()) return; - // Done with lootMatrix table - auto* lootTableTable = CDClientManager::Instance()->GetTable("LootTable"); + auto* lootTableTable = CDClientManager::Instance().GetTable(); + auto* itemComponentTable = CDClientManager::Instance().GetTable(); + auto* compRegistryTable = CDClientManager::Instance().GetTable(); for (const auto& lootMatrix : lootMatrices) { int lootTableID = lootMatrix.LootTableIndex; - std::vector vendorItems = lootTableTable->Query([=](CDLootTable entry) { return (entry.LootTableIndex == lootTableID); }); + auto vendorItems = lootTableTable->Query([=](CDLootTable entry) { return (entry.LootTableIndex == lootTableID); }); if (lootMatrix.maxToDrop == 0 || lootMatrix.minToDrop == 0) { - for (CDLootTable item : vendorItems) { - m_Inventory.insert({ item.itemid, item.sortPriority }); + for (const auto& item : vendorItems) { + if (!m_HasStandardCostItems || !m_HasMultiCostItems) { + auto itemComponentID = compRegistryTable->GetByIDAndType(item.itemid, eReplicaComponentType::ITEM, -1); + if (itemComponentID == -1) { + Game::logger->Log("VendorComponent", "Attempted to add item %i with ItemComponent ID -1 to vendor %i inventory. Not adding item!", itemComponentID, m_Parent->GetLOT()); + continue; + } + auto itemComponent = itemComponentTable->GetItemComponentByID(itemComponentID); + if (!m_HasStandardCostItems && itemComponent.baseValue != -1) SetHasStandardCostItems(true); + if (!m_HasMultiCostItems && !itemComponent.currencyCosts.empty()) SetHasMultiCostItems(true); + } + m_Inventory.push_back(SoldItem(item.itemid, item.sortPriority)); } } else { auto randomCount = GeneralUtils::GenerateRandomNumber(lootMatrix.minToDrop, lootMatrix.maxToDrop); for (size_t i = 0; i < randomCount; i++) { if (vendorItems.empty()) break; - auto randomItemIndex = GeneralUtils::GenerateRandomNumber(0, vendorItems.size() - 1); - - const auto& randomItem = vendorItems[randomItemIndex]; - + const auto& randomItem = vendorItems.at(randomItemIndex); vendorItems.erase(vendorItems.begin() + randomItemIndex); - - m_Inventory.insert({ randomItem.itemid, randomItem.sortPriority }); + if (!m_HasStandardCostItems || !m_HasMultiCostItems) { + auto itemComponentID = compRegistryTable->GetByIDAndType(randomItem.itemid, eReplicaComponentType::ITEM, -1); + if (itemComponentID == -1) { + Game::logger->Log("VendorComponent", "Attempted to add item %i with ItemComponent ID -1 to vendor %i inventory. Not adding item!", itemComponentID, m_Parent->GetLOT()); + continue; + } + auto itemComponent = itemComponentTable->GetItemComponentByID(itemComponentID); + if (!m_HasStandardCostItems && itemComponent.baseValue != -1) SetHasStandardCostItems(true); + if (!m_HasMultiCostItems && !itemComponent.currencyCosts.empty()) SetHasMultiCostItems(true); + } + m_Inventory.push_back(SoldItem(randomItem.itemid, randomItem.sortPriority)); } } } - - //Because I want a vendor to sell these cameras - if (m_Parent->GetLOT() == 13569) { - auto randomCamera = GeneralUtils::GenerateRandomNumber(0, 2); - - switch (randomCamera) { - case 0: - m_Inventory.insert({ 16253, 0 }); //Grungagroid - break; - case 1: - m_Inventory.insert({ 16254, 0 }); //Hipstabrick - break; - case 2: - m_Inventory.insert({ 16204, 0 }); //Megabrixel snapshot - break; - default: - break; - } - } + HandleMrReeCameras(); // Callback timer to refresh this inventory. - m_Parent->AddCallbackTimer(m_RefreshTimeSeconds, [this]() { - RefreshInventory(); + if (m_RefreshTimeSeconds > 0.0) { + m_Parent->AddCallbackTimer(m_RefreshTimeSeconds, [this]() { + RefreshInventory(); }); + } + Game::entityManager->SerializeEntity(m_Parent); GameMessages::SendVendorStatusUpdate(m_Parent, UNASSIGNED_SYSTEM_ADDRESS); } void VendorComponent::SetupConstants() { - auto* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - int componentID = compRegistryTable->GetByIDAndType(m_Parent->GetLOT(), COMPONENT_TYPE_VENDOR); + auto* compRegistryTable = CDClientManager::Instance().GetTable(); + int componentID = compRegistryTable->GetByIDAndType(m_Parent->GetLOT(), eReplicaComponentType::VENDOR); - auto* vendorComponentTable = CDClientManager::Instance()->GetTable("VendorComponent"); + auto* vendorComponentTable = CDClientManager::Instance().GetTable(); std::vector vendorComps = vendorComponentTable->Query([=](CDVendorComponent entry) { return (entry.id == componentID); }); if (vendorComps.empty()) return; - m_BuyScalar = vendorComps[0].buyScalar; - m_SellScalar = vendorComps[0].sellScalar; - m_RefreshTimeSeconds = vendorComps[0].refreshTimeSeconds; - m_LootMatrixID = vendorComps[0].LootMatrixIndex; + auto vendorData = vendorComps.at(0); + if (vendorData.buyScalar == 0.0) m_BuyScalar = Game::zoneManager->GetWorldConfig()->vendorBuyMultiplier; + else m_BuyScalar = vendorData.buyScalar; + m_SellScalar = vendorData.sellScalar; + m_RefreshTimeSeconds = vendorData.refreshTimeSeconds; + m_LootMatrixID = vendorData.LootMatrixIndex; +} + +bool VendorComponent::SellsItem(const LOT item) const { + return std::count_if(m_Inventory.begin(), m_Inventory.end(), [item](const SoldItem& lhs) { + return lhs.lot == item; + }) > 0; +} + + +void VendorComponent::SetupMaxCustomVendor(){ + SetHasStandardCostItems(true); + m_Inventory.push_back(SoldItem(11909, 0)); // Top hat w frog + m_Inventory.push_back(SoldItem(7785, 0)); // Flash bulb + m_Inventory.push_back(SoldItem(12764, 0)); // Big fountain soda + m_Inventory.push_back(SoldItem(12241, 0)); // Hot cocoa (from fb) +} + +void VendorComponent::HandleMrReeCameras(){ + if (m_Parent->GetLOT() == 13569) { + SetHasStandardCostItems(true); + auto randomCamera = GeneralUtils::GenerateRandomNumber(0, 2); + + LOT camera = 0; + DluAssert(randomCamera >= 0 && randomCamera <= 2); + switch (randomCamera) { + case 0: + camera = 16253; // Grungagroid + break; + case 1: + camera = 16254; // Hipstabrick + break; + case 2: + camera = 16204; // Megabrixel snapshot + break; + } + m_Inventory.push_back(SoldItem(camera, 0)); + } } diff --git a/dGame/dComponents/VendorComponent.h b/dGame/dComponents/VendorComponent.h index 72a5816d..4a9b582e 100644 --- a/dGame/dComponents/VendorComponent.h +++ b/dGame/dComponents/VendorComponent.h @@ -7,90 +7,56 @@ #include "Entity.h" #include "GameMessages.h" #include "RakNetTypes.h" +#include "eReplicaComponentType.h" + +struct SoldItem { + SoldItem(const LOT lot, const int32_t sortPriority) { + this->lot = lot; + this->sortPriority = sortPriority; + }; + LOT lot = 0; + int32_t sortPriority = 0; +}; -/** - * A component for vendor NPCs. A vendor sells items to the player. - */ class VendorComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_VENDOR; - + inline static const eReplicaComponentType ComponentType = eReplicaComponentType::VENDOR; VendorComponent(Entity* parent); - ~VendorComponent() override; - void Serialize(RakNet::BitStream* outBitStream, bool bIsInitialUpdate, unsigned int& flags); - void OnUse(Entity* originator) override; - - /** - * Gets the buy scaler - * @return the buy scaler - */ - float GetBuyScalar() const; - - /** - * Sets the buy scalar. - * @param value the new value. - */ - void SetBuyScalar(float value); - - /** - * Gets the buy scaler - * @return the buy scaler - */ - float GetSellScalar() const; - - /** - * Sets the sell scalar. - * @param value the new value. - */ - void SetSellScalar(float value); - - /** - * True if the NPC LOT is 13800, the only NPC with a crafting station. - */ - bool HasCraftingStation(); - - /** - * Gets the list if items the vendor sells. - * @return the list of items. - */ - std::map& GetInventory(); - - /** - * Refresh the inventory of this vendor. - */ void RefreshInventory(bool isCreation = false); - - /** - * Called on startup of vendor to setup the variables for the component. - */ void SetupConstants(); + bool SellsItem(const LOT item) const; + float GetBuyScalar() const { return m_BuyScalar; } + float GetSellScalar() const { return m_SellScalar; } + void SetBuyScalar(const float value) { m_BuyScalar = value; } + void SetSellScalar(const float value) { m_SellScalar = value; } + const std::vector& GetInventory() { return m_Inventory; } + + void SetHasMultiCostItems(const bool hasMultiCostItems) { + if (m_HasMultiCostItems == hasMultiCostItems) return; + m_HasMultiCostItems = hasMultiCostItems; + m_DirtyVendor = true; + } + + void SetHasStandardCostItems(const bool hasStandardCostItems) { + if (m_HasStandardCostItems == hasStandardCostItems) return; + m_HasStandardCostItems = hasStandardCostItems; + m_DirtyVendor = true; + } + + private: - /** - * The buy scalar. - */ - float m_BuyScalar; - - /** - * The sell scalar. - */ - float m_SellScalar; - - /** - * The refresh time of this vendors' inventory. - */ - float m_RefreshTimeSeconds; - - /** - * Loot matrix id of this vendor. - */ - uint32_t m_LootMatrixID; - - /** - * The list of items the vendor sells. - */ - std::map m_Inventory; + void SetupMaxCustomVendor(); + void HandleMrReeCameras(); + float m_BuyScalar = 0.0f; + float m_SellScalar = 0.0f; + float m_RefreshTimeSeconds = 0.0f; + uint32_t m_LootMatrixID = 0; + std::vector m_Inventory; + bool m_DirtyVendor = false; + bool m_HasStandardCostItems = false; + bool m_HasMultiCostItems = false; }; #endif // VENDORCOMPONENT_H diff --git a/dGame/dGameMessages/DoClientProjectileImpact.h b/dGame/dGameMessages/DoClientProjectileImpact.h new file mode 100644 index 00000000..6b381aa5 --- /dev/null +++ b/dGame/dGameMessages/DoClientProjectileImpact.h @@ -0,0 +1,80 @@ +#ifndef __DOCLIENTPROJECTILEIMPACT__H__ +#define __DOCLIENTPROJECTILEIMPACT__H__ + +#include "dCommonVars.h" + +/* Tell a client local projectile to impact */ +class DoClientProjectileImpact { +public: + DoClientProjectileImpact() { + i64OrgID = LWOOBJID_EMPTY; + i64OwnerID = LWOOBJID_EMPTY; + i64TargetID = LWOOBJID_EMPTY; + } + + DoClientProjectileImpact(std::string _sBitStream, LWOOBJID _i64OrgID = LWOOBJID_EMPTY, LWOOBJID _i64OwnerID = LWOOBJID_EMPTY, LWOOBJID _i64TargetID = LWOOBJID_EMPTY) { + i64OrgID = _i64OrgID; + i64OwnerID = _i64OwnerID; + i64TargetID = _i64TargetID; + sBitStream = _sBitStream; + } + + DoClientProjectileImpact(RakNet::BitStream* stream) : DoClientProjectileImpact() { + Deserialize(stream); + } + + ~DoClientProjectileImpact() { + } + + void Serialize(RakNet::BitStream* stream) { + stream->Write(eGameMessageType::DO_CLIENT_PROJECTILE_IMPACT); + + stream->Write(i64OrgID != LWOOBJID_EMPTY); + if (i64OrgID != LWOOBJID_EMPTY) stream->Write(i64OrgID); + + stream->Write(i64OwnerID != LWOOBJID_EMPTY); + if (i64OwnerID != LWOOBJID_EMPTY) stream->Write(i64OwnerID); + + stream->Write(i64TargetID != LWOOBJID_EMPTY); + if (i64TargetID != LWOOBJID_EMPTY) stream->Write(i64TargetID); + + uint32_t sBitStreamLength = sBitStream.length(); + stream->Write(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + stream->Write(sBitStream[k]); + } + + } + + bool Deserialize(RakNet::BitStream* stream) { + bool i64OrgIDIsDefault{}; + stream->Read(i64OrgIDIsDefault); + if (i64OrgIDIsDefault != 0) stream->Read(i64OrgID); + + bool i64OwnerIDIsDefault{}; + stream->Read(i64OwnerIDIsDefault); + if (i64OwnerIDIsDefault != 0) stream->Read(i64OwnerID); + + bool i64TargetIDIsDefault{}; + stream->Read(i64TargetIDIsDefault); + if (i64TargetIDIsDefault != 0) stream->Read(i64TargetID); + + uint32_t sBitStreamLength{}; + stream->Read(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + unsigned char character; + stream->Read(character); + sBitStream.push_back(character); + } + + + return true; + } + + LWOOBJID i64OrgID; + LWOOBJID i64OwnerID; + LWOOBJID i64TargetID; + std::string sBitStream; +}; + +#endif //!__DOCLIENTPROJECTILEIMPACT__H__ diff --git a/dGame/dGameMessages/EchoStartSkill.h b/dGame/dGameMessages/EchoStartSkill.h new file mode 100644 index 00000000..f5dee816 --- /dev/null +++ b/dGame/dGameMessages/EchoStartSkill.h @@ -0,0 +1,130 @@ +#ifndef __ECHOSTARTSKILL__H__ +#define __ECHOSTARTSKILL__H__ + +#include "dCommonVars.h" +#include "NiPoint3.h" +#include "NiQuaternion.h" +#include "eGameMessageType.h" + +/* Same as start skill but with different network options. An echo down to other clients that need to play the skill. */ +class EchoStartSkill { +public: + EchoStartSkill() { + bUsedMouse = false; + fCasterLatency = 0.0f; + iCastType = 0; + lastClickedPosit = NiPoint3::ZERO; + optionalTargetID = LWOOBJID_EMPTY; + originatorRot = NiQuaternion::IDENTITY; + uiSkillHandle = 0; + } + + EchoStartSkill(LWOOBJID _optionalOriginatorID, std::string _sBitStream, TSkillID _skillID, bool _bUsedMouse = false, float _fCasterLatency = 0.0f, int32_t _iCastType = 0, NiPoint3 _lastClickedPosit = NiPoint3::ZERO, LWOOBJID _optionalTargetID = LWOOBJID_EMPTY, NiQuaternion _originatorRot = NiQuaternion::IDENTITY, uint32_t _uiSkillHandle = 0) { + bUsedMouse = _bUsedMouse; + fCasterLatency = _fCasterLatency; + iCastType = _iCastType; + lastClickedPosit = _lastClickedPosit; + optionalOriginatorID = _optionalOriginatorID; + optionalTargetID = _optionalTargetID; + originatorRot = _originatorRot; + sBitStream = _sBitStream; + skillID = _skillID; + uiSkillHandle = _uiSkillHandle; + } + + EchoStartSkill(RakNet::BitStream* stream) : EchoStartSkill() { + Deserialize(stream); + } + + ~EchoStartSkill() { + } + + void Serialize(RakNet::BitStream* stream) { + stream->Write(eGameMessageType::ECHO_START_SKILL); + + stream->Write(bUsedMouse); + + stream->Write(fCasterLatency != 0.0f); + if (fCasterLatency != 0.0f) stream->Write(fCasterLatency); + + stream->Write(iCastType != 0); + if (iCastType != 0) stream->Write(iCastType); + + stream->Write(lastClickedPosit != NiPoint3::ZERO); + if (lastClickedPosit != NiPoint3::ZERO) stream->Write(lastClickedPosit); + + stream->Write(optionalOriginatorID); + + stream->Write(optionalTargetID != LWOOBJID_EMPTY); + if (optionalTargetID != LWOOBJID_EMPTY) stream->Write(optionalTargetID); + + stream->Write(originatorRot != NiQuaternion::IDENTITY); + if (originatorRot != NiQuaternion::IDENTITY) stream->Write(originatorRot); + + uint32_t sBitStreamLength = sBitStream.length(); + stream->Write(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + stream->Write(sBitStream[k]); + } + + stream->Write(skillID); + + stream->Write(uiSkillHandle != 0); + if (uiSkillHandle != 0) stream->Write(uiSkillHandle); + } + + bool Deserialize(RakNet::BitStream* stream) { + stream->Read(bUsedMouse); + + bool fCasterLatencyIsDefault{}; + stream->Read(fCasterLatencyIsDefault); + if (fCasterLatencyIsDefault != 0) stream->Read(fCasterLatency); + + bool iCastTypeIsDefault{}; + stream->Read(iCastTypeIsDefault); + if (iCastTypeIsDefault != 0) stream->Read(iCastType); + + bool lastClickedPositIsDefault{}; + stream->Read(lastClickedPositIsDefault); + if (lastClickedPositIsDefault != 0) stream->Read(lastClickedPosit); + + stream->Read(optionalOriginatorID); + + bool optionalTargetIDIsDefault{}; + stream->Read(optionalTargetIDIsDefault); + if (optionalTargetIDIsDefault != 0) stream->Read(optionalTargetID); + + bool originatorRotIsDefault{}; + stream->Read(originatorRotIsDefault); + if (originatorRotIsDefault != 0) stream->Read(originatorRot); + + uint32_t sBitStreamLength{}; + stream->Read(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + unsigned char character; + stream->Read(character); + sBitStream.push_back(character); + } + + stream->Read(skillID); + + bool uiSkillHandleIsDefault{}; + stream->Read(uiSkillHandleIsDefault); + if (uiSkillHandleIsDefault != 0) stream->Read(uiSkillHandle); + + return true; + } + + bool bUsedMouse; + float fCasterLatency; + int32_t iCastType; + NiPoint3 lastClickedPosit; + LWOOBJID optionalOriginatorID; + LWOOBJID optionalTargetID; + NiQuaternion originatorRot; + std::string sBitStream; + TSkillID skillID; + uint32_t uiSkillHandle; +}; + +#endif //!__ECHOSTARTSKILL__H__ diff --git a/dGame/dGameMessages/EchoSyncSkill.h b/dGame/dGameMessages/EchoSyncSkill.h new file mode 100644 index 00000000..ab5a3f2b --- /dev/null +++ b/dGame/dGameMessages/EchoSyncSkill.h @@ -0,0 +1,68 @@ +#ifndef __ECHOSYNCSKILL__H__ +#define __ECHOSYNCSKILL__H__ + +#include + +#include "BitStream.h" +#include "eGameMessageType.h" + + +/* Message to synchronize a skill cast */ +class EchoSyncSkill { +public: + EchoSyncSkill() { + bDone = false; + } + + EchoSyncSkill(std::string _sBitStream, uint32_t _uiBehaviorHandle, uint32_t _uiSkillHandle, bool _bDone = false) { + bDone = _bDone; + sBitStream = _sBitStream; + uiBehaviorHandle = _uiBehaviorHandle; + uiSkillHandle = _uiSkillHandle; + } + + EchoSyncSkill(RakNet::BitStream* stream) : EchoSyncSkill() { + Deserialize(stream); + } + + ~EchoSyncSkill() { + } + + void Serialize(RakNet::BitStream* stream) { + stream->Write(eGameMessageType::ECHO_SYNC_SKILL); + + stream->Write(bDone); + uint32_t sBitStreamLength = sBitStream.length(); + stream->Write(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + stream->Write(sBitStream[k]); + } + + stream->Write(uiBehaviorHandle); + stream->Write(uiSkillHandle); + } + + bool Deserialize(RakNet::BitStream* stream) { + stream->Read(bDone); + + uint32_t sBitStreamLength{}; + stream->Read(sBitStreamLength); + for (unsigned int k = 0; k < sBitStreamLength; k++) { + unsigned char character; + stream->Read(character); + sBitStream.push_back(character); + } + + stream->Read(uiBehaviorHandle); + stream->Read(uiSkillHandle); + + return true; + } + + bool bDone{}; + std::string sBitStream{}; + uint32_t uiBehaviorHandle{}; + uint32_t uiSkillHandle{}; +}; + +#endif //!__ECHOSYNCSKILL__H__ diff --git a/dGame/dGameMessages/GameMessageHandler.cpp b/dGame/dGameMessages/GameMessageHandler.cpp index 9cf09bbc..99592ed2 100644 --- a/dGame/dGameMessages/GameMessageHandler.cpp +++ b/dGame/dGameMessages/GameMessageHandler.cpp @@ -26,70 +26,83 @@ #include "CDSkillBehaviorTable.h" #include "SkillComponent.h" #include "RacingControlComponent.h" +#include "RequestServerProjectileImpact.h" +#include "SyncSkill.h" +#include "StartSkill.h" +#include "EchoStartSkill.h" +#include "EchoSyncSkill.h" +#include "eMissionTaskType.h" +#include "eReplicaComponentType.h" +#include "eConnectionType.h" using namespace std; -void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const SystemAddress& sysAddr, LWOOBJID objectID, GAME_MSG messageID) { +void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const SystemAddress& sysAddr, LWOOBJID objectID, eGameMessageType messageID) { CBITSTREAM; // Get the entity - Entity* entity = EntityManager::Instance()->GetEntity(objectID); + Entity* entity = Game::entityManager->GetEntity(objectID); User* usr = UserManager::Instance()->GetUser(sysAddr); if (!entity) { Game::logger->Log("GameMessageHandler", "Failed to find associated entity (%llu), aborting GM (%X)!", objectID, messageID); - return; } + if (messageID != eGameMessageType::READY_FOR_UPDATES) Game::logger->LogDebug("GameMessageHandler", "received game message ID: %i", messageID); + switch (messageID) { - case GAME_MSG_PLAY_EMOTE: { + case eGameMessageType::UN_USE_BBB_MODEL: { + GameMessages::HandleUnUseModel(inStream, entity, sysAddr); + break; + } + case eGameMessageType::PLAY_EMOTE: { GameMessages::HandlePlayEmote(inStream, entity); break; } - case GAME_MSG_MOVE_ITEM_IN_INVENTORY: { + case eGameMessageType::MOVE_ITEM_IN_INVENTORY: { GameMessages::HandleMoveItemInInventory(inStream, entity); break; } - case GAME_MSG_REMOVE_ITEM_FROM_INVENTORY: { + case eGameMessageType::REMOVE_ITEM_FROM_INVENTORY: { GameMessages::HandleRemoveItemFromInventory(inStream, entity, sysAddr); break; } - case GAME_MSG_EQUIP_ITEM: + case eGameMessageType::EQUIP_INVENTORY: GameMessages::HandleEquipItem(inStream, entity); break; - case GAME_MSG_UN_EQUIP_ITEM: + case eGameMessageType::UN_EQUIP_INVENTORY: GameMessages::HandleUnequipItem(inStream, entity); break; - case GAME_MSG_RESPOND_TO_MISSION: { + case eGameMessageType::RESPOND_TO_MISSION: { GameMessages::HandleRespondToMission(inStream, entity); break; } - case GAME_MSG_REQUEST_USE: { + case eGameMessageType::REQUEST_USE: { GameMessages::HandleRequestUse(inStream, entity, sysAddr); break; } - case GAME_MSG_SET_FLAG: { + case eGameMessageType::SET_FLAG: { GameMessages::HandleSetFlag(inStream, entity); break; } - case GAME_MSG_HAS_BEEN_COLLECTED: { + case eGameMessageType::HAS_BEEN_COLLECTED: { GameMessages::HandleHasBeenCollected(inStream, entity); break; } - case GAME_MSG_PLAYER_LOADED: { + case eGameMessageType::PLAYER_LOADED: { GameMessages::SendRestoreToPostLoadStats(entity, sysAddr); entity->SetPlayerReadyForUpdates(); @@ -110,9 +123,9 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System auto* destroyable = entity->GetComponent(); destroyable->SetImagination(destroyable->GetImagination()); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); - std::vector racingControllers = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_RACING_CONTROL); + std::vector racingControllers = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::RACING_CONTROL); for (Entity* racingController : racingControllers) { auto* racingComponent = racingController->GetComponent(); if (racingComponent != nullptr) { @@ -120,12 +133,12 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System } } - Entity* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); + Entity* zoneControl = Game::entityManager->GetZoneControlEntity(); for (CppScripts::Script* script : CppScripts::GetEntityScripts(zoneControl)) { script->OnPlayerLoaded(zoneControl, player); } - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPT); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPT); for (Entity* scriptEntity : scriptedActs) { if (scriptEntity->GetObjectID() != zoneControl->GetObjectID()) { // Don't want to trigger twice on instance worlds for (CppScripts::Script* script : CppScripts::GetEntityScripts(scriptEntity)) { @@ -157,97 +170,97 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System Game::logger->Log("GameMessageHandler", "Player %s (%llu) loaded.", entity->GetCharacter()->GetName().c_str(), entity->GetObjectID()); // After we've done our thing, tell the client they're ready - GameMessages::SendPlayerReady(dZoneManager::Instance()->GetZoneControlObject(), sysAddr); GameMessages::SendPlayerReady(entity, sysAddr); + GameMessages::SendPlayerReady(Game::zoneManager->GetZoneControlObject(), sysAddr); break; } - case GAME_MSG_REQUEST_LINKED_MISSION: { + case eGameMessageType::REQUEST_LINKED_MISSION: { GameMessages::HandleRequestLinkedMission(inStream, entity); break; } - case GAME_MSG_MISSION_DIALOGUE_OK: { + case eGameMessageType::MISSION_DIALOGUE_OK: { GameMessages::HandleMissionDialogOK(inStream, entity); break; } - case GAME_MSG_MISSION_DIALOGUE_CANCELLED: { + case eGameMessageType::MISSION_DIALOGUE_CANCELLED: { //This message is pointless for our implementation, as the client just carries on after //rejecting a mission offer. We dont need to do anything. This is just here to remove a warning in our logs :) break; } - case GAME_MSG_REQUEST_PLATFORM_RESYNC: { + case eGameMessageType::REQUEST_PLATFORM_RESYNC: { GameMessages::HandleRequestPlatformResync(inStream, entity, sysAddr); break; } - case GAME_MSG_FIRE_EVENT_SERVER_SIDE: { + case eGameMessageType::FIRE_EVENT_SERVER_SIDE: { GameMessages::HandleFireEventServerSide(inStream, entity, sysAddr); break; } - case GAME_MSG_SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA: { + case eGameMessageType::SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA: { GameMessages::HandleActivitySummaryLeaderboardData(inStream, entity, sysAddr); break; } - case GAME_MSG_REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA: { + case eGameMessageType::REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA: { GameMessages::HandleRequestActivitySummaryLeaderboardData(inStream, entity, sysAddr); break; } - case GAME_MSG_ACTIVITY_STATE_CHANGE_REQUEST: { + case eGameMessageType::ACTIVITY_STATE_CHANGE_REQUEST: { GameMessages::HandleActivityStateChangeRequest(inStream, entity); break; } - case GAME_MSG_PARSE_CHAT_MESSAGE: { + case eGameMessageType::PARSE_CHAT_MESSAGE: { GameMessages::HandleParseChatMessage(inStream, entity, sysAddr); break; } - case GAME_MSG_NOTIFY_SERVER_LEVEL_PROCESSING_COMPLETE: { + case eGameMessageType::NOTIFY_SERVER_LEVEL_PROCESSING_COMPLETE: { GameMessages::HandleNotifyServerLevelProcessingComplete(inStream, entity); break; } - case GAME_MSG_PICKUP_CURRENCY: { + case eGameMessageType::PICKUP_CURRENCY: { GameMessages::HandlePickupCurrency(inStream, entity); break; } - case GAME_MSG_PICKUP_ITEM: { + case eGameMessageType::PICKUP_ITEM: { GameMessages::HandlePickupItem(inStream, entity); break; } - case GAME_MSG_RESURRECT: { + case eGameMessageType::RESURRECT: { GameMessages::HandleResurrect(inStream, entity); break; } - case GAME_MSG_REQUEST_RESURRECT: { + case eGameMessageType::REQUEST_RESURRECT: { GameMessages::SendResurrect(entity); - /*auto* dest = static_cast(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + /*auto* dest = static_cast(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (dest) { dest->SetHealth(4); dest->SetArmor(0); dest->SetImagination(6); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); }*/ break; } - case GAME_MSG_HANDLE_HOT_PROPERTY_DATA: { + case eGameMessageType::GET_HOT_PROPERTY_DATA: { GameMessages::HandleGetHotPropertyData(inStream, entity, sysAddr); break; } - case GAME_MSG_REQUEST_SERVER_PROJECTILE_IMPACT: + case eGameMessageType::REQUEST_SERVER_PROJECTILE_IMPACT: { - auto message = GameMessages::RequestServerProjectileImpact(); + auto message = RequestServerProjectileImpact(); message.Deserialize(inStream); @@ -264,18 +277,18 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System break; } - case GAME_MSG_START_SKILL: { - GameMessages::StartSkill startSkill = GameMessages::StartSkill(); + case eGameMessageType::START_SKILL: { + StartSkill startSkill = StartSkill(); startSkill.Deserialize(inStream); // inStream replaces &bitStream if (startSkill.skillID == 1561 || startSkill.skillID == 1562 || startSkill.skillID == 1541) return; MissionComponent* comp = entity->GetComponent(); if (comp) { - comp->Progress(MissionTaskType::MISSION_TASK_TYPE_SKILL, startSkill.skillID); + comp->Progress(eMissionTaskType::USE_SKILL, startSkill.skillID); } - CDSkillBehaviorTable* skillTable = CDClientManager::Instance()->GetTable("SkillBehavior"); + CDSkillBehaviorTable* skillTable = CDClientManager::Instance().GetTable(); unsigned int behaviorId = skillTable->GetSkillByID(startSkill.skillID).behaviorID; bool success = false; @@ -302,10 +315,10 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System if (success) { //Broadcast our startSkill: RakNet::BitStream bitStreamLocal; - PacketUtils::WriteHeader(bitStreamLocal, CLIENT, MSG_CLIENT_GAME_MSG); + PacketUtils::WriteHeader(bitStreamLocal, eConnectionType::CLIENT, eClientMessageType::GAME_MSG); bitStreamLocal.Write(entity->GetObjectID()); - GameMessages::EchoStartSkill echoStartSkill; + EchoStartSkill echoStartSkill; echoStartSkill.bUsedMouse = startSkill.bUsedMouse; echoStartSkill.fCasterLatency = startSkill.fCasterLatency; echoStartSkill.iCastType = startSkill.iCastType; @@ -322,14 +335,14 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System } } break; - case GAME_MSG_SYNC_SKILL: { + case eGameMessageType::SYNC_SKILL: { RakNet::BitStream bitStreamLocal; - PacketUtils::WriteHeader(bitStreamLocal, CLIENT, MSG_CLIENT_GAME_MSG); + PacketUtils::WriteHeader(bitStreamLocal, eConnectionType::CLIENT, eClientMessageType::GAME_MSG); bitStreamLocal.Write(entity->GetObjectID()); - //bitStreamLocal.Write((unsigned short)GAME_MSG_ECHO_SYNC_SKILL); + //bitStreamLocal.Write((unsigned short)eGameMessageType::ECHO_SYNC_SKILL); //bitStreamLocal.Write(inStream); - GameMessages::SyncSkill sync = GameMessages::SyncSkill(inStream); // inStream replaced &bitStream + SyncSkill sync = SyncSkill(inStream); // inStream replaced &bitStream //sync.Serialize(&bitStreamLocal); ostringstream buffer; @@ -352,7 +365,7 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System delete bs; } - GameMessages::EchoSyncSkill echo = GameMessages::EchoSyncSkill(); + EchoSyncSkill echo = EchoSyncSkill(); echo.bDone = sync.bDone; echo.sBitStream = sync.sBitStream; echo.uiBehaviorHandle = sync.uiBehaviorHandle; @@ -363,302 +376,325 @@ void GameMessageHandler::HandleMessage(RakNet::BitStream* inStream, const System Game::server->Send(&bitStreamLocal, sysAddr, true); } break; - case GAME_MSG_REQUEST_SMASH_PLAYER: + case eGameMessageType::REQUEST_SMASH_PLAYER: entity->Smash(entity->GetObjectID()); break; - case GAME_MSG_MOVE_ITEM_BETWEEN_INVENTORY_TYPES: + case eGameMessageType::MOVE_ITEM_BETWEEN_INVENTORY_TYPES: GameMessages::HandleMoveItemBetweenInventoryTypes(inStream, entity, sysAddr); break; - case GAME_MSG_MODULAR_BUILD_FINISH: + case eGameMessageType::MODULAR_BUILD_FINISH: GameMessages::HandleModularBuildFinish(inStream, entity, sysAddr); break; - case GAME_MSG_PUSH_EQUIPPED_ITEMS_STATE: + case eGameMessageType::PUSH_EQUIPPED_ITEMS_STATE: GameMessages::HandlePushEquippedItemsState(inStream, entity); break; - case GAME_MSG_POP_EQUIPPED_ITEMS_STATE: + case eGameMessageType::POP_EQUIPPED_ITEMS_STATE: GameMessages::HandlePopEquippedItemsState(inStream, entity); break; - case GAME_MSG_BUY_FROM_VENDOR: + case eGameMessageType::BUY_FROM_VENDOR: GameMessages::HandleBuyFromVendor(inStream, entity, sysAddr); break; - case GAME_MSG_SELL_TO_VENDOR: + case eGameMessageType::SELL_TO_VENDOR: GameMessages::HandleSellToVendor(inStream, entity, sysAddr); break; - case GAME_MSG_BUYBACK_FROM_VENDOR: + case eGameMessageType::BUYBACK_FROM_VENDOR: GameMessages::HandleBuybackFromVendor(inStream, entity, sysAddr); break; - case GAME_MSG_MODULAR_BUILD_MOVE_AND_EQUIP: + case eGameMessageType::MODULAR_BUILD_MOVE_AND_EQUIP: GameMessages::HandleModularBuildMoveAndEquip(inStream, entity, sysAddr); break; - case GAME_MSG_DONE_ARRANGING_WITH_ITEM: + case eGameMessageType::DONE_ARRANGING_WITH_ITEM: GameMessages::HandleDoneArrangingWithItem(inStream, entity, sysAddr); break; - case GAME_MSG_MODULAR_BUILD_CONVERT_MODEL: + case eGameMessageType::MODULAR_BUILD_CONVERT_MODEL: GameMessages::HandleModularBuildConvertModel(inStream, entity, sysAddr); break; - case GAME_MSG_BUILD_MODE_SET: + case eGameMessageType::BUILD_MODE_SET: GameMessages::HandleBuildModeSet(inStream, entity); break; - case GAME_MSG_REBUILD_CANCEL: + case eGameMessageType::REBUILD_CANCEL: GameMessages::HandleRebuildCancel(inStream, entity); break; - case GAME_MSG_MATCH_REQUEST: + case eGameMessageType::MATCH_REQUEST: GameMessages::HandleMatchRequest(inStream, entity); break; - case GAME_MSG_USE_NON_EQUIPMENT_ITEM: + case eGameMessageType::USE_NON_EQUIPMENT_ITEM: GameMessages::HandleUseNonEquipmentItem(inStream, entity); break; - case GAME_MSG_CLIENT_ITEM_CONSUMED: + case eGameMessageType::CLIENT_ITEM_CONSUMED: GameMessages::HandleClientItemConsumed(inStream, entity); break; - case GAME_MSG_SET_CONSUMABLE_ITEM: + case eGameMessageType::SET_CONSUMABLE_ITEM: GameMessages::HandleSetConsumableItem(inStream, entity, sysAddr); break; - case GAME_MSG_VERIFY_ACK: + case eGameMessageType::VERIFY_ACK: GameMessages::HandleVerifyAck(inStream, entity, sysAddr); break; // Trading - case GAME_MSG_CLIENT_TRADE_REQUEST: + case eGameMessageType::CLIENT_TRADE_REQUEST: GameMessages::HandleClientTradeRequest(inStream, entity, sysAddr); break; - case GAME_MSG_CLIENT_TRADE_CANCEL: + case eGameMessageType::CLIENT_TRADE_CANCEL: GameMessages::HandleClientTradeCancel(inStream, entity, sysAddr); break; - case GAME_MSG_CLIENT_TRADE_ACCEPT: + case eGameMessageType::CLIENT_TRADE_ACCEPT: GameMessages::HandleClientTradeAccept(inStream, entity, sysAddr); break; - case GAME_MSG_CLIENT_TRADE_UPDATE: + case eGameMessageType::CLIENT_TRADE_UPDATE: GameMessages::HandleClientTradeUpdate(inStream, entity, sysAddr); break; // Pets - case GAME_MSG_PET_TAMING_TRY_BUILD: + case eGameMessageType::PET_TAMING_TRY_BUILD: GameMessages::HandlePetTamingTryBuild(inStream, entity, sysAddr); break; - case GAME_MSG_NOTIFY_TAMING_BUILD_SUCCESS: + case eGameMessageType::NOTIFY_TAMING_BUILD_SUCCESS: GameMessages::HandleNotifyTamingBuildSuccess(inStream, entity, sysAddr); break; - case GAME_MSG_REQUEST_SET_PET_NAME: + case eGameMessageType::REQUEST_SET_PET_NAME: GameMessages::HandleRequestSetPetName(inStream, entity, sysAddr); break; - case GAME_MSG_START_SERVER_PET_MINIGAME_TIMER: + case eGameMessageType::START_SERVER_PET_MINIGAME_TIMER: GameMessages::HandleStartServerPetMinigameTimer(inStream, entity, sysAddr); break; - case GAME_MSG_CLIENT_EXIT_TAMING_MINIGAME: + case eGameMessageType::CLIENT_EXIT_TAMING_MINIGAME: GameMessages::HandleClientExitTamingMinigame(inStream, entity, sysAddr); break; - case GAME_MSG_COMMAND_PET: + case eGameMessageType::COMMAND_PET: GameMessages::HandleCommandPet(inStream, entity, sysAddr); break; - case GAME_MSG_DESPAWN_PET: + case eGameMessageType::DESPAWN_PET: GameMessages::HandleDespawnPet(inStream, entity, sysAddr); break; - case GAME_MSG_MESSAGE_BOX_RESPOND: + case eGameMessageType::MESSAGE_BOX_RESPOND: GameMessages::HandleMessageBoxResponse(inStream, entity, sysAddr); break; - case GAME_MSG_CHOICE_BOX_RESPOND: + case eGameMessageType::CHOICE_BOX_RESPOND: GameMessages::HandleChoiceBoxRespond(inStream, entity, sysAddr); break; // Property - case GAME_MSG_QUERY_PROPERTY_DATA: + case eGameMessageType::QUERY_PROPERTY_DATA: GameMessages::HandleQueryPropertyData(inStream, entity, sysAddr); break; - case GAME_MSG_START_BUILDING_WITH_ITEM: + case eGameMessageType::START_BUILDING_WITH_ITEM: GameMessages::HandleStartBuildingWithItem(inStream, entity, sysAddr); break; - case GAME_MSG_SET_BUILD_MODE: + case eGameMessageType::SET_BUILD_MODE: GameMessages::HandleSetBuildMode(inStream, entity, sysAddr); break; - case GAME_MSG_PROPERTY_EDITOR_BEGIN: + case eGameMessageType::PROPERTY_EDITOR_BEGIN: GameMessages::HandlePropertyEditorBegin(inStream, entity, sysAddr); break; - case GAME_MSG_PROPERTY_EDITOR_END: + case eGameMessageType::PROPERTY_EDITOR_END: GameMessages::HandlePropertyEditorEnd(inStream, entity, sysAddr); break; - case GAME_MSG_PROPERTY_CONTENTS_FROM_CLIENT: + case eGameMessageType::PROPERTY_CONTENTS_FROM_CLIENT: GameMessages::HandlePropertyContentsFromClient(inStream, entity, sysAddr); break; - case GAME_MSG_ZONE_PROPERTY_MODEL_EQUIPPED: + case eGameMessageType::ZONE_PROPERTY_MODEL_EQUIPPED: GameMessages::HandlePropertyModelEquipped(inStream, entity, sysAddr); break; - case GAME_MSG_PLACE_PROPERTY_MODEL: + case eGameMessageType::PLACE_PROPERTY_MODEL: GameMessages::HandlePlacePropertyModel(inStream, entity, sysAddr); break; - case GAME_MSG_UPDATE_MODEL_FROM_CLIENT: + case eGameMessageType::UPDATE_MODEL_FROM_CLIENT: GameMessages::HandleUpdatePropertyModel(inStream, entity, sysAddr); break; - case GAME_MSG_DELETE_MODEL_FROM_CLIENT: + case eGameMessageType::DELETE_MODEL_FROM_CLIENT: GameMessages::HandleDeletePropertyModel(inStream, entity, sysAddr); break; - case GAME_MSG_BBB_LOAD_ITEM_REQUEST: + case eGameMessageType::BBB_LOAD_ITEM_REQUEST: GameMessages::HandleBBBLoadItemRequest(inStream, entity, sysAddr); break; - case GAME_MSG_BBB_SAVE_REQUEST: + case eGameMessageType::BBB_SAVE_REQUEST: GameMessages::HandleBBBSaveRequest(inStream, entity, sysAddr); break; - case GAME_MSG_CONTROL_BEHAVIOR: + case eGameMessageType::CONTROL_BEHAVIORS: GameMessages::HandleControlBehaviors(inStream, entity, sysAddr); break; - case GAME_MSG_PROPERTY_ENTRANCE_SYNC: + case eGameMessageType::PROPERTY_ENTRANCE_SYNC: GameMessages::HandlePropertyEntranceSync(inStream, entity, sysAddr); break; - case GAME_MSG_ENTER_PROPERTY1: + case eGameMessageType::ENTER_PROPERTY1: GameMessages::HandleEnterProperty(inStream, entity, sysAddr); break; - case GAME_MSG_ZONE_PROPERTY_MODEL_ROTATED: - EntityManager::Instance()->GetZoneControlEntity()->OnZonePropertyModelRotated(usr->GetLastUsedChar()->GetEntity()); + case eGameMessageType::ZONE_PROPERTY_MODEL_ROTATED: + Game::entityManager->GetZoneControlEntity()->OnZonePropertyModelRotated(usr->GetLastUsedChar()->GetEntity()); break; - case GAME_MSG_UPDATE_PROPERTY_OR_MODEL_FOR_FILTER_CHECK: + case eGameMessageType::UPDATE_PROPERTY_OR_MODEL_FOR_FILTER_CHECK: GameMessages::HandleUpdatePropertyOrModelForFilterCheck(inStream, entity, sysAddr); break; - case GAME_MSG_SET_PROPERTY_ACCESS: + case eGameMessageType::SET_PROPERTY_ACCESS: GameMessages::HandleSetPropertyAccess(inStream, entity, sysAddr); break; // Racing - case GAME_MSG_MODULE_ASSEMBLY_QUERY_DATA: + case eGameMessageType::MODULE_ASSEMBLY_QUERY_DATA: GameMessages::HandleModuleAssemblyQueryData(inStream, entity, sysAddr); break; - case GAME_MSG_ACKNOWLEDGE_POSSESSION: + case eGameMessageType::ACKNOWLEDGE_POSSESSION: GameMessages::HandleAcknowledgePossession(inStream, entity, sysAddr); break; - case GAME_MSG_VEHICLE_SET_WHEEL_LOCK_STATE: + case eGameMessageType::VEHICLE_SET_WHEEL_LOCK_STATE: GameMessages::HandleVehicleSetWheelLockState(inStream, entity, sysAddr); break; - case GAME_MSG_MODULAR_ASSEMBLY_NIF_COMPLETED: + case eGameMessageType::MODULAR_ASSEMBLY_NIF_COMPLETED: GameMessages::HandleModularAssemblyNIFCompleted(inStream, entity, sysAddr); break; - case GAME_MSG_RACING_CLIENT_READY: + case eGameMessageType::RACING_CLIENT_READY: GameMessages::HandleRacingClientReady(inStream, entity, sysAddr); break; - case GAME_MSG_REQUEST_DIE: + case eGameMessageType::REQUEST_DIE: GameMessages::HandleRequestDie(inStream, entity, sysAddr); break; - case GAME_MSG_VEHICLE_NOTIFY_SERVER_ADD_PASSIVE_BOOST_ACTION: + case eGameMessageType::NOTIFY_SERVER_VEHICLE_ADD_PASSIVE_BOOST_ACTION: GameMessages::HandleVehicleNotifyServerAddPassiveBoostAction(inStream, entity, sysAddr); break; - case GAME_MSG_VEHICLE_NOTIFY_SERVER_REMOVE_PASSIVE_BOOST_ACTION: + case eGameMessageType::NOTIFY_SERVER_VEHICLE_REMOVE_PASSIVE_BOOST_ACTION: GameMessages::HandleVehicleNotifyServerRemovePassiveBoostAction(inStream, entity, sysAddr); break; - case GAME_MSG_RACING_PLAYER_INFO_RESET_FINISHED: + case eGameMessageType::RACING_PLAYER_INFO_RESET_FINISHED: GameMessages::HandleRacingPlayerInfoResetFinished(inStream, entity, sysAddr); break; - case GAME_MSG_VEHICLE_NOTIFY_HIT_IMAGINATION_SERVER: + case eGameMessageType::VEHICLE_NOTIFY_HIT_IMAGINATION_SERVER: GameMessages::HandleVehicleNotifyHitImaginationServer(inStream, entity, sysAddr); break; - case GAME_MSG_UPDATE_PROPERTY_PERFORMANCE_COST: + case eGameMessageType::UPDATE_PROPERTY_PERFORMANCE_COST: GameMessages::HandleUpdatePropertyPerformanceCost(inStream, entity, sysAddr); break; // SG - case GAME_MSG_UPDATE_SHOOTING_GALLERY_ROTATION: + case eGameMessageType::UPDATE_SHOOTING_GALLERY_ROTATION: GameMessages::HandleUpdateShootingGalleryRotation(inStream, entity, sysAddr); break; // NT - case GAME_MSG_REQUEST_MOVE_ITEM_BETWEEN_INVENTORY_TYPES: + case eGameMessageType::REQUEST_MOVE_ITEM_BETWEEN_INVENTORY_TYPES: GameMessages::HandleRequestMoveItemBetweenInventoryTypes(inStream, entity, sysAddr); break; - case GAME_MSG_TOGGLE_GHOST_REFERENCE_OVERRIDE: + case eGameMessageType::TOGGLE_GHOST_REFERENCE_OVERRIDE: GameMessages::HandleToggleGhostReferenceOverride(inStream, entity, sysAddr); break; - case GAME_MSG_SET_GHOST_REFERENCE_POSITION: + case eGameMessageType::SET_GHOST_REFERENCE_POSITION: GameMessages::HandleSetGhostReferencePosition(inStream, entity, sysAddr); break; - case GAME_MSG_READY_FOR_UPDATES: + case eGameMessageType::READY_FOR_UPDATES: //We don't really care about this message, as it's simply here to inform us that the client is done loading an object. //In the event we _do_ send an update to an object that hasn't finished loading, the client will handle it anyway. break; - case GAME_MSG_REPORT_BUG: + case eGameMessageType::REPORT_BUG: GameMessages::HandleReportBug(inStream, entity); break; - case GAME_MSG_CLIENT_RAIL_MOVEMENT_READY: + case eGameMessageType::CLIENT_RAIL_MOVEMENT_READY: GameMessages::HandleClientRailMovementReady(inStream, entity, sysAddr); break; - case GAME_MSG_CANCEL_RAIL_MOVEMENT: + case eGameMessageType::CANCEL_RAIL_MOVEMENT: GameMessages::HandleCancelRailMovement(inStream, entity, sysAddr); break; - case GAME_MSG_PLAYER_RAIL_ARRIVED_NOTIFICATION: + case eGameMessageType::PLAYER_RAIL_ARRIVED_NOTIFICATION: GameMessages::HandlePlayerRailArrivedNotification(inStream, entity, sysAddr); break; - case GAME_MSG_CINEMATIC_UPDATE: + case eGameMessageType::CINEMATIC_UPDATE: GameMessages::HandleCinematicUpdate(inStream, entity, sysAddr); break; - case GAME_MSG_MODIFY_PLAYER_ZONE_STATISTIC: + case eGameMessageType::MODIFY_PLAYER_ZONE_STATISTIC: GameMessages::HandleModifyPlayerZoneStatistic(inStream, entity); break; - case GAME_MSG_UPDATE_PLAYER_STATISTIC: + case eGameMessageType::UPDATE_PLAYER_STATISTIC: GameMessages::HandleUpdatePlayerStatistic(inStream, entity); break; - case GAME_MSG_DISMOUNT_COMPLETE: + case eGameMessageType::DISMOUNT_COMPLETE: GameMessages::HandleDismountComplete(inStream, entity, sysAddr); break; - + case eGameMessageType::DECTIVATE_BUBBLE_BUFF: + GameMessages::HandleDeactivateBubbleBuff(inStream, entity); + break; + case eGameMessageType::ACTIVATE_BUBBLE_BUFF: + GameMessages::HandleActivateBubbleBuff(inStream, entity); + break; + case eGameMessageType::ZONE_SUMMARY_DISMISSED: + GameMessages::HandleZoneSummaryDismissed(inStream, entity); + break; + case eGameMessageType::REQUEST_ACTIVITY_EXIT: + GameMessages::HandleRequestActivityExit(inStream, entity); + break; + case eGameMessageType::ADD_DONATION_ITEM: + GameMessages::HandleAddDonationItem(inStream, entity, sysAddr); + break; + case eGameMessageType::REMOVE_DONATION_ITEM: + GameMessages::HandleRemoveDonationItem(inStream, entity, sysAddr); + break; + case eGameMessageType::CONFIRM_DONATION_ON_PLAYER: + GameMessages::HandleConfirmDonationOnPlayer(inStream, entity); + break; + case eGameMessageType::CANCEL_DONATION_ON_PLAYER: + GameMessages::HandleCancelDonationOnPlayer(inStream, entity); + break; default: - //Game::logger->Log("GameMessageHandler", "Unknown game message ID: %X", messageID); + Game::logger->LogDebug("GameMessageHandler", "Unknown game message ID: %i", messageID); break; } } diff --git a/dGame/dGameMessages/GameMessageHandler.h b/dGame/dGameMessages/GameMessageHandler.h index 063e97f6..8b6685cb 100644 --- a/dGame/dGameMessages/GameMessageHandler.h +++ b/dGame/dGameMessages/GameMessageHandler.h @@ -7,7 +7,6 @@ #define GAMEMESSAGEHANDLER_H #include "RakNetTypes.h" -#include "dMessageIdentifiers.h" #include "dCommonVars.h" #include #include @@ -21,8 +20,10 @@ #include "GameMessages.h" #include "../dDatabase/CDClientDatabase.h" +enum class eGameMessageType : uint16_t; + namespace GameMessageHandler { - void HandleMessage(RakNet::BitStream* inStream, const SystemAddress& sysAddr, LWOOBJID objectID, GAME_MSG messageID); + void HandleMessage(RakNet::BitStream* inStream, const SystemAddress& sysAddr, LWOOBJID objectID, eGameMessageType messageID); }; #endif // GAMEMESSAGEHANDLER_H diff --git a/dGame/dGameMessages/GameMessages.cpp b/dGame/dGameMessages/GameMessages.cpp index 772cb691..020e0abe 100644 --- a/dGame/dGameMessages/GameMessages.cpp +++ b/dGame/dGameMessages/GameMessages.cpp @@ -4,7 +4,6 @@ #include "PacketUtils.h" #include "BitStream.h" #include "Game.h" -#include "dMessageIdentifiers.h" #include "SlashCommandHandler.h" #include "NiPoint3.h" #include "NiQuaternion.h" @@ -25,9 +24,24 @@ #include "dConfig.h" #include "TeamManager.h" #include "ChatPackets.h" -#include "GameConfig.h" #include "RocketLaunchLupComponent.h" #include "eUnequippableActiveType.h" +#include "eMovementPlatformState.h" +#include "LeaderboardManager.h" +#include "Amf3.h" +#include "Loot.h" +#include "eRacingTaskParam.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" +#include "eObjectBits.h" +#include "eTriggerEventType.h" +#include "eMatchUpdate.h" +#include "eCyclingMode.h" +#include "eCinematicEvent.h" +#include "eQuickBuildFailReason.h" +#include "eControlScheme.h" +#include "eStateChangeType.h" +#include "eConnectionType.h" #include #include @@ -49,17 +63,20 @@ #include "ScriptComponent.h" #include "RebuildComponent.h" #include "VendorComponent.h" +#include "InventoryComponent.h" #include "RocketLaunchpadControlComponent.h" #include "PropertyEntranceComponent.h" #include "MovingPlatformComponent.h" #include "PetComponent.h" #include "ModuleAssemblyComponent.h" #include "VehiclePhysicsComponent.h" +#include "RenderComponent.h" #include "PossessableComponent.h" #include "PossessorComponent.h" #include "RacingControlComponent.h" #include "RailActivatorComponent.h" #include "LevelProgressionComponent.h" +#include "DonationVendorComponent.h" // Message includes: #include "dZoneManager.h" @@ -68,13 +85,25 @@ #include "PropertyVendorComponent.h" #include "PropertySelectQueryProperty.h" #include "TradingManager.h" +#include "ControlBehaviors.h" +#include "AMFDeserialize.h" +#include "eBlueprintSaveResponseType.h" +#include "eAninmationFlags.h" +#include "AmfSerialize.h" +#include "eReplicaComponentType.h" +#include "eClientMessageType.h" +#include "eGameMessageType.h" +#include "ActivityManager.h" + +#include "CDComponentsRegistryTable.h" +#include "CDObjectsTable.h" void GameMessages::SendFireEventClientSide(const LWOOBJID& objectID, const SystemAddress& sysAddr, std::u16string args, const LWOOBJID& object, int64_t param1, int param2, const LWOOBJID& sender) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_FIRE_EVENT_CLIENT_SIDE); + bitStream.Write(eGameMessageType::FIRE_EVENT_CLIENT_SIDE); //bitStream.Write(args); uint32_t argSize = args.size(); @@ -92,11 +121,11 @@ void GameMessages::SendFireEventClientSide(const LWOOBJID& objectID, const Syste SEND_PACKET; } -void GameMessages::SendTeleport(const LWOOBJID& objectID, const NiPoint3& pos, const NiQuaternion& rot, const SystemAddress& sysAddr, bool bSetRotation, bool noGravTeleport) { +void GameMessages::SendTeleport(const LWOOBJID& objectID, const NiPoint3& pos, const NiQuaternion& rot, const SystemAddress& sysAddr, bool bSetRotation) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_TELEPORT); + bitStream.Write(eGameMessageType::TELEPORT); bool bIgnoreY = (pos.y == 0.0f); bool bUseNavmesh = false; @@ -113,7 +142,6 @@ void GameMessages::SendTeleport(const LWOOBJID& objectID, const NiPoint3& pos, c bitStream.Write(pos.y); bitStream.Write(pos.z); bitStream.Write(bUseNavmesh); - bitStream.Write(noGravTeleport); bitStream.Write(rot.w != 1.0f); if (rot.w != 1.0f) bitStream.Write(rot.w); @@ -134,7 +162,6 @@ void GameMessages::SendPlayAnimation(Entity* entity, const std::u16string& anima //Stolen from the old DLU codebase as the new one's autogenerated code doesn't work properly for animationIDs longer than 6 characters. CBITSTREAM; CMSGHEADER; - uint16_t gameMsgID = GAME_MSG_PLAY_ANIMATION; std::string sAnimationID = GeneralUtils::UTF16ToWTF8(animationName); uint32_t animationIDLength = sAnimationID.size(); @@ -143,7 +170,7 @@ void GameMessages::SendPlayAnimation(Entity* entity, const std::u16string& anima bool bTriggerOnCompleteMsg = false; bitStream.Write(entity->GetObjectID()); - bitStream.Write(gameMsgID); + bitStream.Write(eGameMessageType::PLAY_ANIMATION); bitStream.Write(animationIDLength); PacketUtils::WriteWString(bitStream, animationName, animationIDLength); @@ -167,7 +194,7 @@ void GameMessages::SendPlayerReady(Entity* entity, const SystemAddress& sysAddr) CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_PLAYER_READY); + bitStream.Write(eGameMessageType::PLAYER_READY); SEND_PACKET; } @@ -176,7 +203,7 @@ void GameMessages::SendPlayerAllowedRespawn(LWOOBJID entityID, bool doNotPromptR CMSGHEADER; bitStream.Write(entityID); - bitStream.Write(GAME_MSG::GAME_MSG_SET_PLAYER_ALLOWED_RESPAWN); + bitStream.Write(eGameMessageType::SET_PLAYER_ALLOWED_RESPAWN); bitStream.Write(doNotPromptRespawn); SEND_PACKET; @@ -187,7 +214,7 @@ void GameMessages::SendInvalidZoneTransferList(Entity* entity, const SystemAddre CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_INVALID_ZONE_TRANSFER_LIST); + bitStream.Write(eGameMessageType::INVALID_ZONE_TRANSFER_LIST); uint32_t CustomerFeedbackURLLength = feedbackURL.size(); bitStream.Write(CustomerFeedbackURLLength); @@ -212,7 +239,7 @@ void GameMessages::SendKnockback(const LWOOBJID& objectID, const LWOOBJID& caste CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG_KNOCKBACK); + bitStream.Write(eGameMessageType::KNOCKBACK); bool casterFlag = caster != LWOOBJID_EMPTY; bool originatorFlag = originator != LWOOBJID_EMPTY; @@ -248,7 +275,7 @@ void GameMessages::SendStartArrangingWithItem( CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_START_ARRANGING_WITH_ITEM); + bitStream.Write(eGameMessageType::START_ARRANGING_WITH_ITEM); bitStream.Write(bFirstTime); bitStream.Write(buildAreaID != LWOOBJID_EMPTY); @@ -272,13 +299,13 @@ void GameMessages::SendPlayerSetCameraCyclingMode(const LWOOBJID& objectID, cons CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG_PLAYER_SET_CAMERA_CYCLING_MODE); + bitStream.Write(eGameMessageType::PLAYER_SET_CAMERA_CYCLING_MODE); bitStream.Write(bAllowCyclingWhileDeadOnly); - bitStream.Write(cyclingMode != ALLOW_CYCLE_TEAMMATES); - if (cyclingMode != ALLOW_CYCLE_TEAMMATES) { - bitStream.Write(cyclingMode); + bitStream.Write(cyclingMode != eCyclingMode::ALLOW_CYCLE_TEAMMATES); + if (cyclingMode != eCyclingMode::ALLOW_CYCLE_TEAMMATES) { + bitStream.Write(cyclingMode); } SEND_PACKET; @@ -289,7 +316,7 @@ void GameMessages::SendPlayNDAudioEmitter(Entity* entity, const SystemAddress& s CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_PLAY_ND_AUDIO_EMITTER); + bitStream.Write((uint16_t)eGameMessageType::PLAY_ND_AUDIO_EMITTER); bitStream.Write0(); bitStream.Write0(); @@ -318,14 +345,14 @@ void GameMessages::SendStartPathing(Entity* entity) { CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG_START_PATHING); + bitStream.Write(eGameMessageType::START_PATHING); SEND_PACKET_BROADCAST; } void GameMessages::SendPlatformResync(Entity* entity, const SystemAddress& sysAddr, bool bStopAtDesiredWaypoint, int iIndex, int iDesiredWaypointIndex, int nextIndex, - MovementPlatformState movementState) { + eMovementPlatformState movementState) { CBITSTREAM; CMSGHEADER; @@ -336,11 +363,11 @@ void GameMessages::SendPlatformResync(Entity* entity, const SystemAddress& sysAd iIndex = 0; nextIndex = 0; bStopAtDesiredWaypoint = true; - movementState = MovementPlatformState::Stationary; + movementState = eMovementPlatformState::Stationary; } bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_PLATFORM_RESYNC); + bitStream.Write((uint16_t)eGameMessageType::PLATFORM_RESYNC); bool bReverse = false; int eCommand = 0; @@ -381,7 +408,7 @@ void GameMessages::SendRestoreToPostLoadStats(Entity* entity, const SystemAddres CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_RESTORE_TO_POST_LOAD_STATS); + bitStream.Write(eGameMessageType::RESTORE_TO_POST_LOAD_STATS); SEND_PACKET; } @@ -389,24 +416,24 @@ void GameMessages::SendServerDoneLoadingAllObjects(Entity* entity, const SystemA CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_DONE_LOADING_ALL_OBJECTS); + bitStream.Write(eGameMessageType::SERVER_DONE_LOADING_ALL_OBJECTS); SEND_PACKET; } -void GameMessages::SendChatModeUpdate(const LWOOBJID& objectID, uint8_t level) { +void GameMessages::SendChatModeUpdate(const LWOOBJID& objectID, eGameMasterLevel level) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write((uint16_t)GAME_MSG_UPDATE_CHAT_MODE); + bitStream.Write((uint16_t)eGameMessageType::UPDATE_CHAT_MODE); bitStream.Write(level); SEND_PACKET_BROADCAST; } -void GameMessages::SendGMLevelBroadcast(const LWOOBJID& objectID, uint8_t level) { +void GameMessages::SendGMLevelBroadcast(const LWOOBJID& objectID, eGameMasterLevel level) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write((uint16_t)GAME_MSG_SET_GM_LEVEL); + bitStream.Write((uint16_t)eGameMessageType::SET_GM_LEVEL); bitStream.Write1(); bitStream.Write(level); SEND_PACKET_BROADCAST; @@ -417,13 +444,13 @@ void GameMessages::SendAddItemToInventoryClientSync(Entity* entity, const System CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(static_cast(GAME_MSG_ADD_ITEM_TO_INVENTORY_CLIENT_SYNC)); + bitStream.Write(eGameMessageType::ADD_ITEM_TO_INVENTORY_CLIENT_SYNC); bitStream.Write(item->GetBound()); bitStream.Write(item->GetInfo().isBOE); bitStream.Write(item->GetInfo().isBOP); - bitStream.Write(lootSourceType != eLootSourceType::LOOT_SOURCE_NONE); // Loot source - if (lootSourceType != eLootSourceType::LOOT_SOURCE_NONE) bitStream.Write(lootSourceType); + bitStream.Write(lootSourceType != eLootSourceType::NONE); // Loot source + if (lootSourceType != eLootSourceType::NONE) bitStream.Write(lootSourceType); LWONameValue extraInfo; auto config = item->GetConfig(); @@ -471,24 +498,24 @@ void GameMessages::SendAddItemToInventoryClientSync(Entity* entity, const System SEND_PACKET; } -void GameMessages::SendNotifyClientFlagChange(const LWOOBJID& objectID, int iFlagID, bool bFlag, const SystemAddress& sysAddr) { +void GameMessages::SendNotifyClientFlagChange(const LWOOBJID& objectID, uint32_t iFlagID, bool bFlag, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write((uint16_t)GAME_MSG_NOTIFY_CLIENT_FLAG_CHANGE); + bitStream.Write((uint16_t)eGameMessageType::NOTIFY_CLIENT_FLAG_CHANGE); bitStream.Write(bFlag); bitStream.Write(iFlagID); SEND_PACKET; } -void GameMessages::SendChangeObjectWorldState(const LWOOBJID& objectID, int state, const SystemAddress& sysAddr) { +void GameMessages::SendChangeObjectWorldState(const LWOOBJID& objectID, eObjectWorldState state, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write((uint16_t)GAME_MSG_CHANGE_OBJECT_WORLD_STATE); + bitStream.Write((uint16_t)eGameMessageType::CHANGE_OBJECT_WORLD_STATE); bitStream.Write(state); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST @@ -506,7 +533,7 @@ void GameMessages::SendOfferMission(const LWOOBJID& entity, const SystemAddress& CMSGHEADER; bitStream.Write(offererID); - bitStream.Write(uint16_t(GAME_MSG_OFFER_MISSION)); + bitStream.Write(eGameMessageType::OFFER_MISSION); bitStream.Write(missionID); bitStream.Write(offererID); @@ -517,7 +544,7 @@ void GameMessages::SendOfferMission(const LWOOBJID& entity, const SystemAddress& CMSGHEADER; bitStream.Write(entity); - bitStream.Write(uint16_t(GAME_MSG_OFFER_MISSION)); + bitStream.Write(eGameMessageType::OFFER_MISSION); bitStream.Write(missionID); bitStream.Write(offererID); @@ -530,7 +557,7 @@ void GameMessages::SendNotifyMission(Entity* entity, const SystemAddress& sysAdd CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_NOTIFY_MISSION)); + bitStream.Write(eGameMessageType::NOTIFY_MISSION); bitStream.Write(missionID); bitStream.Write(missionState); bitStream.Write(sendingRewards); @@ -543,7 +570,7 @@ void GameMessages::SendNotifyMissionTask(Entity* entity, const SystemAddress& sy CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_NOTIFY_MISSION_TASK); + bitStream.Write((uint16_t)eGameMessageType::NOTIFY_MISSION_TASK); bitStream.Write(missionID); bitStream.Write(taskMask); @@ -561,23 +588,23 @@ void GameMessages::SendModifyLEGOScore(Entity* entity, const SystemAddress& sysA CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_MODIFY_LEGO_SCORE); + bitStream.Write((uint16_t)eGameMessageType::MODIFY_LEGO_SCORE); bitStream.Write(score); - bitStream.Write(sourceType != eLootSourceType::LOOT_SOURCE_NONE); - if (sourceType != eLootSourceType::LOOT_SOURCE_NONE) bitStream.Write(sourceType); + bitStream.Write(sourceType != eLootSourceType::NONE); + if (sourceType != eLootSourceType::NONE) bitStream.Write(sourceType); SEND_PACKET; } -void GameMessages::SendUIMessageServerToSingleClient(Entity* entity, const SystemAddress& sysAddr, const std::string& message, NDGFxValue args) { +void GameMessages::SendUIMessageServerToSingleClient(Entity* entity, const SystemAddress& sysAddr, const std::string& message, AMFBaseValue& args) { CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_UI_MESSAGE_SERVER_TO_SINGLE_CLIENT); + bitStream.Write((uint16_t)eGameMessageType::UI_MESSAGE_SERVER_TO_SINGLE_CLIENT); - bitStream.Write(args); + bitStream.Write(args); uint32_t strMessageNameLength = message.size(); bitStream.Write(strMessageNameLength); @@ -588,15 +615,15 @@ void GameMessages::SendUIMessageServerToSingleClient(Entity* entity, const Syste SEND_PACKET; } -void GameMessages::SendUIMessageServerToAllClients(const std::string& message, NDGFxValue args) { +void GameMessages::SendUIMessageServerToAllClients(const std::string& message, AMFBaseValue& args) { CBITSTREAM; CMSGHEADER; LWOOBJID empty = 0; bitStream.Write(empty); - bitStream.Write((uint16_t)GAME_MSG_UI_MESSAGE_SERVER_TO_ALL_CLIENTS); + bitStream.Write((uint16_t)eGameMessageType::UI_MESSAGE_SERVER_TO_ALL_CLIENTS); - bitStream.Write(args); + bitStream.Write(args); uint32_t strMessageNameLength = message.size(); bitStream.Write(strMessageNameLength); @@ -612,7 +639,7 @@ void GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(Entity* entity, CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_PLAY_EMBEDDED_EFFECT_ON_ALL_CLIENTS_NEAR_OBJECT); + bitStream.Write((uint16_t)eGameMessageType::PLAY_EMBEDDED_EFFECT_ON_ALL_CLIENTS_NEAR_OBJECT); bitStream.Write(static_cast(effectName.length())); for (uint32_t k = 0; k < effectName.length(); k++) { @@ -633,7 +660,7 @@ void GameMessages::SendPlayFXEffect(const LWOOBJID& entity, int32_t effectID, co CMSGHEADER; bitStream.Write(entity); - bitStream.Write((uint16_t)GAME_MSG::GAME_MSG_PLAY_FX_EFFECT); + bitStream.Write((uint16_t)eGameMessageType::PLAY_FX_EFFECT); bitStream.Write(effectID != -1); if (effectID != -1) bitStream.Write(effectID); @@ -667,7 +694,7 @@ void GameMessages::SendStopFXEffect(Entity* entity, bool killImmediate, std::str CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_STOP_FX_EFFECT); + bitStream.Write(eGameMessageType::STOP_FX_EFFECT); bitStream.Write(killImmediate); bitStream.Write(name.size()); @@ -681,7 +708,7 @@ void GameMessages::SendBroadcastTextToChatbox(Entity* entity, const SystemAddres CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG::GAME_MSG_BROADCAST_TEXT_TO_CHATBOX); + bitStream.Write((uint16_t)eGameMessageType::BROADCAST_TEXT_TO_CHATBOX); LWONameValue attribs; attribs.name = attrs; @@ -707,7 +734,7 @@ void GameMessages::SendSetCurrency(Entity* entity, int64_t currency, int lootTyp CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_SET_CURRENCY)); + bitStream.Write(eGameMessageType::SET_CURRENCY); bitStream.Write(currency); @@ -725,19 +752,19 @@ void GameMessages::SendSetCurrency(Entity* entity, int64_t currency, int lootTyp bitStream.Write(sourceTradeID != LWOOBJID_EMPTY); if (sourceTradeID != LWOOBJID_EMPTY) bitStream.Write(sourceTradeID); - bitStream.Write(sourceType != LOOTTYPE_NONE); - if (sourceType != LOOTTYPE_NONE) bitStream.Write(sourceType); + bitStream.Write(sourceType != eLootSourceType::NONE); + if (sourceType != eLootSourceType::NONE) bitStream.Write(sourceType); SystemAddress sysAddr = entity->GetSystemAddress(); SEND_PACKET; } -void GameMessages::SendRebuildNotifyState(Entity* entity, int prevState, int state, const LWOOBJID& playerID) { +void GameMessages::SendRebuildNotifyState(Entity* entity, eRebuildState prevState, eRebuildState state, const LWOOBJID& playerID) { CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_REBUILD_NOTIFY_STATE); + bitStream.Write((uint16_t)eGameMessageType::REBUILD_NOTIFY_STATE); bitStream.Write(prevState); bitStream.Write(state); @@ -746,19 +773,19 @@ void GameMessages::SendRebuildNotifyState(Entity* entity, int prevState, int sta SEND_PACKET_BROADCAST; } -void GameMessages::SendEnableRebuild(Entity* entity, bool enable, bool fail, bool success, int failReason, float duration, const LWOOBJID& playerID) { +void GameMessages::SendEnableRebuild(Entity* entity, bool enable, bool fail, bool success, eQuickBuildFailReason failReason, float duration, const LWOOBJID& playerID) { CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_ENABLE_REBUILD); + bitStream.Write((uint16_t)eGameMessageType::ENABLE_REBUILD); bitStream.Write(enable); bitStream.Write(fail); bitStream.Write(success); - bitStream.Write(failReason != eFailReason::REASON_NOT_GIVEN); - if (failReason != eFailReason::REASON_NOT_GIVEN) bitStream.Write(failReason); + bitStream.Write(failReason != eQuickBuildFailReason::NOT_GIVEN); + if (failReason != eQuickBuildFailReason::NOT_GIVEN) bitStream.Write(failReason); bitStream.Write(duration); bitStream.Write(playerID); @@ -771,7 +798,7 @@ void GameMessages::SendTerminateInteraction(const LWOOBJID& objectID, eTerminate CMSGHEADER; bitStream.Write(objectID); - bitStream.Write((uint16_t)GAME_MSG_TERMINATE_INTERACTION); + bitStream.Write((uint16_t)eGameMessageType::TERMINATE_INTERACTION); bitStream.Write(terminator); bitStream.Write(type); @@ -784,7 +811,7 @@ void GameMessages::SendDieNoImplCode(Entity* entity, const LWOOBJID& killerID, c CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_DIE)); + bitStream.Write(eGameMessageType::DIE); bitStream.Write(bClientDeath); bitStream.Write(bSpawnLoot); bitStream.Write(deathType); @@ -807,7 +834,7 @@ void GameMessages::SendDie(Entity* entity, const LWOOBJID& killerID, const LWOOB bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_DIE); + bitStream.Write((uint16_t)eGameMessageType::DIE); bitStream.Write(bClientDeath); bitStream.Write(bSpawnLoot); @@ -825,8 +852,8 @@ void GameMessages::SendDie(Entity* entity, const LWOOBJID& killerID, const LWOOB bitStream.Write(directionRelative_AngleY); bitStream.Write(directionRelative_Force); - bitStream.Write(killType != VIOLENT); - if (killType != VIOLENT) bitStream.Write(killType); + bitStream.Write(killType != eKillType::VIOLENT); + if (killType != eKillType::VIOLENT) bitStream.Write(killType); bitStream.Write(killerID); @@ -843,7 +870,7 @@ void GameMessages::SendSetInventorySize(Entity* entity, int invType, int size) { CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_SET_INVENTORY_SIZE)); + bitStream.Write(eGameMessageType::SET_INVENTORY_SIZE); bitStream.Write(invType); bitStream.Write(size); @@ -856,7 +883,7 @@ void GameMessages::SendSetEmoteLockState(Entity* entity, bool bLock, int emoteID CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_SET_EMOTE_LOCK_STATE)); + bitStream.Write(eGameMessageType::SET_EMOTE_LOCK_STATE); bitStream.Write(bLock); bitStream.Write(emoteID); @@ -877,7 +904,7 @@ void GameMessages::SendSetJetPackMode(Entity* entity, bool use, bool bypassCheck CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_SET_JET_PACK_MODE)); + bitStream.Write(eGameMessageType::SET_JET_PACK_MODE); bitStream.Write(bypassChecks); bitStream.Write(doHover); @@ -902,17 +929,30 @@ void GameMessages::SendSetJetPackMode(Entity* entity, bool use, bool bypassCheck } void GameMessages::SendResurrect(Entity* entity) { - DestroyableComponent* dest = static_cast(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + // Restore the players health after the animation for respawning has finished. + // This is when the health appered back in live, not immediately upon requesting respawn + // Add a half second in case someone decides to cheat and move during the death animation + // and just make sure the client has time to be ready. + constexpr float respawnTime = 3.66700005531311f + 0.5f; + entity->AddCallbackTimer(respawnTime, [=]() { + auto* destroyableComponent = entity->GetComponent(); - if (dest != nullptr && entity->GetLOT() == 1) { - auto* levelComponent = entity->GetComponent(); - if (levelComponent) { - dest->SetHealth(levelComponent->GetLevel() >= 45 ? 8 : 4); - dest->SetImagination(levelComponent->GetLevel() >= 45 ? 20 : 6); + if (destroyableComponent != nullptr && entity->GetLOT() == 1) { + auto* levelComponent = entity->GetComponent(); + if (levelComponent) { + int32_t healthToRestore = levelComponent->GetLevel() >= 45 ? 8 : 4; + if (healthToRestore > destroyableComponent->GetMaxHealth()) healthToRestore = destroyableComponent->GetMaxHealth(); + destroyableComponent->SetHealth(healthToRestore); + + int32_t imaginationToRestore = levelComponent->GetLevel() >= 45 ? 20 : 6; + if (imaginationToRestore > destroyableComponent->GetMaxImagination()) imaginationToRestore = destroyableComponent->GetMaxImagination(); + destroyableComponent->SetImagination(imaginationToRestore); + } } - } + }); - auto cont = static_cast(entity->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); + + auto cont = static_cast(entity->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); if (cont && entity->GetLOT() == 1) { cont->SetPosition(entity->GetRespawnPosition()); cont->SetRotation(entity->GetRespawnRotation()); @@ -924,7 +964,7 @@ void GameMessages::SendResurrect(Entity* entity) { bool bRezImmediately = false; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_RESURRECT)); + bitStream.Write(eGameMessageType::RESURRECT); bitStream.Write(bRezImmediately); SEND_PACKET_BROADCAST; @@ -935,7 +975,7 @@ void GameMessages::SendStop2DAmbientSound(Entity* entity, bool force, std::strin CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_PLAY2_DAMBIENT_SOUND); + bitStream.Write((uint16_t)eGameMessageType::STOP2_D_AMBIENT_SOUND); uint32_t audioGUIDSize = audioGUID.size(); @@ -958,7 +998,7 @@ void GameMessages::SendPlay2DAmbientSound(Entity* entity, std::string audioGUID, CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_PLAY2_DAMBIENT_SOUND); + bitStream.Write((uint16_t)eGameMessageType::PLAY2_D_AMBIENT_SOUND); uint32_t audioGUIDSize = audioGUID.size(); @@ -977,7 +1017,7 @@ void GameMessages::SendSetNetworkScriptVar(Entity* entity, const SystemAddress& CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_SET_NETWORK_SCRIPT_VAR); + bitStream.Write((uint16_t)eGameMessageType::SCRIPT_NETWORK_VAR_UPDATE); // FIXME: this is a bad place to need to do a conversion because we have no clue whether data is utf8 or plain ascii // an this has performance implications @@ -995,7 +1035,7 @@ void GameMessages::SendSetNetworkScriptVar(Entity* entity, const SystemAddress& } void GameMessages::SendDropClientLoot(Entity* entity, const LWOOBJID& sourceID, LOT item, int currency, NiPoint3 spawnPos, int count) { - if (GameConfig::GetValue("no_drops") == 1) { + if (Game::config->GetValue("disable_drops") == "1") { return; } @@ -1036,7 +1076,7 @@ void GameMessages::SendDropClientLoot(Entity* entity, const LWOOBJID& sourceID, CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_DROP_CLIENT_LOOT)); + bitStream.Write(eGameMessageType::DROP_CLIENT_LOOT); bitStream.Write(bUsePosition); @@ -1056,13 +1096,13 @@ void GameMessages::SendDropClientLoot(Entity* entity, const LWOOBJID& sourceID, // Currency and powerups should not sync if (team != nullptr && currency == 0) { - CDObjectsTable* objectsTable = CDClientManager::Instance()->GetTable("Objects"); + CDObjectsTable* objectsTable = CDClientManager::Instance().GetTable(); const CDObjects& object = objectsTable->GetByID(item); if (object.type != "Powerup") { for (const auto memberId : team->members) { - auto* member = EntityManager::Instance()->GetEntity(memberId); + auto* member = Game::entityManager->GetEntity(memberId); if (member == nullptr) continue; @@ -1078,7 +1118,7 @@ void GameMessages::SendDropClientLoot(Entity* entity, const LWOOBJID& sourceID, SEND_PACKET; } -void GameMessages::SendSetPlayerControlScheme(Entity* entity, eControlSceme controlScheme) { +void GameMessages::SendSetPlayerControlScheme(Entity* entity, eControlScheme controlScheme) { CBITSTREAM; CMSGHEADER; @@ -1086,13 +1126,13 @@ void GameMessages::SendSetPlayerControlScheme(Entity* entity, eControlSceme cont bool bSwitchCam = true; bitStream.Write(entity->GetObjectID()); - bitStream.Write(uint16_t(GAME_MSG_SET_PLAYER_CONTROL_SCHEME)); + bitStream.Write(eGameMessageType::SET_PLAYER_CONTROL_SCHEME); bitStream.Write(bDelayCamSwitchIfInCinematic); bitStream.Write(bSwitchCam); - bitStream.Write(controlScheme != SCHEME_A); - if (controlScheme != SCHEME_A) bitStream.Write(controlScheme); + bitStream.Write(controlScheme != eControlScheme::SCHEME_A); + if (controlScheme != eControlScheme::SCHEME_A) bitStream.Write(controlScheme); SystemAddress sysAddr = entity->GetSystemAddress(); SEND_PACKET; @@ -1103,13 +1143,13 @@ void GameMessages::SendPlayerReachedRespawnCheckpoint(Entity* entity, const NiPo CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_PLAYER_REACHED_RESPAWN_CHECKPOINT); + bitStream.Write((uint16_t)eGameMessageType::PLAYER_REACHED_RESPAWN_CHECKPOINT); bitStream.Write(position.x); bitStream.Write(position.y); bitStream.Write(position.z); - auto con = static_cast(entity->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); + auto con = static_cast(entity->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); if (con) { auto rot = con->GetRotation(); bitStream.Write(rot.x); @@ -1137,7 +1177,7 @@ void GameMessages::SendAddSkill(Entity* entity, TSkillID skillID, int slotID) { CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write((uint16_t)GAME_MSG_ADD_SKILL); + bitStream.Write((uint16_t)eGameMessageType::ADD_SKILL); bitStream.Write(AICombatWeight != 0); if (AICombatWeight != 0) bitStream.Write(AICombatWeight); @@ -1169,7 +1209,7 @@ void GameMessages::SendRemoveSkill(Entity* entity, TSkillID skillID) { CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG_REMOVE_SKILL); + bitStream.Write(eGameMessageType::REMOVE_SKILL); bitStream.Write(false); bitStream.Write(skillID); @@ -1198,7 +1238,7 @@ void GameMessages::SendFinishArrangingWithItem(Entity* entity, const LWOOBJID& b bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_FINISH_ARRANGING_WITH_ITEM); + bitStream.Write(eGameMessageType::FINISH_ARRANGING_WITH_ITEM); bitStream.Write(buildAreaID != LWOOBJID_EMPTY); if (buildAreaID != LWOOBJID_EMPTY) bitStream.Write(buildAreaID); @@ -1224,7 +1264,7 @@ void GameMessages::SendModularBuildEnd(Entity* entity) { CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_MODULAR_BUILD_END); + bitStream.Write(eGameMessageType::MODULAR_BUILD_END); SystemAddress sysAddr = entity->GetSystemAddress(); SEND_PACKET; @@ -1235,7 +1275,7 @@ void GameMessages::SendVendorOpenWindow(Entity* entity, const SystemAddress& sys CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_VENDOR_OPEN_WINDOW); + bitStream.Write(eGameMessageType::VENDOR_OPEN_WINDOW); SEND_PACKET; } @@ -1244,24 +1284,25 @@ void GameMessages::SendVendorStatusUpdate(Entity* entity, const SystemAddress& s CBITSTREAM; CMSGHEADER; - VendorComponent* vendor = static_cast(entity->GetComponent(COMPONENT_TYPE_VENDOR)); + VendorComponent* vendor = static_cast(entity->GetComponent(eReplicaComponentType::VENDOR)); if (!vendor) return; - std::map vendorItems = vendor->GetInventory(); + auto vendorItems = vendor->GetInventory(); bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_VENDOR_STATUS_UPDATE); + bitStream.Write(eGameMessageType::VENDOR_STATUS_UPDATE); bitStream.Write(bUpdateOnly); - bitStream.Write(static_cast(vendorItems.size())); + bitStream.Write(vendorItems.size()); - for (std::pair item : vendorItems) { - bitStream.Write(static_cast(item.first)); - bitStream.Write(static_cast(item.second)); + + for (const auto& item : vendorItems) { + bitStream.Write(item.lot); + bitStream.Write(item.sortPriority); } - if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST - SEND_PACKET; + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; + SEND_PACKET; } void GameMessages::SendVendorTransactionResult(Entity* entity, const SystemAddress& sysAddr) { @@ -1271,7 +1312,7 @@ void GameMessages::SendVendorTransactionResult(Entity* entity, const SystemAddre int iResult = 0x02; // success, seems to be the only relevant one bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_VENDOR_TRANSACTION_RESULT); + bitStream.Write(eGameMessageType::VENDOR_TRANSACTION_RESULT); bitStream.Write(iResult); SEND_PACKET; @@ -1297,7 +1338,7 @@ void GameMessages::SendRemoveItemFromInventory(Entity* entity, const SystemAddre LWOOBJID iTradeID = LWOOBJID_EMPTY; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_REMOVE_ITEM_FROM_INVENTORY); + bitStream.Write(eGameMessageType::REMOVE_ITEM_FROM_INVENTORY); bitStream.Write(bConfirmed); bitStream.Write(bDeleteItem); bitStream.Write(bOutSuccess); @@ -1329,7 +1370,7 @@ void GameMessages::SendConsumeClientItem(Entity* entity, bool bSuccess, LWOOBJID CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG_CONSUME_CLIENT_ITEM); + bitStream.Write(eGameMessageType::CONSUME_CLIENT_ITEM); bitStream.Write(bSuccess); bitStream.Write(item); @@ -1342,7 +1383,7 @@ void GameMessages::SendUseItemResult(Entity* entity, LOT templateID, bool useIte CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG_USE_ITEM_RESULT); + bitStream.Write(eGameMessageType::USE_ITEM_RESULT); bitStream.Write(templateID); bitStream.Write(useItemResult); @@ -1350,12 +1391,12 @@ void GameMessages::SendUseItemResult(Entity* entity, LOT templateID, bool useIte SEND_PACKET; } -void GameMessages::SendUseItemRequirementsResponse(LWOOBJID objectID, const SystemAddress& sysAddr, UseItemResponse itemResponse) { +void GameMessages::SendUseItemRequirementsResponse(LWOOBJID objectID, const SystemAddress& sysAddr, eUseItemResponse itemResponse) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_USE_ITEM_REQUIREMENTS_RESPONSE); + bitStream.Write(eGameMessageType::USE_ITEM_REQUIREMENTS_RESPONSE); bitStream.Write(itemResponse); @@ -1366,7 +1407,7 @@ void GameMessages::SendMoveInventoryBatch(Entity* entity, uint32_t stackCount, i CBITSTREAM; CMSGHEADER; - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; Item* itemStack = inv->FindItemById(iObjID); @@ -1417,18 +1458,18 @@ void GameMessages::SendMatchResponse(Entity* entity, const SystemAddress& sysAdd CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_MATCH_RESPONSE); + bitStream.Write(eGameMessageType::MATCH_RESPONSE); bitStream.Write(response); SEND_PACKET; } -void GameMessages::SendMatchUpdate(Entity* entity, const SystemAddress& sysAddr, std::string data, int type) { +void GameMessages::SendMatchUpdate(Entity* entity, const SystemAddress& sysAddr, std::string data, eMatchUpdate type) { CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_MATCH_UPDATE); + bitStream.Write(eGameMessageType::MATCH_UPDATE); bitStream.Write(uint32_t(data.size())); for (char character : data) { bitStream.Write(uint16_t(character)); @@ -1447,7 +1488,7 @@ void GameMessages::SendRequestActivitySummaryLeaderboardData(const LWOOBJID& obj CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA); + bitStream.Write(eGameMessageType::REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA); bitStream.Write(gameID != 0); if (gameID != 0) { @@ -1480,7 +1521,7 @@ void GameMessages::SendActivityPause(LWOOBJID objectId, bool pause, const System CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ACTIVITY_PAUSE); + bitStream.Write(eGameMessageType::ACTIVITY_PAUSE); bitStream.Write(pause); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; @@ -1492,7 +1533,7 @@ void GameMessages::SendStartActivityTime(LWOOBJID objectId, float_t startTime, c CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_START_ACTIVITY_TIME); + bitStream.Write(eGameMessageType::START_ACTIVITY_TIME); bitStream.Write(startTime); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; @@ -1504,7 +1545,7 @@ void GameMessages::SendRequestActivityEnter(LWOOBJID objectId, const SystemAddre CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_REQUEST_ACTIVITY_ENTER); + bitStream.Write(eGameMessageType::REQUEST_ACTIVITY_ENTER); bitStream.Write(bStart); bitStream.Write(userID); @@ -1517,7 +1558,7 @@ void GameMessages::NotifyLevelRewards(LWOOBJID objectID, const SystemAddress& sy CMSGHEADER; bitStream.Write(objectID); - bitStream.Write((uint16_t)GAME_MSG::GAME_MSG_NOTIFY_LEVEL_REWARDS); + bitStream.Write((uint16_t)eGameMessageType::NOTIFY_LEVEL_REWARDS); bitStream.Write(level); bitStream.Write(sending_rewards); @@ -1538,7 +1579,7 @@ void GameMessages::SendSetShootingGalleryParams(LWOOBJID objectId, const SystemA CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SET_SHOOTING_GALLERY_PARAMS); + bitStream.Write(eGameMessageType::SET_SHOOTING_GALLERY_PARAMS); /* bitStream.Write(cameraFOV); bitStream.Write(cooldown); @@ -1573,7 +1614,7 @@ void GameMessages::SendNotifyClientShootingGalleryScore(LWOOBJID objectId, const CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_CLIENT_SHOOTING_GALLERY_SCORE); + bitStream.Write(eGameMessageType::NOTIFY_CLIENT_SHOOTING_GALLERY_SCORE); bitStream.Write(addTime); bitStream.Write(score); bitStream.Write(target); @@ -1604,22 +1645,9 @@ void GameMessages::SendActivitySummaryLeaderboardData(const LWOOBJID& objectID, CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA); - - bitStream.Write(leaderboard->GetGameID()); - bitStream.Write(leaderboard->GetInfoType()); - - // Leaderboard is written back as LDF string - const auto leaderboardString = leaderboard->ToString(); - bitStream.Write(leaderboardString.size()); - for (const auto c : leaderboardString) { - bitStream.Write(c); - } - if (!leaderboardString.empty()) bitStream.Write(uint16_t(0)); - - bitStream.Write0(); - bitStream.Write0(); + bitStream.Write(eGameMessageType::SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA); + leaderboard->Serialize(&bitStream); SEND_PACKET; } @@ -1627,8 +1655,8 @@ void GameMessages::HandleRequestActivitySummaryLeaderboardData(RakNet::BitStream int32_t gameID = 0; if (inStream->ReadBit()) inStream->Read(gameID); - int32_t queryType = 1; - if (inStream->ReadBit()) inStream->Read(queryType); + Leaderboard::InfoType queryType = Leaderboard::InfoType::MyStanding; + if (inStream->ReadBit()) inStream->Read(queryType); int32_t resultsEnd = 10; if (inStream->ReadBit()) inStream->Read(resultsEnd); @@ -1641,9 +1669,7 @@ void GameMessages::HandleRequestActivitySummaryLeaderboardData(RakNet::BitStream bool weekly = inStream->ReadBit(); - const auto* leaderboard = LeaderboardManager::GetLeaderboard(gameID, (InfoType)queryType, weekly, entity->GetObjectID()); - SendActivitySummaryLeaderboardData(entity->GetObjectID(), leaderboard, sysAddr); - delete leaderboard; + LeaderboardManager::SendLeaderboard(gameID, queryType, weekly, entity->GetObjectID(), entity->GetObjectID(), resultsStart, resultsEnd); } void GameMessages::HandleActivityStateChangeRequest(RakNet::BitStream* inStream, Entity* entity) { @@ -1666,11 +1692,11 @@ void GameMessages::HandleActivityStateChangeRequest(RakNet::BitStream* inStream, stringValue.push_back(character); } - auto* assosiate = EntityManager::Instance()->GetEntity(objectID); + auto* assosiate = Game::entityManager->GetEntity(objectID); Game::logger->Log("Activity State Change", "%s [%i, %i] from %i to %i", GeneralUtils::UTF16ToWTF8(stringValue).c_str(), value1, value2, entity->GetLOT(), assosiate != nullptr ? assosiate->GetLOT() : 0); - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SHOOTING_GALLERY); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SHOOTING_GALLERY); for (Entity* scriptEntity : scriptedActs) { scriptEntity->OnActivityStateChangeRequest(objectID, value1, value2, stringValue); } @@ -1683,7 +1709,7 @@ void GameMessages::SendStartCelebrationEffect(Entity* entity, const SystemAddres CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_START_CELEBRATION_EFFECT); + bitStream.Write(eGameMessageType::START_CELEBRATION_EFFECT); bitStream.Write(0); //animation bitStream.Write0(); //No custom bg obj @@ -1714,7 +1740,7 @@ void GameMessages::SendSetRailMovement(const LWOOBJID& objectID, bool pathGoForw CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_SET_RAIL_MOVEMENT); + bitStream.Write(eGameMessageType::SET_RAIL_MOVEMENT); bitStream.Write(pathGoForward); @@ -1748,7 +1774,7 @@ void GameMessages::SendStartRailMovement(const LWOOBJID& objectID, std::u16strin CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_START_RAIL_MOVEMENT); + bitStream.Write(eGameMessageType::START_RAIL_MOVEMENT); bitStream.Write(damageImmune); bitStream.Write(noAggro); @@ -1808,7 +1834,7 @@ void GameMessages::SendNotifyClientObject(const LWOOBJID& objectID, std::u16stri CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_CLIENT_OBJECT); + bitStream.Write(eGameMessageType::NOTIFY_CLIENT_OBJECT); bitStream.Write(uint32_t(name.size())); for (auto character : name) { @@ -1837,7 +1863,7 @@ void GameMessages::SendNotifyClientZoneObject(const LWOOBJID& objectID, const st CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_CLIENT_ZONE_OBJECT); + bitStream.Write(eGameMessageType::NOTIFY_CLIENT_ZONE_OBJECT); bitStream.Write(uint32_t(name.size())); for (const auto& character : name) { @@ -1863,7 +1889,7 @@ void GameMessages::SendNotifyClientFailedPrecondition(LWOOBJID objectId, const S CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_CLIENT_FAILED_PRECONDITION); + bitStream.Write(eGameMessageType::NOTIFY_CLIENT_FAILED_PRECONDITION); bitStream.Write(uint32_t(failedReason.size())); for (uint16_t character : failedReason) { @@ -1881,7 +1907,7 @@ void GameMessages::SendToggleGMInvis(LWOOBJID objectId, bool enabled, const Syst CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_TOGGLE_GM_INVIS); + bitStream.Write(eGameMessageType::TOGGLE_GM_INVIS); bitStream.Write(enabled); // does not matter? if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; @@ -1893,7 +1919,7 @@ void GameMessages::SendSetName(LWOOBJID objectID, std::u16string name, const Sys CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_SET_NAME); + bitStream.Write(eGameMessageType::SET_NAME); bitStream.Write(name.size()); @@ -1909,7 +1935,7 @@ void GameMessages::SendBBBSaveResponse(const LWOOBJID& objectId, const LWOOBJID& CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_BBB_SAVE_RESPONSE); + bitStream.Write(eGameMessageType::BBB_SAVE_RESPONSE); bitStream.Write(localID); @@ -1926,7 +1952,7 @@ void GameMessages::SendBBBSaveResponse(const LWOOBJID& objectId, const LWOOBJID& bitStream.Write(buffer[i]); SEND_PACKET; - PacketUtils::SavePacket("GAME_MSG_BBB_SAVE_RESPONSE.bin", (char*)bitStream.GetData(), bitStream.GetNumberOfBytesUsed()); + PacketUtils::SavePacket("eGameMessageType::BBB_SAVE_RESPONSE.bin", (char*)bitStream.GetData(), bitStream.GetNumberOfBytesUsed()); } // Property @@ -1936,7 +1962,7 @@ void GameMessages::SendOpenPropertyVendor(const LWOOBJID objectId, const SystemA CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_OPEN_PROPERTY_VENDOR); + bitStream.Write(eGameMessageType::OPEN_PROPERTY_VENDOR); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -1947,7 +1973,7 @@ void GameMessages::SendOpenPropertyManagment(const LWOOBJID objectId, const Syst CMSGHEADER; bitStream.Write(PropertyManagementComponent::Instance()->GetParent()->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_OPEN_PROPERTY_MANAGEMENT); + bitStream.Write(eGameMessageType::OPEN_PROPERTY_MANAGEMENT); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -1958,7 +1984,7 @@ void GameMessages::SendDownloadPropertyData(const LWOOBJID objectId, const Prope CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_DOWNLOAD_PROPERTY_DATA); + bitStream.Write(eGameMessageType::DOWNLOAD_PROPERTY_DATA); data.Serialize(bitStream); @@ -1973,7 +1999,7 @@ void GameMessages::SendPropertyRentalResponse(const LWOOBJID objectId, const LWO CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PROPERTY_RENTAL_RESPONSE); + bitStream.Write(eGameMessageType::PROPERTY_RENTAL_RESPONSE); bitStream.Write(cloneId); bitStream.Write(code); @@ -1989,7 +2015,7 @@ void GameMessages::SendLockNodeRotation(Entity* entity, std::string nodeName) { CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_LOCK_NODE_ROTATION); + bitStream.Write(eGameMessageType::LOCK_NODE_ROTATION); bitStream.Write(uint32_t(nodeName.size())); for (char character : nodeName) { @@ -2004,7 +2030,7 @@ void GameMessages::SendSetBuildModeConfirmed(LWOOBJID objectId, const SystemAddr CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SET_BUILD_MODE_CONFIRMED); + bitStream.Write(eGameMessageType::SET_BUILD_MODE_CONFIRMED); bitStream.Write(start); bitStream.Write(warnVisitors); @@ -2024,7 +2050,7 @@ void GameMessages::SendGetModelsOnProperty(LWOOBJID objectId, std::map(models.size())); @@ -2044,7 +2070,7 @@ void GameMessages::SendZonePropertyModelEquipped(LWOOBJID objectId, LWOOBJID pla CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ZONE_PROPERTY_MODEL_EQUIPPED); + bitStream.Write(eGameMessageType::ZONE_PROPERTY_MODEL_EQUIPPED); bitStream.Write(playerId); bitStream.Write(propertyId); @@ -2059,7 +2085,7 @@ void GameMessages::SendPlaceModelResponse(LWOOBJID objectId, const SystemAddress CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PLACE_MODEL_RESPONSE); + bitStream.Write(eGameMessageType::PLACE_MODEL_RESPONSE); bitStream.Write(position != NiPoint3::ZERO); if (position != NiPoint3::ZERO) { @@ -2091,7 +2117,7 @@ void GameMessages::SendUGCEquipPreCreateBasedOnEditMode(LWOOBJID objectId, const CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_HANDLE_UGC_EQUIP_PRE_CREATE_BASED_ON_EDIT_MODE); + bitStream.Write(eGameMessageType::HANDLE_UGC_POST_CREATE_BASED_ON_EDIT_MODE); bitStream.Write(modelCount); bitStream.Write(model); @@ -2105,7 +2131,7 @@ void GameMessages::SendUGCEquipPostDeleteBasedOnEditMode(LWOOBJID objectId, cons CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_HANDLE_UGC_EQUIP_POST_DELETE_BASED_ON_EDIT_MODE); + bitStream.Write(eGameMessageType::HANDLE_UGC_POST_DELETE_BASED_ON_EDIT_MODE); bitStream.Write(inventoryItem); @@ -2137,6 +2163,32 @@ void GameMessages::HandleSetPropertyAccess(RakNet::BitStream* inStream, Entity* PropertyManagementComponent::Instance()->SetPrivacyOption(static_cast(accessType)); } +void GameMessages::HandleUnUseModel(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { + bool unknown{}; + LWOOBJID objIdToAddToInventory{}; + inStream->Read(unknown); + inStream->Read(objIdToAddToInventory); + auto* inventoryComponent = entity->GetComponent(); + if (inventoryComponent) { + auto* inventory = inventoryComponent->GetInventory(eInventoryType::MODELS_IN_BBB); + auto* item = inventory->FindItemById(objIdToAddToInventory); + if (item) { + inventoryComponent->MoveItemToInventory(item, eInventoryType::MODELS, 1); + } else { + Game::logger->Log("GameMessages", "item id %llu not found in MODELS_IN_BBB inventory, likely because it does not exist", objIdToAddToInventory); + } + } + + if (unknown) { + CBITSTREAM; + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::BLUEPRINT_SAVE_RESPONSE); + bitStream.Write(LWOOBJID_EMPTY); //always zero so that a check on the client passes + bitStream.Write(eBlueprintSaveResponseType::PlacementFailed); // Sending a non-zero error code here prevents the client from deleting its in progress build for some reason? + bitStream.Write(0); + SEND_PACKET; + } +} + void GameMessages::HandleUpdatePropertyOrModelForFilterCheck(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { bool isProperty{}; LWOOBJID objectId{}; @@ -2173,24 +2225,24 @@ void GameMessages::HandleQueryPropertyData(RakNet::BitStream* inStream, Entity* Game::logger->Log("HandleQueryPropertyData", "Entity (%i) requesting data", entity->GetLOT()); /* - auto entites = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_PROPERTY_VENDOR); + auto entites = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::PROPERTY_VENDOR); entity = entites[0]; */ - auto* propertyVendorComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_PROPERTY_VENDOR)); + auto* propertyVendorComponent = static_cast(entity->GetComponent(eReplicaComponentType::PROPERTY_VENDOR)); if (propertyVendorComponent != nullptr) { propertyVendorComponent->OnQueryPropertyData(entity, sysAddr); } /* - entites = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_PROPERTY_MANAGEMENT); + entites = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::PROPERTY_MANAGEMENT); entity = entites[0]; */ - auto* propertyManagerComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_PROPERTY_MANAGEMENT)); + auto* propertyManagerComponent = static_cast(entity->GetComponent(eReplicaComponentType::PROPERTY_MANAGEMENT)); if (propertyManagerComponent != nullptr) { propertyManagerComponent->OnQueryPropertyData(entity, sysAddr); @@ -2220,7 +2272,7 @@ void GameMessages::HandleSetBuildMode(RakNet::BitStream* inStream, Entity* entit if (inStream->ReadBit()) inStream->Read(startPosition); - auto* player = EntityManager::Instance()->GetEntity(playerId); + auto* player = Game::entityManager->GetEntity(playerId); if (startPosition == NiPoint3::ZERO) { startPosition = player->GetPosition(); @@ -2234,7 +2286,7 @@ void GameMessages::HandleSetBuildMode(RakNet::BitStream* inStream, Entity* entit } void GameMessages::HandleStartBuildingWithItem(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - if (!entity->HasComponent(COMPONENT_TYPE_PROPERTY_MANAGEMENT)) { + if (!entity->HasComponent(eReplicaComponentType::PROPERTY_MANAGEMENT)) { return; } @@ -2268,7 +2320,7 @@ void GameMessages::HandleStartBuildingWithItem(RakNet::BitStream* inStream, Enti auto* user = UserManager::Instance()->GetUser(sysAddr); - auto* player = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + auto* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); SendStartArrangingWithItem( player, @@ -2290,25 +2342,25 @@ void GameMessages::HandleStartBuildingWithItem(RakNet::BitStream* inStream, Enti void GameMessages::HandlePropertyEditorBegin(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { PropertyManagementComponent::Instance()->OnStartBuilding(); - dZoneManager::Instance()->GetZoneControlObject()->OnZonePropertyEditBegin(); + Game::zoneManager->GetZoneControlObject()->OnZonePropertyEditBegin(); } void GameMessages::HandlePropertyEditorEnd(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { PropertyManagementComponent::Instance()->OnFinishBuilding(); - dZoneManager::Instance()->GetZoneControlObject()->OnZonePropertyEditEnd(); + Game::zoneManager->GetZoneControlObject()->OnZonePropertyEditEnd(); } void GameMessages::HandlePropertyContentsFromClient(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { User* user = UserManager::Instance()->GetUser(sysAddr); - Entity* player = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); SendGetModelsOnProperty(player->GetObjectID(), PropertyManagementComponent::Instance()->GetModels(), UNASSIGNED_SYSTEM_ADDRESS); } void GameMessages::HandlePropertyModelEquipped(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - dZoneManager::Instance()->GetZoneControlObject()->OnZonePropertyModelEquipped(); + Game::zoneManager->GetZoneControlObject()->OnZonePropertyModelEquipped(); } void GameMessages::HandlePlacePropertyModel(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { @@ -2351,16 +2403,40 @@ void GameMessages::HandleDeletePropertyModel(RakNet::BitStream* inStream, Entity } void GameMessages::HandleBBBLoadItemRequest(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - LWOOBJID itemID = LWOOBJID_EMPTY; - inStream->Read(itemID); + LWOOBJID previousItemID = LWOOBJID_EMPTY; + inStream->Read(previousItemID); - Game::logger->Log("BBB", "Load item request for: %lld", itemID); + Game::logger->Log("BBB", "Load item request for: %lld", previousItemID); + LWOOBJID newId = previousItemID; + auto* inventoryComponent = entity->GetComponent(); + if (inventoryComponent) { + auto* inventory = inventoryComponent->GetInventory(eInventoryType::MODELS); + auto* itemToMove = inventory->FindItemById(previousItemID); + if (itemToMove) { + LOT previousLot = itemToMove->GetLot(); + inventoryComponent->MoveItemToInventory(itemToMove, eInventoryType::MODELS_IN_BBB, 1, false); + + auto* destinationInventory = inventoryComponent->GetInventory(eInventoryType::MODELS_IN_BBB); + if (destinationInventory) { + auto* movedItem = destinationInventory->FindItemByLot(previousLot); + if (movedItem) newId = movedItem->GetId(); + } + } else { + Game::logger->Log("GameMessages", "item id %llu not found in MODELS inventory, likely because it does not exist", previousItemID); + } + } + + // Second argument always true (successful) for now + SendBlueprintLoadItemResponse(sysAddr, true, previousItemID, newId); +} + +void GameMessages::SendBlueprintLoadItemResponse(const SystemAddress& sysAddr, bool success, LWOOBJID oldItemId, LWOOBJID newItemId) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_BLUEPRINT_LOAD_RESPONSE_ITEMID); - bitStream.Write(static_cast(1)); - bitStream.Write(itemID); - bitStream.Write(itemID); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::BLUEPRINT_LOAD_RESPONSE_ITEMID); + bitStream.Write(static_cast(success)); + bitStream.Write(oldItemId); + bitStream.Write(newItemId); SEND_PACKET; } @@ -2369,7 +2445,7 @@ void GameMessages::SendSmash(Entity* entity, float force, float ghostOpacity, LW CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_SMASH); + bitStream.Write(eGameMessageType::SMASH); bitStream.Write(ignoreObjectVisibility); bitStream.Write(force); @@ -2384,7 +2460,7 @@ void GameMessages::SendUnSmash(Entity* entity, LWOOBJID builderID, float duratio CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_UNSMASH); + bitStream.Write(eGameMessageType::UN_SMASH); bitStream.Write(builderID != LWOOBJID_EMPTY); if (builderID != LWOOBJID_EMPTY) bitStream.Write(builderID); @@ -2396,8 +2472,24 @@ void GameMessages::SendUnSmash(Entity* entity, LWOOBJID builderID, float duratio } void GameMessages::HandleControlBehaviors(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - // TODO - Game::logger->Log("GameMessages", "Recieved Control Behavior GameMessage, but property behaviors are unimplemented."); + AMFDeserialize reader; + std::unique_ptr amfArguments(reader.Read(inStream)); + if (amfArguments->GetValueType() != eAmf::Array) return; + + uint32_t commandLength{}; + inStream->Read(commandLength); + + std::string command; + for (uint32_t i = 0; i < commandLength; i++) { + unsigned char character; + inStream->Read(character); + command.push_back(character); + } + + auto owner = PropertyManagementComponent::Instance()->GetOwner(); + if (!owner) return; + + ControlBehaviors::Instance().ProcessCommand(entity, sysAddr, static_cast(amfArguments.get()), command, owner); } void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { @@ -2413,7 +2505,6 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent /__\/// _ \ \ /\ / / _` | '__/ _ \/ / / \/ \ __/\ V V / (_| | | | __/\_/ \_____/\___| \_/\_/ \__,_|_| \___\/ - <>=======() (/\___ /|\\ ()==========<>_ \_/ | \\ //|\ ______/ \) @@ -2430,34 +2521,25 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent ______/ / '------' */ - - //First, we have Wincent's clean methods of reading in the data received from the client. LWOOBJID localId; - uint32_t timeTaken; inStream->Read(localId); - uint32_t ld0Size; - inStream->Read(ld0Size); - for (auto i = 0; i < 5; ++i) { - uint8_t c; - inStream->Read(c); - } + uint32_t sd0Size; + inStream->Read(sd0Size); + std::shared_ptr sd0Data(new char[sd0Size]); - uint32_t lxfmlSize; - inStream->Read(lxfmlSize); - uint8_t* inData = static_cast(std::malloc(lxfmlSize)); - - if (inData == nullptr) { + if (sd0Data == nullptr) { return; } - for (uint32_t i = 0; i < lxfmlSize; ++i) { + for (uint32_t i = 0; i < sd0Size; ++i) { uint8_t c; inStream->Read(c); - inData[i] = c; + sd0Data[i] = c; } + uint32_t timeTaken; inStream->Read(timeTaken); /* @@ -2469,6 +2551,8 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent Note, in the live client it'll still display the bricks going out as they're being used, but on relog/world change, they reappear as we didn't take them. + + TODO Apparently the bricks are supposed to be taken via MoveInventoryBatch? */ ////Decompress the SD0 from the client so we can process the lxfml properly @@ -2487,22 +2571,22 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent //We runs this in async because the http library here is blocking, meaning it'll halt the thread. //But we don't want the server to go unresponsive, because then the client would disconnect. - std::async(std::launch::async, [&]() { + auto returnVal = std::async(std::launch::async, [&]() { //We need to get a new ID for our model first: ObjectIDManager::Instance()->RequestPersistentID([=](uint32_t newID) { LWOOBJID newIDL = newID; - newIDL = GeneralUtils::SetBit(newIDL, OBJECT_BIT_CHARACTER); - newIDL = GeneralUtils::SetBit(newIDL, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(newIDL, eObjectBits::CHARACTER); + GeneralUtils::SetBit(newIDL, eObjectBits::PERSISTENT); ObjectIDManager::Instance()->RequestPersistentID([=](uint32_t blueprintIDSmall) { blueprintIDSmall = ObjectIDManager::Instance()->GenerateRandomObjectID(); LWOOBJID blueprintID = blueprintIDSmall; - blueprintID = GeneralUtils::SetBit(blueprintID, OBJECT_BIT_CHARACTER); - blueprintID = GeneralUtils::SetBit(blueprintID, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(blueprintID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(blueprintID, eObjectBits::PERSISTENT); //We need to get the propertyID: (stolen from Wincent's propertyManagementComp) - const auto& worldId = dZoneManager::Instance()->GetZone()->GetZoneID(); + const auto& worldId = Game::zoneManager->GetZone()->GetZoneID(); const auto zoneId = worldId.GetMapID(); const auto cloneId = worldId.GetCloneID(); @@ -2513,9 +2597,7 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent auto result = query.execQuery(); - if (result.eof() || result.fieldIsNull(0)) { - return; - } + if (result.eof() || result.fieldIsNull(0)) return; int templateId = result.getIntField(0); @@ -2533,6 +2615,7 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent propertyId = propertyEntry->getUInt64(1); } + delete propertyEntry; delete propertyLookup; //Insert into ugc: @@ -2543,7 +2626,7 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent ugcs->setInt(4, 0); //whacky stream biz - std::string s((char*)inData, lxfmlSize); + std::string s(sd0Data.get(), sd0Size); std::istringstream iss(s); std::istream& stream = iss; @@ -2554,18 +2637,25 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent delete ugcs; //Insert into the db as a BBB model: - auto* stmt = Database::CreatePreppedStmt("INSERT INTO `properties_contents`(`id`, `property_id`, `ugc_id`, `lot`, `x`, `y`, `z`, `rx`, `ry`, `rz`, `rw`) VALUES (?,?,?,?,?,?,?,?,?,?,?)"); + auto* stmt = Database::CreatePreppedStmt("INSERT INTO `properties_contents` VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)"); stmt->setUInt64(1, newIDL); stmt->setUInt64(2, propertyId); stmt->setUInt(3, blueprintIDSmall); - stmt->setUInt(4, 14); //14 is the lot the BBB models use - stmt->setDouble(5, 0.0f); //x - stmt->setDouble(6, 0.0f); //y - stmt->setDouble(7, 0.0f); //z - stmt->setDouble(8, 0.0f); - stmt->setDouble(9, 0.0f); - stmt->setDouble(10, 0.0f); - stmt->setDouble(11, 0.0f); + stmt->setUInt(4, 14); // 14 is the lot the BBB models use + stmt->setDouble(5, 0.0f); // x + stmt->setDouble(6, 0.0f); // y + stmt->setDouble(7, 0.0f); // z + stmt->setDouble(8, 0.0f); // rx + stmt->setDouble(9, 0.0f); // ry + stmt->setDouble(10, 0.0f); // rz + stmt->setDouble(11, 0.0f); // rw + stmt->setString(12, "Objects_14_name"); // Model name. TODO make this customizable + stmt->setString(13, ""); // Model description. TODO implement this. + stmt->setDouble(14, 0); // behavior 1. TODO implement this. + stmt->setDouble(15, 0); // behavior 2. TODO implement this. + stmt->setDouble(16, 0); // behavior 3. TODO implement this. + stmt->setDouble(17, 0); // behavior 4. TODO implement this. + stmt->setDouble(18, 0); // behavior 5. TODO implement this. stmt->execute(); delete stmt; @@ -2591,38 +2681,21 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent //Tell the client their model is saved: (this causes us to actually pop out of our current state): CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_BLUEPRINT_SAVE_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::BLUEPRINT_SAVE_RESPONSE); bitStream.Write(localId); - bitStream.Write(0); - bitStream.Write(1); + bitStream.Write(eBlueprintSaveResponseType::EverythingWorked); + bitStream.Write(1); bitStream.Write(blueprintID); - bitStream.Write(lxfmlSize + 9); + bitStream.Write(sd0Size); - //Write a fake sd0 header: - bitStream.Write(0x73); //s - bitStream.Write(0x64); //d - bitStream.Write(0x30); //0 - bitStream.Write(0x01); //1 - bitStream.Write(0xFF); //end magic - - bitStream.Write(lxfmlSize); - - for (size_t i = 0; i < lxfmlSize; ++i) - bitStream.Write(inData[i]); + for (size_t i = 0; i < sd0Size; ++i) { + bitStream.Write(sd0Data[i]); + } SEND_PACKET; - PacketUtils::SavePacket("MSG_CLIENT_BLUEPRINT_SAVE_RESPONSE.bin", (char*)bitStream.GetData(), bitStream.GetNumberOfBytesUsed()); //Now we have to construct this object: - /* - * This needs to be sent as config data, but I don't know how to right now. - 'blueprintid': (9, 1152921508346399522), - 'componentWhitelist': (1, 1), - 'modelType': (1, 2), - 'propertyObjectID': (7, True), - 'userModelID': (9, 1152921510759098799) - */ EntityInfo info; info.lot = 14; @@ -2644,15 +2717,16 @@ void GameMessages::HandleBBBSaveRequest(RakNet::BitStream* inStream, Entity* ent info.settings.push_back(propertyObjectID); info.settings.push_back(userModelID); - Entity* newEntity = EntityManager::Instance()->CreateEntity(info, nullptr); + Entity* newEntity = Game::entityManager->CreateEntity(info, nullptr); if (newEntity) { - EntityManager::Instance()->ConstructEntity(newEntity); + Game::entityManager->ConstructEntity(newEntity); //Make sure the propMgmt doesn't delete our model after the server dies //Trying to do this after the entity is constructed. Shouldn't really change anything but //there was an issue with builds not appearing since it was placed above ConstructEntity. PropertyManagementComponent::Instance()->AddModel(newEntity->GetObjectID(), newIDL); } + }); }); }); @@ -2741,21 +2815,21 @@ void GameMessages::HandleSetConsumableItem(RakNet::BitStream* inStream, Entity* void GameMessages::SendPlayCinematic(LWOOBJID objectId, std::u16string pathName, const SystemAddress& sysAddr, bool allowGhostUpdates, bool bCloseMultiInteract, bool bSendServerNotify, bool bUseControlledObjectForAudioListener, - int endBehavior, bool hidePlayerDuringCine, float leadIn, bool leavePlayerLockedWhenFinished, + eEndBehavior endBehavior, bool hidePlayerDuringCine, float leadIn, bool leavePlayerLockedWhenFinished, bool lockPlayer, bool result, bool skipIfSamePath, float startTimeAdvance) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PLAY_CINEMATIC); + bitStream.Write(eGameMessageType::PLAY_CINEMATIC); bitStream.Write(allowGhostUpdates); bitStream.Write(bCloseMultiInteract); bitStream.Write(bSendServerNotify); bitStream.Write(bUseControlledObjectForAudioListener); - bitStream.Write(endBehavior != 0); - if (endBehavior != 0) bitStream.Write(endBehavior); + bitStream.Write(endBehavior != eEndBehavior::RETURN); + if (endBehavior != eEndBehavior::RETURN) bitStream.Write(endBehavior); bitStream.Write(hidePlayerDuringCine); @@ -2786,7 +2860,7 @@ void GameMessages::SendEndCinematic(LWOOBJID objectId, std::u16string pathName, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_END_CINEMATIC); + bitStream.Write(eGameMessageType::END_CINEMATIC); bitStream.Write(leadOut != -1); if (leadOut != -1) bitStream.Write(leadOut); @@ -2805,7 +2879,7 @@ void GameMessages::SendEndCinematic(LWOOBJID objectId, std::u16string pathName, void GameMessages::HandleCinematicUpdate(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { eCinematicEvent event; if (!inStream->ReadBit()) { - event = STARTED; + event = eCinematicEvent::STARTED; } else { inStream->Read(event); } @@ -2841,13 +2915,13 @@ void GameMessages::HandleCinematicUpdate(RakNet::BitStream* inStream, Entity* en inStream->Read(waypoint); } - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPT); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPT); for (Entity* scriptEntity : scriptedActs) { scriptEntity->OnCinematicUpdate(scriptEntity, entity, event, pathName, pathTime, overallTime, waypoint); } } -void GameMessages::SendSetStunned(LWOOBJID objectId, eStunState stateChangeType, const SystemAddress& sysAddr, +void GameMessages::SendSetStunned(LWOOBJID objectId, eStateChangeType stateChangeType, const SystemAddress& sysAddr, LWOOBJID originator, bool bCantAttack, bool bCantEquip, bool bCantInteract, bool bCantJump, bool bCantMove, bool bCantTurn, bool bCantUseItem, bool bDontTerminateInteract, bool bIgnoreImmunity, @@ -2859,7 +2933,7 @@ void GameMessages::SendSetStunned(LWOOBJID objectId, eStunState stateChangeType, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SET_STUNNED); + bitStream.Write(eGameMessageType::SET_STUNNED); bitStream.Write(originator != LWOOBJID_EMPTY); if (originator != LWOOBJID_EMPTY) bitStream.Write(originator); @@ -2895,13 +2969,76 @@ void GameMessages::SendSetStunned(LWOOBJID objectId, eStunState stateChangeType, SEND_PACKET; } +void GameMessages::SendSetStunImmunity(LWOOBJID target, eStateChangeType state, const SystemAddress& sysAddr, + LWOOBJID originator, + bool bImmuneToStunAttack, + bool bImmuneToStunEquip, + bool bImmuneToStunInteract, + bool bImmuneToStunJump, + bool bImmuneToStunMove, + bool bImmuneToStunTurn, + bool bImmuneToStunUseItem) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(target); + bitStream.Write(eGameMessageType::SET_STUN_IMMUNITY); + + bitStream.Write(originator != LWOOBJID_EMPTY); + if (originator != LWOOBJID_EMPTY) bitStream.Write(originator); + + bitStream.Write(state); + + bitStream.Write(bImmuneToStunAttack); + bitStream.Write(bImmuneToStunEquip); + bitStream.Write(bImmuneToStunInteract); + bitStream.Write(bImmuneToStunJump); + bitStream.Write(bImmuneToStunMove); + bitStream.Write(bImmuneToStunTurn); + bitStream.Write(bImmuneToStunUseItem); + + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; + SEND_PACKET; +} + +void GameMessages::SendSetStatusImmunity(LWOOBJID objectId, eStateChangeType state, const SystemAddress& sysAddr, + bool bImmuneToBasicAttack, + bool bImmuneToDamageOverTime, + bool bImmuneToKnockback, + bool bImmuneToInterrupt, + bool bImmuneToSpeed, + bool bImmuneToImaginationGain, + bool bImmuneToImaginationLoss, + bool bImmuneToQuickbuildInterrupt, + bool bImmuneToPullToPoint) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(objectId); + bitStream.Write(eGameMessageType::SET_STATUS_IMMUNITY); + + bitStream.Write(state); + + bitStream.Write(bImmuneToBasicAttack); + bitStream.Write(bImmuneToDamageOverTime); + bitStream.Write(bImmuneToKnockback); + bitStream.Write(bImmuneToInterrupt); + bitStream.Write(bImmuneToSpeed); + bitStream.Write(bImmuneToImaginationGain); + bitStream.Write(bImmuneToImaginationLoss); + bitStream.Write(bImmuneToQuickbuildInterrupt); + bitStream.Write(bImmuneToPullToPoint); + + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; + SEND_PACKET; +} void GameMessages::SendOrientToAngle(LWOOBJID objectId, bool bRelativeToCurrent, float fAngle, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ORIENT_TO_ANGLE); + bitStream.Write(eGameMessageType::ORIENT_TO_ANGLE); bitStream.Write(bRelativeToCurrent); bitStream.Write(fAngle); @@ -2916,7 +3053,7 @@ void GameMessages::SendAddRunSpeedModifier(LWOOBJID objectId, LWOOBJID caster, u CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ADD_RUN_SPEED_MODIFIER); + bitStream.Write(eGameMessageType::ADD_RUN_SPEED_MODIFIER); bitStream.Write(caster != LWOOBJID_EMPTY); if (caster != LWOOBJID_EMPTY) bitStream.Write(caster); @@ -2933,7 +3070,7 @@ void GameMessages::SendRemoveRunSpeedModifier(LWOOBJID objectId, uint32_t modifi CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_REMOVE_RUN_SPEED_MODIFIER); + bitStream.Write(eGameMessageType::REMOVE_RUN_SPEED_MODIFIER); bitStream.Write(modifier != 500); if (modifier != 500) bitStream.Write(modifier); @@ -2947,7 +3084,7 @@ void GameMessages::SendPropertyEntranceBegin(LWOOBJID objectId, const SystemAddr CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PROPERTY_ENTRANCE_BEGIN); + bitStream.Write(eGameMessageType::PROPERTY_ENTRANCE_BEGIN); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -2958,7 +3095,7 @@ void GameMessages::SendPropertySelectQuery(LWOOBJID objectId, int32_t navOffset, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PROPERTY_SELECT_QUERY); + bitStream.Write(eGameMessageType::PROPERTY_SELECT_QUERY); bitStream.Write(navOffset); bitStream.Write(thereAreMore); @@ -2981,7 +3118,7 @@ void GameMessages::SendNotifyObject(LWOOBJID objectId, LWOOBJID objIDSender, std CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_OBJECT); + bitStream.Write(eGameMessageType::NOTIFY_OBJECT); bitStream.Write(objIDSender); bitStream.Write(static_cast(name.size())); @@ -3021,7 +3158,7 @@ void GameMessages::SendTeamPickupItem(LWOOBJID objectId, LWOOBJID lootID, LWOOBJ CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_TEAM_PICKUP_ITEM); + bitStream.Write(eGameMessageType::TEAM_PICKUP_ITEM); bitStream.Write(lootID); bitStream.Write(lootOwnerID); @@ -3037,7 +3174,7 @@ void GameMessages::SendServerTradeInvite(LWOOBJID objectId, bool bNeedInvitePopU CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_TRADE_INVITE); + bitStream.Write(eGameMessageType::SERVER_TRADE_INVITE); bitStream.Write(bNeedInvitePopUp); bitStream.Write(i64Requestor); @@ -3055,7 +3192,7 @@ void GameMessages::SendServerTradeInitialReply(LWOOBJID objectId, LWOOBJID i64In CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_TRADE_INITIAL_REPLY); + bitStream.Write(eGameMessageType::SERVER_TRADE_INITIAL_REPLY); bitStream.Write(i64Invitee); bitStream.Write(resultType); @@ -3073,7 +3210,7 @@ void GameMessages::SendServerTradeFinalReply(LWOOBJID objectId, bool bResult, LW CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_TRADE_FINAL_REPLY); + bitStream.Write(eGameMessageType::SERVER_TRADE_FINAL_REPLY); bitStream.Write(bResult); bitStream.Write(i64Invitee); @@ -3091,7 +3228,7 @@ void GameMessages::SendServerTradeAccept(LWOOBJID objectId, bool bFirst, const S CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_TRADE_ACCEPT); + bitStream.Write(eGameMessageType::SERVER_TRADE_ACCEPT); bitStream.Write(bFirst); @@ -3104,7 +3241,7 @@ void GameMessages::SendServerTradeCancel(LWOOBJID objectId, const SystemAddress& CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_TRADE_CANCEL); + bitStream.Write(eGameMessageType::SERVER_TRADE_CANCEL); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -3115,7 +3252,7 @@ void GameMessages::SendServerTradeUpdate(LWOOBJID objectId, uint64_t coins, cons CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SERVER_TRADE_UPDATE); + bitStream.Write(eGameMessageType::SERVER_TRADE_UPDATE); bitStream.Write(false); bitStream.Write(coins); @@ -3143,7 +3280,7 @@ void GameMessages::HandleClientTradeRequest(RakNet::BitStream* inStream, Entity* // Check if the player has restricted trade access auto* character = entity->GetCharacter(); - if (character->HasPermission(PermissionMap::RestrictedTradeAccess)) { + if (character->HasPermission(ePermissionMap::RestrictedTradeAccess)) { // Send a message to the player ChatPackets::SendSystemMessage( sysAddr, @@ -3158,12 +3295,12 @@ void GameMessages::HandleClientTradeRequest(RakNet::BitStream* inStream, Entity* inStream->Read(i64Invitee); - auto* invitee = EntityManager::Instance()->GetEntity(i64Invitee); + auto* invitee = Game::entityManager->GetEntity(i64Invitee); if (invitee != nullptr && invitee->IsPlayer()) { character = invitee->GetCharacter(); - if (character->HasPermission(PermissionMap::RestrictedTradeAccess)) { + if (character->HasPermission(ePermissionMap::RestrictedTradeAccess)) { // Send a message to the player ChatPackets::SendSystemMessage( sysAddr, @@ -3288,12 +3425,12 @@ void GameMessages::HandleClientTradeUpdate(RakNet::BitStream* inStream, Entity* //Pets: -void GameMessages::SendNotifyPetTamingMinigame(LWOOBJID objectId, LWOOBJID petId, LWOOBJID playerTamingId, bool bForceTeleport, uint32_t notifyType, NiPoint3 petsDestPos, NiPoint3 telePos, NiQuaternion teleRot, const SystemAddress& sysAddr) { +void GameMessages::SendNotifyPetTamingMinigame(LWOOBJID objectId, LWOOBJID petId, LWOOBJID playerTamingId, bool bForceTeleport, ePetTamingNotifyType notifyType, NiPoint3 petsDestPos, NiPoint3 telePos, NiQuaternion teleRot, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_PET_TAMING_MINIGAME); + bitStream.Write(eGameMessageType::NOTIFY_PET_TAMING_MINIGAME); bitStream.Write(petId); bitStream.Write(playerTamingId); @@ -3315,18 +3452,18 @@ void GameMessages::SendNotifyTamingModelLoadedOnServer(LWOOBJID objectId, const CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_TAMING_MODEL_LOADED_ON_SERVER); + bitStream.Write(eGameMessageType::NOTIFY_TAMING_MODEL_LOADED_ON_SERVER); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; } -void GameMessages::SendNotifyPetTamingPuzzleSelected(LWOOBJID objectId, std::vector& bricks, const SystemAddress& sysAddr) { +void GameMessages::SendNotifyPetTamingPuzzleSelected(LWOOBJID objectId, const std::vector& bricks, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_PET_TAMING_PUZZLE_SELECTED); + bitStream.Write(eGameMessageType::NOTIFY_TAMING_PUZZLE_SELECTED); bitStream.Write(static_cast(bricks.size())); for (const auto& brick : bricks) { @@ -3343,7 +3480,7 @@ void GameMessages::SendPetTamingTryBuildResult(LWOOBJID objectId, bool bSuccess, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PET_TAMING_TRY_BUILD_RESULT); + bitStream.Write(eGameMessageType::PET_TAMING_TRY_BUILD_RESULT); bitStream.Write(bSuccess); bitStream.Write(iNumCorrect != 0); @@ -3358,7 +3495,7 @@ void GameMessages::SendPetResponse(LWOOBJID objectId, LWOOBJID objIDPet, int32_t CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PET_RESPONSE); + bitStream.Write(eGameMessageType::PET_RESPONSE); bitStream.Write(objIDPet); bitStream.Write(iPetCommandType); @@ -3374,7 +3511,7 @@ void GameMessages::SendAddPetToPlayer(LWOOBJID objectId, int32_t iElementalType, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ADD_PET_TO_PLAYER); + bitStream.Write(eGameMessageType::ADD_PET_TO_PLAYER); bitStream.Write(iElementalType); bitStream.Write(static_cast(name.size())); @@ -3394,7 +3531,7 @@ void GameMessages::SendRegisterPetID(LWOOBJID objectId, LWOOBJID objID, const Sy CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_REGISTER_PET_ID); + bitStream.Write(eGameMessageType::REGISTER_PET_ID); bitStream.Write(objID); @@ -3407,7 +3544,7 @@ void GameMessages::SendRegisterPetDBID(LWOOBJID objectId, LWOOBJID petDBID, cons CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_REGISTER_PET_DBID); + bitStream.Write(eGameMessageType::REGISTER_PET_DBID); bitStream.Write(petDBID); @@ -3420,7 +3557,7 @@ void GameMessages::SendMarkInventoryItemAsActive(LWOOBJID objectId, bool bActive CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_MARK_INVENTORY_ITEM_AS_ACTIVE); + bitStream.Write(eGameMessageType::MARK_INVENTORY_ITEM_AS_ACTIVE); bitStream.Write(bActive); @@ -3439,7 +3576,7 @@ void GameMessages::SendClientExitTamingMinigame(LWOOBJID objectId, bool bVolunta CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_CLIENT_EXIT_TAMING_MINIGAME); + bitStream.Write(eGameMessageType::CLIENT_EXIT_TAMING_MINIGAME); bitStream.Write(bVoluntaryExit); @@ -3452,7 +3589,7 @@ void GameMessages::SendShowPetActionButton(LWOOBJID objectId, int32_t buttonLabe CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SHOW_PET_ACTION_BUTTON); + bitStream.Write(eGameMessageType::SHOW_PET_ACTION_BUTTON); bitStream.Write(buttonLabel); bitStream.Write(bShow); @@ -3466,7 +3603,7 @@ void GameMessages::SendPlayEmote(LWOOBJID objectId, int32_t emoteID, LWOOBJID ta CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PLAY_EMOTE); + bitStream.Write(eGameMessageType::PLAY_EMOTE); bitStream.Write(emoteID); bitStream.Write(target); @@ -3475,13 +3612,27 @@ void GameMessages::SendPlayEmote(LWOOBJID objectId, int32_t emoteID, LWOOBJID ta SEND_PACKET; } +void GameMessages::SendRemoveBuff(Entity* entity, bool fromUnEquip, bool removeImmunity, uint32_t buffId) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(entity->GetObjectID()); + bitStream.Write(eGameMessageType::REMOVE_BUFF); + + bitStream.Write(false); // bFromRemoveBehavior but setting this to true makes the GM not do anything on the client? + bitStream.Write(fromUnEquip); + bitStream.Write(removeImmunity); + bitStream.Write(buffId); + + SEND_PACKET_BROADCAST; +} void GameMessages::SendBouncerActiveStatus(LWOOBJID objectId, bool bActive, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_BOUNCER_ACTIVE_STATUS); + bitStream.Write(eGameMessageType::BOUNCER_ACTIVE_STATUS); bitStream.Write(bActive); @@ -3495,7 +3646,7 @@ void GameMessages::SendSetPetName(LWOOBJID objectId, std::u16string name, LWOOBJ CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SET_PET_NAME); + bitStream.Write(eGameMessageType::SET_PET_NAME); bitStream.Write(static_cast(name.size())); for (const auto character : name) { @@ -3515,7 +3666,7 @@ void GameMessages::SendSetPetNameModerated(LWOOBJID objectId, LWOOBJID petDBID, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SET_PET_NAME_MODERATED); + bitStream.Write(eGameMessageType::SET_PET_NAME_MODERATED); bitStream.Write(petDBID != LWOOBJID_EMPTY); if (petDBID != LWOOBJID_EMPTY) bitStream.Write(petDBID); @@ -3532,7 +3683,7 @@ void GameMessages::SendPetNameChanged(LWOOBJID objectId, int32_t moderationStatu CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_PET_NAME_CHANGED); + bitStream.Write(eGameMessageType::PET_NAME_CHANGED); bitStream.Write(moderationStatus); @@ -3728,10 +3879,10 @@ void GameMessages::HandleMessageBoxResponse(RakNet::BitStream* inStream, Entity* auto* racingControlComponent = entity->GetComponent(); if (racingControlComponent != nullptr) { - racingControlComponent->HandleMessageBoxResponse(userEntity, GeneralUtils::UTF16ToWTF8(identifier)); + racingControlComponent->HandleMessageBoxResponse(userEntity, iButton, GeneralUtils::UTF16ToWTF8(identifier)); } - for (auto* shootingGallery : EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SHOOTING_GALLERY)) { + for (auto* shootingGallery : Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SHOOTING_GALLERY)) { shootingGallery->OnMessageBoxResponse(userEntity, iButton, identifier, userData); } } @@ -3781,7 +3932,7 @@ void GameMessages::SendDisplayZoneSummary(LWOOBJID objectId, const SystemAddress CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_DISPLAY_ZONE_SUMMARY); + bitStream.Write(eGameMessageType::DISPLAY_ZONE_SUMMARY); bitStream.Write(isPropertyMap); bitStream.Write(isZoneStart); @@ -3799,7 +3950,7 @@ void GameMessages::SendNotifyNotEnoughInvSpace(LWOOBJID objectId, uint32_t freeS CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_VEHICLE_NOTIFY_FINISHED_RACE); + bitStream.Write(eGameMessageType::VEHICLE_NOTIFY_FINISHED_RACE); bitStream.Write(freeSlotsNeeded); bitStream.Write(inventoryType != 0); @@ -3814,7 +3965,7 @@ void GameMessages::SendDisplayMessageBox(LWOOBJID objectId, bool bShow, LWOOBJID CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_DISPLAY_MESSAGE_BOX); + bitStream.Write(eGameMessageType::DISPLAY_MESSAGE_BOX); bitStream.Write(bShow); bitStream.Write(callbackClient); @@ -3841,12 +3992,12 @@ void GameMessages::SendDisplayMessageBox(LWOOBJID objectId, bool bShow, LWOOBJID } void GameMessages::SendDisplayChatBubble(LWOOBJID objectId, const std::u16string& text, const SystemAddress& sysAddr) { - // GAME_MSG_DISPLAY_CHAT_BUBBLE + // eGameMessageType::DISPLAY_CHAT_BUBBLE CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_DISPLAY_CHAT_BUBBLE); + bitStream.Write(eGameMessageType::DISPLAY_CHAT_BUBBLE); bitStream.Write(static_cast(text.size())); for (const auto character : text) { @@ -3857,13 +4008,27 @@ void GameMessages::SendDisplayChatBubble(LWOOBJID objectId, const std::u16string SEND_PACKET; } + +void GameMessages::SendChangeIdleFlags(LWOOBJID objectId, eAnimationFlags flagsOn, eAnimationFlags flagsOff, const SystemAddress& sysAddr) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(objectId); + bitStream.Write(eGameMessageType::CHANGE_IDLE_FLAGS); + bitStream.Write(flagsOff != eAnimationFlags::IDLE_NONE); + if (flagsOff != eAnimationFlags::IDLE_NONE) bitStream.Write(flagsOff); + bitStream.Write(flagsOn != eAnimationFlags::IDLE_NONE); + if (flagsOn != eAnimationFlags::IDLE_NONE) bitStream.Write(flagsOn); + + SEND_PACKET_BROADCAST; +} // Mounts void GameMessages::SendSetMountInventoryID(Entity* entity, const LWOOBJID& objectID, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_SET_MOUNT_INVENTORY_ID); + bitStream.Write(eGameMessageType::SET_MOUNT_INVENTORY_ID); bitStream.Write(objectID); SEND_PACKET_BROADCAST; @@ -3878,7 +4043,7 @@ void GameMessages::HandleDismountComplete(RakNet::BitStream* inStream, Entity* e // If we aren't possessing somethings, the don't do anything if (objectId != LWOOBJID_EMPTY) { auto* possessorComponent = entity->GetComponent(); - auto* mount = EntityManager::Instance()->GetEntity(objectId); + auto* mount = Game::entityManager->GetEntity(objectId); // make sure we have the things we need and they aren't null if (possessorComponent && mount) { if (!possessorComponent->GetIsDismounting()) return; @@ -3903,21 +4068,21 @@ void GameMessages::HandleDismountComplete(RakNet::BitStream* inStream, Entity* e if (possessableComponent) possessableComponent->Dismount(); // Update the entity that was possessing - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); // We aren't mounted so remove the stun - GameMessages::SendSetStunned(entity->GetObjectID(), eStunState::POP, UNASSIGNED_SYSTEM_ADDRESS, LWOOBJID_EMPTY, true, false, true, false, false, false, false, true, true, true, true, true, true, true, true, true); + GameMessages::SendSetStunned(entity->GetObjectID(), eStateChangeType::POP, UNASSIGNED_SYSTEM_ADDRESS, LWOOBJID_EMPTY, true, false, true, false, false, false, false, true, true, true, true, true, true, true, true, true); } } } void GameMessages::HandleAcknowledgePossession(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); LWOOBJID objectId{}; inStream->Read(objectId); - auto* mount = EntityManager::Instance()->GetEntity(objectId); - if (mount) EntityManager::Instance()->SerializeEntity(mount); + auto* mount = Game::entityManager->GetEntity(objectId); + if (mount) Game::entityManager->SerializeEntity(mount); } //Racing @@ -3953,13 +4118,13 @@ void GameMessages::HandleRacingClientReady(RakNet::BitStream* inStream, Entity* inStream->Read(playerID); - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) { return; } - auto* racingControlComponent = dZoneManager::Instance()->GetZoneControlObject()->GetComponent(); + auto* racingControlComponent = Game::zoneManager->GetZoneControlObject()->GetComponent(); if (racingControlComponent == nullptr) { return; @@ -3976,7 +4141,7 @@ void GameMessages::HandleRequestDie(RakNet::BitStream* inStream, Entity* entity, float directionRelativeAngleXZ; float directionRelativeAngleY; float directionRelativeForce; - int32_t killType = VIOLENT; + eKillType killType = eKillType::VIOLENT; LWOOBJID killerID; LWOOBJID lootOwnerID = LWOOBJID_EMPTY; @@ -4007,7 +4172,7 @@ void GameMessages::HandleRequestDie(RakNet::BitStream* inStream, Entity* entity, inStream->Read(lootOwnerID); } - auto* zoneController = dZoneManager::Instance()->GetZoneControlObject(); + auto* zoneController = Game::zoneManager->GetZoneControlObject(); auto* racingControlComponent = zoneController->GetComponent(); @@ -4017,7 +4182,7 @@ void GameMessages::HandleRequestDie(RakNet::BitStream* inStream, Entity* entity, auto* possessableComponent = entity->GetComponent(); if (possessableComponent != nullptr) { - entity = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); + entity = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); if (entity == nullptr) { return; @@ -4044,13 +4209,13 @@ void GameMessages::HandleRacingPlayerInfoResetFinished(RakNet::BitStream* inStre inStream->Read(playerID); - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) { return; } - auto* zoneController = dZoneManager::Instance()->GetZoneControlObject(); + auto* zoneController = Game::zoneManager->GetZoneControlObject(); auto* racingControlComponent = zoneController->GetComponent(); @@ -4066,7 +4231,7 @@ void GameMessages::SendUpdateReputation(const LWOOBJID objectId, const int64_t r CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_UPDATE_REPUTATION); + bitStream.Write(eGameMessageType::UPDATE_REPUTATION); bitStream.Write(reputation); @@ -4080,7 +4245,7 @@ void GameMessages::HandleUpdatePropertyPerformanceCost(RakNet::BitStream* inStre if (performanceCost == 0.0f) return; - auto zone = dZoneManager::Instance()->GetZone(); + auto zone = Game::zoneManager->GetZone(); const auto& worldId = zone->GetZoneID(); const auto cloneId = worldId.GetCloneID(); const auto zoneId = worldId.GetMapID(); @@ -4108,7 +4273,7 @@ void GameMessages::HandleVehicleNotifyHitImaginationServer(RakNet::BitStream* in if (inStream->ReadBit()) inStream->Read(pickupSpawnerIndex); if (inStream->ReadBit()) inStream->Read(vehiclePosition); - auto* pickup = EntityManager::Instance()->GetEntity(pickupObjID); + auto* pickup = Game::entityManager->GetEntity(pickupObjID); if (pickup == nullptr) { return; @@ -4117,7 +4282,7 @@ void GameMessages::HandleVehicleNotifyHitImaginationServer(RakNet::BitStream* in auto* possessableComponent = entity->GetComponent(); if (possessableComponent != nullptr) { - entity = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); + entity = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); if (entity == nullptr) { return; @@ -4140,7 +4305,7 @@ void GameMessages::SendModuleAssemblyDBDataForClient(LWOOBJID objectId, LWOOBJID CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_MODULE_ASSEMBLY_DB_DATA_FOR_CLIENT); + bitStream.Write(eGameMessageType::MODULE_ASSEMBLY_DB_DATA_FOR_CLIENT); bitStream.Write(assemblyID); @@ -4159,7 +4324,7 @@ void GameMessages::SendNotifyVehicleOfRacingObject(LWOOBJID objectId, LWOOBJID r CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_VEHICLE_OF_RACING_OBJECT); + bitStream.Write(eGameMessageType::NOTIFY_VEHICLE_OF_RACING_OBJECT); bitStream.Write(racingObjectID != LWOOBJID_EMPTY); if (racingObjectID != LWOOBJID_EMPTY) bitStream.Write(racingObjectID); @@ -4174,7 +4339,7 @@ void GameMessages::SendRacingPlayerLoaded(LWOOBJID objectId, LWOOBJID playerID, CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_RACING_PLAYER_LOADED); + bitStream.Write(eGameMessageType::RACING_PLAYER_LOADED); bitStream.Write(playerID); bitStream.Write(vehicleID); @@ -4189,7 +4354,7 @@ void GameMessages::SendVehicleUnlockInput(LWOOBJID objectId, bool bLockWheels, c CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_VEHICLE_UNLOCK_INPUT); + bitStream.Write(eGameMessageType::VEHICLE_UNLOCK_INPUT); bitStream.Write(bLockWheels); @@ -4203,7 +4368,7 @@ void GameMessages::SendVehicleSetWheelLockState(LWOOBJID objectId, bool bExtraFr CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_VEHICLE_SET_WHEEL_LOCK_STATE); + bitStream.Write(eGameMessageType::VEHICLE_SET_WHEEL_LOCK_STATE); bitStream.Write(bExtraFriction); bitStream.Write(bLocked); @@ -4218,7 +4383,7 @@ void GameMessages::SendRacingSetPlayerResetInfo(LWOOBJID objectId, int32_t curre CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_RACING_SET_PLAYER_RESET_INFO); + bitStream.Write(eGameMessageType::RACING_SET_PLAYER_RESET_INFO); bitStream.Write(currentLap); bitStream.Write(furthestResetPlane); @@ -4236,7 +4401,7 @@ void GameMessages::SendRacingResetPlayerToLastReset(LWOOBJID objectId, LWOOBJID CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_RACING_RESET_PLAYER_TO_LAST_RESET); + bitStream.Write(eGameMessageType::RACING_RESET_PLAYER_TO_LAST_RESET); bitStream.Write(playerID); @@ -4244,13 +4409,43 @@ void GameMessages::SendRacingResetPlayerToLastReset(LWOOBJID objectId, LWOOBJID SEND_PACKET; } +void GameMessages::SendVehicleStopBoost(Entity* targetEntity, const SystemAddress& playerSysAddr, bool affectPassive) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(targetEntity->GetObjectID()); + bitStream.Write(eGameMessageType::VEHICLE_STOP_BOOST); + + bitStream.Write(affectPassive); + + SEND_PACKET_BROADCAST; +} + +void GameMessages::SendSetResurrectRestoreValues(Entity* targetEntity, int32_t armorRestore, int32_t healthRestore, int32_t imaginationRestore) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(targetEntity->GetObjectID()); + bitStream.Write(eGameMessageType::SET_RESURRECT_RESTORE_VALUES); + + bitStream.Write(armorRestore != -1); + if (armorRestore != -1) bitStream.Write(armorRestore); + + bitStream.Write(healthRestore != -1); + if (healthRestore != -1) bitStream.Write(healthRestore); + + bitStream.Write(imaginationRestore != -1); + if (imaginationRestore != -1) bitStream.Write(imaginationRestore); + + SEND_PACKET_BROADCAST; +} void GameMessages::SendNotifyRacingClient(LWOOBJID objectId, int32_t eventType, int32_t param1, LWOOBJID paramObj, std::u16string paramStr, LWOOBJID singleClient, const SystemAddress& sysAddr) { CBITSTREAM; CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_NOTIFY_RACING_CLIENT); + bitStream.Write(eGameMessageType::NOTIFY_RACING_CLIENT); bitStream.Write(eventType != 0); if (eventType != 0) bitStream.Write(eventType); @@ -4276,7 +4471,7 @@ void GameMessages::SendActivityEnter(LWOOBJID objectId, const SystemAddress& sys CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ACTIVITY_ENTER); + bitStream.Write(eGameMessageType::ACTIVITY_ENTER); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -4288,7 +4483,7 @@ void GameMessages::SendActivityStart(LWOOBJID objectId, const SystemAddress& sys CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ACTIVITY_START); + bitStream.Write(eGameMessageType::ACTIVITY_START); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -4300,7 +4495,7 @@ void GameMessages::SendActivityExit(LWOOBJID objectId, const SystemAddress& sysA CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ACTIVITY_EXIT); + bitStream.Write(eGameMessageType::ACTIVITY_EXIT); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -4312,7 +4507,7 @@ void GameMessages::SendActivityStop(LWOOBJID objectId, bool bExit, bool bUserCan CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_ACTIVITY_STOP); + bitStream.Write(eGameMessageType::ACTIVITY_STOP); bitStream.Write(bExit); bitStream.Write(bUserCancel); @@ -4327,7 +4522,7 @@ void GameMessages::SendVehicleAddPassiveBoostAction(LWOOBJID objectId, const Sys CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_VEHICLE_ADD_PASSIVE_BOOST_ACTION); + bitStream.Write(eGameMessageType::VEHICLE_ADD_PASSIVE_BOOST_ACTION); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -4339,7 +4534,7 @@ void GameMessages::SendVehicleRemovePassiveBoostAction(LWOOBJID objectId, const CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_VEHICLE_REMOVE_PASSIVE_BOOST_ACTION); + bitStream.Write(eGameMessageType::VEHICLE_REMOVE_PASSIVE_BOOST_ACTION); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -4351,7 +4546,7 @@ void GameMessages::SendVehicleNotifyFinishedRace(LWOOBJID objectId, const System CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_VEHICLE_NOTIFY_FINISHED_RACE); + bitStream.Write(eGameMessageType::VEHICLE_NOTIFY_FINISHED_RACE); if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; SEND_PACKET; @@ -4365,7 +4560,7 @@ void GameMessages::SendAddBuff(LWOOBJID& objectID, const LWOOBJID& casterID, uin CMSGHEADER; bitStream.Write(objectID); - bitStream.Write(GAME_MSG::GAME_MSG_ADD_BUFF); + bitStream.Write(eGameMessageType::ADD_BUFF); bitStream.Write(false); // Added by teammate bitStream.Write(false); // Apply on teammates @@ -4397,13 +4592,13 @@ void GameMessages::SendAddBuff(LWOOBJID& objectID, const LWOOBJID& casterID, uin // NT void GameMessages::HandleRequestMoveItemBetweenInventoryTypes(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - bool bAllowPartial; + bool bAllowPartial{}; int32_t destSlot = -1; int32_t iStackCount = 1; eInventoryType invTypeDst = ITEMS; eInventoryType invTypeSrc = ITEMS; LWOOBJID itemID = LWOOBJID_EMPTY; - bool showFlyingLoot; + bool showFlyingLoot{}; LWOOBJID subkey = LWOOBJID_EMPTY; LOT itemLOT = 0; @@ -4427,12 +4622,12 @@ void GameMessages::HandleRequestMoveItemBetweenInventoryTypes(RakNet::BitStream* if (itemID != LWOOBJID_EMPTY) { auto* item = inventoryComponent->FindItemById(itemID); - if (item == nullptr) { - return; - } + if (!item) return; - if (inventoryComponent->IsPet(item->GetSubKey()) || !item->GetConfig().empty()) { - return; + // Despawn the pet if we are moving that pet to the vault. + auto* petComponent = PetComponent::GetActivePet(entity->GetObjectID()); + if (petComponent && petComponent->GetDatabaseId() == item->GetSubKey()) { + inventoryComponent->DespawnPet(); } inventoryComponent->MoveItemToInventory(item, invTypeDst, iStackCount, showFlyingLoot, false, false, destSlot); @@ -4446,7 +4641,7 @@ void GameMessages::SendShowActivityCountdown(LWOOBJID objectId, bool bPlayAdditi CMSGHEADER; bitStream.Write(objectId); - bitStream.Write(GAME_MSG::GAME_MSG_SHOW_ACTIVITY_COUNTDOWN); + bitStream.Write(eGameMessageType::SHOW_ACTIVITY_COUNTDOWN); bitStream.Write(bPlayAdditionalSound); @@ -4478,7 +4673,7 @@ void GameMessages::HandleToggleGhostReferenceOverride(RakNet::BitStream* inStrea if (player != nullptr) { player->SetGhostOverride(bOverride); - EntityManager::Instance()->UpdateGhosting(player); + Game::entityManager->UpdateGhosting(player); } } @@ -4493,7 +4688,7 @@ void GameMessages::HandleSetGhostReferencePosition(RakNet::BitStream* inStream, if (player != nullptr) { player->SetGhostOverridePoint(position); - EntityManager::Instance()->UpdateGhosting(player); + Game::entityManager->UpdateGhosting(player); } } @@ -4511,10 +4706,10 @@ void GameMessages::HandleBuyFromVendor(RakNet::BitStream* inStream, Entity* enti User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* player = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!player) return; - auto* propertyVendorComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_PROPERTY_VENDOR)); + auto* propertyVendorComponent = static_cast(entity->GetComponent(eReplicaComponentType::PROPERTY_VENDOR)); if (propertyVendorComponent != nullptr) { propertyVendorComponent->OnBuyFromVendor(player, bConfirmed, item, count); @@ -4524,16 +4719,21 @@ void GameMessages::HandleBuyFromVendor(RakNet::BitStream* inStream, Entity* enti const auto isCommendationVendor = entity->GetLOT() == 13806; - VendorComponent* vend = static_cast(entity->GetComponent(COMPONENT_TYPE_VENDOR)); + auto* vend = entity->GetComponent(); if (!vend && !isCommendationVendor) return; - InventoryComponent* inv = static_cast(player->GetComponent(COMPONENT_TYPE_INVENTORY)); + auto* inv = player->GetComponent(); if (!inv) return; - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - CDItemComponentTable* itemComponentTable = CDClientManager::Instance()->GetTable("ItemComponent"); + if (!isCommendationVendor && !vend->SellsItem(item)) { + Game::logger->Log("GameMessages", "User %llu %s tried to buy an item %i from a vendor when they do not sell said item", player->GetObjectID(), user->GetUsername().c_str(), item); + return; + } - int itemCompID = compRegistryTable->GetByIDAndType(item, COMPONENT_TYPE_ITEM); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + CDItemComponentTable* itemComponentTable = CDClientManager::Instance().GetTable(); + + int itemCompID = compRegistryTable->GetByIDAndType(item, eReplicaComponentType::ITEM); CDItemComponent itemComp = itemComponentTable->GetItemComponentByID(itemCompID); Character* character = player->GetCharacter(); @@ -4558,25 +4758,10 @@ void GameMessages::HandleBuyFromVendor(RakNet::BitStream* inStream, Entity* enti LOT tokenId = -1; - if (missionComponent->GetMissionState(545) == MissionState::MISSION_STATE_COMPLETE) // "Join Assembly!" - { - tokenId = 8318; // "Assembly Token" - } - - if (missionComponent->GetMissionState(556) == MissionState::MISSION_STATE_COMPLETE) // "Join Venture League!" - { - tokenId = 8321; // "Venture League Token" - } - - if (missionComponent->GetMissionState(567) == MissionState::MISSION_STATE_COMPLETE) // "Join The Sentinels!" - { - tokenId = 8319; // "Sentinels Token" - } - - if (missionComponent->GetMissionState(578) == MissionState::MISSION_STATE_COMPLETE) // "Join Paradox!" - { - tokenId = 8320; // "Paradox Token" - } + if (missionComponent->GetMissionState(545) == eMissionState::COMPLETE) tokenId = 8318; // "Assembly Token" + if (missionComponent->GetMissionState(556) == eMissionState::COMPLETE) tokenId = 8321; // "Venture League Token" + if (missionComponent->GetMissionState(567) == eMissionState::COMPLETE) tokenId = 8319; // "Sentinels Token" + if (missionComponent->GetMissionState(578) == eMissionState::COMPLETE) tokenId = 8320; // "Paradox Token" const uint32_t altCurrencyCost = itemComp.commendationCost * count; @@ -4586,7 +4771,7 @@ void GameMessages::HandleBuyFromVendor(RakNet::BitStream* inStream, Entity* enti inv->RemoveItem(tokenId, altCurrencyCost); - inv->AddItem(item, count, eLootSourceType::LOOT_SOURCE_VENDOR); + inv->AddItem(item, count, eLootSourceType::VENDOR); } else { float buyScalar = vend->GetBuyScalar(); @@ -4606,8 +4791,8 @@ void GameMessages::HandleBuyFromVendor(RakNet::BitStream* inStream, Entity* enti inv->RemoveItem(itemComp.currencyLOT, altCurrencyCost); } - character->SetCoins(character->GetCoins() - (coinCost), eLootSourceType::LOOT_SOURCE_VENDOR); - inv->AddItem(item, count, eLootSourceType::LOOT_SOURCE_VENDOR); + character->SetCoins(character->GetCoins() - (coinCost), eLootSourceType::VENDOR); + inv->AddItem(item, count, eLootSourceType::VENDOR); } GameMessages::SendVendorTransactionResult(entity, sysAddr); @@ -4624,23 +4809,23 @@ void GameMessages::HandleSellToVendor(RakNet::BitStream* inStream, Entity* entit User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* player = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!player) return; Character* character = player->GetCharacter(); if (!character) return; - InventoryComponent* inv = static_cast(player->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(player->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; - VendorComponent* vend = static_cast(entity->GetComponent(COMPONENT_TYPE_VENDOR)); + VendorComponent* vend = static_cast(entity->GetComponent(eReplicaComponentType::VENDOR)); if (!vend) return; Item* item = inv->FindItemById(iObjID); if (!item) return; - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - CDItemComponentTable* itemComponentTable = CDClientManager::Instance()->GetTable("ItemComponent"); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + CDItemComponentTable* itemComponentTable = CDClientManager::Instance().GetTable(); - int itemCompID = compRegistryTable->GetByIDAndType(item->GetLot(), COMPONENT_TYPE_ITEM); + int itemCompID = compRegistryTable->GetByIDAndType(item->GetLot(), eReplicaComponentType::ITEM); CDItemComponent itemComp = itemComponentTable->GetItemComponentByID(itemCompID); // Items with a base value of 0 or max int are special items that should not be sold if they're not sub items @@ -4648,14 +4833,14 @@ void GameMessages::HandleSellToVendor(RakNet::BitStream* inStream, Entity* entit float sellScalar = vend->GetSellScalar(); if (Inventory::IsValidItem(itemComp.currencyLOT)) { - const auto altCurrency = (itemComp.altCurrencyCost * sellScalar) * count; - inv->AddItem(itemComp.currencyLOT, std::floor(altCurrency), eLootSourceType::LOOT_SOURCE_VENDOR); // Return alt currencies like faction tokens. + const auto altCurrency = static_cast(itemComp.altCurrencyCost * sellScalar) * count; + inv->AddItem(itemComp.currencyLOT, std::floor(altCurrency), eLootSourceType::VENDOR); // Return alt currencies like faction tokens. } //inv->RemoveItem(count, -1, iObjID); inv->MoveItemToInventory(item, eInventoryType::VENDOR_BUYBACK, count, true, false, true); - character->SetCoins(std::floor(character->GetCoins() + ((itemComp.baseValue * sellScalar) * count)), eLootSourceType::LOOT_SOURCE_VENDOR); - //EntityManager::Instance()->SerializeEntity(player); // so inventory updates + character->SetCoins(std::floor(character->GetCoins() + (static_cast(itemComp.baseValue * sellScalar) * count)), eLootSourceType::VENDOR); + //Game::entityManager->SerializeEntity(player); // so inventory updates GameMessages::SendVendorTransactionResult(entity, sysAddr); } @@ -4674,23 +4859,23 @@ void GameMessages::HandleBuybackFromVendor(RakNet::BitStream* inStream, Entity* User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* player = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!player) return; Character* character = player->GetCharacter(); if (!character) return; - InventoryComponent* inv = static_cast(player->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(player->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; - VendorComponent* vend = static_cast(entity->GetComponent(COMPONENT_TYPE_VENDOR)); + VendorComponent* vend = static_cast(entity->GetComponent(eReplicaComponentType::VENDOR)); if (!vend) return; Item* item = inv->FindItemById(iObjID); if (!item) return; - CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - CDItemComponentTable* itemComponentTable = CDClientManager::Instance()->GetTable("ItemComponent"); + CDComponentsRegistryTable* compRegistryTable = CDClientManager::Instance().GetTable(); + CDItemComponentTable* itemComponentTable = CDClientManager::Instance().GetTable(); - int itemCompID = compRegistryTable->GetByIDAndType(item->GetLot(), COMPONENT_TYPE_ITEM); + int itemCompID = compRegistryTable->GetByIDAndType(item->GetLot(), eReplicaComponentType::ITEM); CDItemComponent itemComp = itemComponentTable->GetItemComponentByID(itemCompID); float sellScalar = vend->GetSellScalar(); @@ -4713,8 +4898,8 @@ void GameMessages::HandleBuybackFromVendor(RakNet::BitStream* inStream, Entity* //inv->RemoveItem(count, -1, iObjID); inv->MoveItemToInventory(item, Inventory::FindInventoryTypeForLot(item->GetLot()), count, true, false); - character->SetCoins(character->GetCoins() - cost, eLootSourceType::LOOT_SOURCE_VENDOR); - //EntityManager::Instance()->SerializeEntity(player); // so inventory updates + character->SetCoins(character->GetCoins() - cost, eLootSourceType::VENDOR); + //Game::entityManager->SerializeEntity(player); // so inventory updates GameMessages::SendVendorTransactionResult(entity, sysAddr); } @@ -4761,7 +4946,7 @@ void GameMessages::HandleFireEventServerSide(RakNet::BitStream* inStream, Entity if (param3IsDefault) inStream->Read(param3); inStream->Read(senderID); - auto* sender = EntityManager::Instance()->GetEntity(senderID); + auto* sender = Game::entityManager->GetEntity(senderID); auto* player = Player::GetPlayer(sysAddr); if (!player) { @@ -4792,7 +4977,7 @@ void GameMessages::HandleFireEventServerSide(RakNet::BitStream* inStream, Entity } if (mapId == 0) { - mapId = dZoneManager::Instance()->GetZoneID().GetMapID(); // Fallback to sending the player back to the same zone. + mapId = Game::zoneManager->GetZoneID().GetMapID(); // Fallback to sending the player back to the same zone. } Game::logger->Log("FireEventServerSide", "Player %llu has requested zone transfer to (%i, %i).", sender->GetObjectID(), (int)mapId, (int)cloneId); @@ -4832,10 +5017,10 @@ void GameMessages::HandleRebuildCancel(RakNet::BitStream* inStream, Entity* enti inStream->Read(bEarlyRelease); inStream->Read(userID); - RebuildComponent* rebComp = static_cast(entity->GetComponent(COMPONENT_TYPE_REBUILD)); + RebuildComponent* rebComp = static_cast(entity->GetComponent(eReplicaComponentType::QUICK_BUILD)); if (!rebComp) return; - rebComp->CancelRebuild(EntityManager::Instance()->GetEntity(userID), eFailReason::REASON_CANCELED_EARLY); + rebComp->CancelRebuild(Game::entityManager->GetEntity(userID), eQuickBuildFailReason::CANCELED_EARLY); } void GameMessages::HandleRequestUse(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { @@ -4851,7 +5036,7 @@ void GameMessages::HandleRequestUse(RakNet::BitStream* inStream, Entity* entity, inStream->Read(objectID); inStream->Read(secondary); - Entity* interactedObject = EntityManager::Instance()->GetEntity(objectID); + Entity* interactedObject = Game::entityManager->GetEntity(objectID); if (interactedObject == nullptr) { Game::logger->Log("GameMessages", "Object %llu tried to interact, but doesn't exist!", objectID); @@ -4865,7 +5050,7 @@ void GameMessages::HandleRequestUse(RakNet::BitStream* inStream, Entity* entity, if (bIsMultiInteractUse) { if (multiInteractType == 0) { - auto* missionOfferComponent = static_cast(interactedObject->GetComponent(COMPONENT_TYPE_MISSION_OFFER)); + auto* missionOfferComponent = static_cast(interactedObject->GetComponent(eReplicaComponentType::MISSION_OFFER)); if (missionOfferComponent != nullptr) { missionOfferComponent->OfferMissions(entity, multiInteractID); @@ -4878,12 +5063,12 @@ void GameMessages::HandleRequestUse(RakNet::BitStream* inStream, Entity* entity, } //Perform use task if possible: - auto missionComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + auto missionComponent = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (missionComponent == nullptr) return; - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MISSION_INTERACTION, interactedObject->GetLOT(), interactedObject->GetObjectID()); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_NON_MISSION_INTERACTION, interactedObject->GetLOT(), interactedObject->GetObjectID()); + missionComponent->Progress(eMissionTaskType::TALK_TO_NPC, interactedObject->GetLOT(), interactedObject->GetObjectID()); + missionComponent->Progress(eMissionTaskType::INTERACT, interactedObject->GetLOT(), interactedObject->GetObjectID()); } void GameMessages::HandlePlayEmote(RakNet::BitStream* inStream, Entity* entity) { @@ -4893,28 +5078,28 @@ void GameMessages::HandlePlayEmote(RakNet::BitStream* inStream, Entity* entity) inStream->Read(emoteID); inStream->Read(targetID); - Game::logger->Log("GameMessages", "Emote (%i) (%llu)", emoteID, targetID); + Game::logger->LogDebug("GameMessages", "Emote (%i) (%llu)", emoteID, targetID); //TODO: If targetID != 0, and we have one of the "perform emote" missions, complete them. if (emoteID == 0) return; std::string sAnimationName = "deaded"; //Default name in case we fail to get the emote - MissionComponent* mission = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); - if (mission) { - mission->Progress(MissionTaskType::MISSION_TASK_TYPE_EMOTE, emoteID, targetID); - } + MissionComponent* missionComponent = entity->GetComponent(); + if (!missionComponent) return; if (targetID != LWOOBJID_EMPTY) { - auto* targetEntity = EntityManager::Instance()->GetEntity(targetID); + auto* targetEntity = Game::entityManager->GetEntity(targetID); - Game::logger->Log("GameMessages", "Emote target found (%d)", targetEntity != nullptr); + Game::logger->LogDebug("GameMessages", "Emote target found (%d)", targetEntity != nullptr); if (targetEntity != nullptr) { targetEntity->OnEmoteReceived(emoteID, entity); + missionComponent->Progress(eMissionTaskType::EMOTE, emoteID, targetID); } } else { - const auto scriptedEntities = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPT); + Game::logger->LogDebug("GameMessages", "Target ID is empty, using backup"); + const auto scriptedEntities = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPT); const auto& referencePoint = entity->GetPosition(); @@ -4922,16 +5107,17 @@ void GameMessages::HandlePlayEmote(RakNet::BitStream* inStream, Entity* entity) if (Vector3::DistanceSquared(scripted->GetPosition(), referencePoint) > 5.0f * 5.0f) continue; scripted->OnEmoteReceived(emoteID, entity); + missionComponent->Progress(eMissionTaskType::EMOTE, emoteID, scripted->GetObjectID()); } } - CDEmoteTableTable* emotes = CDClientManager::Instance()->GetTable("EmoteTable"); + CDEmoteTableTable* emotes = CDClientManager::Instance().GetTable(); if (emotes) { CDEmoteTable* emote = emotes->GetEmote(emoteID); if (emote) sAnimationName = emote->animationName; } - GameMessages::SendPlayAnimation(entity, GeneralUtils::ASCIIToUTF16(sAnimationName)); + RenderComponent::PlayAnimation(entity, sAnimationName); } void GameMessages::HandleModularBuildConvertModel(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { @@ -4941,9 +5127,9 @@ void GameMessages::HandleModularBuildConvertModel(RakNet::BitStream* inStream, E User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* character = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* character = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!character) return; - InventoryComponent* inv = static_cast(character->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(character->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; auto* item = inv->FindItemById(modelID); @@ -4954,12 +5140,12 @@ void GameMessages::HandleModularBuildConvertModel(RakNet::BitStream* inStream, E item->Disassemble(TEMP_MODELS); - item->SetCount(item->GetCount() - 1, false, false, true, eLootSourceType::LOOT_SOURCE_QUICKBUILD); + item->SetCount(item->GetCount() - 1, false, false, true, eLootSourceType::QUICKBUILD); } void GameMessages::HandleSetFlag(RakNet::BitStream* inStream, Entity* entity) { bool bFlag{}; - int iFlagID{}; + int32_t iFlagID{}; inStream->Read(bFlag); inStream->Read(iFlagID); @@ -4986,7 +5172,7 @@ void GameMessages::HandleRespondToMission(RakNet::BitStream* inStream, Entity* e inStream->Read(isDefaultReward); if (isDefaultReward) inStream->Read(reward); - MissionComponent* missionComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + MissionComponent* missionComponent = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (!missionComponent) { Game::logger->Log("GameMessages", "Unable to get mission component for entity %llu to handle RespondToMission", playerID); return; @@ -4999,7 +5185,7 @@ void GameMessages::HandleRespondToMission(RakNet::BitStream* inStream, Entity* e Game::logger->Log("GameMessages", "Unable to get mission %i for entity %llu to update reward in RespondToMission", missionID, playerID); } - Entity* offerer = EntityManager::Instance()->GetEntity(receiverID); + Entity* offerer = Game::entityManager->GetEntity(receiverID); if (offerer == nullptr) { Game::logger->Log("GameMessages", "Unable to get receiver entity %llu for RespondToMission", receiverID); @@ -5007,13 +5193,13 @@ void GameMessages::HandleRespondToMission(RakNet::BitStream* inStream, Entity* e } for (CppScripts::Script* script : CppScripts::GetEntityScripts(offerer)) { - script->OnRespondToMission(offerer, missionID, EntityManager::Instance()->GetEntity(playerID), reward); + script->OnRespondToMission(offerer, missionID, Game::entityManager->GetEntity(playerID), reward); } } void GameMessages::HandleMissionDialogOK(RakNet::BitStream* inStream, Entity* entity) { bool bIsComplete{}; - MissionState iMissionState{}; + eMissionState iMissionState{}; int missionID{}; LWOOBJID responder{}; Entity* player = nullptr; @@ -5022,22 +5208,22 @@ void GameMessages::HandleMissionDialogOK(RakNet::BitStream* inStream, Entity* en inStream->Read(iMissionState); inStream->Read(missionID); inStream->Read(responder); - player = EntityManager::Instance()->GetEntity(responder); + player = Game::entityManager->GetEntity(responder); for (CppScripts::Script* script : CppScripts::GetEntityScripts(entity)) { script->OnMissionDialogueOK(entity, player, missionID, iMissionState); } // Get the player's mission component - MissionComponent* missionComponent = static_cast(player->GetComponent(COMPONENT_TYPE_MISSION)); + MissionComponent* missionComponent = static_cast(player->GetComponent(eReplicaComponentType::MISSION)); if (!missionComponent) { Game::logger->Log("GameMessages", "Unable to get mission component for entity %llu to handle MissionDialogueOK", player->GetObjectID()); return; } - if (iMissionState == MissionState::MISSION_STATE_AVAILABLE || iMissionState == MissionState::MISSION_STATE_COMPLETE_AVAILABLE) { + if (iMissionState == eMissionState::AVAILABLE || iMissionState == eMissionState::COMPLETE_AVAILABLE) { missionComponent->AcceptMission(missionID); - } else if (iMissionState == MissionState::MISSION_STATE_READY_TO_COMPLETE || iMissionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { + } else if (iMissionState == eMissionState::READY_TO_COMPLETE || iMissionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { missionComponent->CompleteMission(missionID); } } @@ -5051,9 +5237,9 @@ void GameMessages::HandleRequestLinkedMission(RakNet::BitStream* inStream, Entit inStream->Read(missionId); inStream->Read(bMissionOffered); - auto* player = EntityManager::Instance()->GetEntity(playerId); + auto* player = Game::entityManager->GetEntity(playerId); - auto* missionOfferComponent = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION_OFFER)); + auto* missionOfferComponent = static_cast(entity->GetComponent(eReplicaComponentType::MISSION_OFFER)); if (missionOfferComponent != nullptr) { missionOfferComponent->OfferMissions(player, 0); @@ -5064,12 +5250,12 @@ void GameMessages::HandleHasBeenCollected(RakNet::BitStream* inStream, Entity* e LWOOBJID playerID; inStream->Read(playerID); - Entity* player = EntityManager::Instance()->GetEntity(playerID); + Entity* player = Game::entityManager->GetEntity(playerID); if (!player || !entity || entity->GetCollectibleID() == 0) return; - MissionComponent* missionComponent = static_cast(player->GetComponent(COMPONENT_TYPE_MISSION)); + MissionComponent* missionComponent = static_cast(player->GetComponent(eReplicaComponentType::MISSION)); if (missionComponent) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ENVIRONMENT, entity->GetLOT(), entity->GetObjectID()); + missionComponent->Progress(eMissionTaskType::COLLECTION, entity->GetLOT(), entity->GetObjectID()); } } @@ -5120,7 +5306,7 @@ void GameMessages::HandlePickupCurrency(RakNet::BitStream* inStream, Entity* ent auto* ch = entity->GetCharacter(); if (entity->CanPickupCoins(currency)) { - ch->SetCoins(ch->GetCoins() + currency, eLootSourceType::LOOT_SOURCE_PICKUP); + ch->SetCoins(ch->GetCoins() + currency, eLootSourceType::PICKUP); } } @@ -5170,7 +5356,7 @@ void GameMessages::HandleEquipItem(RakNet::BitStream* inStream, Entity* entity) inStream->Read(immediate); //twice? inStream->Read(objectID); - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; Item* item = inv->FindItemById(objectID); @@ -5178,7 +5364,7 @@ void GameMessages::HandleEquipItem(RakNet::BitStream* inStream, Entity* entity) item->Equip(); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } void GameMessages::HandleUnequipItem(RakNet::BitStream* inStream, Entity* entity) { @@ -5189,7 +5375,7 @@ void GameMessages::HandleUnequipItem(RakNet::BitStream* inStream, Entity* entity inStream->Read(immediate); inStream->Read(objectID); - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; auto* item = inv->FindItemById(objectID); @@ -5198,7 +5384,7 @@ void GameMessages::HandleUnequipItem(RakNet::BitStream* inStream, Entity* entity item->UnEquip(); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } void GameMessages::HandleRemoveItemFromInventory(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { @@ -5264,7 +5450,7 @@ void GameMessages::HandleRemoveItemFromInventory(RakNet::BitStream* inStream, En inStream->Read(iTradeIDIsDefault); if (iTradeIDIsDefault) inStream->Read(iTradeID); - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; auto* item = inv->FindItemById(iObjID); @@ -5283,12 +5469,12 @@ void GameMessages::HandleRemoveItemFromInventory(RakNet::BitStream* inStream, En } item->SetCount(item->GetCount() - iStackCount, true); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); auto* missionComponent = entity->GetComponent(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, item->GetLot(), LWOOBJID_EMPTY, "", -iStackCount); + missionComponent->Progress(eMissionTaskType::GATHER, item->GetLot(), LWOOBJID_EMPTY, "", -iStackCount); } } } @@ -5307,7 +5493,7 @@ void GameMessages::HandleMoveItemInInventory(RakNet::BitStream* inStream, Entity inStream->Read(responseCode); inStream->Read(slot); - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; auto* item = inv->FindItemById(iObjID); @@ -5317,7 +5503,7 @@ void GameMessages::HandleMoveItemInInventory(RakNet::BitStream* inStream, Entity } inv->MoveStack(item, static_cast(destInvType), slot); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } void GameMessages::HandleMoveItemBetweenInventoryTypes(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { @@ -5363,7 +5549,7 @@ void GameMessages::HandleMoveItemBetweenInventoryTypes(RakNet::BitStream* inStre } inv->MoveItemToInventory(item, inventoryTypeB, stackCount, showFlyingLoot); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } void GameMessages::HandleBuildModeSet(RakNet::BitStream* inStream, Entity* entity) { @@ -5382,9 +5568,9 @@ void GameMessages::HandleBuildModeSet(RakNet::BitStream* inStream, Entity* entit void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* character = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* character = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!character) return; - InventoryComponent* inv = static_cast(character->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(character->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; Game::logger->Log("GameMessages", "Build finished"); @@ -5393,7 +5579,7 @@ void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* GameMessages::SendModularBuildEnd(character); // i dont know if this does anything but DLUv2 did it //inv->UnequipItem(inv->GetItemStackByLOT(6086, eInventoryType::ITEMS)); // take off the thinking cap - //EntityManager::Instance()->SerializeEntity(entity); + //Game::entityManager->SerializeEntity(entity); uint8_t count; // 3 for rockets, 7 for cars @@ -5401,7 +5587,8 @@ void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* auto* temp = inv->GetInventory(TEMP_MODELS); std::vector modList; - + auto& oldPartList = character->GetVar(u"currentModifiedBuild"); + bool everyPieceSwapped = !oldPartList.empty(); // If the player didn't put a build in initially, then they should not get this achievement. if (count >= 3) { std::u16string modules; @@ -5409,7 +5596,8 @@ void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* uint32_t mod; inStream->Read(mod); modList.push_back(mod); - modules += u"1:" + (GeneralUtils::to_u16string(mod)); + auto modToStr = GeneralUtils::to_u16string(mod); + modules += u"1:" + (modToStr); if (k + 1 != count) modules += u"+"; if (temp->GetLotCount(mod) > 0) { @@ -5417,6 +5605,13 @@ void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* } else { inv->RemoveItem(mod, 1); } + + // Doing this check for 1 singular mission that needs to know when you've swapped every part out during a car modular build. + // since all 8129's are the same, skip checking that + if (mod != 8129) { + if (oldPartList.find(GeneralUtils::UTF16ToWTF8(modToStr)) != std::string::npos) everyPieceSwapped = false; + + } } const auto moduleAssembly = new LDFData(u"assemblyPartLOTs", modules); @@ -5425,21 +5620,22 @@ void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* config.push_back(moduleAssembly); if (count == 3) { - inv->AddItem(6416, 1, eLootSourceType::LOOT_SOURCE_QUICKBUILD, eInventoryType::MODELS, config); + inv->AddItem(6416, 1, eLootSourceType::QUICKBUILD, eInventoryType::MODELS, config); } else if (count == 7) { - inv->AddItem(8092, 1, eLootSourceType::LOOT_SOURCE_QUICKBUILD, eInventoryType::MODELS, config); + inv->AddItem(8092, 1, eLootSourceType::QUICKBUILD, eInventoryType::MODELS, config); } auto* missionComponent = character->GetComponent(); if (entity->GetLOT() != 9980 || Game::server->GetZoneID() != 1200) { if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, entity->GetLOT(), entity->GetObjectID()); + missionComponent->Progress(eMissionTaskType::SCRIPT, entity->GetLOT(), entity->GetObjectID()); + if (count >= 7 && everyPieceSwapped) missionComponent->Progress(eMissionTaskType::RACING, LWOOBJID_EMPTY, (LWOOBJID)eRacingTaskParam::MODULAR_BUILDING); } } } - ScriptComponent* script = static_cast(entity->GetComponent(COMPONENT_TYPE_SCRIPT)); + ScriptComponent* script = static_cast(entity->GetComponent(eReplicaComponentType::SCRIPT)); for (CppScripts::Script* script : CppScripts::GetEntityScripts(entity)) { script->OnModularBuildExit(entity, character, count >= 3, modList); @@ -5460,9 +5656,9 @@ void GameMessages::HandleModularBuildFinish(RakNet::BitStream* inStream, Entity* void GameMessages::HandleDoneArrangingWithItem(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* character = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* character = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!character) return; - InventoryComponent* inv = static_cast(character->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(character->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; /** @@ -5513,9 +5709,9 @@ void GameMessages::HandleDoneArrangingWithItem(RakNet::BitStream* inStream, Enti */ if (PropertyManagementComponent::Instance() != nullptr) { - const auto& buildAreas = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_BUILD_BORDER); + const auto& buildAreas = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::BUILD_BORDER); - const auto& entities = EntityManager::Instance()->GetEntitiesInGroup("PropertyPlaque"); + const auto& entities = Game::entityManager->GetEntitiesInGroup("PropertyPlaque"); Entity* buildArea; @@ -5571,7 +5767,7 @@ void GameMessages::HandleDoneArrangingWithItem(RakNet::BitStream* inStream, Enti void GameMessages::HandleModularBuildMoveAndEquip(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { User* user = UserManager::Instance()->GetUser(sysAddr); if (!user) return; - Entity* character = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + Entity* character = Game::entityManager->GetEntity(user->GetLoggedInChar()); if (!character) return; Game::logger->Log("GameMessages", "Build and move"); @@ -5580,7 +5776,7 @@ void GameMessages::HandleModularBuildMoveAndEquip(RakNet::BitStream* inStream, E inStream->Read(templateID); - InventoryComponent* inv = static_cast(character->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(character->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; auto* item = inv->FindItemByLot(templateID, TEMP_MODELS); @@ -5604,7 +5800,7 @@ void GameMessages::HandlePickupItem(RakNet::BitStream* inStream, Entity* entity) if (team != nullptr) { for (const auto memberId : team->members) { - auto* member = EntityManager::Instance()->GetEntity(memberId); + auto* member = Game::entityManager->GetEntity(memberId); if (member == nullptr || memberId == playerID) continue; @@ -5616,12 +5812,12 @@ void GameMessages::HandlePickupItem(RakNet::BitStream* inStream, Entity* entity) void GameMessages::HandleResurrect(RakNet::BitStream* inStream, Entity* entity) { bool immediate = inStream->ReadBit(); - Entity* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); + Entity* zoneControl = Game::entityManager->GetZoneControlEntity(); for (CppScripts::Script* script : CppScripts::GetEntityScripts(zoneControl)) { script->OnPlayerResurrected(zoneControl, entity); } - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPTED_ACTIVITY); for (Entity* scriptEntity : scriptedActs) { if (scriptEntity->GetObjectID() != zoneControl->GetObjectID()) { // Don't want to trigger twice on instance worlds for (CppScripts::Script* script : CppScripts::GetEntityScripts(scriptEntity)) { @@ -5632,16 +5828,16 @@ void GameMessages::HandleResurrect(RakNet::BitStream* inStream, Entity* entity) } void GameMessages::HandlePushEquippedItemsState(RakNet::BitStream* inStream, Entity* entity) { - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; inv->PushEquippedItems(); } void GameMessages::HandlePopEquippedItemsState(RakNet::BitStream* inStream, Entity* entity) { - InventoryComponent* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; inv->PopEquippedItems(); - EntityManager::Instance()->SerializeEntity(entity); // so it updates on client side + Game::entityManager->SerializeEntity(entity); // so it updates on client side } @@ -5650,23 +5846,23 @@ void GameMessages::HandleClientItemConsumed(RakNet::BitStream* inStream, Entity* inStream->Read(itemConsumed); - auto* inventory = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + auto* inventory = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (inventory == nullptr) { return; } auto* item = inventory->FindItemById(itemConsumed); - if (item == nullptr) { return; } + LOT itemLot = item->GetLot(); item->Consume(); - auto* missions = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + auto* missions = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (missions != nullptr) { - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_FOOD, item->GetLot()); + missions->Progress(eMissionTaskType::USE_ITEM, itemLot); } } @@ -5676,17 +5872,13 @@ void GameMessages::HandleUseNonEquipmentItem(RakNet::BitStream* inStream, Entity inStream->Read(itemConsumed); - auto* inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + auto* inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; auto* item = inv->FindItemById(itemConsumed); - if (item == nullptr) { - return; - } - - item->UseNonEquip(); + if (item) item->UseNonEquip(item); } void GameMessages::HandleMatchRequest(RakNet::BitStream* inStream, Entity* entity) { @@ -5711,11 +5903,11 @@ void GameMessages::HandleMatchRequest(RakNet::BitStream* inStream, Entity* entit inStream->Read(type); inStream->Read(value); - std::vector scriptedActs = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY); + std::vector scriptedActs = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPTED_ACTIVITY); if (type == 0) { // join if (value != 0) { for (Entity* scriptedAct : scriptedActs) { - ScriptedActivityComponent* comp = static_cast(scriptedAct->GetComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY)); + ScriptedActivityComponent* comp = static_cast(scriptedAct->GetComponent(eReplicaComponentType::SCRIPTED_ACTIVITY)); if (!comp) continue; if (comp->GetActivityID() == value) { comp->PlayerJoin(entity); @@ -5726,7 +5918,7 @@ void GameMessages::HandleMatchRequest(RakNet::BitStream* inStream, Entity* entit } } else if (type == 1) { // ready/unready for (Entity* scriptedAct : scriptedActs) { - ScriptedActivityComponent* comp = static_cast(scriptedAct->GetComponent(COMPONENT_TYPE_SCRIPTED_ACTIVITY)); + ScriptedActivityComponent* comp = static_cast(scriptedAct->GetComponent(eReplicaComponentType::SCRIPTED_ACTIVITY)); if (!comp) continue; if (comp->PlayerIsInQueue(entity)) { comp->PlayerReady(entity, value); @@ -5759,7 +5951,7 @@ void GameMessages::SendGetHotPropertyData(RakNet::BitStream* inStream, Entity* e // TODO This needs to be implemented when reputation is implemented for getting hot properties. /** bitStream.Write(entity->GetObjectID()); - bitStream.Write(GAME_MSG::GAME_MSG_SEND_HOT_PROPERTY_DATA); + bitStream.Write(eGameMessageType::SEND_HOT_PROPERTY_DATA); std::vector t = {25166, 25188, 25191, 25194}; bitStream.Write(4); for (uint8_t i = 0; i < 4; i++) { @@ -5844,7 +6036,7 @@ void GameMessages::HandleReportBug(RakNet::BitStream* inStream, Entity* entity) void GameMessages::HandleClientRailMovementReady(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { - const auto possibleRails = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_RAIL_ACTIVATOR); + const auto possibleRails = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::RAIL_ACTIVATOR); for (const auto* possibleRail : possibleRails) { const auto* rail = possibleRail->GetComponent(); if (rail != nullptr) { @@ -5856,7 +6048,7 @@ GameMessages::HandleClientRailMovementReady(RakNet::BitStream* inStream, Entity* void GameMessages::HandleCancelRailMovement(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { const auto immediate = inStream->ReadBit(); - const auto possibleRails = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_RAIL_ACTIVATOR); + const auto possibleRails = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::RAIL_ACTIVATOR); for (const auto* possibleRail : possibleRails) { auto* rail = possibleRail->GetComponent(); if (rail != nullptr) { @@ -5880,7 +6072,7 @@ void GameMessages::HandlePlayerRailArrivedNotification(RakNet::BitStream* inStre int32_t waypointNumber; inStream->Read(waypointNumber); - const auto possibleRails = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_RAIL_ACTIVATOR); + const auto possibleRails = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::RAIL_ACTIVATOR); for (auto* possibleRail : possibleRails) { for (CppScripts::Script* script : CppScripts::GetEntityScripts(possibleRail)) { script->OnPlayerRailArrived(possibleRail, entity, pathName, waypointNumber); @@ -5929,3 +6121,192 @@ void GameMessages::HandleUpdatePlayerStatistic(RakNet::BitStream* inStream, Enti characterComponent->UpdatePlayerStatistic((StatisticID)updateID, (uint64_t)std::max(updateValue, int64_t(0))); } } + +void GameMessages::HandleDeactivateBubbleBuff(RakNet::BitStream* inStream, Entity* entity) { + auto controllablePhysicsComponent = entity->GetComponent(); + if (controllablePhysicsComponent) controllablePhysicsComponent->DeactivateBubbleBuff(); +} + +void GameMessages::HandleActivateBubbleBuff(RakNet::BitStream* inStream, Entity* entity) { + bool specialAnimations; + if (!inStream->Read(specialAnimations)) return; + + std::u16string type = GeneralUtils::ReadWString(inStream); + auto bubbleType = eBubbleType::DEFAULT; + if (type == u"skunk") bubbleType = eBubbleType::SKUNK; + else if (type == u"energy") bubbleType = eBubbleType::ENERGY; + + auto controllablePhysicsComponent = entity->GetComponent(); + if (controllablePhysicsComponent) controllablePhysicsComponent->ActivateBubbleBuff(bubbleType, specialAnimations); +} + +void GameMessages::SendActivateBubbleBuffFromServer(LWOOBJID objectId, const SystemAddress& sysAddr) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(objectId); + bitStream.Write(eGameMessageType::ACTIVATE_BUBBLE_BUFF_FROM_SERVER); + + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; + SEND_PACKET; +} + +void GameMessages::SendDeactivateBubbleBuffFromServer(LWOOBJID objectId, const SystemAddress& sysAddr) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(objectId); + bitStream.Write(eGameMessageType::DEACTIVATE_BUBBLE_BUFF_FROM_SERVER); + + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST; + SEND_PACKET; +} + +void GameMessages::HandleZoneSummaryDismissed(RakNet::BitStream* inStream, Entity* entity) { + LWOOBJID player_id; + inStream->Read(player_id); + auto target = Game::entityManager->GetEntity(player_id); + entity->TriggerEvent(eTriggerEventType::ZONE_SUMMARY_DISMISSED, target); +}; + +void GameMessages::SendSetNamebillboardState(const SystemAddress& sysAddr, LWOOBJID objectId) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(objectId); + bitStream.Write(eGameMessageType::SET_NAME_BILLBOARD_STATE); + + // Technically these bits would be written, however the client does not + // contain a deserialize method to actually deserialize, so we are leaving it out. + // As such this GM only turns the billboard off. + + // bitStream.Write(overrideDefault); + // bitStream.Write(state); + + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST + else SEND_PACKET +} + +void GameMessages::SendShowBillboardInteractIcon(const SystemAddress& sysAddr, LWOOBJID objectId) { + CBITSTREAM; + CMSGHEADER; + + bitStream.Write(objectId); + bitStream.Write(eGameMessageType::SHOW_BILLBOARD_INTERACT_ICON); + + if (sysAddr == UNASSIGNED_SYSTEM_ADDRESS) SEND_PACKET_BROADCAST + else SEND_PACKET +} + +void GameMessages::HandleRequestActivityExit(RakNet::BitStream* inStream, Entity* entity) { + bool canceled = false; + inStream->Read(canceled); + if (!canceled) return; + + LWOOBJID player_id = LWOOBJID_EMPTY; + inStream->Read(player_id); + auto player = Game::entityManager->GetEntity(player_id); + if (!entity || !player) return; + entity->RequestActivityExit(entity, player_id, canceled); +} + +void GameMessages::HandleAddDonationItem(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { + uint32_t count = 1; + bool hasCount = false; + inStream->Read(hasCount); + if (hasCount) inStream->Read(count); + LWOOBJID itemId = LWOOBJID_EMPTY; + inStream->Read(itemId); + if (!itemId) return; + + auto* donationVendorComponent = entity->GetComponent(); + if (!donationVendorComponent) return; + if (donationVendorComponent->GetActivityID() == 0) { + Game::logger->Log("GameMessages", "WARNING: Trying to dontate to a vendor with no activity"); + return; + } + User* user = UserManager::Instance()->GetUser(sysAddr); + if (!user) return; + Entity* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); + if (!player) return; + auto* characterComponent = player->GetComponent(); + if (!characterComponent) return; + auto* inventoryComponent = player->GetComponent(); + if (!inventoryComponent) return; + Item* item = inventoryComponent->FindItemById(itemId); + if (!item) return; + if (item->GetCount() < count) return; + characterComponent->SetCurrentInteracting(entity->GetObjectID()); + inventoryComponent->MoveItemToInventory(item, eInventoryType::DONATION, count, true, false, true); +} + +void GameMessages::HandleRemoveDonationItem(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr) { + bool confirmed = false; + inStream->Read(confirmed); + uint32_t count = 1; + bool hasCount = false; + inStream->Read(hasCount); + if (hasCount) inStream->Read(count); + LWOOBJID itemId = LWOOBJID_EMPTY; + inStream->Read(itemId); + if (!itemId) return; + + User* user = UserManager::Instance()->GetUser(sysAddr); + if (!user) return; + Entity* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); + if (!player) return; + + auto* inventoryComponent = player->GetComponent(); + if (!inventoryComponent) return; + + Item* item = inventoryComponent->FindItemById(itemId); + if (!item) return; + if (item->GetCount() < count) return; + inventoryComponent->MoveItemToInventory(item, eInventoryType::BRICKS, count, true, false, true); +} + +void GameMessages::HandleConfirmDonationOnPlayer(RakNet::BitStream* inStream, Entity* entity) { + auto* inventoryComponent = entity->GetComponent(); + if (!inventoryComponent) return; + auto* missionComponent = entity->GetComponent(); + if (!missionComponent) return; + auto* characterComponent = entity->GetComponent(); + if (!characterComponent || !characterComponent->GetCurrentInteracting()) return; + auto* donationEntity = Game::entityManager->GetEntity(characterComponent->GetCurrentInteracting()); + if (!donationEntity) return; + auto* donationVendorComponent = donationEntity->GetComponent(); + if(!donationVendorComponent) return; + if (donationVendorComponent->GetActivityID() == 0) { + Game::logger->Log("GameMessages", "WARNING: Trying to dontate to a vendor with no activity"); + return; + } + auto* inventory = inventoryComponent->GetInventory(eInventoryType::DONATION); + if (!inventory) return; + auto items = inventory->GetItems(); + if (!items.empty()) { + uint32_t count = 0; + for (auto& [itemID, item] : items){ + count += item->GetCount(); + item->RemoveFromInventory(); + } + missionComponent->Progress(eMissionTaskType::DONATION, 0, LWOOBJID_EMPTY, "", count); + LeaderboardManager::SaveScore(entity->GetObjectID(), donationVendorComponent->GetActivityID(), count); + donationVendorComponent->SubmitDonation(count); + Game::entityManager->SerializeEntity(donationEntity); + } + characterComponent->SetCurrentInteracting(LWOOBJID_EMPTY); +} + +void GameMessages::HandleCancelDonationOnPlayer(RakNet::BitStream* inStream, Entity* entity) { + auto* inventoryComponent = entity->GetComponent(); + if (!inventoryComponent) return; + auto* inventory = inventoryComponent->GetInventory(eInventoryType::DONATION); + if (!inventory) return; + auto items = inventory->GetItems(); + for (auto& [itemID, item] : items){ + inventoryComponent->MoveItemToInventory(item, eInventoryType::BRICKS, item->GetCount(), false, false, true); + } + auto* characterComponent = entity->GetComponent(); + if (!characterComponent) return; + characterComponent->SetCurrentInteracting(LWOOBJID_EMPTY); +} diff --git a/dGame/dGameMessages/GameMessages.h b/dGame/dGameMessages/GameMessages.h index 31bdebb3..3a9963b4 100644 --- a/dGame/dGameMessages/GameMessages.h +++ b/dGame/dGameMessages/GameMessages.h @@ -1,36 +1,53 @@ - #ifndef GAMEMESSAGES_H #define GAMEMESSAGES_H #include "dCommonVars.h" -#include "RakNetTypes.h" +#include #include -#include "InventoryComponent.h" -#include "dMessageIdentifiers.h" -#include "AMFFormat.h" -#include "AMFFormat_BitStream.h" -#include "NiQuaternion.h" -#include "PropertySelectQueryProperty.h" -#include "TradingManager.h" -#include "LeaderboardManager.h" -#include "MovingPlatformComponent.h" +#include +#include "eMovementPlatformState.h" +#include "NiPoint3.h" +#include "eEndBehavior.h" +#include "eCyclingMode.h" +#include "eLootSourceType.h" +#include "Brick.h" +class AMFBaseValue; +class Entity; +class Item; class NiQuaternion; class User; -class Entity; -class NiPoint3; +class Leaderboard; +class PropertySelectQueryProperty; +class TradeItem; + +enum class eAnimationFlags : uint32_t; + enum class eUnequippableActiveType; +enum eInventoryType : uint32_t; +enum class eGameMasterLevel : uint8_t; +enum class eMatchUpdate : int32_t; +enum class eKillType : uint32_t; +enum class eObjectWorldState : uint32_t; +enum class eTerminateType : uint32_t; +enum class eControlScheme : uint32_t; +enum class eStateChangeType : uint32_t; +enum class ePetTamingNotifyType : uint32_t; +enum class eUseItemResponse : uint32_t; +enum class eQuickBuildFailReason : uint32_t; +enum class eRebuildState : uint32_t; namespace GameMessages { class PropertyDataMessage; void SendFireEventClientSide(const LWOOBJID& objectID, const SystemAddress& sysAddr, std::u16string args, const LWOOBJID& object, int64_t param1, int param2, const LWOOBJID& sender); - void SendTeleport(const LWOOBJID& objectID, const NiPoint3& pos, const NiQuaternion& rot, const SystemAddress& sysAddr, bool bSetRotation = false, bool noGravTeleport = true); + void SendTeleport(const LWOOBJID& objectID, const NiPoint3& pos, const NiQuaternion& rot, const SystemAddress& sysAddr, bool bSetRotation = false); void SendPlayAnimation(Entity* entity, const std::u16string& animationName, float fPriority = 0.0f, float fScale = 1.0f); void SendPlayerReady(Entity* entity, const SystemAddress& sysAddr); void SendPlayerAllowedRespawn(LWOOBJID entityID, bool doNotPromptRespawn, const SystemAddress& systemAddress); void SendInvalidZoneTransferList(Entity* entity, const SystemAddress& sysAddr, const std::u16string& feedbackURL, const std::u16string& invalidMapTransferList, bool feedbackOnExit, bool feedbackOnInvalidTransfer); void SendKnockback(const LWOOBJID& objectID, const LWOOBJID& caster, const LWOOBJID& originator, int knockBackTimeMS, const NiPoint3& vector); - + // https://lcdruniverse.org/lu_packets/lu_packets/world/gm/client/struct.VehicleStopBoost.html + void SendVehicleStopBoost(Entity* targetEntity, const SystemAddress& playerSysAddr, bool affectPassive); void SendStartArrangingWithItem( Entity* entity, const SystemAddress& sysAddr, @@ -47,24 +64,23 @@ namespace GameMessages { int targetTYPE = 0 ); - void SendPlayerSetCameraCyclingMode(const LWOOBJID& objectID, const SystemAddress& sysAddr, - bool bAllowCyclingWhileDeadOnly = true, eCyclingMode cyclingMode = ALLOW_CYCLE_TEAMMATES); + void SendPlayerSetCameraCyclingMode(const LWOOBJID& objectID, const SystemAddress& sysAddr, bool bAllowCyclingWhileDeadOnly = true, eCyclingMode cyclingMode = eCyclingMode::ALLOW_CYCLE_TEAMMATES); void SendPlayNDAudioEmitter(Entity* entity, const SystemAddress& sysAddr, std::string audioGUID); void SendStartPathing(Entity* entity); void SendPlatformResync(Entity* entity, const SystemAddress& sysAddr, bool bStopAtDesiredWaypoint = false, int iIndex = 0, int iDesiredWaypointIndex = 1, int nextIndex = 1, - MovementPlatformState movementState = MovementPlatformState::Moving); + eMovementPlatformState movementState = eMovementPlatformState::Moving); void SendRestoreToPostLoadStats(Entity* entity, const SystemAddress& sysAddr); void SendServerDoneLoadingAllObjects(Entity* entity, const SystemAddress& sysAddr); - void SendGMLevelBroadcast(const LWOOBJID& objectID, uint8_t level); - void SendChatModeUpdate(const LWOOBJID& objectID, uint8_t level); + void SendGMLevelBroadcast(const LWOOBJID& objectID, eGameMasterLevel level); + void SendChatModeUpdate(const LWOOBJID& objectID, eGameMasterLevel level); - void SendAddItemToInventoryClientSync(Entity* entity, const SystemAddress& sysAddr, Item* item, const LWOOBJID& objectID, bool showFlyingLoot, int itemCount, LWOOBJID subKey = LWOOBJID_EMPTY, eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE); - void SendNotifyClientFlagChange(const LWOOBJID& objectID, int iFlagID, bool bFlag, const SystemAddress& sysAddr); - void SendChangeObjectWorldState(const LWOOBJID& objectID, int state, const SystemAddress& sysAddr); + void SendAddItemToInventoryClientSync(Entity* entity, const SystemAddress& sysAddr, Item* item, const LWOOBJID& objectID, bool showFlyingLoot, int itemCount, LWOOBJID subKey = LWOOBJID_EMPTY, eLootSourceType lootSourceType = eLootSourceType::NONE); + void SendNotifyClientFlagChange(const LWOOBJID& objectID, uint32_t iFlagID, bool bFlag, const SystemAddress& sysAddr); + void SendChangeObjectWorldState(const LWOOBJID& objectID, eObjectWorldState state, const SystemAddress& sysAddr); void SendOfferMission(const LWOOBJID& entity, const SystemAddress& sysAddr, int32_t missionID, const LWOOBJID& offererID); void SendNotifyMission(Entity* entity, const SystemAddress& sysAddr, int missionID, int missionState, bool sendingRewards); @@ -72,8 +88,8 @@ namespace GameMessages { void NotifyLevelRewards(LWOOBJID objectID, const SystemAddress& sysAddr, int level, bool sending_rewards); void SendModifyLEGOScore(Entity* entity, const SystemAddress& sysAddr, int64_t score, eLootSourceType sourceType); - void SendUIMessageServerToSingleClient(Entity* entity, const SystemAddress& sysAddr, const std::string& message, NDGFxValue args); - void SendUIMessageServerToAllClients(const std::string& message, NDGFxValue args); + void SendUIMessageServerToSingleClient(Entity* entity, const SystemAddress& sysAddr, const std::string& message, AMFBaseValue& args); + void SendUIMessageServerToAllClients(const std::string& message, AMFBaseValue& args); void SendPlayEmbeddedEffectOnAllClientsNearObject(Entity* entity, std::u16string effectName, const LWOOBJID& fromObjectID, float radius); void SendPlayFXEffect(Entity* entity, int32_t effectID, const std::u16string& effectType, const std::string& name, LWOOBJID secondary, float priority = 1, float scale = 1, bool serialize = true); @@ -82,8 +98,8 @@ namespace GameMessages { void SendBroadcastTextToChatbox(Entity* entity, const SystemAddress& sysAddr, const std::u16string& attrs, const std::u16string& wsText); void SendSetCurrency(Entity* entity, int64_t currency, int lootType, const LWOOBJID& sourceID, const LOT& sourceLOT, int sourceTradeID, bool overrideCurrent, eLootSourceType sourceType); - void SendRebuildNotifyState(Entity* entity, int prevState, int state, const LWOOBJID& playerID); - void SendEnableRebuild(Entity* entity, bool enable, bool fail, bool success, int failReason, float duration, const LWOOBJID& playerID); + void SendRebuildNotifyState(Entity* entity, eRebuildState prevState, eRebuildState state, const LWOOBJID& playerID); + void SendEnableRebuild(Entity* entity, bool enable, bool fail, bool success, eQuickBuildFailReason failReason, float duration, const LWOOBJID& playerID); void AddActivityOwner(Entity* entity, LWOOBJID& ownerID); void SendTerminateInteraction(const LWOOBJID& objectID, eTerminateType type, const LWOOBJID& terminator); @@ -100,7 +116,7 @@ namespace GameMessages { void SendSetNetworkScriptVar(Entity* entity, const SystemAddress& sysAddr, std::string data); void SendDropClientLoot(Entity* entity, const LWOOBJID& sourceID, LOT item, int currency, NiPoint3 spawnPos = NiPoint3::ZERO, int count = 1); - void SendSetPlayerControlScheme(Entity* entity, eControlSceme controlScheme); + void SendSetPlayerControlScheme(Entity* entity, eControlScheme controlScheme); void SendPlayerReachedRespawnCheckpoint(Entity* entity, const NiPoint3& position, const NiQuaternion& rotation); void SendAddSkill(Entity* entity, TSkillID skillID, int slotID); @@ -119,10 +135,14 @@ namespace GameMessages { void SendMoveInventoryBatch(Entity* entity, uint32_t stackCount, int srcInv, int dstInv, const LWOOBJID& iObjID); void SendMatchResponse(Entity* entity, const SystemAddress& sysAddr, int response); - void SendMatchUpdate(Entity* entity, const SystemAddress& sysAddr, std::string data, int type); + void SendMatchUpdate(Entity* entity, const SystemAddress& sysAddr, std::string data, eMatchUpdate type); + void HandleUnUseModel(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); void SendStartCelebrationEffect(Entity* entity, const SystemAddress& sysAddr, int celebrationID); + // https://lcdruniverse.org/lu_packets/lu_packets/world/gm/client/struct.SetResurrectRestoreValues.html + void SendSetResurrectRestoreValues(Entity* targetEntity, int32_t armorRestore, int32_t healthRestore, int32_t imaginationRestore); + /** * Sends a message to an Entity to smash itself, but not delete or destroy itself from the world * @@ -197,6 +217,16 @@ namespace GameMessages { void SendDownloadPropertyData(LWOOBJID objectId, const PropertyDataMessage& data, const SystemAddress& sysAddr); + /** + * @brief Send an updated item id to the client when they load a blueprint in brick build mode + * + * @param sysAddr SystemAddress to respond to + * @param oldItemId The item ID that was requested to be loaded + * @param newItemId The new item ID of the loaded item + * + */ + void SendBlueprintLoadItemResponse(const SystemAddress& sysAddr, bool success, LWOOBJID oldItemId, LWOOBJID newItemId); + void SendPropertyRentalResponse(LWOOBJID objectId, LWOCLONEID cloneId, uint32_t code, LWOOBJID propertyId, int64_t rentDue, const SystemAddress& sysAddr); void SendLockNodeRotation(Entity* entity, std::string nodeName); @@ -249,14 +279,14 @@ namespace GameMessages { void SendPlayCinematic(LWOOBJID objectId, std::u16string pathName, const SystemAddress& sysAddr, bool allowGhostUpdates = true, bool bCloseMultiInteract = true, bool bSendServerNotify = false, bool bUseControlledObjectForAudioListener = false, - int endBehavior = 0, bool hidePlayerDuringCine = false, float leadIn = -1, bool leavePlayerLockedWhenFinished = false, + eEndBehavior endBehavior = eEndBehavior::RETURN, bool hidePlayerDuringCine = false, float leadIn = -1, bool leavePlayerLockedWhenFinished = false, bool lockPlayer = true, bool result = false, bool skipIfSamePath = false, float startTimeAdvance = 0); void SendEndCinematic(LWOOBJID objectID, std::u16string pathName, const SystemAddress& sysAddr = UNASSIGNED_SYSTEM_ADDRESS, float leadOut = -1.0f, bool leavePlayerLocked = false); void HandleCinematicUpdate(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); - void SendSetStunned(LWOOBJID objectId, eStunState stateChangeType, const SystemAddress& sysAddr, + void SendSetStunned(LWOOBJID objectId, eStateChangeType stateChangeType, const SystemAddress& sysAddr, LWOOBJID originator = LWOOBJID_EMPTY, bool bCantAttack = false, bool bCantEquip = false, bool bCantInteract = false, bool bCantJump = false, bool bCantMove = false, bool bCantTurn = false, bool bCantUseItem = false, bool bDontTerminateInteract = false, bool bIgnoreImmunity = true, @@ -265,6 +295,35 @@ namespace GameMessages { bool bCantMoveOutChangeWasApplied = false, bool bCantTurnOutChangeWasApplied = false, bool bCantUseItemOutChangeWasApplied = false); + void SendSetStunImmunity( + LWOOBJID target, + eStateChangeType state, + const SystemAddress& sysAddr, + LWOOBJID originator = LWOOBJID_EMPTY, + bool bImmuneToStunAttack = false, + bool bImmuneToStunEquip = false, + bool bImmuneToStunInteract = false, + bool bImmuneToStunJump = false, + bool bImmuneToStunMove = false, + bool bImmuneToStunTurn = false, + bool bImmuneToStunUseItem = false + ); + + void SendSetStatusImmunity( + LWOOBJID objectId, + eStateChangeType state, + const SystemAddress& sysAddr, + bool bImmuneToBasicAttack = false, + bool bImmuneToDamageOverTime = false, + bool bImmuneToKnockback = false, + bool bImmuneToInterrupt = false, + bool bImmuneToSpeed = false, + bool bImmuneToImaginationGain = false, + bool bImmuneToImaginationLoss = false, + bool bImmuneToQuickbuildInterrupt = false, + bool bImmuneToPullToPoint = false + ); + void SendOrientToAngle(LWOOBJID objectId, bool bRelativeToCurrent, float fAngle, const SystemAddress& sysAddr); void SendAddRunSpeedModifier(LWOOBJID objectId, LWOOBJID caster, uint32_t modifier, const SystemAddress& sysAddr); @@ -303,9 +362,9 @@ namespace GameMessages { void HandleClientTradeUpdate(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); //Pets: - void SendNotifyPetTamingMinigame(LWOOBJID objectId, LWOOBJID petId, LWOOBJID playerTamingId, bool bForceTeleport, uint32_t notifyType, NiPoint3 petsDestPos, NiPoint3 telePos, NiQuaternion teleRot, const SystemAddress& sysAddr); + void SendNotifyPetTamingMinigame(LWOOBJID objectId, LWOOBJID petId, LWOOBJID playerTamingId, bool bForceTeleport, ePetTamingNotifyType notifyType, NiPoint3 petsDestPos, NiPoint3 telePos, NiQuaternion teleRot, const SystemAddress& sysAddr); - void SendNotifyPetTamingPuzzleSelected(LWOOBJID objectId, std::vector& bricks, const SystemAddress& sysAddr); + void SendNotifyPetTamingPuzzleSelected(LWOOBJID objectId, const std::vector& bricks, const SystemAddress& sysAddr); void SendNotifyTamingModelLoadedOnServer(LWOOBJID objectId, const SystemAddress& sysAddr); @@ -362,6 +421,15 @@ namespace GameMessages { void SendDisplayChatBubble(LWOOBJID objectId, const std::u16string& text, const SystemAddress& sysAddr); + /** + * @brief + * + * @param objectId ID of the entity to set idle flags + * @param FlagsOn Flag to turn on + * @param FlagsOff Flag to turn off + */ + void SendChangeIdleFlags(LWOOBJID objectId, eAnimationFlags FlagsOn, eAnimationFlags FlagsOff, const SystemAddress& sysAddr); + // Mounts /** * @brief Set the Inventory LWOOBJID of the mount @@ -464,7 +532,7 @@ namespace GameMessages { void SendActivityPause(LWOOBJID objectId, bool pause = false, const SystemAddress& sysAddr = UNASSIGNED_SYSTEM_ADDRESS); void SendStartActivityTime(LWOOBJID objectId, float_t startTime, const SystemAddress& sysAddr = UNASSIGNED_SYSTEM_ADDRESS); void SendRequestActivityEnter(LWOOBJID objectId, const SystemAddress& sysAddr, bool bStart, LWOOBJID userID); - void SendUseItemRequirementsResponse(LWOOBJID objectID, const SystemAddress& sysAddr, UseItemResponse itemResponse); + void SendUseItemRequirementsResponse(LWOOBJID objectID, const SystemAddress& sysAddr, eUseItemResponse itemResponse); // SG: @@ -520,6 +588,8 @@ namespace GameMessages { void HandleToggleGhostReferenceOverride(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); void HandleSetGhostReferencePosition(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); + void SendSetNamebillboardState(const SystemAddress& sysAddr, LWOOBJID objectId); + void SendShowBillboardInteractIcon(const SystemAddress& sysAddr, LWOOBJID objectId); void HandleBuyFromVendor(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); void HandleSellToVendor(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); void HandleBuybackFromVendor(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); @@ -566,549 +636,25 @@ namespace GameMessages { void HandleReportBug(RakNet::BitStream* inStream, Entity* entity); - /* Message to synchronize a skill cast */ - class EchoSyncSkill { - static const GAME_MSG MsgID = GAME_MSG_ECHO_SYNC_SKILL; + void SendRemoveBuff(Entity* entity, bool fromUnEquip, bool removeImmunity, uint32_t buffId); - public: - EchoSyncSkill() { - bDone = false; - } + // bubble + void HandleDeactivateBubbleBuff(RakNet::BitStream* inStream, Entity* entity); - EchoSyncSkill(std::string _sBitStream, unsigned int _uiBehaviorHandle, unsigned int _uiSkillHandle, bool _bDone = false) { - bDone = _bDone; - sBitStream = _sBitStream; - uiBehaviorHandle = _uiBehaviorHandle; - uiSkillHandle = _uiSkillHandle; - } + void HandleActivateBubbleBuff(RakNet::BitStream* inStream, Entity* entity); - EchoSyncSkill(RakNet::BitStream* stream) { - bDone = false; + void SendActivateBubbleBuffFromServer(LWOOBJID objectId, const SystemAddress& sysAddr); - Deserialize(stream); - } + void SendDeactivateBubbleBuffFromServer(LWOOBJID objectId, const SystemAddress& sysAddr); - ~EchoSyncSkill() { - } + void HandleZoneSummaryDismissed(RakNet::BitStream* inStream, Entity* entity); + void HandleRequestActivityExit(RakNet::BitStream* inStream, Entity* entity); - void Serialize(RakNet::BitStream* stream) { - stream->Write((unsigned short)MsgID); - - stream->Write(bDone); - uint32_t sBitStreamLength = sBitStream.length(); - stream->Write(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - stream->Write(sBitStream[k]); - } - - stream->Write(uiBehaviorHandle); - stream->Write(uiSkillHandle); - } - - bool Deserialize(RakNet::BitStream* stream) { - stream->Read(bDone); - uint32_t sBitStreamLength{}; - stream->Read(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - unsigned char character; - stream->Read(character); - sBitStream.push_back(character); - } - - stream->Read(uiBehaviorHandle); - stream->Read(uiSkillHandle); - - return true; - } - - bool bDone{}; - std::string sBitStream{}; - unsigned int uiBehaviorHandle{}; - unsigned int uiSkillHandle{}; - }; - - /* Message to synchronize a skill cast */ - class SyncSkill { - static const GAME_MSG MsgID = GAME_MSG_SYNC_SKILL; - - public: - SyncSkill() { - bDone = false; - } - - SyncSkill(std::string _sBitStream, unsigned int _uiBehaviorHandle, unsigned int _uiSkillHandle, bool _bDone = false) { - bDone = _bDone; - sBitStream = _sBitStream; - uiBehaviorHandle = _uiBehaviorHandle; - uiSkillHandle = _uiSkillHandle; - } - - SyncSkill(RakNet::BitStream* stream) { - bDone = false; - Deserialize(stream); - } - - ~SyncSkill() { - } - - void Serialize(RakNet::BitStream* stream) { - stream->Write((unsigned short)MsgID); - - stream->Write(bDone); - uint32_t sBitStreamLength = sBitStream.length(); - stream->Write(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - stream->Write(sBitStream[k]); - } - - stream->Write(uiBehaviorHandle); - stream->Write(uiSkillHandle); - } - - bool Deserialize(RakNet::BitStream* stream) { - stream->Read(bDone); - uint32_t sBitStreamLength{}; - stream->Read(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - unsigned char character; - stream->Read(character); - sBitStream.push_back(character); - } - - stream->Read(uiBehaviorHandle); - stream->Read(uiSkillHandle); - - return true; - } - - bool bDone{}; - std::string sBitStream{}; - unsigned int uiBehaviorHandle{}; - unsigned int uiSkillHandle{}; - }; - - /* Notifying the server that a locally owned projectil impacted. Sent to the caster of the projectile - should always be the local char. */ - class RequestServerProjectileImpact { - static const GAME_MSG MsgID = GAME_MSG_REQUEST_SERVER_PROJECTILE_IMPACT; - - public: - RequestServerProjectileImpact() { - i64LocalID = LWOOBJID_EMPTY; - i64TargetID = LWOOBJID_EMPTY; - } - - RequestServerProjectileImpact(std::string _sBitStream, LWOOBJID _i64LocalID = LWOOBJID_EMPTY, LWOOBJID _i64TargetID = LWOOBJID_EMPTY) { - i64LocalID = _i64LocalID; - i64TargetID = _i64TargetID; - sBitStream = _sBitStream; - } - - RequestServerProjectileImpact(RakNet::BitStream* stream) { - i64LocalID = LWOOBJID_EMPTY; - i64TargetID = LWOOBJID_EMPTY; - - Deserialize(stream); - } - - ~RequestServerProjectileImpact() { - } - - void Serialize(RakNet::BitStream* stream) { - stream->Write((unsigned short)MsgID); - - stream->Write(i64LocalID != LWOOBJID_EMPTY); - if (i64LocalID != LWOOBJID_EMPTY) stream->Write(i64LocalID); - - stream->Write(i64TargetID != LWOOBJID_EMPTY); - if (i64TargetID != LWOOBJID_EMPTY) stream->Write(i64TargetID); - - uint32_t sBitStreamLength = sBitStream.length(); - stream->Write(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - stream->Write(sBitStream[k]); - } - - } - - bool Deserialize(RakNet::BitStream* stream) { - bool i64LocalIDIsDefault{}; - stream->Read(i64LocalIDIsDefault); - if (i64LocalIDIsDefault != 0) stream->Read(i64LocalID); - - bool i64TargetIDIsDefault{}; - stream->Read(i64TargetIDIsDefault); - if (i64TargetIDIsDefault != 0) stream->Read(i64TargetID); - - uint32_t sBitStreamLength{}; - stream->Read(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - unsigned char character; - stream->Read(character); - sBitStream.push_back(character); - } - - - return true; - } - - LWOOBJID i64LocalID; - LWOOBJID i64TargetID; - std::string sBitStream; - }; - - /* Tell a client local projectile to impact */ - class DoClientProjectileImpact { - static const GAME_MSG MsgID = GAME_MSG_DO_CLIENT_PROJECTILE_IMPACT; - - public: - DoClientProjectileImpact() { - i64OrgID = LWOOBJID_EMPTY; - i64OwnerID = LWOOBJID_EMPTY; - i64TargetID = LWOOBJID_EMPTY; - } - - DoClientProjectileImpact(std::string _sBitStream, LWOOBJID _i64OrgID = LWOOBJID_EMPTY, LWOOBJID _i64OwnerID = LWOOBJID_EMPTY, LWOOBJID _i64TargetID = LWOOBJID_EMPTY) { - i64OrgID = _i64OrgID; - i64OwnerID = _i64OwnerID; - i64TargetID = _i64TargetID; - sBitStream = _sBitStream; - } - - DoClientProjectileImpact(RakNet::BitStream* stream) { - i64OrgID = LWOOBJID_EMPTY; - i64OwnerID = LWOOBJID_EMPTY; - i64TargetID = LWOOBJID_EMPTY; - - Deserialize(stream); - } - - ~DoClientProjectileImpact() { - } - - void Serialize(RakNet::BitStream* stream) { - stream->Write((unsigned short)MsgID); - - stream->Write(i64OrgID != LWOOBJID_EMPTY); - if (i64OrgID != LWOOBJID_EMPTY) stream->Write(i64OrgID); - - stream->Write(i64OwnerID != LWOOBJID_EMPTY); - if (i64OwnerID != LWOOBJID_EMPTY) stream->Write(i64OwnerID); - - stream->Write(i64TargetID != LWOOBJID_EMPTY); - if (i64TargetID != LWOOBJID_EMPTY) stream->Write(i64TargetID); - - uint32_t sBitStreamLength = sBitStream.length(); - stream->Write(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - stream->Write(sBitStream[k]); - } - - } - - bool Deserialize(RakNet::BitStream* stream) { - bool i64OrgIDIsDefault{}; - stream->Read(i64OrgIDIsDefault); - if (i64OrgIDIsDefault != 0) stream->Read(i64OrgID); - - bool i64OwnerIDIsDefault{}; - stream->Read(i64OwnerIDIsDefault); - if (i64OwnerIDIsDefault != 0) stream->Read(i64OwnerID); - - bool i64TargetIDIsDefault{}; - stream->Read(i64TargetIDIsDefault); - if (i64TargetIDIsDefault != 0) stream->Read(i64TargetID); - - uint32_t sBitStreamLength{}; - stream->Read(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - unsigned char character; - stream->Read(character); - sBitStream.push_back(character); - } - - - return true; - } - - LWOOBJID i64OrgID; - LWOOBJID i64OwnerID; - LWOOBJID i64TargetID; - std::string sBitStream; - }; - - /* Same as start skill but with different network options. An echo down to other clients that need to play the skill. */ - class EchoStartSkill { - static const GAME_MSG MsgID = GAME_MSG_ECHO_START_SKILL; - - public: - EchoStartSkill() { - bUsedMouse = false; - fCasterLatency = 0.0f; - iCastType = 0; - lastClickedPosit = NiPoint3::ZERO; - optionalTargetID = LWOOBJID_EMPTY; - originatorRot = NiQuaternion::IDENTITY; - uiSkillHandle = 0; - } - - EchoStartSkill(LWOOBJID _optionalOriginatorID, std::string _sBitStream, TSkillID _skillID, bool _bUsedMouse = false, float _fCasterLatency = 0.0f, int _iCastType = 0, NiPoint3 _lastClickedPosit = NiPoint3::ZERO, LWOOBJID _optionalTargetID = LWOOBJID_EMPTY, NiQuaternion _originatorRot = NiQuaternion::IDENTITY, unsigned int _uiSkillHandle = 0) { - bUsedMouse = _bUsedMouse; - fCasterLatency = _fCasterLatency; - iCastType = _iCastType; - lastClickedPosit = _lastClickedPosit; - optionalOriginatorID = _optionalOriginatorID; - optionalTargetID = _optionalTargetID; - originatorRot = _originatorRot; - sBitStream = _sBitStream; - skillID = _skillID; - uiSkillHandle = _uiSkillHandle; - } - - EchoStartSkill(RakNet::BitStream* stream) { - bUsedMouse = false; - fCasterLatency = 0.0f; - iCastType = 0; - lastClickedPosit = NiPoint3::ZERO; - optionalTargetID = LWOOBJID_EMPTY; - originatorRot = NiQuaternion::IDENTITY; - uiSkillHandle = 0; - - Deserialize(stream); - } - - ~EchoStartSkill() { - } - - void Serialize(RakNet::BitStream* stream) { - stream->Write((unsigned short)MsgID); - - stream->Write(bUsedMouse); - - stream->Write(fCasterLatency != 0.0f); - if (fCasterLatency != 0.0f) stream->Write(fCasterLatency); - - stream->Write(iCastType != 0); - if (iCastType != 0) stream->Write(iCastType); - - stream->Write(lastClickedPosit != NiPoint3::ZERO); - if (lastClickedPosit != NiPoint3::ZERO) stream->Write(lastClickedPosit); - - stream->Write(optionalOriginatorID); - - stream->Write(optionalTargetID != LWOOBJID_EMPTY); - if (optionalTargetID != LWOOBJID_EMPTY) stream->Write(optionalTargetID); - - stream->Write(originatorRot != NiQuaternion::IDENTITY); - if (originatorRot != NiQuaternion::IDENTITY) stream->Write(originatorRot); - - uint32_t sBitStreamLength = sBitStream.length(); - stream->Write(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - stream->Write(sBitStream[k]); - } - - stream->Write(skillID); - - stream->Write(uiSkillHandle != 0); - if (uiSkillHandle != 0) stream->Write(uiSkillHandle); - } - - bool Deserialize(RakNet::BitStream* stream) { - stream->Read(bUsedMouse); - - bool fCasterLatencyIsDefault{}; - stream->Read(fCasterLatencyIsDefault); - if (fCasterLatencyIsDefault != 0) stream->Read(fCasterLatency); - - bool iCastTypeIsDefault{}; - stream->Read(iCastTypeIsDefault); - if (iCastTypeIsDefault != 0) stream->Read(iCastType); - - bool lastClickedPositIsDefault{}; - stream->Read(lastClickedPositIsDefault); - if (lastClickedPositIsDefault != 0) stream->Read(lastClickedPosit); - - stream->Read(optionalOriginatorID); - - bool optionalTargetIDIsDefault{}; - stream->Read(optionalTargetIDIsDefault); - if (optionalTargetIDIsDefault != 0) stream->Read(optionalTargetID); - - bool originatorRotIsDefault{}; - stream->Read(originatorRotIsDefault); - if (originatorRotIsDefault != 0) stream->Read(originatorRot); - - uint32_t sBitStreamLength{}; - stream->Read(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - unsigned char character; - stream->Read(character); - sBitStream.push_back(character); - } - - stream->Read(skillID); - - bool uiSkillHandleIsDefault{}; - stream->Read(uiSkillHandleIsDefault); - if (uiSkillHandleIsDefault != 0) stream->Read(uiSkillHandle); - - return true; - } - - bool bUsedMouse; - float fCasterLatency; - int iCastType; - NiPoint3 lastClickedPosit; - LWOOBJID optionalOriginatorID; - LWOOBJID optionalTargetID; - NiQuaternion originatorRot; - std::string sBitStream; - TSkillID skillID; - unsigned int uiSkillHandle; - }; - - /* Same as sync skill but with different network options. An echo down to other clients that need to play the skill. */ - class StartSkill { - static const GAME_MSG MsgID = GAME_MSG_START_SKILL; - - public: - StartSkill() { - bUsedMouse = false; - consumableItemID = LWOOBJID_EMPTY; - fCasterLatency = 0.0f; - iCastType = 0; - lastClickedPosit = NiPoint3::ZERO; - optionalTargetID = LWOOBJID_EMPTY; - originatorRot = NiQuaternion::IDENTITY; - uiSkillHandle = 0; - } - - StartSkill(LWOOBJID _optionalOriginatorID, std::string _sBitStream, TSkillID _skillID, bool _bUsedMouse = false, LWOOBJID _consumableItemID = LWOOBJID_EMPTY, float _fCasterLatency = 0.0f, int _iCastType = 0, NiPoint3 _lastClickedPosit = NiPoint3::ZERO, LWOOBJID _optionalTargetID = LWOOBJID_EMPTY, NiQuaternion _originatorRot = NiQuaternion::IDENTITY, unsigned int _uiSkillHandle = 0) { - bUsedMouse = _bUsedMouse; - consumableItemID = _consumableItemID; - fCasterLatency = _fCasterLatency; - iCastType = _iCastType; - lastClickedPosit = _lastClickedPosit; - optionalOriginatorID = _optionalOriginatorID; - optionalTargetID = _optionalTargetID; - originatorRot = _originatorRot; - sBitStream = _sBitStream; - skillID = _skillID; - uiSkillHandle = _uiSkillHandle; - } - - StartSkill(RakNet::BitStream* stream) { - bUsedMouse = false; - consumableItemID = LWOOBJID_EMPTY; - fCasterLatency = 0.0f; - iCastType = 0; - lastClickedPosit = NiPoint3::ZERO; - optionalTargetID = LWOOBJID_EMPTY; - originatorRot = NiQuaternion::IDENTITY; - uiSkillHandle = 0; - - Deserialize(stream); - } - - ~StartSkill() { - } - - void Serialize(RakNet::BitStream* stream) { - stream->Write((unsigned short)MsgID); - - stream->Write(bUsedMouse); - - stream->Write(consumableItemID != LWOOBJID_EMPTY); - if (consumableItemID != LWOOBJID_EMPTY) stream->Write(consumableItemID); - - stream->Write(fCasterLatency != 0.0f); - if (fCasterLatency != 0.0f) stream->Write(fCasterLatency); - - stream->Write(iCastType != 0); - if (iCastType != 0) stream->Write(iCastType); - - stream->Write(lastClickedPosit != NiPoint3::ZERO); - if (lastClickedPosit != NiPoint3::ZERO) stream->Write(lastClickedPosit); - - stream->Write(optionalOriginatorID); - - stream->Write(optionalTargetID != LWOOBJID_EMPTY); - if (optionalTargetID != LWOOBJID_EMPTY) stream->Write(optionalTargetID); - - stream->Write(originatorRot != NiQuaternion::IDENTITY); - if (originatorRot != NiQuaternion::IDENTITY) stream->Write(originatorRot); - - uint32_t sBitStreamLength = sBitStream.length(); - stream->Write(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - stream->Write(sBitStream[k]); - } - - stream->Write(skillID); - - stream->Write(uiSkillHandle != 0); - if (uiSkillHandle != 0) stream->Write(uiSkillHandle); - } - - bool Deserialize(RakNet::BitStream* stream) { - stream->Read(bUsedMouse); - - bool consumableItemIDIsDefault{}; - stream->Read(consumableItemIDIsDefault); - if (consumableItemIDIsDefault != 0) stream->Read(consumableItemID); - - bool fCasterLatencyIsDefault{}; - stream->Read(fCasterLatencyIsDefault); - if (fCasterLatencyIsDefault != 0) stream->Read(fCasterLatency); - - bool iCastTypeIsDefault{}; - stream->Read(iCastTypeIsDefault); - if (iCastTypeIsDefault != 0) stream->Read(iCastType); - - bool lastClickedPositIsDefault{}; - stream->Read(lastClickedPositIsDefault); - if (lastClickedPositIsDefault != 0) stream->Read(lastClickedPosit); - - stream->Read(optionalOriginatorID); - - bool optionalTargetIDIsDefault{}; - stream->Read(optionalTargetIDIsDefault); - if (optionalTargetIDIsDefault != 0) stream->Read(optionalTargetID); - - bool originatorRotIsDefault{}; - stream->Read(originatorRotIsDefault); - if (originatorRotIsDefault != 0) stream->Read(originatorRot); - - uint32_t sBitStreamLength{}; - stream->Read(sBitStreamLength); - for (unsigned int k = 0; k < sBitStreamLength; k++) { - unsigned char character; - stream->Read(character); - sBitStream.push_back(character); - } - - stream->Read(skillID); - - bool uiSkillHandleIsDefault{}; - stream->Read(uiSkillHandleIsDefault); - if (uiSkillHandleIsDefault != 0) stream->Read(uiSkillHandle); - - return true; - } - - bool bUsedMouse = false; - LWOOBJID consumableItemID{}; - float fCasterLatency{}; - int iCastType{}; - NiPoint3 lastClickedPosit{}; - LWOOBJID optionalOriginatorID{}; - LWOOBJID optionalTargetID{}; - NiQuaternion originatorRot{}; - std::string sBitStream = ""; - TSkillID skillID = 0; - unsigned int uiSkillHandle = 0; - }; + // Donation vendor + void HandleAddDonationItem(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); + void HandleRemoveDonationItem(RakNet::BitStream* inStream, Entity* entity, const SystemAddress& sysAddr); + void HandleConfirmDonationOnPlayer(RakNet::BitStream* inStream, Entity* entity); + void HandleCancelDonationOnPlayer(RakNet::BitStream* inStream, Entity* entity); }; #endif // GAMEMESSAGES_H diff --git a/dGame/dGameMessages/PropertyDataMessage.cpp b/dGame/dGameMessages/PropertyDataMessage.cpp index e0b792ad..c3443f31 100644 --- a/dGame/dGameMessages/PropertyDataMessage.cpp +++ b/dGame/dGameMessages/PropertyDataMessage.cpp @@ -6,6 +6,8 @@ #include "dLogger.h" #include "CDClientManager.h" +#include "CDPropertyTemplateTable.h" + void GameMessages::PropertyDataMessage::Serialize(RakNet::BitStream& stream) const { stream.Write(0); // - property id @@ -101,8 +103,7 @@ void GameMessages::PropertyDataMessage::Serialize(RakNet::BitStream& stream) con } GameMessages::PropertyDataMessage::PropertyDataMessage(uint32_t mapID) { - const auto propertyTemplate = CDClientManager::Instance()-> - GetTable("PropertyTemplate")->GetByMapID(mapID); + const auto propertyTemplate = CDClientManager::Instance().GetTable()->GetByMapID(mapID); TemplateID = propertyTemplate.id; ZoneId = propertyTemplate.mapID; diff --git a/dGame/dGameMessages/RequestServerProjectileImpact.h b/dGame/dGameMessages/RequestServerProjectileImpact.h new file mode 100644 index 00000000..090d8274 --- /dev/null +++ b/dGame/dGameMessages/RequestServerProjectileImpact.h @@ -0,0 +1,72 @@ +#ifndef __REQUESTSERVERPROJECTILEIMPACT__H__ +#define __REQUESTSERVERPROJECTILEIMPACT__H__ + +#include "dCommonVars.h" +#include "eGameMessageType.h" + +/* Notifying the server that a locally owned projectile impacted. Sent to the caster of the projectile + should always be the local char. */ +class RequestServerProjectileImpact { +public: + RequestServerProjectileImpact() { + i64LocalID = LWOOBJID_EMPTY; + i64TargetID = LWOOBJID_EMPTY; + } + + RequestServerProjectileImpact(std::string _sBitStream, LWOOBJID _i64LocalID = LWOOBJID_EMPTY, LWOOBJID _i64TargetID = LWOOBJID_EMPTY) { + i64LocalID = _i64LocalID; + i64TargetID = _i64TargetID; + sBitStream = _sBitStream; + } + + RequestServerProjectileImpact(RakNet::BitStream* stream) : RequestServerProjectileImpact() { + Deserialize(stream); + } + + ~RequestServerProjectileImpact() { + } + + void Serialize(RakNet::BitStream* stream) { + stream->Write(eGameMessageType::REQUEST_SERVER_PROJECTILE_IMPACT); + + stream->Write(i64LocalID != LWOOBJID_EMPTY); + if (i64LocalID != LWOOBJID_EMPTY) stream->Write(i64LocalID); + + stream->Write(i64TargetID != LWOOBJID_EMPTY); + if (i64TargetID != LWOOBJID_EMPTY) stream->Write(i64TargetID); + + uint32_t sBitStreamLength = sBitStream.length(); + stream->Write(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + stream->Write(sBitStream[k]); + } + + } + + bool Deserialize(RakNet::BitStream* stream) { + bool i64LocalIDIsDefault{}; + stream->Read(i64LocalIDIsDefault); + if (i64LocalIDIsDefault != 0) stream->Read(i64LocalID); + + bool i64TargetIDIsDefault{}; + stream->Read(i64TargetIDIsDefault); + if (i64TargetIDIsDefault != 0) stream->Read(i64TargetID); + + uint32_t sBitStreamLength{}; + stream->Read(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + unsigned char character; + stream->Read(character); + sBitStream.push_back(character); + } + + + return true; + } + + LWOOBJID i64LocalID; + LWOOBJID i64TargetID; + std::string sBitStream; +}; + +#endif //!__REQUESTSERVERPROJECTILEIMPACT__H__ diff --git a/dGame/dGameMessages/StartSkill.h b/dGame/dGameMessages/StartSkill.h new file mode 100644 index 00000000..40bc210f --- /dev/null +++ b/dGame/dGameMessages/StartSkill.h @@ -0,0 +1,142 @@ +#ifndef __STARTSKILL__H__ +#define __STARTSKILL__H__ + +#include "dCommonVars.h" +#include "NiPoint3.h" +#include "NiQuaternion.h" +#include "eGameMessageType.h" + +/** + * Same as sync skill but with different network options. An echo down to other clients that need to play the skill. + */ +class StartSkill { +public: + StartSkill() { + bUsedMouse = false; + consumableItemID = LWOOBJID_EMPTY; + fCasterLatency = 0.0f; + iCastType = 0; + lastClickedPosit = NiPoint3::ZERO; + optionalTargetID = LWOOBJID_EMPTY; + originatorRot = NiQuaternion::IDENTITY; + uiSkillHandle = 0; + } + + StartSkill(LWOOBJID _optionalOriginatorID, std::string _sBitStream, TSkillID _skillID, bool _bUsedMouse = false, LWOOBJID _consumableItemID = LWOOBJID_EMPTY, float _fCasterLatency = 0.0f, int32_t _iCastType = 0, NiPoint3 _lastClickedPosit = NiPoint3::ZERO, LWOOBJID _optionalTargetID = LWOOBJID_EMPTY, NiQuaternion _originatorRot = NiQuaternion::IDENTITY, uint32_t _uiSkillHandle = 0) { + bUsedMouse = _bUsedMouse; + consumableItemID = _consumableItemID; + fCasterLatency = _fCasterLatency; + iCastType = _iCastType; + lastClickedPosit = _lastClickedPosit; + optionalOriginatorID = _optionalOriginatorID; + optionalTargetID = _optionalTargetID; + originatorRot = _originatorRot; + sBitStream = _sBitStream; + skillID = _skillID; + uiSkillHandle = _uiSkillHandle; + } + + StartSkill(RakNet::BitStream* stream) : StartSkill() { + Deserialize(stream); + } + + ~StartSkill() { + } + + void Serialize(RakNet::BitStream* stream) { + stream->Write(eGameMessageType::START_SKILL); + + stream->Write(bUsedMouse); + + stream->Write(consumableItemID != LWOOBJID_EMPTY); + if (consumableItemID != LWOOBJID_EMPTY) stream->Write(consumableItemID); + + stream->Write(fCasterLatency != 0.0f); + if (fCasterLatency != 0.0f) stream->Write(fCasterLatency); + + stream->Write(iCastType != 0); + if (iCastType != 0) stream->Write(iCastType); + + stream->Write(lastClickedPosit != NiPoint3::ZERO); + if (lastClickedPosit != NiPoint3::ZERO) stream->Write(lastClickedPosit); + + stream->Write(optionalOriginatorID); + + stream->Write(optionalTargetID != LWOOBJID_EMPTY); + if (optionalTargetID != LWOOBJID_EMPTY) stream->Write(optionalTargetID); + + stream->Write(originatorRot != NiQuaternion::IDENTITY); + if (originatorRot != NiQuaternion::IDENTITY) stream->Write(originatorRot); + + uint32_t sBitStreamLength = sBitStream.length(); + stream->Write(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + stream->Write(sBitStream[k]); + } + + stream->Write(skillID); + + stream->Write(uiSkillHandle != 0); + if (uiSkillHandle != 0) stream->Write(uiSkillHandle); + } + + bool Deserialize(RakNet::BitStream* stream) { + stream->Read(bUsedMouse); + + bool consumableItemIDIsDefault{}; + stream->Read(consumableItemIDIsDefault); + if (consumableItemIDIsDefault != 0) stream->Read(consumableItemID); + + bool fCasterLatencyIsDefault{}; + stream->Read(fCasterLatencyIsDefault); + if (fCasterLatencyIsDefault != 0) stream->Read(fCasterLatency); + + bool iCastTypeIsDefault{}; + stream->Read(iCastTypeIsDefault); + if (iCastTypeIsDefault != 0) stream->Read(iCastType); + + bool lastClickedPositIsDefault{}; + stream->Read(lastClickedPositIsDefault); + if (lastClickedPositIsDefault != 0) stream->Read(lastClickedPosit); + + stream->Read(optionalOriginatorID); + + bool optionalTargetIDIsDefault{}; + stream->Read(optionalTargetIDIsDefault); + if (optionalTargetIDIsDefault != 0) stream->Read(optionalTargetID); + + bool originatorRotIsDefault{}; + stream->Read(originatorRotIsDefault); + if (originatorRotIsDefault != 0) stream->Read(originatorRot); + + uint32_t sBitStreamLength{}; + stream->Read(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + unsigned char character; + stream->Read(character); + sBitStream.push_back(character); + } + + stream->Read(skillID); + + bool uiSkillHandleIsDefault{}; + stream->Read(uiSkillHandleIsDefault); + if (uiSkillHandleIsDefault != 0) stream->Read(uiSkillHandle); + + return true; + } + + bool bUsedMouse = false; + LWOOBJID consumableItemID{}; + float fCasterLatency{}; + int32_t iCastType{}; + NiPoint3 lastClickedPosit{}; + LWOOBJID optionalOriginatorID{}; + LWOOBJID optionalTargetID{}; + NiQuaternion originatorRot{}; + std::string sBitStream = ""; + TSkillID skillID = 0; + uint32_t uiSkillHandle = 0; +}; + +#endif //!__STARTSKILL__H__ diff --git a/dGame/dGameMessages/SyncSkill.h b/dGame/dGameMessages/SyncSkill.h new file mode 100644 index 00000000..6485199e --- /dev/null +++ b/dGame/dGameMessages/SyncSkill.h @@ -0,0 +1,67 @@ +#ifndef __SYNCSKILL__H__ +#define __SYNCSKILL__H__ + +#include +#include + +#include "BitStream.h" +#include "eGameMessageType.h" + +/* Message to synchronize a skill cast */ +class SyncSkill { +public: + SyncSkill() { + bDone = false; + } + + SyncSkill(std::string _sBitStream, uint32_t _uiBehaviorHandle, uint32_t _uiSkillHandle, bool _bDone = false) { + bDone = _bDone; + sBitStream = _sBitStream; + uiBehaviorHandle = _uiBehaviorHandle; + uiSkillHandle = _uiSkillHandle; + } + + SyncSkill(RakNet::BitStream* stream) : SyncSkill() { + Deserialize(stream); + } + + ~SyncSkill() { + } + + void Serialize(RakNet::BitStream* stream) { + stream->Write(eGameMessageType::SYNC_SKILL); + + stream->Write(bDone); + uint32_t sBitStreamLength = sBitStream.length(); + stream->Write(sBitStreamLength); + for (unsigned int k = 0; k < sBitStreamLength; k++) { + stream->Write(sBitStream[k]); + } + + stream->Write(uiBehaviorHandle); + stream->Write(uiSkillHandle); + } + + bool Deserialize(RakNet::BitStream* stream) { + stream->Read(bDone); + uint32_t sBitStreamLength{}; + stream->Read(sBitStreamLength); + for (uint32_t k = 0; k < sBitStreamLength; k++) { + unsigned char character; + stream->Read(character); + sBitStream.push_back(character); + } + + stream->Read(uiBehaviorHandle); + stream->Read(uiSkillHandle); + + return true; + } + + bool bDone{}; + std::string sBitStream{}; + uint32_t uiBehaviorHandle{}; + uint32_t uiSkillHandle{}; +}; + +#endif //!__SYNCSKILL__H__ diff --git a/dGame/dInventory/Inventory.cpp b/dGame/dInventory/Inventory.cpp index 151e3164..3d2c82ae 100644 --- a/dGame/dInventory/Inventory.cpp +++ b/dGame/dInventory/Inventory.cpp @@ -2,7 +2,11 @@ #include "GameMessages.h" #include "Game.h" #include "Item.h" +#include "InventoryComponent.h" #include "eItemType.h" +#include "eReplicaComponentType.h" + +#include "CDComponentsRegistryTable.h" std::vector Inventory::m_GameMasterRestrictedItems = { 1727, // GM Only - JetPack @@ -230,52 +234,52 @@ eInventoryType Inventory::FindInventoryTypeForLot(const LOT lot) { const auto itemType = static_cast(itemComponent.itemType); switch (itemType) { - case eItemType::ITEM_TYPE_BRICK: + case eItemType::BRICK: return BRICKS; - case eItemType::ITEM_TYPE_BEHAVIOR: + case eItemType::BEHAVIOR: return BEHAVIORS; - case eItemType::ITEM_TYPE_PROPERTY: + case eItemType::PROPERTY: return PROPERTY_DEEDS; - case eItemType::ITEM_TYPE_MODEL: - case eItemType::ITEM_TYPE_VEHICLE: - case eItemType::ITEM_TYPE_LOOT_MODEL: - case eItemType::ITEM_TYPE_MOUNT: + case eItemType::MODEL: + case eItemType::PET_INVENTORY_ITEM: + case eItemType::LOOT_MODEL: + case eItemType::VEHICLE: + case eItemType::MOUNT: return MODELS; - case eItemType::ITEM_TYPE_HAT: - case eItemType::ITEM_TYPE_HAIR: - case eItemType::ITEM_TYPE_NECK: - case eItemType::ITEM_TYPE_LEFT_HAND: - case eItemType::ITEM_TYPE_RIGHT_HAND: - case eItemType::ITEM_TYPE_LEGS: - case eItemType::ITEM_TYPE_LEFT_TRINKET: - case eItemType::ITEM_TYPE_RIGHT_TRINKET: - case eItemType::ITEM_TYPE_COLLECTIBLE: - case eItemType::ITEM_TYPE_CONSUMABLE: - case eItemType::ITEM_TYPE_CHEST: - case eItemType::ITEM_TYPE_EGG: - case eItemType::ITEM_TYPE_PET_FOOD: - case eItemType::ITEM_TYPE_PET_INVENTORY_ITEM: - case eItemType::ITEM_TYPE_PACKAGE: - case eItemType::ITEM_TYPE_CURRENCY: + case eItemType::HAT: + case eItemType::HAIR: + case eItemType::NECK: + case eItemType::LEFT_HAND: + case eItemType::RIGHT_HAND: + case eItemType::LEGS: + case eItemType::LEFT_TRINKET: + case eItemType::RIGHT_TRINKET: + case eItemType::COLLECTIBLE: + case eItemType::CONSUMABLE: + case eItemType::CHEST: + case eItemType::EGG: + case eItemType::PET_FOOD: + case eItemType::PACKAGE: + case eItemType::LUP_MODEL: return ITEMS; - case eItemType::ITEM_TYPE_QUEST_OBJECT: - case eItemType::ITEM_TYPE_UNKNOWN: + case eItemType::QUEST_OBJECT: + case eItemType::UNKNOWN: default: - return HIDDEN; + return QUEST; } } const CDItemComponent& Inventory::FindItemComponent(const LOT lot) { - auto* registry = CDClientManager::Instance()->GetTable("ComponentsRegistry"); + auto* registry = CDClientManager::Instance().GetTable(); - auto* itemComponents = CDClientManager::Instance()->GetTable("ItemComponent"); + auto* itemComponents = CDClientManager::Instance().GetTable(); - const auto componentId = registry->GetByIDAndType(lot, COMPONENT_TYPE_ITEM); + const auto componentId = registry->GetByIDAndType(lot, eReplicaComponentType::ITEM); if (componentId == 0) { Game::logger->Log("Inventory", "Failed to find item component for (%i)!", lot); @@ -289,9 +293,9 @@ const CDItemComponent& Inventory::FindItemComponent(const LOT lot) { } bool Inventory::IsValidItem(const LOT lot) { - auto* registry = CDClientManager::Instance()->GetTable("ComponentsRegistry"); + auto* registry = CDClientManager::Instance().GetTable(); - const auto componentId = registry->GetByIDAndType(lot, COMPONENT_TYPE_ITEM); + const auto componentId = registry->GetByIDAndType(lot, eReplicaComponentType::ITEM); return componentId != 0; } @@ -300,6 +304,12 @@ const std::vector& Inventory::GetAllGMItems() { return m_GameMasterRestrictedItems; } +void Inventory::DeleteAllItems() { + while (!this->items.empty()) { + if (items.begin()->second) items.begin()->second->SetCount(0); + } +} + Inventory::~Inventory() { for (auto item : items) { delete item.second; diff --git a/dGame/dInventory/Inventory.h b/dGame/dInventory/Inventory.h index 30f753da..5e0ac8d6 100644 --- a/dGame/dInventory/Inventory.h +++ b/dGame/dInventory/Inventory.h @@ -6,10 +6,11 @@ #include #include - +#include "CDItemComponentTable.h" #include "CDClientManager.h" #include "dCommonVars.h" +enum eInventoryType : uint32_t; class Item; class InventoryComponent; @@ -95,7 +96,7 @@ public: * @param lot the lot to find items for * @param ignoreEquipped ignores equipped items * @param ignoreBound ignores bound items - * @return item in the inventory for the provided LOT + * @return item with the lowest stack count in the inventory for the provided LOT */ Item* FindItemByLot(LOT lot, bool ignoreEquipped = false, bool ignoreBound = false) const; @@ -152,6 +153,11 @@ public: */ static const std::vector& GetAllGMItems(); + /** + * Remove ALL Items from this inventory. + */ + void DeleteAllItems(); + ~Inventory(); private: diff --git a/dGame/dInventory/Item.cpp b/dGame/dInventory/Item.cpp index 695ab47b..64f1dfbd 100644 --- a/dGame/dInventory/Item.cpp +++ b/dGame/dInventory/Item.cpp @@ -13,9 +13,17 @@ #include "PossessableComponent.h" #include "CharacterComponent.h" #include "eItemType.h" +#include "AssetManager.h" +#include "InventoryComponent.h" +#include "Loot.h" +#include "eObjectBits.h" +#include "eReplicaComponentType.h" +#include "eUseItemResponse.h" -class Inventory; - +#include "CDBrickIDTableTable.h" +#include "CDObjectSkillsTable.h" +#include "CDComponentsRegistryTable.h" +#include "CDPackageComponentTable.h" Item::Item(const LWOOBJID id, const LOT lot, Inventory* inventory, const uint32_t slot, const uint32_t count, const bool bound, const std::vector& config, const LWOOBJID parent, LWOOBJID subKey, eLootSourceType lootSourceType) { if (!Inventory::IsValidItem(lot)) { @@ -71,13 +79,13 @@ Item::Item( LWOOBJID id = ObjectIDManager::GenerateRandomObjectID(); - id = GeneralUtils::SetBit(id, OBJECT_BIT_CHARACTER); - id = GeneralUtils::SetBit(id, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(id, eObjectBits::CHARACTER); + GeneralUtils::SetBit(id, eObjectBits::PERSISTENT); const auto type = static_cast(info->itemType); - if (type == eItemType::ITEM_TYPE_MOUNT) { - id = GeneralUtils::SetBit(id, OBJECT_BIT_CLIENT); + if (type == eItemType::MOUNT) { + GeneralUtils::SetBit(id, eObjectBits::CLIENT); } this->id = id; @@ -92,7 +100,7 @@ Item::Item( Game::logger->Log("Item", "Move and equipped (%i) from (%i)", this->lot, this->inventory->GetType()); - EntityManager::Instance()->SerializeEntity(inventory->GetComponent()->GetParent()); + Game::entityManager->SerializeEntity(inventory->GetComponent()->GetParent()); } } @@ -237,7 +245,7 @@ bool Item::IsEquipped() const { } bool Item::Consume() { - auto* skillsTable = CDClientManager::Instance()->GetTable("ObjectSkills"); + auto* skillsTable = CDClientManager::Instance().GetTable(); auto skills = skillsTable->Query([=](const CDObjectSkills entry) { return entry.objectTemplate == static_cast(lot); @@ -252,7 +260,7 @@ bool Item::Consume() { } } - Game::logger->Log("Item", "Consumed (%i) / (%llu) with (%d)", lot, id, success); + Game::logger->LogDebug("Item", "Consumed LOT (%i) itemID (%llu). Success=(%d)", lot, id, success); GameMessages::SendUseItemResult(inventory->GetComponent()->GetParent(), lot, success); @@ -263,33 +271,83 @@ bool Item::Consume() { return success; } -void Item::UseNonEquip() { - const auto type = static_cast(info->itemType); - if (type == eItemType::ITEM_TYPE_MOUNT) { - GetInventory()->GetComponent()->HandlePossession(this); - } else if (type == eItemType::ITEM_TYPE_PET_INVENTORY_ITEM && subKey != LWOOBJID_EMPTY) { - const auto& databasePet = GetInventory()->GetComponent()->GetDatabasePet(subKey); - if (databasePet.lot != LOT_NULL) { - GetInventory()->GetComponent()->SpawnPet(this); - } - } else if (type == eItemType::ITEM_TYPE_PACKAGE) { - auto* compRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - const auto packageComponentId = compRegistryTable->GetByIDAndType(lot, COMPONENT_TYPE_PACKAGE); - auto* packCompTable = CDClientManager::Instance()->GetTable("PackageComponent"); - auto packages = packCompTable->Query([=](const CDPackageComponent entry) {return entry.id == static_cast(packageComponentId); }); +void Item::UseNonEquip(Item* item) { + LOT thisLot = this->GetLot(); + if (!GetInventory()) { + Game::logger->LogDebug("Item", "item %i has no inventory??", this->GetLot()); + return; + } - const auto success = !packages.empty(); - if (success) { - auto* entityParent = inventory->GetComponent()->GetParent(); - for (auto& pack : packages) { - std::unordered_map result{}; - result = LootGenerator::Instance().RollLootMatrix(entityParent, pack.LootMatrixIndex); - if (!inventory->GetComponent()->HasSpaceForLoot(result)) { - } - LootGenerator::Instance().GiveLoot(inventory->GetComponent()->GetParent(), result, eLootSourceType::LOOT_SOURCE_CONSUMPTION); - } - inventory->GetComponent()->RemoveItem(lot, 1); + auto* playerInventoryComponent = GetInventory()->GetComponent(); + if (!playerInventoryComponent) { + Game::logger->LogDebug("Item", "no inventory component attached to item id %llu lot %i", this->GetId(), this->GetLot()); + return; + } + + auto* playerEntity = playerInventoryComponent->GetParent(); + if (!playerEntity) { + Game::logger->LogDebug("Item", "no player entity attached to inventory? item id is %llu", this->GetId()); + return; + } + + const auto type = static_cast(info->itemType); + if (type == eItemType::MOUNT) { + playerInventoryComponent->HandlePossession(this); + // TODO Check if mounts are allowed to be spawned + } else if (type == eItemType::PET_INVENTORY_ITEM && subKey != LWOOBJID_EMPTY) { + const auto& databasePet = playerInventoryComponent->GetDatabasePet(subKey); + if (databasePet.lot != LOT_NULL) { + playerInventoryComponent->SpawnPet(this); } + // This precondition response is taken care of in SpawnPet(). + } else { + bool success = false; + auto inventory = item->GetInventory(); + if (inventory && inventory->GetType() == eInventoryType::ITEMS) { + auto* compRegistryTable = CDClientManager::Instance().GetTable(); + const auto packageComponentId = compRegistryTable->GetByIDAndType(lot, eReplicaComponentType::PACKAGE); + + if (packageComponentId == 0) return; + + auto* packCompTable = CDClientManager::Instance().GetTable(); + auto packages = packCompTable->Query([=](const CDPackageComponent entry) {return entry.id == static_cast(packageComponentId); }); + + auto success = !packages.empty(); + if (success) { + if (this->GetPreconditionExpression()->Check(playerInventoryComponent->GetParent())) { + auto* entityParent = playerInventoryComponent->GetParent(); + // Roll the loot for all the packages then see if it all fits. If it fits, give it to the player, otherwise don't. + std::unordered_map rolledLoot{}; + for (auto& pack : packages) { + auto thisPackage = LootGenerator::Instance().RollLootMatrix(entityParent, pack.LootMatrixIndex); + for (auto& loot : thisPackage) { + // If we already rolled this lot, add it to the existing one, otherwise create a new entry. + auto existingLoot = rolledLoot.find(loot.first); + if (existingLoot == rolledLoot.end()) { + rolledLoot.insert(loot); + } else { + existingLoot->second += loot.second; + } + } + } + if (playerInventoryComponent->HasSpaceForLoot(rolledLoot)) { + LootGenerator::Instance().GiveLoot(playerInventoryComponent->GetParent(), rolledLoot, eLootSourceType::CONSUMPTION); + item->SetCount(item->GetCount() - 1); + } else { + success = false; + } + } else { + GameMessages::SendUseItemRequirementsResponse( + playerInventoryComponent->GetParent()->GetObjectID(), + playerInventoryComponent->GetParent()->GetSystemAddress(), + eUseItemResponse::FailedPrecondition + ); + success = false; + } + } + } + Game::logger->LogDebug("Item", "Player %llu %s used item %i", playerEntity->GetObjectID(), success ? "successfully" : "unsuccessfully", thisLot); + GameMessages::SendUseItemResult(playerInventoryComponent->GetParent(), thisLot, success); } } @@ -298,6 +356,15 @@ void Item::Disassemble(const eInventoryType inventoryType) { if (data->GetKey() == u"assemblyPartLOTs") { auto modStr = data->GetValueAsString(); + // This shouldn't be null but always check your pointers. + if (GetInventory()) { + auto inventoryComponent = GetInventory()->GetComponent(); + if (inventoryComponent) { + auto entity = inventoryComponent->GetParent(); + if (entity) entity->SetVar(u"currentModifiedBuild", modStr); + } + } + std::vector modArray; std::stringstream ssData(modStr); @@ -313,42 +380,55 @@ void Item::Disassemble(const eInventoryType inventoryType) { } for (const auto mod : modArray) { - inventory->GetComponent()->AddItem(mod, 1, eLootSourceType::LOOT_SOURCE_DELETION, inventoryType); + inventory->GetComponent()->AddItem(mod, 1, eLootSourceType::DELETION, inventoryType); } } } } void Item::DisassembleModel() { - auto* table = CDClientManager::Instance()->GetTable("ComponentsRegistry"); + auto* table = CDClientManager::Instance().GetTable(); - const auto componentId = table->GetByIDAndType(GetLot(), COMPONENT_TYPE_RENDER); + const auto componentId = table->GetByIDAndType(GetLot(), eReplicaComponentType::RENDER); auto query = CDClientDatabase::CreatePreppedStmt( - "SELECT render_asset FROM RenderComponent WHERE id = ?;"); + "SELECT render_asset, LXFMLFolder FROM RenderComponent WHERE id = ?;"); query.bind(1, (int)componentId); auto result = query.execQuery(); - if (result.eof()) { + if (result.eof() || result.fieldIsNull(0)) { return; } - std::string renderAsset = result.fieldIsNull(0) ? "" : std::string(result.getStringField(0)); - std::vector renderAssetSplit = GeneralUtils::SplitString(renderAsset, '\\'); + std::string renderAsset = std::string(result.getStringField(0)); + std::string lxfmlFolderName = std::string(result.getStringField(1)); - std::string lxfmlPath = "res/BrickModels/" + GeneralUtils::SplitString(renderAssetSplit.back(), '.')[0] + ".lxfml"; - std::ifstream file(lxfmlPath); + std::vector renderAssetSplit = GeneralUtils::SplitString(renderAsset, '\\'); + if (renderAssetSplit.size() == 0) return; + + std::string lxfmlPath = "BrickModels/" + lxfmlFolderName + "/" + GeneralUtils::SplitString(renderAssetSplit.back(), '.').at(0) + ".lxfml"; + auto buffer = Game::assetManager->GetFileAsBuffer(lxfmlPath.c_str()); + + if (!buffer.m_Success) { + Game::logger->Log("Item", "Failed to load %s to disassemble model into bricks, check that this file exists", lxfmlPath.c_str()); + return; + } + + std::istream file(&buffer); result.finalize(); if (!file.good()) { + buffer.close(); return; } std::stringstream data; data << file.rdbuf(); + buffer.close(); + if (data.str().empty()) { return; } @@ -387,7 +467,7 @@ void Item::DisassembleModel() { currentBrick = currentBrick->NextSiblingElement(searchTerm.c_str()); } - auto* brickIDTable = CDClientManager::Instance()->GetTable("BrickIDTable"); + auto* brickIDTable = CDClientManager::Instance().GetTable(); for (unsigned int part : parts) { const auto brickID = brickIDTable->Query([=](const CDBrickIDTable& entry) { @@ -398,7 +478,7 @@ void Item::DisassembleModel() { continue; } - GetInventory()->GetComponent()->AddItem(brickID[0].NDObjectID, 1, eLootSourceType::LOOT_SOURCE_DELETION); + GetInventory()->GetComponent()->AddItem(brickID[0].NDObjectID, 1, eLootSourceType::DELETION); } } diff --git a/dGame/dInventory/Item.h b/dGame/dInventory/Item.h index db7e246a..be2359ef 100644 --- a/dGame/dInventory/Item.h +++ b/dGame/dInventory/Item.h @@ -6,6 +6,8 @@ #include "CDClientManager.h" #include "dLogger.h" #include "Preconditions.h" +#include "eInventoryType.h" +#include "eLootSourceType.h" /** * An item that can be stored in an inventory and optionally consumed or equipped @@ -37,7 +39,7 @@ public: const std::vector& config, LWOOBJID parent, LWOOBJID subKey, - eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE + eLootSourceType lootSourceType = eLootSourceType::NONE ); /** @@ -64,7 +66,7 @@ public: bool isModMoveAndEquip = false, LWOOBJID subKey = LWOOBJID_EMPTY, bool bound = false, - eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE + eLootSourceType lootSourceType = eLootSourceType::NONE ); ~Item(); @@ -88,7 +90,7 @@ public: * @param disassemble if items were removed, this returns all the sub parts of the item individually if it had assembly part lots * @param showFlyingLoot shows flying loot to the client, if not silent */ - void SetCount(uint32_t value, bool silent = false, bool disassemble = true, bool showFlyingLoot = true, eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE); + void SetCount(uint32_t value, bool silent = false, bool disassemble = true, bool showFlyingLoot = true, eLootSourceType lootSourceType = eLootSourceType::NONE); /** * Returns the number of items this item represents (e.g. for stacks) @@ -194,7 +196,7 @@ public: /** * Uses this item if its non equip, essentially an interface for the linked GM */ - void UseNonEquip(); + void UseNonEquip(Item* item); /** * Disassembles the part LOTs of this item back into the inventory, if it has any diff --git a/dGame/dInventory/ItemSet.cpp b/dGame/dInventory/ItemSet.cpp index 5063139e..a8e58739 100644 --- a/dGame/dInventory/ItemSet.cpp +++ b/dGame/dInventory/ItemSet.cpp @@ -6,8 +6,11 @@ #include "CDClientDatabase.h" #include "Game.h" #include "MissionComponent.h" +#include "eMissionTaskType.h" #include +#include "CDSkillBehaviorTable.h" + ItemSet::ItemSet(const uint32_t id, InventoryComponent* inventoryComponent) { this->m_ID = id; this->m_InventoryComponent = inventoryComponent; @@ -126,11 +129,11 @@ void ItemSet::OnEquip(const LOT lot) { auto* missionComponent = m_InventoryComponent->GetParent()->GetComponent(); for (const auto skill : skillSet) { - auto* skillTable = CDClientManager::Instance()->GetTable("SkillBehavior"); + auto* skillTable = CDClientManager::Instance().GetTable(); const auto behaviorId = skillTable->GetSkillByID(skill).behaviorID; - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SKILL, skill); + missionComponent->Progress(eMissionTaskType::USE_SKILL, skill); skillComponent->HandleUnmanaged(behaviorId, m_InventoryComponent->GetParent()->GetObjectID()); } @@ -158,7 +161,7 @@ void ItemSet::OnUnEquip(const LOT lot) { const auto& skillComponent = m_InventoryComponent->GetParent()->GetComponent(); for (const auto skill : skillSet) { - auto* skillTable = CDClientManager::Instance()->GetTable("SkillBehavior"); + auto* skillTable = CDClientManager::Instance().GetTable(); const auto behaviorId = skillTable->GetSkillByID(skill).behaviorID; @@ -180,9 +183,9 @@ void ItemSet::Update(float deltaTime) { } } -void ItemSet::TriggerPassiveAbility(PassiveAbilityTrigger trigger) { +void ItemSet::TriggerPassiveAbility(PassiveAbilityTrigger trigger, Entity* target) { for (auto& passiveAbility : m_PassiveAbilities) { - passiveAbility.Trigger(trigger); + passiveAbility.Trigger(trigger, target); } } diff --git a/dGame/dInventory/ItemSet.h b/dGame/dInventory/ItemSet.h index 7c713f03..d167cfaa 100644 --- a/dGame/dInventory/ItemSet.h +++ b/dGame/dInventory/ItemSet.h @@ -52,7 +52,7 @@ public: * Triggers all the passive abilities in this item set that match this trigger * @param trigger the trigger to use to trigger passive abilities */ - void TriggerPassiveAbility(PassiveAbilityTrigger trigger); + void TriggerPassiveAbility(PassiveAbilityTrigger trigger, Entity* target = nullptr); /** * Returns the skills that can be equipped for a specified amount of equipped items diff --git a/dGame/dInventory/ItemSetPassiveAbility.cpp b/dGame/dInventory/ItemSetPassiveAbility.cpp index d90f6e4e..3030904d 100644 --- a/dGame/dInventory/ItemSetPassiveAbility.cpp +++ b/dGame/dInventory/ItemSetPassiveAbility.cpp @@ -3,7 +3,7 @@ #include "DestroyableComponent.h" #include "SkillComponent.h" #include "ItemSet.h" -#include "ItemSetPassiveAbilityID.h" +#include "eItemSetPassiveAbilityID.h" ItemSetPassiveAbility::ItemSetPassiveAbility(PassiveAbilityTrigger trigger, Entity* parent, ItemSet* itemSet) { m_Trigger = trigger; @@ -16,12 +16,12 @@ ItemSetPassiveAbility::ItemSetPassiveAbility(PassiveAbilityTrigger trigger, Enti ItemSetPassiveAbility::~ItemSetPassiveAbility() { } -void ItemSetPassiveAbility::Trigger(PassiveAbilityTrigger trigger) { +void ItemSetPassiveAbility::Trigger(PassiveAbilityTrigger trigger, Entity* target) { if (m_Trigger != trigger || m_Cooldown > 0.0f) { return; } - Activate(); + Activate(target); } void ItemSetPassiveAbility::Update(float deltaTime) { @@ -30,9 +30,9 @@ void ItemSetPassiveAbility::Update(float deltaTime) { } } -void ItemSetPassiveAbility::Activate() { +void ItemSetPassiveAbility::Activate(Entity* target) { if (m_Trigger == PassiveAbilityTrigger::EnemySmashed) { - OnEnemySmshed(); + OnEnemySmshed(target); return; } @@ -44,33 +44,33 @@ void ItemSetPassiveAbility::Activate() { return; } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); - const auto id = static_cast(m_ItemSet->GetID()); + const auto id = static_cast(m_ItemSet->GetID()); const auto parentID = m_Parent->GetObjectID(); const auto equippedCount = m_ItemSet->GetEquippedCount(); switch (id) { // Assembly - case ItemSetPassiveAbilityID::InventorRank1: - case ItemSetPassiveAbilityID::SummonerRank1: - case ItemSetPassiveAbilityID::EngineerRank1: { + case eItemSetPassiveAbilityID::InventorRank1: + case eItemSetPassiveAbilityID::SummonerRank1: + case eItemSetPassiveAbilityID::EngineerRank1: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(394, 4401, parentID); break; } - case ItemSetPassiveAbilityID::InventorRank2: - case ItemSetPassiveAbilityID::SummonerRank2: - case ItemSetPassiveAbilityID::EngineerRank2: { + case eItemSetPassiveAbilityID::InventorRank2: + case eItemSetPassiveAbilityID::SummonerRank2: + case eItemSetPassiveAbilityID::EngineerRank2: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(581, 9433, parentID); break; } - case ItemSetPassiveAbilityID::InventorRank3: - case ItemSetPassiveAbilityID::SummonerRank3: - case ItemSetPassiveAbilityID::EngineerRank3: { + case eItemSetPassiveAbilityID::InventorRank3: + case eItemSetPassiveAbilityID::SummonerRank3: + case eItemSetPassiveAbilityID::EngineerRank3: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(582, 9435, parentID); @@ -78,57 +78,57 @@ void ItemSetPassiveAbility::Activate() { } // Sentinel - case ItemSetPassiveAbilityID::KnightRank1: { + case eItemSetPassiveAbilityID::KnightRank1: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(559, 8884, parentID); break; } - case ItemSetPassiveAbilityID::KnightRank2: { + case eItemSetPassiveAbilityID::KnightRank2: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(560, 8885, parentID); break; } - case ItemSetPassiveAbilityID::KnightRank3: { + case eItemSetPassiveAbilityID::KnightRank3: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(561, 8890, parentID); break; } - case ItemSetPassiveAbilityID::SpaceRangerRank1: { + case eItemSetPassiveAbilityID::SpaceRangerRank1: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(1101, 24612, parentID); break; } - case ItemSetPassiveAbilityID::SpaceRangerRank2: { + case eItemSetPassiveAbilityID::SpaceRangerRank2: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(1102, 24617, parentID); break; } - case ItemSetPassiveAbilityID::SpaceRangerRank3: { + case eItemSetPassiveAbilityID::SpaceRangerRank3: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(1103, 24622, parentID); break; } - case ItemSetPassiveAbilityID::SamuraiRank1: { + case eItemSetPassiveAbilityID::SamuraiRank1: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(562, 8899, parentID); break; } - case ItemSetPassiveAbilityID::SamuraiRank2: { + case eItemSetPassiveAbilityID::SamuraiRank2: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(563, 8904, parentID); break; } - case ItemSetPassiveAbilityID::SamuraiRank3: { + case eItemSetPassiveAbilityID::SamuraiRank3: { if (equippedCount < 4) return; m_Cooldown = 11.0f; skillComponent->CalculateBehavior(564, 8909, parentID); @@ -143,47 +143,47 @@ void ItemSetPassiveAbility::Activate() { std::vector ItemSetPassiveAbility::FindAbilities(uint32_t itemSetID, Entity* parent, ItemSet* itemSet) { std::vector abilities; - switch (static_cast(itemSetID)) { + switch (static_cast(itemSetID)) { // Assembly - case ItemSetPassiveAbilityID::SummonerRank1: - case ItemSetPassiveAbilityID::SummonerRank2: - case ItemSetPassiveAbilityID::SummonerRank3: - case ItemSetPassiveAbilityID::InventorRank1: - case ItemSetPassiveAbilityID::InventorRank2: - case ItemSetPassiveAbilityID::InventorRank3: - case ItemSetPassiveAbilityID::EngineerRank1: - case ItemSetPassiveAbilityID::EngineerRank2: - case ItemSetPassiveAbilityID::EngineerRank3: { + case eItemSetPassiveAbilityID::SummonerRank1: + case eItemSetPassiveAbilityID::SummonerRank2: + case eItemSetPassiveAbilityID::SummonerRank3: + case eItemSetPassiveAbilityID::InventorRank1: + case eItemSetPassiveAbilityID::InventorRank2: + case eItemSetPassiveAbilityID::InventorRank3: + case eItemSetPassiveAbilityID::EngineerRank1: + case eItemSetPassiveAbilityID::EngineerRank2: + case eItemSetPassiveAbilityID::EngineerRank3: { abilities.emplace_back(PassiveAbilityTrigger::AssemblyImagination, parent, itemSet); break; } // Sentinel - case ItemSetPassiveAbilityID::KnightRank1: - case ItemSetPassiveAbilityID::KnightRank2: - case ItemSetPassiveAbilityID::KnightRank3: - case ItemSetPassiveAbilityID::SpaceRangerRank1: - case ItemSetPassiveAbilityID::SpaceRangerRank2: - case ItemSetPassiveAbilityID::SpaceRangerRank3: - case ItemSetPassiveAbilityID::SamuraiRank1: - case ItemSetPassiveAbilityID::SamuraiRank2: - case ItemSetPassiveAbilityID::SamuraiRank3: { + case eItemSetPassiveAbilityID::KnightRank1: + case eItemSetPassiveAbilityID::KnightRank2: + case eItemSetPassiveAbilityID::KnightRank3: + case eItemSetPassiveAbilityID::SpaceRangerRank1: + case eItemSetPassiveAbilityID::SpaceRangerRank2: + case eItemSetPassiveAbilityID::SpaceRangerRank3: + case eItemSetPassiveAbilityID::SamuraiRank1: + case eItemSetPassiveAbilityID::SamuraiRank2: + case eItemSetPassiveAbilityID::SamuraiRank3: { abilities.emplace_back(PassiveAbilityTrigger::SentinelArmor, parent, itemSet); abilities.emplace_back(PassiveAbilityTrigger::EnemySmashed, parent, itemSet); break; } // Paradox - case ItemSetPassiveAbilityID::BatLord: - case ItemSetPassiveAbilityID::SpaceMarauderRank1: - case ItemSetPassiveAbilityID::SpaceMarauderRank2: - case ItemSetPassiveAbilityID::SpaceMarauderRank3: - case ItemSetPassiveAbilityID::SorcererRank1: - case ItemSetPassiveAbilityID::SorcererRank2: - case ItemSetPassiveAbilityID::SorcererRank3: - case ItemSetPassiveAbilityID::ShinobiRank1: - case ItemSetPassiveAbilityID::ShinobiRank2: - case ItemSetPassiveAbilityID::ShinobiRank3: { + case eItemSetPassiveAbilityID::BatLord: + case eItemSetPassiveAbilityID::SpaceMarauderRank1: + case eItemSetPassiveAbilityID::SpaceMarauderRank2: + case eItemSetPassiveAbilityID::SpaceMarauderRank3: + case eItemSetPassiveAbilityID::SorcererRank1: + case eItemSetPassiveAbilityID::SorcererRank2: + case eItemSetPassiveAbilityID::SorcererRank3: + case eItemSetPassiveAbilityID::ShinobiRank1: + case eItemSetPassiveAbilityID::ShinobiRank2: + case eItemSetPassiveAbilityID::ShinobiRank3: { abilities.emplace_back(PassiveAbilityTrigger::EnemySmashed, parent, itemSet); break; @@ -195,7 +195,7 @@ std::vector ItemSetPassiveAbility::FindAbilities(uint32_t return abilities; } -void ItemSetPassiveAbility::OnEnemySmshed() { +void ItemSetPassiveAbility::OnEnemySmshed(Entity* target) { auto* destroyableComponent = m_Parent->GetComponent(); auto* skillComponent = m_Parent->GetComponent(); @@ -203,112 +203,112 @@ void ItemSetPassiveAbility::OnEnemySmshed() { return; } - EntityManager::Instance()->SerializeEntity(m_Parent); + Game::entityManager->SerializeEntity(m_Parent); - const auto id = static_cast(m_ItemSet->GetID()); + const auto id = static_cast(m_ItemSet->GetID()); const auto parentID = m_Parent->GetObjectID(); const auto equippedCount = m_ItemSet->GetEquippedCount(); switch (id) { // Bat Lord - case ItemSetPassiveAbilityID::BatLord: { + case eItemSetPassiveAbilityID::BatLord: { if (equippedCount < 5) return; destroyableComponent->Heal(3); break; } // Sentinel - case ItemSetPassiveAbilityID::KnightRank1: { + case eItemSetPassiveAbilityID::KnightRank1: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::KnightRank2: { + case eItemSetPassiveAbilityID::KnightRank2: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::KnightRank3: { + case eItemSetPassiveAbilityID::KnightRank3: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::SpaceRangerRank1: { + case eItemSetPassiveAbilityID::SpaceRangerRank1: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::SpaceRangerRank2: { + case eItemSetPassiveAbilityID::SpaceRangerRank2: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::SpaceRangerRank3: { + case eItemSetPassiveAbilityID::SpaceRangerRank3: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::SamuraiRank1: { + case eItemSetPassiveAbilityID::SamuraiRank1: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::SamuraiRank2: { + case eItemSetPassiveAbilityID::SamuraiRank2: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } - case ItemSetPassiveAbilityID::SamuraiRank3: { + case eItemSetPassiveAbilityID::SamuraiRank3: { if (equippedCount < 5) return; destroyableComponent->Repair(1); break; } // Paradox - case ItemSetPassiveAbilityID::SpaceMarauderRank1: { + case eItemSetPassiveAbilityID::SpaceMarauderRank1: { if (equippedCount < 4) return; destroyableComponent->Imagine(1); break; } - case ItemSetPassiveAbilityID::SpaceMarauderRank2: { + case eItemSetPassiveAbilityID::SpaceMarauderRank2: { if (equippedCount < 4) return; destroyableComponent->Imagine(2); break; } - case ItemSetPassiveAbilityID::SpaceMarauderRank3: { + case eItemSetPassiveAbilityID::SpaceMarauderRank3: { if (equippedCount < 4) return; destroyableComponent->Imagine(3); break; } - case ItemSetPassiveAbilityID::ShinobiRank1: { + case eItemSetPassiveAbilityID::ShinobiRank1: { if (equippedCount < 4) return; destroyableComponent->Imagine(1); break; } - case ItemSetPassiveAbilityID::ShinobiRank2: { + case eItemSetPassiveAbilityID::ShinobiRank2: { if (equippedCount < 4) return; destroyableComponent->Imagine(2); break; } - case ItemSetPassiveAbilityID::ShinobiRank3: { + case eItemSetPassiveAbilityID::ShinobiRank3: { if (equippedCount < 4) return; destroyableComponent->Imagine(3); break; } - case ItemSetPassiveAbilityID::SorcererRank1: { + case eItemSetPassiveAbilityID::SorcererRank1: { if (equippedCount < 4) return; destroyableComponent->Imagine(1); break; } - case ItemSetPassiveAbilityID::SorcererRank2: { + case eItemSetPassiveAbilityID::SorcererRank2: { if (equippedCount < 4) return; destroyableComponent->Imagine(2); break; } - case ItemSetPassiveAbilityID::SorcererRank3: { + case eItemSetPassiveAbilityID::SorcererRank3: { if (equippedCount < 4) return; destroyableComponent->Imagine(3); break; diff --git a/dGame/dInventory/ItemSetPassiveAbility.h b/dGame/dInventory/ItemSetPassiveAbility.h index ff945df2..8735e695 100644 --- a/dGame/dInventory/ItemSetPassiveAbility.h +++ b/dGame/dInventory/ItemSetPassiveAbility.h @@ -30,12 +30,12 @@ public: * Attempts to trigger a passive ability for this item set, if this is the wrong trigger this is a no-op * @param trigger the trigger to attempt to fire */ - void Trigger(PassiveAbilityTrigger trigger); + void Trigger(PassiveAbilityTrigger trigger, Entity* target = nullptr); /** * Activates the passive ability */ - void Activate(); + void Activate(Entity* target = nullptr); /** * Finds all the passive abilities associated with a certain item set @@ -47,7 +47,7 @@ public: static std::vector FindAbilities(uint32_t itemSetID, Entity* parent, ItemSet* itemSet); private: - void OnEnemySmshed(); + void OnEnemySmshed(Entity* target = nullptr); /** * The means of triggering this ability diff --git a/dGame/dInventory/ItemSetPassiveAbilityID.h b/dGame/dInventory/ItemSetPassiveAbilityID.h deleted file mode 100644 index 92caea19..00000000 --- a/dGame/dInventory/ItemSetPassiveAbilityID.h +++ /dev/null @@ -1,105 +0,0 @@ -#pragma once - -/** -2 Engineer (Rank 1) Item Set -3 Engineer (Rank 2) Item Set -4 Engineer (Rank 3) Item Set -7 Knight (Rank 1) Item Set -8 Knight (Rank 2) Item Set -9 Knight (Rank 3) Item Set -10 Space Ranger (Rank 1) Item Set -11 Space Ranger (Rank 2) Item Set -12 Space Ranger (Rank 3) Item Set -13 Samurai (Rank 1) Item Set -14 Samurai (Rank 2) Item Set -15 Samurai (Rank 3) Item Set -16 Sorcerer (Rank 1) Item Set -17 Sorcerer (Rank 2) Item Set -18 Sorcerer (Rank 3) Item Set -19 Space Marauder (Rank 1) Item Set -20 Space Marauder (Rank 2) Item Set -21 Space Marauder (Rank 3) Item Set -22 Shinobi (Rank 1) Item Set -23 Shinobi (Rank 2) Item Set -24 Shinobi (Rank 3) Item Set -25 Inventor (Rank 1) Item Set -26 Inventor (Rank 2) Item Set -27 Inventor (Rank 3) Item Set -28 Summoner (Rank 1) Item Set -29 Summoner (Rank 2) Item Set -30 Summoner (Rank 3) Item Set -31 Adventurer (Rank 1) Item Set -32 Adventurer (Rank 2) Item Set -33 Adventurer (Rank 3) Item Set -34 Daredevil (Rank 1) Item Set -35 Daredevil (Rank 2) Item Set -36 Daredevil (Rank 3) Item Set -37 Buccaneer (Rank 1) Item Set -38 Buccaneer (Rank 2) Item Set -39 Buccaneer (Rank 3) Item Set -40 Bone Suit Item Set -41 Imagination Spinjitzu Item Set -42 Bat Lord Item Set -43 Mosaic Jester Item Set -44 Explorien Bot Item Set -45 [Unnamed] Item Set -46 [Unnamed] Item Set -47 [Unnamed] Item Set -48 Earth Spinjitzu Item Set -49 [Unnamed] Item Set -50 Fire Spinjitzu Item Set -51 Ice Spinjitzu Item Set -52 Lightning Spinjitzu Item Set - */ -enum class ItemSetPassiveAbilityID -{ - EngineerRank1 = 2, - EngineerRank2 = 3, - EngineerRank3 = 4, - KnightRank1 = 7, - KnightRank2 = 8, - KnightRank3 = 9, - SpaceRangerRank1 = 10, - SpaceRangerRank2 = 11, - SpaceRangerRank3 = 12, - SamuraiRank1 = 13, - SamuraiRank2 = 14, - SamuraiRank3 = 15, - SorcererRank1 = 16, - SorcererRank2 = 17, - SorcererRank3 = 18, - SpaceMarauderRank1 = 19, - SpaceMarauderRank2 = 20, - SpaceMarauderRank3 = 21, - ShinobiRank1 = 22, - ShinobiRank2 = 23, - ShinobiRank3 = 24, - InventorRank1 = 25, - InventorRank2 = 26, - InventorRank3 = 27, - SummonerRank1 = 28, - SummonerRank2 = 29, - SummonerRank3 = 30, - AdventurerRank1 = 31, - AdventurerRank2 = 32, - AdventurerRank3 = 33, - DaredevilRank1 = 34, - DaredevilRank2 = 35, - DaredevilRank3 = 36, - BuccaneerRank1 = 37, - BuccaneerRank2 = 38, - BuccaneerRank3 = 39, - BoneSuit = 40, - ImaginationSpinjitzu = 41, - BatLord = 42, - MosaicJester = 43, - ExplorienBot = 44, - Unnamed1 = 45, - Unnamed2 = 46, - Unnamed3 = 47, - EarthSpinjitzu = 48, - Unnamed4 = 49, - FireSpinjitzu = 50, - IceSpinjitzu = 51, - LightningSpinjitzu = 52 -}; diff --git a/dGame/dMission/Mission.cpp b/dGame/dMission/Mission.cpp index 6020e51c..34641335 100644 --- a/dGame/dMission/Mission.cpp +++ b/dGame/dMission/Mission.cpp @@ -12,12 +12,20 @@ #include "GameMessages.h" #include "Mail.h" #include "MissionComponent.h" -#include "RacingTaskParam.h" -#include "dLocale.h" +#include "eRacingTaskParam.h" #include "dLogger.h" #include "dServer.h" #include "dZoneManager.h" +#include "InventoryComponent.h" +#include "User.h" #include "Database.h" +#include "WorldConfig.h" +#include "eMissionState.h" +#include "eMissionTaskType.h" +#include "eMissionLockState.h" +#include "eReplicaComponentType.h" + +#include "CDMissionEmailTable.h" Mission::Mission(MissionComponent* missionComponent, const uint32_t missionId) { m_MissionComponent = missionComponent; @@ -26,13 +34,13 @@ Mission::Mission(MissionComponent* missionComponent, const uint32_t missionId) { m_Timestamp = 0; - m_UniqueMissionID = dZoneManager::Instance()->GetUniqueMissionIdStartingValue(); + m_UniqueMissionID = Game::zoneManager->GetUniqueMissionIdStartingValue(); m_Reward = 0; - m_State = MissionState::MISSION_STATE_UNKNOWN; + m_State = eMissionState::UNKNOWN; - auto* missionsTable = CDClientManager::Instance()->GetTable("Missions"); + auto* missionsTable = CDClientManager::Instance().GetTable(); info = missionsTable->GetPtrByMissionID(missionId); @@ -42,7 +50,7 @@ Mission::Mission(MissionComponent* missionComponent, const uint32_t missionId) { return; } - auto* tasksTable = CDClientManager::Instance()->GetTable("MissionTasks"); + auto* tasksTable = CDClientManager::Instance().GetTable(); auto tasks = tasksTable->GetByMissionID(missionId); @@ -58,7 +66,7 @@ Mission::Mission(MissionComponent* missionComponent, const uint32_t missionId) { void Mission::LoadFromXml(tinyxml2::XMLElement* element) { // Start custom XML if (element->Attribute("state") != nullptr) { - m_State = static_cast(std::stoul(element->Attribute("state"))); + m_State = static_cast(std::stoul(element->Attribute("state"))); } // End custom XML @@ -83,8 +91,8 @@ void Mission::LoadFromXml(tinyxml2::XMLElement* element) { const auto type = m_Tasks[index]->GetType(); - if (type == MissionTaskType::MISSION_TASK_TYPE_ENVIRONMENT || - type == MissionTaskType::MISSION_TASK_TYPE_VISIT_PROPERTY) { + if (type == eMissionTaskType::COLLECTION || + type == eMissionTaskType::VISIT_PROPERTY) { std::vector uniques; const auto value = std::stoul(task->Attribute("v")); @@ -98,7 +106,7 @@ void Mission::LoadFromXml(tinyxml2::XMLElement* element) { uniques.push_back(unique); - if (m_MissionComponent != nullptr && type == MissionTaskType::MISSION_TASK_TYPE_ENVIRONMENT) { + if (m_MissionComponent != nullptr && type == eMissionTaskType::COLLECTION) { m_MissionComponent->AddCollectible(unique); } @@ -142,8 +150,8 @@ void Mission::UpdateXml(tinyxml2::XMLElement* element) { } for (auto* task : m_Tasks) { - if (task->GetType() == MissionTaskType::MISSION_TASK_TYPE_ENVIRONMENT || - task->GetType() == MissionTaskType::MISSION_TASK_TYPE_VISIT_PROPERTY) { + if (task->GetType() == eMissionTaskType::COLLECTION || + task->GetType() == eMissionTaskType::VISIT_PROPERTY) { auto* child = element->GetDocument()->NewElement("sv"); @@ -170,7 +178,7 @@ void Mission::UpdateXml(tinyxml2::XMLElement* element) { } bool Mission::IsValidMission(const uint32_t missionId) { - auto* table = CDClientManager::Instance()->GetTable("Missions"); + auto* table = CDClientManager::Instance().GetTable(); const auto missions = table->Query([=](const CDMissions& entry) { return entry.id == static_cast(missionId); @@ -180,7 +188,7 @@ bool Mission::IsValidMission(const uint32_t missionId) { } bool Mission::IsValidMission(const uint32_t missionId, CDMissions& info) { - auto* table = CDClientManager::Instance()->GetTable("Missions"); + auto* table = CDClientManager::Instance().GetTable(); const auto missions = table->Query([=](const CDMissions& entry) { return entry.id == static_cast(missionId); @@ -227,7 +235,7 @@ std::vector Mission::GetTasks() const { return m_Tasks; } -MissionState Mission::GetMissionState() const { +eMissionState Mission::GetMissionState() const { return m_State; } @@ -244,47 +252,47 @@ bool Mission::IsRepeatable() const { } bool Mission::IsComplete() const { - return m_State == MissionState::MISSION_STATE_COMPLETE; + return m_State == eMissionState::COMPLETE; } bool Mission::IsActive() const { - return m_State == MissionState::MISSION_STATE_ACTIVE || m_State == MissionState::MISSION_STATE_COMPLETE_AVAILABLE; + return m_State == eMissionState::ACTIVE || m_State == eMissionState::COMPLETE_AVAILABLE; } void Mission::MakeActive() { - SetMissionState(m_Completions == 0 ? MissionState::MISSION_STATE_ACTIVE : MissionState::MISSION_STATE_COMPLETE_ACTIVE); + SetMissionState(m_Completions == 0 ? eMissionState::ACTIVE : eMissionState::COMPLETE_ACTIVE); } bool Mission::IsReadyToComplete() const { - return m_State == MissionState::MISSION_STATE_READY_TO_COMPLETE || m_State == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE; + return m_State == eMissionState::READY_TO_COMPLETE || m_State == eMissionState::COMPLETE_READY_TO_COMPLETE; } void Mission::MakeReadyToComplete() { - SetMissionState(m_Completions == 0 ? MissionState::MISSION_STATE_READY_TO_COMPLETE : MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE); + SetMissionState(m_Completions == 0 ? eMissionState::READY_TO_COMPLETE : eMissionState::COMPLETE_READY_TO_COMPLETE); } bool Mission::IsAvalible() const { - return m_State == MissionState::MISSION_STATE_AVAILABLE || m_State == MissionState::MISSION_STATE_COMPLETE_AVAILABLE; + return m_State == eMissionState::AVAILABLE || m_State == eMissionState::COMPLETE_AVAILABLE; } bool Mission::IsFetchMission() const { - return m_Tasks.size() == 1 && m_Tasks[0]->GetType() == MissionTaskType::MISSION_TASK_TYPE_MISSION_INTERACTION; + return m_Tasks.size() == 1 && m_Tasks[0]->GetType() == eMissionTaskType::TALK_TO_NPC; } void Mission::MakeAvalible() { - SetMissionState(m_Completions == 0 ? MissionState::MISSION_STATE_AVAILABLE : MissionState::MISSION_STATE_COMPLETE_AVAILABLE); + SetMissionState(m_Completions == 0 ? eMissionState::AVAILABLE : eMissionState::COMPLETE_AVAILABLE); } void Mission::Accept() { - SetMissionTypeState(MissionLockState::MISSION_LOCK_NEW, info->defined_type, info->defined_subtype); + SetMissionTypeState(eMissionLockState::NEW, info->defined_type, info->defined_subtype); - SetMissionState(m_Completions > 0 ? MissionState::MISSION_STATE_COMPLETE_ACTIVE : MissionState::MISSION_STATE_ACTIVE); + SetMissionState(m_Completions > 0 ? eMissionState::COMPLETE_ACTIVE : eMissionState::ACTIVE); Catchup(); } void Mission::Complete(const bool yieldRewards) { - if (m_State != MissionState::MISSION_STATE_ACTIVE && m_State != MissionState::MISSION_STATE_COMPLETE_ACTIVE) { + if (m_State != eMissionState::ACTIVE && m_State != eMissionState::COMPLETE_ACTIVE) { // If we are accepting a mission here there is no point to giving it a unique ID since we just complete it immediately. Accept(); } @@ -293,13 +301,13 @@ void Mission::Complete(const bool yieldRewards) { task->Complete(); } - SetMissionState(MissionState::MISSION_STATE_REWARDING, true); + SetMissionState(eMissionState::REWARDING, true); if (yieldRewards) { YieldRewards(); } - SetMissionState(MissionState::MISSION_STATE_COMPLETE); + SetMissionState(eMissionState::COMPLETE); m_Completions++; @@ -318,13 +326,13 @@ void Mission::Complete(const bool yieldRewards) { auto* missionComponent = entity->GetComponent(); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MISSION_COMPLETE, info->id); + missionComponent->Progress(eMissionTaskType::META, info->id); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, info->id, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_COMPLETE_ANY_RACING_TASK); + missionComponent->Progress(eMissionTaskType::RACING, info->id, (LWOOBJID)eRacingTaskParam::COMPLETE_ANY_RACING_TASK); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, info->id, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_COMPLETE_TRACK_TASKS); + missionComponent->Progress(eMissionTaskType::RACING, info->id, (LWOOBJID)eRacingTaskParam::COMPLETE_TRACK_TASKS); - auto* missionEmailTable = CDClientManager::Instance()->GetTable("MissionEmail"); + auto* missionEmailTable = CDClientManager::Instance().GetTable(); const auto missionId = GetMissionId(); @@ -335,13 +343,10 @@ void Mission::Complete(const bool yieldRewards) { for (const auto& email : missionEmails) { const auto missionEmailBase = "MissionEmail_" + std::to_string(email.ID) + "_"; - const auto senderLocale = missionEmailBase + "senderName"; - const auto announceLocale = missionEmailBase + "announceText"; - - if (email.messageType == 1 && Game::locale->HasPhrase(senderLocale)) { - const auto subject = dLocale::GetTemplate(missionEmailBase + "subjectText"); - const auto body = dLocale::GetTemplate(missionEmailBase + "bodyText"); - const auto sender = dLocale::GetTemplate(senderLocale); + if (email.messageType == 1) { + const auto subject = "%[" + missionEmailBase + "subjectText]"; + const auto body = "%[" + missionEmailBase + "bodyText]"; + const auto sender = "%[" + missionEmailBase + "senderName]"; Mail::SendMail(LWOOBJID_EMPTY, sender, GetAssociate(), subject, body, email.attachmentLOT, 1); } @@ -367,12 +372,12 @@ void Mission::CheckCompletion() { void Mission::Catchup() { auto* entity = GetAssociate(); - auto* inventory = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + auto* inventory = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); for (auto* task : m_Tasks) { const auto type = task->GetType(); - if (type == MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION) { + if (type == eMissionTaskType::GATHER) { for (auto target : task->GetAllTargets()) { const auto count = inventory->GetLotCountNonTransfer(target); @@ -382,8 +387,8 @@ void Mission::Catchup() { } } - if (type == MissionTaskType::MISSION_TASK_TYPE_PLAYER_FLAG) { - for (auto target : task->GetAllTargets()) { + if (type == eMissionTaskType::PLAYER_FLAG) { + for (int32_t target : task->GetAllTargets()) { const auto flag = GetUser()->GetLastUsedChar()->GetPlayerFlag(target); if (!flag) { @@ -417,7 +422,7 @@ void Mission::YieldRewards() { // Remove mission items for (auto* task : m_Tasks) { - if (task->GetType() != MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION) { + if (task->GetType() != eMissionTaskType::GATHER) { continue; } @@ -428,19 +433,19 @@ void Mission::YieldRewards() { for (const auto target : task->GetAllTargets()) { // This is how live did it. ONLY remove item collection items from the items and hidden inventories and none of the others. inventoryComponent->RemoveItem(target, task->GetClientInfo().targetValue, eInventoryType::ITEMS); - inventoryComponent->RemoveItem(target, task->GetClientInfo().targetValue, eInventoryType::HIDDEN); + inventoryComponent->RemoveItem(target, task->GetClientInfo().targetValue, eInventoryType::QUEST); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, target, LWOOBJID_EMPTY, "", -task->GetClientInfo().targetValue); + missionComponent->Progress(eMissionTaskType::GATHER, target, LWOOBJID_EMPTY, "", -task->GetClientInfo().targetValue); } } } int32_t coinsToSend = 0; if (info->LegoScore > 0) { - eLootSourceType lootSource = info->isMission ? eLootSourceType::LOOT_SOURCE_MISSION : eLootSourceType::LOOT_SOURCE_ACHIEVEMENT; - if (levelComponent->GetLevel() >= dZoneManager::Instance()->GetMaxLevel()) { + eLootSourceType lootSource = info->isMission ? eLootSourceType::MISSION : eLootSourceType::ACHIEVEMENT; + if (levelComponent->GetLevel() >= Game::zoneManager->GetWorldConfig()->levelCap) { // Since the character is at the level cap we reward them with coins instead of UScore. - coinsToSend += info->LegoScore * dZoneManager::Instance()->GetLevelCapCurrencyConversion(); + coinsToSend += info->LegoScore * Game::zoneManager->GetWorldConfig()->levelCapCurrencyConversion; } else { characterComponent->SetUScore(characterComponent->GetUScore() + info->LegoScore); GameMessages::SendModifyLEGOScore(entity, entity->GetSystemAddress(), info->LegoScore, lootSource); @@ -470,11 +475,11 @@ void Mission::YieldRewards() { count = 0; } - inventoryComponent->AddItem(pair.first, count, IsMission() ? eLootSourceType::LOOT_SOURCE_MISSION : eLootSourceType::LOOT_SOURCE_ACHIEVEMENT); + inventoryComponent->AddItem(pair.first, count, IsMission() ? eLootSourceType::MISSION : eLootSourceType::ACHIEVEMENT); } if (info->reward_currency_repeatable > 0 || coinsToSend > 0) { - eLootSourceType lootSource = info->isMission ? eLootSourceType::LOOT_SOURCE_MISSION : eLootSourceType::LOOT_SOURCE_ACHIEVEMENT; + eLootSourceType lootSource = info->isMission ? eLootSourceType::MISSION : eLootSourceType::ACHIEVEMENT; character->SetCoins(character->GetCoins() + info->reward_currency_repeatable + coinsToSend, lootSource); } @@ -503,11 +508,11 @@ void Mission::YieldRewards() { count = 0; } - inventoryComponent->AddItem(pair.first, count, IsMission() ? eLootSourceType::LOOT_SOURCE_MISSION : eLootSourceType::LOOT_SOURCE_ACHIEVEMENT); + inventoryComponent->AddItem(pair.first, count, IsMission() ? eLootSourceType::MISSION : eLootSourceType::ACHIEVEMENT); } if (info->reward_currency > 0 || coinsToSend > 0) { - eLootSourceType lootSource = info->isMission ? eLootSourceType::LOOT_SOURCE_MISSION : eLootSourceType::LOOT_SOURCE_ACHIEVEMENT; + eLootSourceType lootSource = info->isMission ? eLootSourceType::MISSION : eLootSourceType::ACHIEVEMENT; character->SetCoins(character->GetCoins() + info->reward_currency + coinsToSend, lootSource); } @@ -526,7 +531,7 @@ void Mission::YieldRewards() { } if (info->reward_reputation > 0) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_EARN_REPUTATION, 0, 0L, "", info->reward_reputation); + missionComponent->Progress(eMissionTaskType::EARN_REPUTATION, 0, 0L, "", info->reward_reputation); auto character = entity->GetComponent(); if (character) { character->SetReputation(character->GetReputation() + info->reward_reputation); @@ -542,7 +547,7 @@ void Mission::YieldRewards() { destroyableComponent->SetMaxImagination(destroyableComponent->GetMaxImagination() + static_cast(info->reward_maximagination), true); } - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); if (info->reward_emote > 0) { character->UnlockEmote(info->reward_emote); @@ -561,7 +566,7 @@ void Mission::YieldRewards() { } } -void Mission::Progress(MissionTaskType type, int32_t value, LWOOBJID associate, const std::string& targets, int32_t count) { +void Mission::Progress(eMissionTaskType type, int32_t value, LWOOBJID associate, const std::string& targets, int32_t count) { const auto isRemoval = count < 0; if (isRemoval && (IsComplete() || IsAchievement())) { @@ -585,7 +590,7 @@ void Mission::Progress(MissionTaskType type, int32_t value, LWOOBJID associate, } } -void Mission::SetMissionState(const MissionState state, const bool sendingRewards) { +void Mission::SetMissionState(const eMissionState state, const bool sendingRewards) { this->m_State = state; auto* entity = GetAssociate(); @@ -597,7 +602,7 @@ void Mission::SetMissionState(const MissionState state, const bool sendingReward GameMessages::SendNotifyMission(entity, entity->GetParentUser()->GetSystemAddress(), info->id, static_cast(state), sendingRewards); } -void Mission::SetMissionTypeState(MissionLockState state, const std::string& type, const std::string& subType) { +void Mission::SetMissionTypeState(eMissionLockState state, const std::string& type, const std::string& subType) { // TODO } diff --git a/dGame/dMission/Mission.h b/dGame/dMission/Mission.h index b8892f3d..b04c3548 100644 --- a/dGame/dMission/Mission.h +++ b/dGame/dMission/Mission.h @@ -10,9 +10,12 @@ #include "MissionTask.h" #include "dCommonVars.h" #include "Entity.h" -#include "MissionState.h" -#include "MissionLockState.h" +namespace tinyxml2 { + class XMLElement; +}; +enum class eMissionState : int; +enum class eMissionLockState : int; class MissionComponent; /** @@ -49,7 +52,7 @@ public: * Returns the current state of this mission * @return the current state of this mission */ - MissionState GetMissionState() const; + eMissionState GetMissionState() const; /** * Returns the database information that represents to this mission. @@ -98,12 +101,12 @@ public: * @param state the mission state to set * @param sendingRewards a flag indicating to the client that rewards wil lfollow */ - void SetMissionState(MissionState state, bool sendingRewards = false); + void SetMissionState(eMissionState state, bool sendingRewards = false); /** * Currently unimplemented */ - void SetMissionTypeState(MissionLockState state, const std::string& type, const std::string& subType); + void SetMissionTypeState(eMissionLockState state, const std::string& type, const std::string& subType); /** * Returns whether this mission is an achievement @@ -204,7 +207,7 @@ public: * @param targets optional multiple targets that need to be met for progression * @param count optional count to progress with */ - void Progress(MissionTaskType type, int32_t value, LWOOBJID associate = 0, const std::string& targets = "", int32_t count = 1); + void Progress(eMissionTaskType type, int32_t value, LWOOBJID associate = 0, const std::string& targets = "", int32_t count = 1); /** * Returns if the mission ID that's given belongs to an existing mission @@ -223,7 +226,7 @@ public: /** * @brief Returns the unique mission order ID - * + * * @return The unique order ID */ uint32_t GetUniqueMissionOrderID() { return m_UniqueMissionID; }; @@ -247,7 +250,7 @@ private: /** * The current state this mission is in */ - MissionState m_State; + eMissionState m_State; /** * The number of times the entity has completed this mission diff --git a/dGame/dMission/MissionLockState.h b/dGame/dMission/MissionLockState.h deleted file mode 100644 index 9fd2252a..00000000 --- a/dGame/dMission/MissionLockState.h +++ /dev/null @@ -1,13 +0,0 @@ -#pragma once - -#ifndef MISSIONLOCKSTATE_H -#define MISSIONLOCKSTATE_H - -enum class MissionLockState : int -{ - MISSION_LOCK_LOCKED, - MISSION_LOCK_NEW, - MISSION_LOCK_UNLOCKED, -}; - -#endif diff --git a/dGame/dMission/MissionPrerequisites.cpp b/dGame/dMission/MissionPrerequisites.cpp index 46fa73a6..ec4522b9 100644 --- a/dGame/dMission/MissionPrerequisites.cpp +++ b/dGame/dMission/MissionPrerequisites.cpp @@ -106,8 +106,8 @@ bool PrerequisiteExpression::Execute(const std::unordered_mapsub != 0) { // Special case for one Wisp Lee repeatable mission. a = mission->GetClientInfo().id == 1883 ? - mission->GetMissionState() == static_cast(this->sub) : - mission->GetMissionState() >= static_cast(this->sub); + mission->GetMissionState() == static_cast(this->sub) : + mission->GetMissionState() >= static_cast(this->sub); } else if (mission->IsComplete()) { a = true; } @@ -163,7 +163,7 @@ bool MissionPrerequisites::CheckPrerequisites(uint32_t missionId, const std::uno return index->second->Execute(missions); } - auto* missionsTable = CDClientManager::Instance()->GetTable("Missions"); + auto* missionsTable = CDClientManager::Instance().GetTable(); const auto missionEntries = missionsTable->Query([=](const CDMissions& entry) { return entry.id == static_cast(missionId); }); diff --git a/dGame/dMission/MissionState.h b/dGame/dMission/MissionState.h deleted file mode 100644 index fb1841d3..00000000 --- a/dGame/dMission/MissionState.h +++ /dev/null @@ -1,56 +0,0 @@ -#pragma once - -#ifndef __MISSIONSTATE__H__ -#define __MISSIONSTATE__H__ - -/** - * Represents the possible states a mission can be in - */ -enum class MissionState : int { - /** - * The mission state is unknown - */ - MISSION_STATE_UNKNOWN = -1, - - /** - * The mission is yielding rewards - */ - MISSION_STATE_REWARDING = 0, - - /** - * The mission can be accepted - */ - MISSION_STATE_AVAILABLE = 1, - - /** - * The mission has been accepted but not yet completed - */ - MISSION_STATE_ACTIVE = 2, - - /** - * All the tasks for the mission have been completed and the entity can turn the mission in to complete it - */ - MISSION_STATE_READY_TO_COMPLETE = 4, //!< The mission is ready to complete - - /** - * The mission has been completed - */ - MISSION_STATE_COMPLETE = 8, - - /** - * The mission is available again and has been completed before. Used for daily missions. - */ - MISSION_STATE_COMPLETE_AVAILABLE = 9, - - /** - * The mission is active and has been completed before. Used for daily missions. - */ - MISSION_STATE_COMPLETE_ACTIVE = 10, - - /** - * The mission has been completed before and has now been completed again. Used for daily missions. - */ - MISSION_STATE_COMPLETE_READY_TO_COMPLETE = 12 -}; - -#endif //!__MISSIONSTATE__H__ diff --git a/dGame/dMission/MissionTask.cpp b/dGame/dMission/MissionTask.cpp index dc2cb149..997fd34e 100644 --- a/dGame/dMission/MissionTask.cpp +++ b/dGame/dMission/MissionTask.cpp @@ -11,8 +11,10 @@ #include "ScriptedActivityComponent.h" #include "GameMessages.h" #include "dZoneManager.h" +#include "InventoryComponent.h" #include "MissionComponent.h" - +#include "eMissionTaskType.h" +#include "eReplicaComponentType.h" MissionTask::MissionTask(Mission* mission, CDMissionTasks* info, uint32_t mask) { this->info = info; @@ -42,8 +44,8 @@ MissionTask::MissionTask(Mission* mission, CDMissionTasks* info, uint32_t mask) } -MissionTaskType MissionTask::GetType() const { - return static_cast(info->taskType); +eMissionTaskType MissionTask::GetType() const { + return static_cast(info->taskType); } @@ -187,7 +189,7 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& const auto type = GetType(); if (count < 0) { - if (mission->IsMission() && type == MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION && InAllTargets(value)) { + if (mission->IsMission() && type == eMissionTaskType::GATHER && InAllTargets(value)) { if (parameters.size() > 0 && (parameters[0] & 1) != 0) { return; } @@ -218,17 +220,17 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& std::vector settings; switch (type) { - case MissionTaskType::MISSION_TASK_TYPE_UNKNOWN: + case eMissionTaskType::UNKNOWN: break; - case MissionTaskType::MISSION_TASK_TYPE_ACTIVITY: + case eMissionTaskType::ACTIVITY: { if (InAllTargets(value)) { AddProgress(count); break; } - entity = EntityManager::Instance()->GetEntity(associate); + entity = Game::entityManager->GetEntity(associate); if (entity == nullptr) { if (associate != LWOOBJID_EMPTY) { Game::logger->Log("MissionTask", "Failed to find associated entity (%llu)!", associate); @@ -236,7 +238,7 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - activity = static_cast(entity->GetComponent(COMPONENT_TYPE_REBUILD)); + activity = static_cast(entity->GetComponent(eReplicaComponentType::QUICK_BUILD)); if (activity == nullptr) { break; } @@ -256,8 +258,8 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_FOOD: - case MissionTaskType::MISSION_TASK_TYPE_MISSION_INTERACTION: + case eMissionTaskType::USE_ITEM: + case eMissionTaskType::TALK_TO_NPC: { if (GetTarget() != value) break; @@ -266,11 +268,11 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_EMOTE: + case eMissionTaskType::EMOTE: { if (!InParameters(value)) break; - entity = EntityManager::Instance()->GetEntity(associate); + entity = Game::entityManager->GetEntity(associate); if (entity == nullptr) { Game::logger->Log("MissionTask", "Failed to find associated entity (%llu)!", associate); @@ -287,7 +289,7 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_SKILL: + case eMissionTaskType::USE_SKILL: { // This is a complicated check because for some missions we need to check for the associate being in the parameters instead of the value being in the parameters. if (associate == LWOOBJID_EMPTY && GetAllTargets().size() == 1 && GetAllTargets()[0] == -1) { @@ -298,9 +300,9 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_MINIGAME: + case eMissionTaskType::PERFORM_ACTIVITY: { - auto* minigameManager = EntityManager::Instance()->GetEntity(associate); + auto* minigameManager = Game::entityManager->GetEntity(associate); if (minigameManager == nullptr) break; @@ -327,7 +329,7 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_VISIT_PROPERTY: + case eMissionTaskType::VISIT_PROPERTY: { if (!InAllTargets(value)) break; @@ -340,11 +342,11 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_ENVIRONMENT: + case eMissionTaskType::COLLECTION: { if (!InAllTargets(value)) break; - entity = EntityManager::Instance()->GetEntity(associate); + entity = Game::entityManager->GetEntity(associate); if (entity == nullptr) { Game::logger->Log("MissionTask", "Failed to find associated entity (%llu)!", associate); @@ -375,7 +377,7 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_LOCATION: + case eMissionTaskType::EXPLORE: { if (info->targetGroup != targets) break; @@ -384,12 +386,12 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_RACING: + case eMissionTaskType::RACING: { - // The meaning of associate can be found in RacingTaskParam.h + // The meaning of associate can be found in eRacingTaskParam.h if (parameters.empty()) break; - if (!InAllTargets(dZoneManager::Instance()->GetZone()->GetWorldID()) && !(parameters[0] == 4 || parameters[0] == 5) && !InAllTargets(value)) break; + if (!InAllTargets(Game::zoneManager->GetZone()->GetWorldID()) && !(parameters[0] == 4 || parameters[0] == 5) && !InAllTargets(value)) break; if (parameters[0] != associate) break; @@ -426,15 +428,15 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_PET_TAMING: - case MissionTaskType::MISSION_TASK_TYPE_SCRIPT: - case MissionTaskType::MISSION_TASK_TYPE_NON_MISSION_INTERACTION: - case MissionTaskType::MISSION_TASK_TYPE_MISSION_COMPLETE: - case MissionTaskType::MISSION_TASK_TYPE_POWERUP: - case MissionTaskType::MISSION_TASK_TYPE_SMASH: - case MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION: - case MissionTaskType::MISSION_TASK_TYPE_PLAYER_FLAG: - case MissionTaskType::MISSION_TASK_TYPE_EARN_REPUTATION: + case eMissionTaskType::PET_TAMING: + case eMissionTaskType::SCRIPT: + case eMissionTaskType::INTERACT: + case eMissionTaskType::META: + case eMissionTaskType::POWERUP: + case eMissionTaskType::SMASH: + case eMissionTaskType::GATHER: + case eMissionTaskType::PLAYER_FLAG: + case eMissionTaskType::EARN_REPUTATION: { if (!InAllTargets(value)) break; @@ -442,11 +444,14 @@ void MissionTask::Progress(int32_t value, LWOOBJID associate, const std::string& break; } - case MissionTaskType::MISSION_TASK_TYPE_PLACE_MODEL: + case eMissionTaskType::PLACE_MODEL: { AddProgress(count); break; } + case eMissionTaskType::DONATION: + AddProgress(count); + break; default: Game::logger->Log("MissionTask", "Invalid mission task type (%i)!", static_cast(type)); return; diff --git a/dGame/dMission/MissionTask.h b/dGame/dMission/MissionTask.h index ea4b8a05..f867b632 100644 --- a/dGame/dMission/MissionTask.h +++ b/dGame/dMission/MissionTask.h @@ -4,9 +4,9 @@ #define MISSIONTASK_H #include "CDMissionTasksTable.h" -#include "MissionTaskType.h" #include "dCommonVars.h" +enum class eMissionTaskType : int; class Mission; /** @@ -57,7 +57,7 @@ public: * Returns the type of this task * @return the type of this task */ - MissionTaskType GetType() const; + eMissionTaskType GetType() const; /** * Returns the value that should be progressed to, to complete the mission (the target value) diff --git a/dGame/dMission/MissionTaskType.h b/dGame/dMission/MissionTaskType.h deleted file mode 100644 index 6c9b2668..00000000 --- a/dGame/dMission/MissionTaskType.h +++ /dev/null @@ -1,31 +0,0 @@ -#pragma once - -#ifndef MISSIONTASKTYPE_H -#define MISSIONTASKTYPE_H - -//! An enum for mission task types -enum class MissionTaskType : int { - MISSION_TASK_TYPE_UNKNOWN = -1, //!< The task type is unknown - MISSION_TASK_TYPE_SMASH = 0, //!< A task for smashing something - MISSION_TASK_TYPE_SCRIPT = 1, //!< A task handled by a server LUA script - MISSION_TASK_TYPE_ACTIVITY = 2, //!< A task for completing a quickbuild - MISSION_TASK_TYPE_ENVIRONMENT = 3, //!< A task for something in the environment - MISSION_TASK_TYPE_MISSION_INTERACTION = 4, //!< A task for interacting with a mission - MISSION_TASK_TYPE_EMOTE = 5, //!< A task for playing an emote - MISSION_TASK_TYPE_FOOD = 9, //!< A task for eating food - MISSION_TASK_TYPE_SKILL = 10, //!< A task for performing a skill - MISSION_TASK_TYPE_ITEM_COLLECTION = 11, //!< A task for collecting an item - MISSION_TASK_TYPE_LOCATION = 12, //!< A task for finding a location - MISSION_TASK_TYPE_MINIGAME = 14, //!< A task for doing something in a minigame - MISSION_TASK_TYPE_NON_MISSION_INTERACTION = 15, //!< A task for interacting with a non-mission - MISSION_TASK_TYPE_MISSION_COMPLETE = 16, //!< A task for completing a mission - MISSION_TASK_TYPE_EARN_REPUTATION = 17, //!< A task for earning reputation - MISSION_TASK_TYPE_POWERUP = 21, //!< A task for collecting a powerup - MISSION_TASK_TYPE_PET_TAMING = 22, //!< A task for taming a pet - MISSION_TASK_TYPE_RACING = 23, //!< A task for racing - MISSION_TASK_TYPE_PLAYER_FLAG = 24, //!< A task for setting a player flag - MISSION_TASK_TYPE_PLACE_MODEL = 25, //!< A task for picking up a model - MISSION_TASK_TYPE_VISIT_PROPERTY = 30 //!< A task for visiting a property -}; - -#endif diff --git a/dGame/dMission/RacingTaskParam.h b/dGame/dMission/RacingTaskParam.h deleted file mode 100644 index 5958cb5a..00000000 --- a/dGame/dMission/RacingTaskParam.h +++ /dev/null @@ -1,20 +0,0 @@ -#pragma once - -#include - -enum class RacingTaskParam : int32_t { - RACING_TASK_PARAM_FINISH_WITH_PLACEMENT = 1, // + +enum class BehaviorState : uint32_t { + HOME_STATE = 0, //!< The HOME behavior state + CIRCLE_STATE, //!< The CIRCLE behavior state + SQUARE_STATE, //!< The SQUARE behavior state + DIAMOND_STATE, //!< The DIAMOND behavior state + TRIANGLE_STATE, //!< The TRIANGLE behavior state + STAR_STATE //!< The STAR behavior state +}; + +#endif //!__BEHAVIORSTATES__H__ diff --git a/dGame/dPropertyBehaviors/BlockDefinition.cpp b/dGame/dPropertyBehaviors/BlockDefinition.cpp new file mode 100644 index 00000000..2950ac82 --- /dev/null +++ b/dGame/dPropertyBehaviors/BlockDefinition.cpp @@ -0,0 +1,9 @@ +#include "BlockDefinition.h" + +BlockDefinition BlockDefinition::blockDefinitionDefault{}; + +BlockDefinition::BlockDefinition(std::string defaultValue, float minimumValue, float maximumValue) { + this->defaultValue = defaultValue; + this->minimumValue = minimumValue; + this->maximumValue = maximumValue; +} diff --git a/dGame/dPropertyBehaviors/BlockDefinition.h b/dGame/dPropertyBehaviors/BlockDefinition.h new file mode 100644 index 00000000..3a5a6bf1 --- /dev/null +++ b/dGame/dPropertyBehaviors/BlockDefinition.h @@ -0,0 +1,25 @@ +#ifndef __BLOCKDEFINITION__H__ +#define __BLOCKDEFINITION__H__ + +#include + +class AMFArrayValue; + +class BlockDefinition { +public: + BlockDefinition(std::string defaultValue = "", float minimumValue = 0.0f, float maximumValue = 0.0f); + static BlockDefinition blockDefinitionDefault; + + std::string& GetDefaultValue() { return defaultValue; }; + float GetMinimumValue() { return minimumValue; }; + float GetMaximumValue() { return maximumValue; }; + void SetDefaultValue(std::string value) { defaultValue = value; }; + void SetMinimumValue(float value) { minimumValue = value; }; + void SetMaximumValue(float value) { maximumValue = value; }; +private: + std::string defaultValue; + float minimumValue; + float maximumValue; +}; + +#endif //!__BLOCKDEFINITION__H__ diff --git a/dGame/dPropertyBehaviors/CMakeLists.txt b/dGame/dPropertyBehaviors/CMakeLists.txt new file mode 100644 index 00000000..5e33a5f5 --- /dev/null +++ b/dGame/dPropertyBehaviors/CMakeLists.txt @@ -0,0 +1,12 @@ +set(DGAME_DPROPERTYBEHAVIORS_SOURCES + "BlockDefinition.cpp" + "ControlBehaviors.cpp" +) + +add_subdirectory(ControlBehaviorMessages) + +foreach(file ${DGAME_DPROPERTYBEHAVIORS_CONTROLBEHAVIORMESSAGES}) + set(DGAME_DPROPERTYBEHAVIORS_SOURCES ${DGAME_DPROPERTYBEHAVIORS_SOURCES} "ControlBehaviorMessages/${file}") +endforeach() + +set(DGAME_DPROPERTYBEHAVIORS_SOURCES ${DGAME_DPROPERTYBEHAVIORS_SOURCES} PARENT_SCOPE) diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/Action.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/Action.cpp new file mode 100644 index 00000000..eabe76b7 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/Action.cpp @@ -0,0 +1,31 @@ +#include "Action.h" + +Action::Action() { + type = ""; + valueParameterName = ""; + valueParameterString = ""; + valueParameterDouble = 0.0; +} + +Action::Action(AMFArrayValue* arguments) { + type = ""; + valueParameterName = ""; + valueParameterString = ""; + valueParameterDouble = 0.0; + for (auto& typeValueMap : arguments->GetAssociative()) { + if (typeValueMap.first == "Type") { + if (typeValueMap.second->GetValueType() != eAmf::String) continue; + type = static_cast(typeValueMap.second)->GetValue(); + } else { + valueParameterName = typeValueMap.first; + // Message is the only known string parameter + if (valueParameterName == "Message") { + if (typeValueMap.second->GetValueType() != eAmf::String) continue; + valueParameterString = static_cast(typeValueMap.second)->GetValue(); + } else { + if (typeValueMap.second->GetValueType() != eAmf::Double) continue; + valueParameterDouble = static_cast(typeValueMap.second)->GetValue(); + } + } + } +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/Action.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/Action.h new file mode 100644 index 00000000..c97b4050 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/Action.h @@ -0,0 +1,25 @@ +#ifndef __ACTION__H__ +#define __ACTION__H__ + +#include "BehaviorMessageBase.h" + +/** + * @brief Sent if a ControlBehavior message has an Action associated with it + * + */ +class Action { +public: + Action(); + Action(AMFArrayValue* arguments); + const std::string& GetType() { return type; }; + const std::string& GetValueParameterName() { return valueParameterName; }; + const std::string& GetValueParameterString() { return valueParameterString; }; + const double GetValueParameterDouble() { return valueParameterDouble; }; +private: + std::string type; + std::string valueParameterName; + std::string valueParameterString; + double valueParameterDouble; +}; + +#endif //!__ACTION__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/ActionContext.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/ActionContext.cpp new file mode 100644 index 00000000..c2ba2eeb --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/ActionContext.cpp @@ -0,0 +1,31 @@ +#include "ActionContext.h" + +#include + +#include "Amf3.h" + +ActionContext::ActionContext() { + stripId = 0; + stateId = BehaviorState::HOME_STATE; +} + +ActionContext::ActionContext(AMFArrayValue* arguments, std::string customStateKey, std::string customStripKey) { + stripId = 0; + stateId = BehaviorState::HOME_STATE; + stripId = GetStripIdFromArgument(arguments, customStripKey); + stateId = GetBehaviorStateFromArgument(arguments, customStateKey); +} + +BehaviorState ActionContext::GetBehaviorStateFromArgument(AMFArrayValue* arguments, const std::string& key) { + auto* stateIDValue = arguments->Get(key); + if (!stateIDValue) throw std::invalid_argument("Unable to find behavior state from argument \"" + key + "\""); + + return static_cast(stateIDValue->GetValue()); +} + +StripId ActionContext::GetStripIdFromArgument(AMFArrayValue* arguments, const std::string& key) { + auto* stripIdValue = arguments->Get(key); + if (!stripIdValue) throw std::invalid_argument("Unable to find strip ID from argument \"" + key + "\""); + + return static_cast(stripIdValue->GetValue()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/ActionContext.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/ActionContext.h new file mode 100644 index 00000000..5f46fd8c --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/ActionContext.h @@ -0,0 +1,26 @@ +#ifndef __ACTIONCONTEXT__H__ +#define __ACTIONCONTEXT__H__ + +#include "BehaviorStates.h" +#include "dCommonVars.h" + +class AMFArrayValue; + +/** + * @brief Sent if contextual State and Strip informationis needed for a ControlBehaviors message + * + */ +class ActionContext { +public: + ActionContext(); + ActionContext(AMFArrayValue* arguments, std::string customStateKey = "stateID", std::string customStripKey = "stripID"); + const StripId GetStripId() { return stripId; }; + const BehaviorState GetStateId() { return stateId; }; +private: + BehaviorState GetBehaviorStateFromArgument(AMFArrayValue* arguments, const std::string& key); + StripId GetStripIdFromArgument(AMFArrayValue* arguments, const std::string& key); + StripId stripId; + BehaviorState stateId; +}; + +#endif //!__ACTIONCONTEXT__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddActionMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddActionMessage.cpp new file mode 100644 index 00000000..98672909 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddActionMessage.cpp @@ -0,0 +1,13 @@ +#include "AddActionMessage.h" + +AddActionMessage::AddActionMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + actionContext = ActionContext(arguments); + actionIndex = GetActionIndexFromArgument(arguments); + + auto* actionValue = arguments->GetArray("action"); + if (!actionValue) return; + + action = Action(actionValue); + + Game::logger->LogDebug("AddActionMessage", "actionIndex %i stripId %i stateId %i type %s valueParameterName %s valueParameterString %s valueParameterDouble %f behaviorId %i", actionIndex, actionContext.GetStripId(), actionContext.GetStateId(), action.GetType().c_str(), action.GetValueParameterName().c_str(), action.GetValueParameterString().c_str(), action.GetValueParameterDouble(), behaviorId); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddActionMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddActionMessage.h new file mode 100644 index 00000000..4faf6a53 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddActionMessage.h @@ -0,0 +1,26 @@ +#ifndef __ADDACTIONMESSAGE__H__ +#define __ADDACTIONMESSAGE__H__ + +#include "Action.h" +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Send if a player takes an Action A from the toolbox and adds it to an already existing strip + * + */ +class AddActionMessage : public BehaviorMessageBase { +public: + AddActionMessage(AMFArrayValue* arguments); + const uint32_t GetActionIndex() { return actionIndex; }; + Action GetAction() { return action; }; + ActionContext GetActionContext() { return actionContext; }; +private: + uint32_t actionIndex; + ActionContext actionContext; + Action action; +}; + +#endif //!__ADDACTIONMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddMessage.cpp new file mode 100644 index 00000000..badee2c2 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddMessage.cpp @@ -0,0 +1,11 @@ +#include "AddMessage.h" + +AddMessage::AddMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + behaviorIndex = 0; + auto* behaviorIndexValue = arguments->Get("BehaviorIndex"); + + if (!behaviorIndexValue) return; + + behaviorIndex = static_cast(behaviorIndexValue->GetValue()); + Game::logger->LogDebug("AddMessage", "behaviorId %i index %i", behaviorId, behaviorIndex); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddMessage.h new file mode 100644 index 00000000..a46d5f98 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddMessage.h @@ -0,0 +1,18 @@ +#ifndef __ADDMESSAGE__H__ +#define __ADDMESSAGE__H__ + +#include "BehaviorMessageBase.h" + +/** + * @brief Sent when a player adds a Behavior A from their inventory to a model. + * + */ +class AddMessage : public BehaviorMessageBase { +public: + AddMessage(AMFArrayValue* arguments); + const uint32_t GetBehaviorIndex() { return behaviorIndex; }; +private: + uint32_t behaviorIndex; +}; + +#endif //!__ADDMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddStripMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddStripMessage.cpp new file mode 100644 index 00000000..261b3462 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddStripMessage.cpp @@ -0,0 +1,24 @@ +#include "AddStripMessage.h" + +#include "Action.h" + +AddStripMessage::AddStripMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + actionContext = ActionContext(arguments); + position = StripUiPosition(arguments); + + auto* strip = arguments->GetArray("strip"); + if (!strip) return; + + auto* actions = strip->GetArray("actions"); + if (!actions) return; + + for (uint32_t actionNumber = 0; actionNumber < actions->GetDense().size(); actionNumber++) { + auto* actionValue = actions->GetArray(actionNumber); + if (!actionValue) continue; + + actionsToAdd.push_back(Action(actionValue)); + + Game::logger->LogDebug("AddStripMessage", "xPosition %f yPosition %f stripId %i stateId %i behaviorId %i t %s valueParameterName %s valueParameterString %s valueParameterDouble %f", position.GetX(), position.GetY(), actionContext.GetStripId(), actionContext.GetStateId(), behaviorId, actionsToAdd.back().GetType().c_str(), actionsToAdd.back().GetValueParameterName().c_str(), actionsToAdd.back().GetValueParameterString().c_str(), actionsToAdd.back().GetValueParameterDouble()); + } + Game::logger->Log("AddStripMessage", "number of actions %i", actionsToAdd.size()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddStripMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddStripMessage.h new file mode 100644 index 00000000..db75aef7 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/AddStripMessage.h @@ -0,0 +1,31 @@ +#ifndef __ADDSTRIPMESSAGE__H__ +#define __ADDSTRIPMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" +#include "StripUiPosition.h" + +#include + +class Action; +class AMFArrayValue; + +/** + * @brief Sent in 2 contexts: + * A player adds an Action A from their toolbox without attaching it to an existing Strip. In this case, only 1 action is sent. + * A player moves a Strip from BehaviorState A directly to BehaviorState B. In this case, a list of actions are sent. + * + */ +class AddStripMessage : public BehaviorMessageBase { +public: + AddStripMessage(AMFArrayValue* arguments); + StripUiPosition GetPosition() { return position; }; + ActionContext GetActionContext() { return actionContext; }; + std::vector GetActionsToAdd() { return actionsToAdd; }; +private: + StripUiPosition position; + ActionContext actionContext; + std::vector actionsToAdd; +}; + +#endif //!__ADDSTRIPMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/BehaviorMessageBase.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/BehaviorMessageBase.cpp new file mode 100644 index 00000000..3286504a --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/BehaviorMessageBase.cpp @@ -0,0 +1,33 @@ +#include "BehaviorMessageBase.h" + +#include "Amf3.h" +#include "BehaviorStates.h" +#include "dCommonVars.h" + +BehaviorMessageBase::BehaviorMessageBase(AMFArrayValue* arguments) { + behaviorId = 0; + behaviorId = GetBehaviorIdFromArgument(arguments); +} + +int32_t BehaviorMessageBase::GetBehaviorIdFromArgument(AMFArrayValue* arguments) { + const auto* key = "BehaviorID"; + auto* behaviorIDValue = arguments->Get(key); + int32_t behaviorID = -1; + + if (behaviorIDValue && behaviorIDValue->GetValueType() == eAmf::String) { + behaviorID = std::stoul(behaviorIDValue->GetValue()); + } else if (arguments->Get(key)->GetValueType() != eAmf::Undefined) { + throw std::invalid_argument("Unable to find behavior ID"); + } + + return behaviorID; +} + +uint32_t BehaviorMessageBase::GetActionIndexFromArgument(AMFArrayValue* arguments, const std::string& keyName) { + auto* actionIndexAmf = arguments->Get(keyName); + if (!actionIndexAmf) { + throw std::invalid_argument("Unable to find actionIndex"); + } + + return static_cast(actionIndexAmf->GetValue()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/BehaviorMessageBase.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/BehaviorMessageBase.h new file mode 100644 index 00000000..8771286c --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/BehaviorMessageBase.h @@ -0,0 +1,29 @@ +#ifndef __BEHAVIORMESSAGEBASE__H__ +#define __BEHAVIORMESSAGEBASE__H__ + +#include +#include + +#include "Amf3.h" +#include "dCommonVars.h" + +#include "Game.h" +#include "dLogger.h" + +enum class BehaviorState : uint32_t; + +/** + * @brief The behaviorID target of this ControlBehaviors message + * + */ +class BehaviorMessageBase { +public: + const uint32_t GetBehaviorId() { return behaviorId; }; +protected: + BehaviorMessageBase(AMFArrayValue* arguments); + int32_t GetBehaviorIdFromArgument(AMFArrayValue* arguments); + uint32_t GetActionIndexFromArgument(AMFArrayValue* arguments, const std::string& keyName = "actionIndex"); + int32_t behaviorId = -1; +}; + +#endif //!__BEHAVIORMESSAGEBASE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/CMakeLists.txt b/dGame/dPropertyBehaviors/ControlBehaviorMessages/CMakeLists.txt new file mode 100644 index 00000000..49b0f460 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/CMakeLists.txt @@ -0,0 +1,20 @@ +set(DGAME_DPROPERTYBEHAVIORS_CONTROLBEHAVIORMESSAGES + "Action.cpp" + "ActionContext.cpp" + "AddActionMessage.cpp" + "AddMessage.cpp" + "AddStripMessage.cpp" + "BehaviorMessageBase.cpp" + "MergeStripsMessage.cpp" + "MigrateActionsMessage.cpp" + "MoveToInventoryMessage.cpp" + "RearrangeStripMessage.cpp" + "RemoveActionsMessage.cpp" + "RemoveStripMessage.cpp" + "RenameMessage.cpp" + "SplitStripMessage.cpp" + "StripUiPosition.cpp" + "UpdateActionMessage.cpp" + "UpdateStripUiMessage.cpp" + PARENT_SCOPE +) diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/MergeStripsMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MergeStripsMessage.cpp new file mode 100644 index 00000000..df50641a --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MergeStripsMessage.cpp @@ -0,0 +1,11 @@ +#include "MergeStripsMessage.h" + +MergeStripsMessage::MergeStripsMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + sourceActionContext = ActionContext(arguments, "srcStateID", "srcStripID"); + + destinationActionContext = ActionContext(arguments, "dstStateID", "dstStripID"); + dstActionIndex = GetActionIndexFromArgument(arguments, "dstActionIndex"); + + Game::logger->LogDebug("MergeStripsMessage", "srcstripId %i dststripId %i srcstateId %i dststateId %i dstactionIndex %i behaviorId %i", sourceActionContext.GetStripId(), destinationActionContext.GetStripId(), sourceActionContext.GetStateId(), destinationActionContext.GetStateId(), dstActionIndex, behaviorId); +} + diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/MergeStripsMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MergeStripsMessage.h new file mode 100644 index 00000000..0aff7f3a --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MergeStripsMessage.h @@ -0,0 +1,25 @@ +#ifndef __MERGESTRIPSMESSAGE__H__ +#define __MERGESTRIPSMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player adds the first Action of Strip A to a Strip B + * + */ +class MergeStripsMessage : public BehaviorMessageBase { +public: + MergeStripsMessage(AMFArrayValue* arguments); + const uint32_t GetDstActionIndex() { return dstActionIndex; }; + ActionContext GetSourceActionContext() { return sourceActionContext; }; + ActionContext GetDestinationActionContext() { return destinationActionContext; }; +private: + ActionContext sourceActionContext; + ActionContext destinationActionContext; + uint32_t dstActionIndex; +}; + +#endif //!__MERGESTRIPSMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/MigrateActionsMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MigrateActionsMessage.cpp new file mode 100644 index 00000000..08f830a4 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MigrateActionsMessage.cpp @@ -0,0 +1,11 @@ +#include "MigrateActionsMessage.h" + +MigrateActionsMessage::MigrateActionsMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + sourceActionContext = ActionContext(arguments, "srcStateID", "srcStripID"); + srcActionIndex = GetActionIndexFromArgument(arguments, "srcActionIndex"); + + destinationActionContext = ActionContext(arguments, "dstStateID", "dstStripID"); + dstActionIndex = GetActionIndexFromArgument(arguments, "dstActionIndex"); + + Game::logger->LogDebug("MigrateActionsMessage", "srcactionIndex %i dstactionIndex %i srcstripId %i dststripId %i srcstateId %i dststateId %i behaviorId %i", srcActionIndex, dstActionIndex, sourceActionContext.GetStripId(), destinationActionContext.GetStripId(), sourceActionContext.GetStateId(), destinationActionContext.GetStateId(), behaviorId); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/MigrateActionsMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MigrateActionsMessage.h new file mode 100644 index 00000000..f60e8748 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MigrateActionsMessage.h @@ -0,0 +1,27 @@ +#ifndef __MIGRATEACTIONSMESSAGE__H__ +#define __MIGRATEACTIONSMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player moves an Action after the first Action to a different Strip + * + */ +class MigrateActionsMessage : public BehaviorMessageBase { +public: + MigrateActionsMessage(AMFArrayValue* arguments); + const uint32_t GetSrcActionIndex() { return srcActionIndex; }; + const uint32_t GetDstActionIndex() { return dstActionIndex; }; + ActionContext GetSourceActionContext() { return sourceActionContext; }; + ActionContext GetDestinationActionContext() { return destinationActionContext; }; +private: + ActionContext sourceActionContext; + ActionContext destinationActionContext; + uint32_t srcActionIndex; + uint32_t dstActionIndex; +}; + +#endif //!__MIGRATEACTIONSMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/MoveToInventoryMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MoveToInventoryMessage.cpp new file mode 100644 index 00000000..5b61ee32 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MoveToInventoryMessage.cpp @@ -0,0 +1,9 @@ +#include "MoveToInventoryMessage.h" + +MoveToInventoryMessage::MoveToInventoryMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + auto* behaviorIndexValue = arguments->Get("BehaviorIndex"); + if (!behaviorIndexValue) return; + + behaviorIndex = static_cast(behaviorIndexValue->GetValue()); + Game::logger->LogDebug("MoveToInventoryMessage", "behaviorId %i behaviorIndex %i", behaviorId, behaviorIndex); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/MoveToInventoryMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MoveToInventoryMessage.h new file mode 100644 index 00000000..c48f7d17 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/MoveToInventoryMessage.h @@ -0,0 +1,21 @@ +#ifndef __MOVETOINVENTORYMESSAGE__H__ +#define __MOVETOINVENTORYMESSAGE__H__ + +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player moves a Behavior A at position B to their inventory. + * + */ +#pragma warning("This Control Behavior Message does not have a test yet. Non-developers can ignore this warning.") +class MoveToInventoryMessage : public BehaviorMessageBase { +public: + MoveToInventoryMessage(AMFArrayValue* arguments); + const uint32_t GetBehaviorIndex() { return behaviorIndex; }; +private: + uint32_t behaviorIndex; +}; + +#endif //!__MOVETOINVENTORYMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RearrangeStripMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RearrangeStripMessage.cpp new file mode 100644 index 00000000..4018a423 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RearrangeStripMessage.cpp @@ -0,0 +1,10 @@ +#include "RearrangeStripMessage.h" + +RearrangeStripMessage::RearrangeStripMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + actionContext = ActionContext(arguments); + srcActionIndex = GetActionIndexFromArgument(arguments, "srcActionIndex"); + + dstActionIndex = GetActionIndexFromArgument(arguments, "dstActionIndex"); + + Game::logger->LogDebug("RearrangeStripMessage", "srcactionIndex %i dstactionIndex %i stripId %i behaviorId %i stateId %i", srcActionIndex, dstActionIndex, actionContext.GetStripId(), behaviorId, actionContext.GetStateId()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RearrangeStripMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RearrangeStripMessage.h new file mode 100644 index 00000000..46819404 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RearrangeStripMessage.h @@ -0,0 +1,23 @@ +#ifndef __REARRANGESTRIPMESSAGE__H__ +#define __REARRANGESTRIPMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +/** + * @brief Sent when a player moves an Action around in the same strip + * + */ +class RearrangeStripMessage : public BehaviorMessageBase { +public: + RearrangeStripMessage(AMFArrayValue* arguments); + const uint32_t GetSrcActionIndex() { return srcActionIndex; }; + const uint32_t GetDstActionIndex() { return dstActionIndex; }; + ActionContext GetActionContext() { return actionContext; }; +private: + ActionContext actionContext; + uint32_t srcActionIndex; + uint32_t dstActionIndex; +}; + +#endif //!__REARRANGESTRIPMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveActionsMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveActionsMessage.cpp new file mode 100644 index 00000000..8f00d2b0 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveActionsMessage.cpp @@ -0,0 +1,8 @@ +#include "RemoveActionsMessage.h" + +RemoveActionsMessage::RemoveActionsMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + actionContext = ActionContext(arguments); + actionIndex = GetActionIndexFromArgument(arguments); + + Game::logger->LogDebug("RemoveActionsMessage", "behaviorId %i actionIndex %i stripId %i stateId %i", behaviorId, actionIndex, actionContext.GetStripId(), actionContext.GetStateId()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveActionsMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveActionsMessage.h new file mode 100644 index 00000000..457ddba8 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveActionsMessage.h @@ -0,0 +1,23 @@ +#ifndef __REMOVEACTIONSMESSAGE__H__ +#define __REMOVEACTIONSMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player removes any Action after the first one from a Strip + * + */ +class RemoveActionsMessage : public BehaviorMessageBase { +public: + RemoveActionsMessage(AMFArrayValue* arguments); + const uint32_t GetActionIndex() { return actionIndex; }; + ActionContext GetActionContext() { return actionContext; }; +private: + ActionContext actionContext; + uint32_t actionIndex; +}; + +#endif //!__REMOVEACTIONSMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveStripMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveStripMessage.cpp new file mode 100644 index 00000000..40288b07 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveStripMessage.cpp @@ -0,0 +1,7 @@ +#include "RemoveStripMessage.h" + +RemoveStripMessage::RemoveStripMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + actionContext = ActionContext(arguments); + + Game::logger->LogDebug("RemoveStripMessage", "stripId %i stateId %i behaviorId %i", actionContext.GetStripId(), actionContext.GetStateId(), behaviorId); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveStripMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveStripMessage.h new file mode 100644 index 00000000..36e2e401 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RemoveStripMessage.h @@ -0,0 +1,19 @@ +#ifndef __REMOVESTRIPMESSAGE__H__ +#define __REMOVESTRIPMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +/** + * @brief Sent when a player removes the first Action from a strip. + * + */ +class RemoveStripMessage : public BehaviorMessageBase { +public: + RemoveStripMessage(AMFArrayValue* arguments); + ActionContext GetActionContext() { return actionContext; }; +private: + ActionContext actionContext; +}; + +#endif //!__REMOVESTRIPMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RenameMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RenameMessage.cpp new file mode 100644 index 00000000..a1c2abbc --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RenameMessage.cpp @@ -0,0 +1,9 @@ +#include "RenameMessage.h" + +RenameMessage::RenameMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + auto* nameAmf = arguments->Get("Name"); + if (!nameAmf) return; + + name = nameAmf->GetValue(); + Game::logger->LogDebug("RenameMessage", "behaviorId %i n %s", behaviorId, name.c_str()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/RenameMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RenameMessage.h new file mode 100644 index 00000000..ba181f63 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/RenameMessage.h @@ -0,0 +1,20 @@ +#ifndef __RENAMEMESSAGE__H__ +#define __RENAMEMESSAGE__H__ + +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player renames this behavior + * + */ +class RenameMessage : public BehaviorMessageBase { +public: + RenameMessage(AMFArrayValue* arguments); + const std::string& GetName() { return name; }; +private: + std::string name; +}; + +#endif //!__RENAMEMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/SplitStripMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/SplitStripMessage.cpp new file mode 100644 index 00000000..b4601a17 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/SplitStripMessage.cpp @@ -0,0 +1,11 @@ +#include "SplitStripMessage.h" + +SplitStripMessage::SplitStripMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + sourceActionContext = ActionContext(arguments, "srcStateID", "srcStripID"); + srcActionIndex = GetActionIndexFromArgument(arguments, "srcActionIndex"); + + destinationActionContext = ActionContext(arguments, "dstStateID", "dstStripID"); + destinationPosition = StripUiPosition(arguments, "dstStripUI"); + + Game::logger->LogDebug("SplitStripMessage", "behaviorId %i xPosition %f yPosition %f sourceStrip %i destinationStrip %i sourceState %i destinationState %i srcActindex %i", behaviorId, destinationPosition.GetX(), destinationPosition.GetY(), sourceActionContext.GetStripId(), destinationActionContext.GetStripId(), sourceActionContext.GetStateId(), destinationActionContext.GetStateId(), srcActionIndex); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/SplitStripMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/SplitStripMessage.h new file mode 100644 index 00000000..9210efb0 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/SplitStripMessage.h @@ -0,0 +1,28 @@ +#ifndef __SPLITSTRIPMESSAGE__H__ +#define __SPLITSTRIPMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" +#include "StripUiPosition.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player takes an Action from Strip A and does not add it to an existing strip + * + */ +class SplitStripMessage : public BehaviorMessageBase { +public: + SplitStripMessage(AMFArrayValue* arguments); + ActionContext GetSourceActionContext() { return sourceActionContext; }; + ActionContext GetDestinationActionContext() { return destinationActionContext; }; + const uint32_t GetSrcActionIndex() { return srcActionIndex; }; + StripUiPosition GetPosition() { return destinationPosition; }; +private: + ActionContext sourceActionContext; + ActionContext destinationActionContext; + uint32_t srcActionIndex; + StripUiPosition destinationPosition; +}; + +#endif //!__SPLITSTRIPMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/StripUiPosition.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/StripUiPosition.cpp new file mode 100644 index 00000000..de612b45 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/StripUiPosition.cpp @@ -0,0 +1,22 @@ +#include "StripUiPosition.h" + +#include "Amf3.h" + +StripUiPosition::StripUiPosition() { + xPosition = 0.0; + yPosition = 0.0; +} + +StripUiPosition::StripUiPosition(AMFArrayValue* arguments, std::string uiKeyName) { + xPosition = 0.0; + yPosition = 0.0; + auto* uiArray = arguments->GetArray(uiKeyName); + if (!uiArray) return; + + auto* xPositionValue = uiArray->Get("x"); + auto* yPositionValue = uiArray->Get("y"); + if (!xPositionValue || !yPositionValue) return; + + yPosition = yPositionValue->GetValue(); + xPosition = xPositionValue->GetValue(); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/StripUiPosition.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/StripUiPosition.h new file mode 100644 index 00000000..809f8890 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/StripUiPosition.h @@ -0,0 +1,21 @@ +#ifndef __STRIPUIPOSITION__H__ +#define __STRIPUIPOSITION__H__ + +class AMFArrayValue; + +/** + * @brief The position of the first Action in a Strip + * + */ +class StripUiPosition { +public: + StripUiPosition(); + StripUiPosition(AMFArrayValue* arguments, std::string uiKeyName = "ui"); + double GetX() { return xPosition; }; + double GetY() { return yPosition; }; +private: + double xPosition; + double yPosition; +}; + +#endif //!__STRIPUIPOSITION__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateActionMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateActionMessage.cpp new file mode 100644 index 00000000..23a0050d --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateActionMessage.cpp @@ -0,0 +1,15 @@ +#include "UpdateActionMessage.h" + +#include "Action.h" + +UpdateActionMessage::UpdateActionMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + actionContext = ActionContext(arguments); + + auto* actionValue = arguments->GetArray("action"); + if (!actionValue) return; + + action = Action(actionValue); + actionIndex = GetActionIndexFromArgument(arguments); + + Game::logger->LogDebug("UpdateActionMessage", "type %s valueParameterName %s valueParameterString %s valueParameterDouble %f behaviorId %i actionIndex %i stripId %i stateId %i", action.GetType().c_str(), action.GetValueParameterName().c_str(), action.GetValueParameterString().c_str(), action.GetValueParameterDouble(), behaviorId, actionIndex, actionContext.GetStripId(), actionContext.GetStateId()); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateActionMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateActionMessage.h new file mode 100644 index 00000000..0a03ce9e --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateActionMessage.h @@ -0,0 +1,26 @@ +#ifndef __UPDATEACTIONMESSAGE__H__ +#define __UPDATEACTIONMESSAGE__H__ + +#include "Action.h" +#include "ActionContext.h" +#include "BehaviorMessageBase.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player updates the value in an Action + * + */ +class UpdateActionMessage : public BehaviorMessageBase { +public: + UpdateActionMessage(AMFArrayValue* arguments); + const uint32_t GetActionIndex() { return actionIndex; }; + ActionContext GetActionContext() { return actionContext; }; + Action GetAction() { return action; }; +private: + uint32_t actionIndex; + ActionContext actionContext; + Action action; +}; + +#endif //!__UPDATEACTIONMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateStripUiMessage.cpp b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateStripUiMessage.cpp new file mode 100644 index 00000000..073db9b4 --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateStripUiMessage.cpp @@ -0,0 +1,8 @@ +#include "UpdateStripUiMessage.h" + +UpdateStripUiMessage::UpdateStripUiMessage(AMFArrayValue* arguments) : BehaviorMessageBase(arguments) { + position = StripUiPosition(arguments); + actionContext = ActionContext(arguments); + + Game::logger->LogDebug("UpdateStripUIMessage", "xPosition %f yPosition %f stripId %i stateId %i behaviorId %i", position.GetX(), position.GetY(), actionContext.GetStripId(), actionContext.GetStateId(), behaviorId); +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateStripUiMessage.h b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateStripUiMessage.h new file mode 100644 index 00000000..6d96f90c --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviorMessages/UpdateStripUiMessage.h @@ -0,0 +1,24 @@ +#ifndef __UPDATESTRIPUIMESSAGE__H__ +#define __UPDATESTRIPUIMESSAGE__H__ + +#include "ActionContext.h" +#include "BehaviorMessageBase.h" +#include "StripUiPosition.h" + +class AMFArrayValue; + +/** + * @brief Sent when a player moves the first Action in a Strip + * + */ +class UpdateStripUiMessage : public BehaviorMessageBase { +public: + UpdateStripUiMessage(AMFArrayValue* arguments); + StripUiPosition GetPosition() { return position; }; + ActionContext GetActionContext() { return actionContext; }; +private: + StripUiPosition position; + ActionContext actionContext; +}; + +#endif //!__UPDATESTRIPUIMESSAGE__H__ diff --git a/dGame/dPropertyBehaviors/ControlBehaviors.cpp b/dGame/dPropertyBehaviors/ControlBehaviors.cpp new file mode 100644 index 00000000..d8a062ca --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviors.cpp @@ -0,0 +1,456 @@ +#include "ControlBehaviors.h" + +#include "Amf3.h" +#include "Entity.h" +#include "Game.h" +#include "GameMessages.h" +#include "ModelComponent.h" +#include "../../dWorldServer/ObjectIDManager.h" +#include "dLogger.h" +#include "BehaviorStates.h" +#include "AssetManager.h" +#include "BlockDefinition.h" +#include "User.h" +#include "tinyxml2.h" +#include "CDClientDatabase.h" + +// Message includes +#include "Action.h" +#include "AddActionMessage.h" +#include "AddStripMessage.h" +#include "AddMessage.h" +#include "MigrateActionsMessage.h" +#include "MoveToInventoryMessage.h" +#include "MergeStripsMessage.h" +#include "RearrangeStripMessage.h" +#include "RemoveActionsMessage.h" +#include "RemoveStripMessage.h" +#include "RenameMessage.h" +#include "SplitStripMessage.h" +#include "UpdateActionMessage.h" +#include "UpdateStripUiMessage.h" + +void ControlBehaviors::RequestUpdatedID(int32_t behaviorID, ModelComponent* modelComponent, Entity* modelOwner, const SystemAddress& sysAddr) { + // auto behavior = modelComponent->FindBehavior(behaviorID); + // if (behavior->GetBehaviorID() == -1 || behavior->GetShouldSetNewID()) { + // ObjectIDManager::Instance()->RequestPersistentID( + // [behaviorID, behavior, modelComponent, modelOwner, sysAddr](uint32_t persistentId) { + // behavior->SetShouldGetNewID(false); + // behavior->SetIsTemplated(false); + // behavior->SetBehaviorID(persistentId); + + // // This updates the behavior ID of the behavior should this be a new behavior + // AMFArrayValue args; + + // AMFStringValue* behaviorIDString = new AMFStringValue(); + // behaviorIDString->SetValue(std::to_string(persistentId)); + // args.InsertValue("behaviorID", behaviorIDString); + + // AMFStringValue* objectIDAsString = new AMFStringValue(); + // objectIDAsString->SetValue(std::to_string(modelComponent->GetParent()->GetObjectID())); + // args.InsertValue("objectID", objectIDAsString); + + // GameMessages::SendUIMessageServerToSingleClient(modelOwner, sysAddr, "UpdateBehaviorID", &args); + // ControlBehaviors::SendBehaviorListToClient(modelComponent->GetParent(), sysAddr, modelOwner); + // }); + // } +} + +void ControlBehaviors::SendBehaviorListToClient(Entity* modelEntity, const SystemAddress& sysAddr, Entity* modelOwner) { + auto* modelComponent = modelEntity->GetComponent(); + + if (!modelComponent) return; + + AMFArrayValue behaviorsToSerialize; + + /** + * The behaviors AMFArray will have up to 5 elements in the dense portion. + * Each element in the dense portion will be made up of another AMFArray + * with the following information mapped in the associative portion + * "id": Behavior ID cast to an AMFString + * "isLocked": AMFTrue or AMFFalse of whether or not the behavior is locked + * "isLoot": AMFTrue or AMFFalse of whether or not the behavior is a custom behavior (true if custom) + * "name": The name of the behavior formatted as an AMFString + */ + + behaviorsToSerialize.Insert("behaviors"); + behaviorsToSerialize.Insert("objectID", std::to_string(modelComponent->GetParent()->GetObjectID())); + + GameMessages::SendUIMessageServerToSingleClient(modelOwner, sysAddr, "UpdateBehaviorList", behaviorsToSerialize); +} + +void ControlBehaviors::ModelTypeChanged(AMFArrayValue* arguments, ModelComponent* ModelComponent) { + auto* modelTypeAmf = arguments->Get("ModelType"); + if (!modelTypeAmf) return; + + uint32_t modelType = static_cast(modelTypeAmf->GetValue()); + + //TODO Update the model type here +} + +void ControlBehaviors::ToggleExecutionUpdates() { + //TODO do something with this info +} + +void ControlBehaviors::AddStrip(AMFArrayValue* arguments) { + AddStripMessage addStripMessage(arguments); +} + +void ControlBehaviors::RemoveStrip(AMFArrayValue* arguments) { + RemoveStripMessage removeStrip(arguments); +} + +void ControlBehaviors::MergeStrips(AMFArrayValue* arguments) { + MergeStripsMessage mergeStripsMessage(arguments); +} + +void ControlBehaviors::SplitStrip(AMFArrayValue* arguments) { + SplitStripMessage splitStripMessage(arguments); +} + +void ControlBehaviors::UpdateStripUI(AMFArrayValue* arguments) { + UpdateStripUiMessage updateStripUiMessage(arguments); +} + +void ControlBehaviors::AddAction(AMFArrayValue* arguments) { + AddActionMessage addActionMessage(arguments); +} + +void ControlBehaviors::MigrateActions(AMFArrayValue* arguments) { + MigrateActionsMessage migrateActionsMessage(arguments); +} + +void ControlBehaviors::RearrangeStrip(AMFArrayValue* arguments) { + RearrangeStripMessage rearrangeStripMessage(arguments); +} + +void ControlBehaviors::Add(AMFArrayValue* arguments) { + AddMessage addMessage(arguments); +} + +void ControlBehaviors::RemoveActions(AMFArrayValue* arguments) { + RemoveActionsMessage removeActionsMessage(arguments); +} + +void ControlBehaviors::Rename(Entity* modelEntity, const SystemAddress& sysAddr, Entity* modelOwner, AMFArrayValue* arguments) { + RenameMessage renameMessage(arguments); +} + +// TODO This is also supposed to serialize the state of the behaviors in progress but those aren't implemented yet +void ControlBehaviors::SendBehaviorBlocksToClient(ModelComponent* modelComponent, const SystemAddress& sysAddr, Entity* modelOwner, AMFArrayValue* arguments) { + // uint32_t behaviorID = ControlBehaviors::GetBehaviorIDFromArgument(arguments); + + // auto modelBehavior = modelComponent->FindBehavior(behaviorID); + + // if (!modelBehavior) return; + + // modelBehavior->VerifyStates(); + + // auto states = modelBehavior->GetBehaviorStates(); + + // // Begin serialization. + + // /** + // * for each state + // * strip id + // * ui info + // * x + // * y + // * actions + // * action1 + // * action2 + // * ... + // * behaviorID of strip + // * objectID of strip + // */ + // LWOOBJID targetObjectID = LWOOBJID_EMPTY; + // behaviorID = 0; + // AMFArrayValue behaviorInfo; + + // AMFArrayValue* stateSerialize = new AMFArrayValue(); + + // for (auto it = states.begin(); it != states.end(); it++) { + // Game::logger->Log("PropertyBehaviors", "Begin serialization of state %i!\n", it->first); + // AMFArrayValue* state = new AMFArrayValue(); + + // AMFDoubleValue* stateAsDouble = new AMFDoubleValue(); + // stateAsDouble->SetValue(it->first); + // state->InsertValue("id", stateAsDouble); + + // AMFArrayValue* strips = new AMFArrayValue(); + // auto stripsInState = it->second->GetStrips(); + // for (auto strip = stripsInState.begin(); strip != stripsInState.end(); strip++) { + // Game::logger->Log("PropertyBehaviors", "Begin serialization of strip %i!\n", strip->first); + // AMFArrayValue* thisStrip = new AMFArrayValue(); + + // AMFDoubleValue* stripID = new AMFDoubleValue(); + // stripID->SetValue(strip->first); + // thisStrip->InsertValue("id", stripID); + + // AMFArrayValue* uiArray = new AMFArrayValue(); + // AMFDoubleValue* yPosition = new AMFDoubleValue(); + // yPosition->SetValue(strip->second->GetYPosition()); + // uiArray->InsertValue("y", yPosition); + + // AMFDoubleValue* xPosition = new AMFDoubleValue(); + // xPosition->SetValue(strip->second->GetXPosition()); + // uiArray->InsertValue("x", xPosition); + + // thisStrip->InsertValue("ui", uiArray); + // targetObjectID = modelComponent->GetParent()->GetObjectID(); + // behaviorID = modelBehavior->GetBehaviorID(); + + // AMFArrayValue* stripSerialize = new AMFArrayValue(); + // for (auto behaviorAction : strip->second->GetActions()) { + // Game::logger->Log("PropertyBehaviors", "Begin serialization of action %s!\n", behaviorAction->actionName.c_str()); + // AMFArrayValue* thisAction = new AMFArrayValue(); + + // AMFStringValue* actionName = new AMFStringValue(); + // actionName->SetValue(behaviorAction->actionName); + // thisAction->InsertValue("Type", actionName); + + // if (behaviorAction->parameterValueString != "") + // { + // AMFStringValue* valueAsString = new AMFStringValue(); + // valueAsString->SetValue(behaviorAction->parameterValueString); + // thisAction->InsertValue(behaviorAction->parameterName, valueAsString); + // } + // else if (behaviorAction->parameterValueDouble != 0.0) + // { + // AMFDoubleValue* valueAsDouble = new AMFDoubleValue(); + // valueAsDouble->SetValue(behaviorAction->parameterValueDouble); + // thisAction->InsertValue(behaviorAction->parameterName, valueAsDouble); + // } + // stripSerialize->PushBackValue(thisAction); + // } + // thisStrip->InsertValue("actions", stripSerialize); + // strips->PushBackValue(thisStrip); + // } + // state->InsertValue("strips", strips); + // stateSerialize->PushBackValue(state); + // } + // behaviorInfo.InsertValue("states", stateSerialize); + + // AMFStringValue* objectidAsString = new AMFStringValue(); + // objectidAsString->SetValue(std::to_string(targetObjectID)); + // behaviorInfo.InsertValue("objectID", objectidAsString); + + // AMFStringValue* behaviorIDAsString = new AMFStringValue(); + // behaviorIDAsString->SetValue(std::to_string(behaviorID)); + // behaviorInfo.InsertValue("BehaviorID", behaviorIDAsString); + + // GameMessages::SendUIMessageServerToSingleClient(modelOwner, sysAddr, "UpdateBehaviorBlocks", &behaviorInfo); +} + +void ControlBehaviors::UpdateAction(AMFArrayValue* arguments) { + UpdateActionMessage updateActionMessage(arguments); + auto* blockDefinition = GetBlockInfo(updateActionMessage.GetAction().GetType()); + + if (!blockDefinition) { + Game::logger->Log("ControlBehaviors", "Received undefined block type %s. Ignoring.", updateActionMessage.GetAction().GetType().c_str()); + return; + } + + if (updateActionMessage.GetAction().GetValueParameterString().size() > 0) { + if (updateActionMessage.GetAction().GetValueParameterString().size() < blockDefinition->GetMinimumValue() || + updateActionMessage.GetAction().GetValueParameterString().size() > blockDefinition->GetMaximumValue()) { + Game::logger->Log("ControlBehaviors", "Updated block %s is out of range. Ignoring update", updateActionMessage.GetAction().GetType().c_str()); + return; + } + } else { + if (updateActionMessage.GetAction().GetValueParameterDouble() < blockDefinition->GetMinimumValue() || + updateActionMessage.GetAction().GetValueParameterDouble() > blockDefinition->GetMaximumValue()) { + Game::logger->Log("ControlBehaviors", "Updated block %s is out of range. Ignoring update", updateActionMessage.GetAction().GetType().c_str()); + return; + } + } +} + +void ControlBehaviors::MoveToInventory(ModelComponent* modelComponent, const SystemAddress& sysAddr, Entity* modelOwner, AMFArrayValue* arguments) { + // This closes the UI menu should it be open while the player is removing behaviors + AMFArrayValue args; + + args.Insert("visible", false); + + GameMessages::SendUIMessageServerToSingleClient(modelOwner, modelOwner->GetParentUser()->GetSystemAddress(), "ToggleBehaviorEditor", args); + + MoveToInventoryMessage moveToInventoryMessage(arguments); + + SendBehaviorListToClient(modelComponent->GetParent(), sysAddr, modelOwner); +} + +void ControlBehaviors::ProcessCommand(Entity* modelEntity, const SystemAddress& sysAddr, AMFArrayValue* arguments, std::string command, Entity* modelOwner) { + if (!isInitialized || !modelEntity || !modelOwner || !arguments) return; + auto* modelComponent = modelEntity->GetComponent(); + + if (!modelComponent) return; + + if (command == "sendBehaviorListToClient") + SendBehaviorListToClient(modelEntity, sysAddr, modelOwner); + else if (command == "modelTypeChanged") + ModelTypeChanged(arguments, modelComponent); + else if (command == "toggleExecutionUpdates") + ToggleExecutionUpdates(); + else if (command == "addStrip") + AddStrip(arguments); + else if (command == "removeStrip") + RemoveStrip(arguments); + else if (command == "mergeStrips") + MergeStrips(arguments); + else if (command == "splitStrip") + SplitStrip(arguments); + else if (command == "updateStripUI") + UpdateStripUI(arguments); + else if (command == "addAction") + AddAction(arguments); + else if (command == "migrateActions") + MigrateActions(arguments); + else if (command == "rearrangeStrip") + RearrangeStrip(arguments); + else if (command == "add") + Add(arguments); + else if (command == "removeActions") + RemoveActions(arguments); + else if (command == "rename") + Rename(modelEntity, sysAddr, modelOwner, arguments); + else if (command == "sendBehaviorBlocksToClient") + SendBehaviorBlocksToClient(modelComponent, sysAddr, modelOwner, arguments); + else if (command == "moveToInventory") + MoveToInventory(modelComponent, sysAddr, modelOwner, arguments); + else if (command == "updateAction") + UpdateAction(arguments); + else + Game::logger->Log("ControlBehaviors", "Unknown behavior command (%s)\n", command.c_str()); +} + +ControlBehaviors::ControlBehaviors() { + auto blocksDefStreamBuffer = Game::assetManager->GetFileAsBuffer("ui\\ingame\\blocksdef.xml"); + if (!blocksDefStreamBuffer.m_Success) { + Game::logger->Log("ControlBehaviors", "failed to open blocksdef"); + return; + } + std::istream blocksBuffer(&blocksDefStreamBuffer); + if (!blocksBuffer.good()) { + Game::logger->Log("ControlBehaviors", "Blocks buffer is not good!"); + return; + } + + tinyxml2::XMLDocument m_Doc; + + std::string read{}; + + std::string buffer{}; + bool commentBlockStart = false; + while (std::getline(blocksBuffer, read)) { + // tinyxml2 should handle comment blocks but the client has one that fails the processing. + // This preprocessing just removes all comments from the read file out of an abundance of caution. + if (read.find("") != std::string::npos) { + commentBlockStart = false; + continue; + } + if (!commentBlockStart) buffer += read; + } + + auto ret = m_Doc.Parse(buffer.c_str()); + if (ret == tinyxml2::XML_SUCCESS) { + Game::logger->LogDebug("ControlBehaviors", "Successfully parsed the blocksdef file!"); + } else { + Game::logger->Log("Character", "Failed to parse BlocksDef xmlData due to error %i!", ret); + return; + } + auto* blockLibrary = m_Doc.FirstChildElement(); + if (!blockLibrary) { + Game::logger->Log("ControlBehaviors", "No Block Library child element found."); + return; + } + + // Now parse the blocksdef for the cheat detection server side. + // The client does these checks, but a bad actor can bypass the client checks + auto* blockSections = blockLibrary->FirstChildElement(); + while (blockSections) { + auto* block = blockSections->FirstChildElement(); + std::string blockName{}; + while (block) { + blockName = block->Name(); + + BlockDefinition* blockDefinition = new BlockDefinition(); + std::string name{}; + std::string typeName{}; + + auto* argument = block->FirstChildElement("Argument"); + if (argument) { + auto* defaultDefinition = argument->FirstChildElement("DefaultValue"); + if (defaultDefinition) blockDefinition->SetDefaultValue(defaultDefinition->GetText()); + + auto* typeDefinition = argument->FirstChildElement("Type"); + if (typeDefinition) typeName = typeDefinition->GetText(); + + auto* nameDefinition = argument->FirstChildElement("Name"); + if (nameDefinition) name = nameDefinition->GetText(); + + // Now we parse the blocksdef file for the relevant information + if (typeName == "String") { + blockDefinition->SetMaximumValue(50); // The client has a hardcoded limit of 50 characters in a string field + } else if (typeName == "Float" || typeName == "Integer") { + auto* maximumDefinition = argument->FirstChildElement("Maximum"); + if (maximumDefinition) blockDefinition->SetMaximumValue(std::stof(maximumDefinition->GetText())); + + auto* minimumDefinition = argument->FirstChildElement("Minimum"); + if (minimumDefinition) blockDefinition->SetMinimumValue(std::stof(minimumDefinition->GetText())); + } else if (typeName == "Enumeration") { + auto* values = argument->FirstChildElement("Values"); + if (values) { + auto* value = values->FirstChildElement("Value"); + while (value) { + if (value->GetText() == blockDefinition->GetDefaultValue()) blockDefinition->GetDefaultValue() = std::to_string(blockDefinition->GetMaximumValue()); + blockDefinition->SetMaximumValue(blockDefinition->GetMaximumValue() + 1); + value = value->NextSiblingElement("Value"); + } + blockDefinition->SetMaximumValue(blockDefinition->GetMaximumValue() - 1); // Maximum value is 0 indexed + } else { + values = argument->FirstChildElement("EnumerationSource"); + if (!values) { + Game::logger->Log("ControlBehaviors", "Failed to parse EnumerationSource from block (%s)", blockName.c_str()); + continue; + } + + auto* serviceNameNode = values->FirstChildElement("ServiceName"); + if (!serviceNameNode) { + Game::logger->Log("ControlBehaviors", "Failed to parse ServiceName from block (%s)", blockName.c_str()); + continue; + } + + std::string serviceName = serviceNameNode->GetText(); + if (serviceName == "GetBehaviorSoundList") { + auto res = CDClientDatabase::ExecuteQuery("SELECT MAX(id) as countSounds FROM UGBehaviorSounds;"); + blockDefinition->SetMaximumValue(res.getIntField("countSounds")); + blockDefinition->SetDefaultValue("0"); + } else { + Game::logger->Log("ControlBehaviors", "Unsupported Enumeration ServiceType (%s)", serviceName.c_str()); + continue; + } + } + } else { + Game::logger->Log("ControlBehaviors", "Unsupported block value type (%s)!", typeName.c_str()); + continue; + } + } + blockTypes.insert(std::make_pair(blockName, blockDefinition)); + block = block->NextSiblingElement(); + } + blockSections = blockSections->NextSiblingElement(); + } + isInitialized = true; + Game::logger->LogDebug("ControlBehaviors", "Created all base block classes"); + for (auto b : blockTypes) { + Game::logger->LogDebug("ControlBehaviors", "block name is %s default %s min %f max %f", b.first.c_str(), b.second->GetDefaultValue().c_str(), b.second->GetMinimumValue(), b.second->GetMaximumValue()); + } +} + +BlockDefinition* ControlBehaviors::GetBlockInfo(const BlockName& blockName) { + auto blockDefinition = blockTypes.find(blockName); + return blockDefinition != blockTypes.end() ? blockDefinition->second : nullptr; +} diff --git a/dGame/dPropertyBehaviors/ControlBehaviors.h b/dGame/dPropertyBehaviors/ControlBehaviors.h new file mode 100644 index 00000000..a562aafe --- /dev/null +++ b/dGame/dPropertyBehaviors/ControlBehaviors.h @@ -0,0 +1,68 @@ +#pragma once + +#ifndef __CONTROLBEHAVIORS__H__ +#define __CONTROLBEHAVIORS__H__ + +#include +#include + +#include "Singleton.h" + +class AMFArrayValue; +class BlockDefinition; +class Entity; +class ModelComponent; +class SystemAddress; + +// Type definition to clarify what is used where +typedef std::string BlockName; //! A block name + +class ControlBehaviors: public Singleton { +public: + ControlBehaviors(); + /** + * @brief Main driver for processing Property Behavior commands + * + * @param modelEntity The model that sent this command + * @param sysAddr The SystemAddress to respond to + * @param arguments The arguments formatted as an AMFArrayValue + * @param command The command to perform + * @param modelOwner The owner of the model which sent this command + */ + void ProcessCommand(Entity* modelEntity, const SystemAddress& sysAddr, AMFArrayValue* arguments, std::string command, Entity* modelOwner); + + /** + * @brief Gets a blocks parameter values by the name + * No exception will be thrown in this function. + * + * @param blockName The block name to get the parameters of + * + * @return A pair of the block parameter name to its typing + */ + BlockDefinition* GetBlockInfo(const BlockName& blockName); +private: + void RequestUpdatedID(int32_t behaviorID, ModelComponent* modelComponent, Entity* modelOwner, const SystemAddress& sysAddr); + void SendBehaviorListToClient(Entity* modelEntity, const SystemAddress& sysAddr, Entity* modelOwner); + void ModelTypeChanged(AMFArrayValue* arguments, ModelComponent* ModelComponent); + void ToggleExecutionUpdates(); + void AddStrip(AMFArrayValue* arguments); + void RemoveStrip(AMFArrayValue* arguments); + void MergeStrips(AMFArrayValue* arguments); + void SplitStrip(AMFArrayValue* arguments); + void UpdateStripUI(AMFArrayValue* arguments); + void AddAction(AMFArrayValue* arguments); + void MigrateActions(AMFArrayValue* arguments); + void RearrangeStrip(AMFArrayValue* arguments); + void Add(AMFArrayValue* arguments); + void RemoveActions(AMFArrayValue* arguments); + void Rename(Entity* modelEntity, const SystemAddress& sysAddr, Entity* modelOwner, AMFArrayValue* arguments); + void SendBehaviorBlocksToClient(ModelComponent* modelComponent, const SystemAddress& sysAddr, Entity* modelOwner, AMFArrayValue* arguments); + void UpdateAction(AMFArrayValue* arguments); + void MoveToInventory(ModelComponent* modelComponent, const SystemAddress& sysAddr, Entity* modelOwner, AMFArrayValue* arguments); + std::map blockTypes{}; + + // If false, property behaviors will not be able to be edited. + bool isInitialized = false; +}; + +#endif //!__CONTROLBEHAVIORS__H__ diff --git a/dGame/dUtilities/BrickDatabase.cpp b/dGame/dUtilities/BrickDatabase.cpp index 7ff0febb..e4a6a714 100644 --- a/dGame/dUtilities/BrickDatabase.cpp +++ b/dGame/dUtilities/BrickDatabase.cpp @@ -3,21 +3,27 @@ #include "BrickDatabase.h" #include "Game.h" +#include "AssetManager.h" +#include "tinyxml2.h" +#include "Brick.h" -std::vector BrickDatabase::emptyCache{}; -BrickDatabase* BrickDatabase::m_Address = nullptr; +const BrickList& BrickDatabase::GetBricks(const LxfmlPath& lxfmlPath) { + static std::unordered_map m_Cache; + static const BrickList emptyCache; -BrickDatabase::BrickDatabase() = default; -BrickDatabase::~BrickDatabase() = default; - -std::vector& BrickDatabase::GetBricks(const std::string& lxfmlPath) { const auto cached = m_Cache.find(lxfmlPath); if (cached != m_Cache.end()) { return cached->second; } - std::ifstream file(lxfmlPath); + AssetMemoryBuffer buffer = Game::assetManager->GetFileAsBuffer((lxfmlPath).c_str()); + + if (!buffer.m_Success) { + return emptyCache; + } + + std::istream file(&buffer); if (!file.good()) { return emptyCache; } @@ -25,16 +31,19 @@ std::vector& BrickDatabase::GetBricks(const std::string& lxfmlPath) { std::stringstream data; data << file.rdbuf(); if (data.str().empty()) { + buffer.close(); return emptyCache; } + buffer.close(); + auto* doc = new tinyxml2::XMLDocument(); if (doc->Parse(data.str().c_str(), data.str().size()) != 0) { delete doc; return emptyCache; } - std::vector parts; + BrickList parts; auto* lxfml = doc->FirstChildElement("LXFML"); auto* bricks = lxfml->FirstChildElement("Bricks"); diff --git a/dGame/dUtilities/BrickDatabase.h b/dGame/dUtilities/BrickDatabase.h index 589d46ae..11c64416 100644 --- a/dGame/dUtilities/BrickDatabase.h +++ b/dGame/dUtilities/BrickDatabase.h @@ -1,29 +1,16 @@ +#ifndef __BRICKDATABASE__H__ +#define __BRICKDATABASE__H__ + #pragma once + #include "Entity.h" -class BrickDatabase -{ -public: - static BrickDatabase* Instance() { - if (m_Address == nullptr) { - m_Address = new BrickDatabase(); - } +class Brick; +using BrickList = std::vector; +using LxfmlPath = std::string; - return m_Address; - } - - std::vector& GetBricks(const std::string& lxfmlPath); - - explicit BrickDatabase(); - - ~BrickDatabase(); - -private: - std::unordered_map> m_Cache; - - static std::vector emptyCache; - - static BrickDatabase* m_Address; //For singleton method - - /* data */ +namespace BrickDatabase { + const BrickList& GetBricks(const LxfmlPath& lxfmlPath); }; + +#endif //!__BRICKDATABASE__H__ diff --git a/dGame/dUtilities/CMakeLists.txt b/dGame/dUtilities/CMakeLists.txt index 0c848bf4..55ca5797 100644 --- a/dGame/dUtilities/CMakeLists.txt +++ b/dGame/dUtilities/CMakeLists.txt @@ -1,6 +1,4 @@ set(DGAME_DUTILITIES_SOURCES "BrickDatabase.cpp" - "dLocale.cpp" - "GameConfig.cpp" "GUID.cpp" "Loot.cpp" "Mail.cpp" diff --git a/dGame/dUtilities/GameConfig.cpp b/dGame/dUtilities/GameConfig.cpp deleted file mode 100644 index ad201e6f..00000000 --- a/dGame/dUtilities/GameConfig.cpp +++ /dev/null @@ -1,50 +0,0 @@ -#include "GameConfig.h" -#include - -std::map GameConfig::m_Config{}; -std::string GameConfig::m_EmptyString{}; - -void GameConfig::Load(const std::string& filepath) { - m_EmptyString = ""; - std::ifstream in(filepath); - if (!in.good()) return; - - std::string line; - while (std::getline(in, line)) { - if (line.length() > 0) { - if (line[0] != '#') ProcessLine(line); - } - } -} - -const std::string& GameConfig::GetValue(const std::string& key) { - const auto& it = m_Config.find(key); - - if (it != m_Config.end()) { - return it->second; - } - - return m_EmptyString; -} - -void GameConfig::SetValue(const std::string& key, const std::string& value) { - m_Config.insert_or_assign(key, value); -} - -void GameConfig::ProcessLine(const std::string& line) { - std::stringstream ss(line); - std::string segment; - std::vector seglist; - - while (std::getline(ss, segment, '=')) { - seglist.push_back(segment); - } - - if (seglist.size() != 2) return; - - //Make sure that on Linux, we remove special characters: - if (!seglist[1].empty() && seglist[1][seglist[1].size() - 1] == '\r') - seglist[1].erase(seglist[1].size() - 1); - - m_Config.insert_or_assign(seglist[0], seglist[1]); -} diff --git a/dGame/dUtilities/GameConfig.h b/dGame/dUtilities/GameConfig.h deleted file mode 100644 index 55089f89..00000000 --- a/dGame/dUtilities/GameConfig.h +++ /dev/null @@ -1,38 +0,0 @@ -#pragma once -#include -#include -#include -#include - -#include "GeneralUtils.h" - -class GameConfig { -public: - static void Load(const std::string& filepath); - - static const std::string& GetValue(const std::string& key); - - static void SetValue(const std::string& key, const std::string& value); - - template - static T GetValue(const std::string& key) { - T value; - - if (GeneralUtils::TryParse(GetValue(key), value)) { - return value; - } - - return T(); - } - - template - static void SetValue(const std::string& key, const T& value) { - SetValue(key, std::to_string(value)); - } - -private: - static void ProcessLine(const std::string& line); - - static std::map m_Config; - static std::string m_EmptyString; -}; diff --git a/dGame/dUtilities/Loot.cpp b/dGame/dUtilities/Loot.cpp index 784d873f..c788c016 100644 --- a/dGame/dUtilities/Loot.cpp +++ b/dGame/dUtilities/Loot.cpp @@ -7,19 +7,23 @@ #include "CDLootMatrixTable.h" #include "CDLootTableTable.h" #include "CDRarityTableTable.h" +#include "CDActivityRewardsTable.h" +#include "CDCurrencyTableTable.h" #include "Character.h" #include "Entity.h" #include "GameMessages.h" #include "GeneralUtils.h" #include "InventoryComponent.h" #include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" LootGenerator::LootGenerator() { - CDLootTableTable* lootTableTable = CDClientManager::Instance()->GetTable("LootTable"); - CDComponentsRegistryTable* componentsRegistryTable = CDClientManager::Instance()->GetTable("ComponentsRegistry"); - CDItemComponentTable* itemComponentTable = CDClientManager::Instance()->GetTable("ItemComponent"); - CDLootMatrixTable* lootMatrixTable = CDClientManager::Instance()->GetTable("LootMatrix"); - CDRarityTableTable* rarityTableTable = CDClientManager::Instance()->GetTable("RarityTable"); + CDLootTableTable* lootTableTable = CDClientManager::Instance().GetTable(); + CDComponentsRegistryTable* componentsRegistryTable = CDClientManager::Instance().GetTable(); + CDItemComponentTable* itemComponentTable = CDClientManager::Instance().GetTable(); + CDLootMatrixTable* lootMatrixTable = CDClientManager::Instance().GetTable(); + CDRarityTableTable* rarityTableTable = CDClientManager::Instance().GetTable(); // ============================== // Cache Item Rarities @@ -36,7 +40,7 @@ LootGenerator::LootGenerator() { uniqueItems.erase(std::unique(uniqueItems.begin(), uniqueItems.end()), uniqueItems.end()); for (const uint32_t itemID : uniqueItems) { - uint32_t itemComponentID = componentsRegistryTable->GetByIDAndType(itemID, COMPONENT_TYPE_ITEM); + uint32_t itemComponentID = componentsRegistryTable->GetByIDAndType(itemID, eReplicaComponentType::ITEM); const CDItemComponent& item = itemComponentTable->GetItemComponentByID(itemComponentID); m_ItemRarities.insert({ itemID, item.rarity }); @@ -186,13 +190,13 @@ std::unordered_map LootGenerator::RollLootMatrix(Entity* player, u // convert faction token proxy if (drop.itemID == 13763) { - if (missionComponent->GetMissionState(545) == MissionState::MISSION_STATE_COMPLETE) + if (missionComponent->GetMissionState(545) == eMissionState::COMPLETE) drop.itemID = 8318; // "Assembly Token" - else if (missionComponent->GetMissionState(556) == MissionState::MISSION_STATE_COMPLETE) + else if (missionComponent->GetMissionState(556) == eMissionState::COMPLETE) drop.itemID = 8321; // "Venture League Token" - else if (missionComponent->GetMissionState(567) == MissionState::MISSION_STATE_COMPLETE) + else if (missionComponent->GetMissionState(567) == eMissionState::COMPLETE) drop.itemID = 8319; // "Sentinels Token" - else if (missionComponent->GetMissionState(578) == MissionState::MISSION_STATE_COMPLETE) + else if (missionComponent->GetMissionState(578) == eMissionState::COMPLETE) drop.itemID = 8320; // "Paradox Token" } @@ -290,7 +294,7 @@ void LootGenerator::GiveLoot(Entity* player, std::unordered_map& r } void LootGenerator::GiveActivityLoot(Entity* player, Entity* source, uint32_t activityID, int32_t rating) { - CDActivityRewardsTable* activityRewardsTable = CDClientManager::Instance()->GetTable("ActivityRewards"); + CDActivityRewardsTable* activityRewardsTable = CDClientManager::Instance().GetTable(); std::vector activityRewards = activityRewardsTable->Query([activityID](CDActivityRewards entry) { return (entry.objectTemplate == activityID); }); const CDActivityRewards* selectedReward = nullptr; @@ -306,7 +310,7 @@ void LootGenerator::GiveActivityLoot(Entity* player, Entity* source, uint32_t ac uint32_t minCoins = 0; uint32_t maxCoins = 0; - CDCurrencyTableTable* currencyTableTable = CDClientManager::Instance()->GetTable("CurrencyTable"); + CDCurrencyTableTable* currencyTableTable = CDClientManager::Instance().GetTable(); std::vector currencyTable = currencyTableTable->Query([selectedReward](CDCurrencyTable entry) { return (entry.currencyIndex == selectedReward->CurrencyIndex && entry.npcminlevel == 1); }); if (currencyTable.size() > 0) { @@ -314,13 +318,13 @@ void LootGenerator::GiveActivityLoot(Entity* player, Entity* source, uint32_t ac maxCoins = currencyTable[0].maxvalue; } - GiveLoot(player, selectedReward->LootMatrixIndex, eLootSourceType::LOOT_SOURCE_ACTIVITY); + GiveLoot(player, selectedReward->LootMatrixIndex, eLootSourceType::ACTIVITY); uint32_t coins = (int)(minCoins + GeneralUtils::GenerateRandomNumber(0, 1) * (maxCoins - minCoins)); auto* character = player->GetCharacter(); - character->SetCoins(character->GetCoins() + coins, eLootSourceType::LOOT_SOURCE_ACTIVITY); + character->SetCoins(character->GetCoins() + coins, eLootSourceType::ACTIVITY); } void LootGenerator::DropLoot(Entity* player, Entity* killedObject, uint32_t matrixIndex, uint32_t minCoins, uint32_t maxCoins) { @@ -360,7 +364,7 @@ void LootGenerator::DropLoot(Entity* player, Entity* killedObject, std::unordere } void LootGenerator::DropActivityLoot(Entity* player, Entity* source, uint32_t activityID, int32_t rating) { - CDActivityRewardsTable* activityRewardsTable = CDClientManager::Instance()->GetTable("ActivityRewards"); + CDActivityRewardsTable* activityRewardsTable = CDClientManager::Instance().GetTable(); std::vector activityRewards = activityRewardsTable->Query([activityID](CDActivityRewards entry) { return (entry.objectTemplate == activityID); }); const CDActivityRewards* selectedReward = nullptr; @@ -377,7 +381,7 @@ void LootGenerator::DropActivityLoot(Entity* player, Entity* source, uint32_t ac uint32_t minCoins = 0; uint32_t maxCoins = 0; - CDCurrencyTableTable* currencyTableTable = CDClientManager::Instance()->GetTable("CurrencyTable"); + CDCurrencyTableTable* currencyTableTable = CDClientManager::Instance().GetTable(); std::vector currencyTable = currencyTableTable->Query([selectedReward](CDCurrencyTable entry) { return (entry.currencyIndex == selectedReward->CurrencyIndex && entry.npcminlevel == 1); }); if (currencyTable.size() > 0) { diff --git a/dGame/dUtilities/Loot.h b/dGame/dUtilities/Loot.h index 7ecc22b8..a1c52b63 100644 --- a/dGame/dUtilities/Loot.h +++ b/dGame/dUtilities/Loot.h @@ -47,8 +47,8 @@ public: std::unordered_map RollLootMatrix(Entity* player, uint32_t matrixIndex); std::unordered_map RollLootMatrix(uint32_t matrixIndex); - void GiveLoot(Entity* player, uint32_t matrixIndex, eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE); - void GiveLoot(Entity* player, std::unordered_map& result, eLootSourceType lootSourceType = eLootSourceType::LOOT_SOURCE_NONE); + void GiveLoot(Entity* player, uint32_t matrixIndex, eLootSourceType lootSourceType = eLootSourceType::NONE); + void GiveLoot(Entity* player, std::unordered_map& result, eLootSourceType lootSourceType = eLootSourceType::NONE); void GiveActivityLoot(Entity* player, Entity* source, uint32_t activityID, int32_t rating = 0); void DropLoot(Entity* player, Entity* killedObject, uint32_t matrixIndex, uint32_t minCoins, uint32_t maxCoins); void DropLoot(Entity* player, Entity* killedObject, std::unordered_map& result, uint32_t minCoins, uint32_t maxCoins); diff --git a/dGame/dUtilities/Mail.cpp b/dGame/dUtilities/Mail.cpp index 8280b627..d33af6dc 100644 --- a/dGame/dUtilities/Mail.cpp +++ b/dGame/dUtilities/Mail.cpp @@ -13,7 +13,6 @@ #include "Entity.h" #include "Character.h" #include "PacketUtils.h" -#include "dMessageIdentifiers.h" #include "dLogger.h" #include "EntityManager.h" #include "InventoryComponent.h" @@ -22,6 +21,11 @@ #include "MissionComponent.h" #include "ChatPackets.h" #include "Character.h" +#include "dZoneManager.h" +#include "WorldConfig.h" +#include "eMissionTaskType.h" +#include "eReplicaComponentType.h" +#include "eConnectionType.h" void Mail::SendMail(const Entity* recipient, const std::string& subject, const std::string& body, const LOT attachment, const uint16_t attachmentCount) { @@ -74,12 +78,12 @@ void Mail::SendMail(const LWOOBJID sender, const std::string& senderName, LWOOBJ auto* ins = Database::CreatePreppedStmt("INSERT INTO `mail`(`sender_id`, `sender_name`, `receiver_id`, `receiver_name`, `time_sent`, `subject`, `body`, `attachment_id`, `attachment_lot`, `attachment_subkey`, `attachment_count`, `was_read`) VALUES (?,?,?,?,?,?,?,?,?,?,?,0)"); ins->setUInt(1, sender); - ins->setString(2, senderName); + ins->setString(2, senderName.c_str()); ins->setUInt(3, recipient); ins->setString(4, recipientName.c_str()); ins->setUInt64(5, time(nullptr)); - ins->setString(6, subject); - ins->setString(7, body); + ins->setString(6, subject.c_str()); + ins->setString(7, body.c_str()); ins->setUInt(8, 0); ins->setInt(9, attachment); ins->setInt(10, 0); @@ -126,7 +130,7 @@ void Mail::HandleMailStuff(RakNet::BitStream* packet, const SystemAddress& sysAd int mailStuffID = 0; packet->Read(mailStuffID); - std::async(std::launch::async, [packet, &sysAddr, entity, mailStuffID]() { + auto returnVal = std::async(std::launch::async, [packet, &sysAddr, entity, mailStuffID]() { Mail::MailMessageID stuffID = MailMessageID(mailStuffID); switch (stuffID) { case MailMessageID::AttachmentCollect: @@ -163,7 +167,7 @@ void Mail::HandleSendMail(RakNet::BitStream* packet, const SystemAddress& sysAdd if (!character) return; - if (character->HasPermission(PermissionMap::RestrictedMailAccess)) { + if (character->HasPermission(ePermissionMap::RestrictedMailAccess)) { // Send a message to the player ChatPackets::SendSystemMessage( sysAddr, @@ -191,15 +195,15 @@ void Mail::HandleSendMail(RakNet::BitStream* packet, const SystemAddress& sysAdd uint32_t itemID = static_cast(attachmentID); LOT itemLOT = 0; //Inventory::InventoryType itemType; - int mailCost = 25; + int mailCost = Game::zoneManager->GetWorldConfig()->mailBaseFee; int stackSize = 0; - auto inv = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + auto inv = static_cast(entity->GetComponent(eReplicaComponentType::INVENTORY)); Item* item = nullptr; if (itemID > 0 && attachmentCount > 0 && inv) { item = inv->FindItemById(attachmentID); if (item) { - mailCost += (item->GetInfo().baseValue * 0.1f); + mailCost += (item->GetInfo().baseValue * Game::zoneManager->GetWorldConfig()->mailPercentAttachmentFee); stackSize = item->GetCount(); itemLOT = item->GetLot(); } else { @@ -255,7 +259,7 @@ void Mail::HandleSendMail(RakNet::BitStream* packet, const SystemAddress& sysAdd } Mail::SendSendResponse(sysAddr, Mail::MailSendResponse::Success); - entity->GetCharacter()->SetCoins(entity->GetCharacter()->GetCoins() - mailCost, eLootSourceType::LOOT_SOURCE_MAIL); + entity->GetCharacter()->SetCoins(entity->GetCharacter()->GetCoins() - mailCost, eLootSourceType::MAIL); Game::logger->Log("Mail", "Seeing if we need to remove item with ID/count/LOT: %i %i %i", itemID, attachmentCount, itemLOT); @@ -266,7 +270,7 @@ void Mail::HandleSendMail(RakNet::BitStream* packet, const SystemAddress& sysAdd auto* missionCompoent = entity->GetComponent(); if (missionCompoent != nullptr) { - missionCompoent->Progress(MissionTaskType::MISSION_TASK_TYPE_ITEM_COLLECTION, itemLOT, LWOOBJID_EMPTY, "", -attachmentCount); + missionCompoent->Progress(eMissionTaskType::GATHER, itemLOT, LWOOBJID_EMPTY, "", -attachmentCount); } } @@ -279,7 +283,7 @@ void Mail::HandleDataRequest(RakNet::BitStream* packet, const SystemAddress& sys sql::ResultSet* res = stmt->executeQuery(); RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAIL); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAIL); bitStream.Write(int(MailMessageID::MailData)); bitStream.Write(int(0)); @@ -352,10 +356,10 @@ void Mail::HandleAttachmentCollect(RakNet::BitStream* packet, const SystemAddres attachmentCount = res->getInt(2); } - auto inv = static_cast(player->GetComponent(COMPONENT_TYPE_INVENTORY)); + auto inv = static_cast(player->GetComponent(eReplicaComponentType::INVENTORY)); if (!inv) return; - inv->AddItem(attachmentLOT, attachmentCount, eLootSourceType::LOOT_SOURCE_MAIL); + inv->AddItem(attachmentLOT, attachmentCount, eLootSourceType::MAIL); Mail::SendAttachmentRemoveConfirm(sysAddr, mailID); @@ -389,7 +393,7 @@ void Mail::HandleMailRead(RakNet::BitStream* packet, const SystemAddress& sysAdd } void Mail::HandleNotificationRequest(const SystemAddress& sysAddr, uint32_t objectID) { - std::async(std::launch::async, [&]() { + auto returnVal = std::async(std::launch::async, [&]() { sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT id FROM mail WHERE receiver_id=? AND was_read=0"); stmt->setUInt(1, objectID); sql::ResultSet* res = stmt->executeQuery(); @@ -402,7 +406,7 @@ void Mail::HandleNotificationRequest(const SystemAddress& sysAddr, uint32_t obje void Mail::SendSendResponse(const SystemAddress& sysAddr, MailSendResponse response) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAIL); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAIL); bitStream.Write(int(MailMessageID::SendResponse)); bitStream.Write(int(response)); Game::server->Send(&bitStream, sysAddr, false); @@ -410,7 +414,7 @@ void Mail::SendSendResponse(const SystemAddress& sysAddr, MailSendResponse respo void Mail::SendNotification(const SystemAddress& sysAddr, int mailCount) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAIL); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAIL); uint64_t messageType = 2; uint64_t s1 = 0; uint64_t s2 = 0; @@ -429,7 +433,7 @@ void Mail::SendNotification(const SystemAddress& sysAddr, int mailCount) { void Mail::SendAttachmentRemoveConfirm(const SystemAddress& sysAddr, uint64_t mailID) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAIL); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAIL); bitStream.Write(int(MailMessageID::AttachmentCollectConfirm)); bitStream.Write(int(0)); //unknown bitStream.Write(mailID); @@ -438,7 +442,7 @@ void Mail::SendAttachmentRemoveConfirm(const SystemAddress& sysAddr, uint64_t ma void Mail::SendDeleteConfirm(const SystemAddress& sysAddr, uint64_t mailID, LWOOBJID playerID) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAIL); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAIL); bitStream.Write(int(MailMessageID::MailDeleteConfirm)); bitStream.Write(int(0)); //unknown bitStream.Write(mailID); @@ -452,7 +456,7 @@ void Mail::SendDeleteConfirm(const SystemAddress& sysAddr, uint64_t mailID, LWOO void Mail::SendReadConfirm(const SystemAddress& sysAddr, uint64_t mailID) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAIL); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAIL); bitStream.Write(int(MailMessageID::MailReadConfirm)); bitStream.Write(int(0)); //unknown bitStream.Write(mailID); diff --git a/dGame/dUtilities/Preconditions.cpp b/dGame/dUtilities/Preconditions.cpp index 1f719433..8602586c 100644 --- a/dGame/dUtilities/Preconditions.cpp +++ b/dGame/dUtilities/Preconditions.cpp @@ -12,7 +12,7 @@ #include "LevelProgressionComponent.h" #include "DestroyableComponent.h" #include "GameMessages.h" - +#include "eMissionState.h" std::map Preconditions::cache = {}; @@ -142,19 +142,19 @@ bool Precondition::CheckValue(Entity* player, const uint32_t value, bool evaluat case PreconditionType::HasAchievement: mission = missionComponent->GetMission(value); - return mission == nullptr || mission->GetMissionState() >= MissionState::MISSION_STATE_COMPLETE; + return mission == nullptr || mission->GetMissionState() >= eMissionState::COMPLETE; case PreconditionType::MissionAvailable: mission = missionComponent->GetMission(value); - return mission == nullptr || mission->GetMissionState() >= MissionState::MISSION_STATE_AVAILABLE; + return mission == nullptr || mission->GetMissionState() >= eMissionState::AVAILABLE; case PreconditionType::OnMission: mission = missionComponent->GetMission(value); - return mission == nullptr || mission->GetMissionState() >= MissionState::MISSION_STATE_ACTIVE; + return mission == nullptr || mission->GetMissionState() >= eMissionState::ACTIVE; case PreconditionType::MissionComplete: mission = missionComponent->GetMission(value); - return mission == nullptr || mission->GetMissionState() >= MissionState::MISSION_STATE_COMPLETE; + return mission == nullptr ? false : mission->GetMissionState() >= eMissionState::COMPLETE; case PreconditionType::PetDeployed: return false; // TODO case PreconditionType::HasFlag: @@ -277,11 +277,6 @@ bool PreconditionExpression::Check(Entity* player, bool evaluateCosts) const { return true; } - if (player->GetGMLevel() >= 9) // Developers can skip this for testing - { - return true; - } - const auto a = Preconditions::Check(player, condition, evaluateCosts); if (!a) { diff --git a/dGame/dUtilities/SlashCommandHandler.cpp b/dGame/dUtilities/SlashCommandHandler.cpp index c6f1da9f..03e3cc89 100644 --- a/dGame/dUtilities/SlashCommandHandler.cpp +++ b/dGame/dUtilities/SlashCommandHandler.cpp @@ -50,6 +50,9 @@ #include "Item.h" #include "PropertyManagementComponent.h" #include "PacketUtils.h" +#include "Loot.h" +#include "EntityInfo.h" +#include "LUTriggers.h" #include "Player.h" #include "PhantomPhysicsComponent.h" #include "ProximityMonitorComponent.h" @@ -60,64 +63,64 @@ #include "BuffComponent.h" #include "SkillComponent.h" #include "VanityUtilities.h" -#include "GameConfig.h" #include "ScriptedActivityComponent.h" #include "LevelProgressionComponent.h" +#include "AssetManager.h" +#include "BinaryPathFinder.h" +#include "dConfig.h" +#include "eBubbleType.h" +#include "Amf3.h" +#include "MovingPlatformComponent.h" +#include "eMissionState.h" +#include "TriggerComponent.h" +#include "eServerDisconnectIdentifiers.h" +#include "eObjectBits.h" +#include "eGameMasterLevel.h" +#include "eReplicaComponentType.h" +#include "RenderComponent.h" +#include "eControlScheme.h" +#include "eConnectionType.h" +#include "eChatInternalMessageType.h" +#include "eMasterMessageType.h" + +#include "CDObjectsTable.h" +#include "CDZoneTableTable.h" void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entity* entity, const SystemAddress& sysAddr) { + auto commandCopy = command; + // Sanity check that a command was given + if (command.empty() || command.front() != u'/') return; + commandCopy.erase(commandCopy.begin()); + + // Split the command by spaces std::string chatCommand; std::vector args; + auto wideCommand = GeneralUtils::SplitString(commandCopy, u' '); + if (wideCommand.empty()) return; - uint32_t breakIndex = 0; - for (uint32_t i = 1; i < command.size(); ++i) { - if (command[i] == L' ') { - breakIndex = i; - break; - } + // Convert the command to lowercase + chatCommand = GeneralUtils::UTF16ToWTF8(wideCommand.front()); + std::transform(chatCommand.begin(), chatCommand.end(), chatCommand.begin(), ::tolower); + wideCommand.erase(wideCommand.begin()); - chatCommand.push_back(static_cast(command[i])); - breakIndex++; - } - - uint32_t index = ++breakIndex; - while (true) { - std::string arg; - - while (index < command.size()) { - if (command[index] == L' ') { - args.push_back(arg); - arg = ""; - index++; - continue; - } - - arg.push_back(static_cast(command[index])); - index++; - } - - if (arg != "") { - args.push_back(arg); - } - - break; - } - - //Game::logger->Log("SlashCommandHandler", "Received chat command \"%s\"", GeneralUtils::UTF16ToWTF8(command).c_str()); + // Convert the arguements to not u16strings + for (auto wideArg : wideCommand) args.push_back(GeneralUtils::UTF16ToWTF8(wideArg)); User* user = UserManager::Instance()->GetUser(sysAddr); - if ((chatCommand == "setgmlevel" || chatCommand == "makegm" || chatCommand == "gmlevel") && user->GetMaxGMLevel() > GAME_MASTER_LEVEL_CIVILIAN) { + if ((chatCommand == "setgmlevel" || chatCommand == "makegm" || chatCommand == "gmlevel") && user->GetMaxGMLevel() > eGameMasterLevel::CIVILIAN) { if (args.size() != 1) return; - uint32_t level; + uint32_t level_intermed = 0; - if (!GeneralUtils::TryParse(args[0], level)) { + if (!GeneralUtils::TryParse(args[0], level_intermed)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid gm level."); return; } + eGameMasterLevel level = static_cast(level_intermed); #ifndef DEVELOPER_SERVER - if (user->GetMaxGMLevel() == GAME_MASTER_LEVEL_JUNIOR_DEVELOPER) { - level = GAME_MASTER_LEVEL_CIVILIAN; + if (user->GetMaxGMLevel() == eGameMasterLevel::JUNIOR_DEVELOPER) { + level = eGameMasterLevel::CIVILIAN; } #endif @@ -130,9 +133,9 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit if (success) { - if (entity->GetGMLevel() > GAME_MASTER_LEVEL_CIVILIAN && level == GAME_MASTER_LEVEL_CIVILIAN) { + if (entity->GetGMLevel() > eGameMasterLevel::CIVILIAN && level == eGameMasterLevel::CIVILIAN) { GameMessages::SendToggleGMInvis(entity->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); - } else if (entity->GetGMLevel() == GAME_MASTER_LEVEL_CIVILIAN && level > GAME_MASTER_LEVEL_CIVILIAN) { + } else if (entity->GetGMLevel() == eGameMasterLevel::CIVILIAN && level > eGameMasterLevel::CIVILIAN) { GameMessages::SendToggleGMInvis(entity->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); } @@ -144,10 +147,10 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } #ifndef DEVELOPER_SERVER - if ((entity->GetGMLevel() > user->GetMaxGMLevel()) || (entity->GetGMLevel() > GAME_MASTER_LEVEL_CIVILIAN && user->GetMaxGMLevel() == GAME_MASTER_LEVEL_JUNIOR_DEVELOPER)) { - WorldPackets::SendGMLevelChange(sysAddr, true, user->GetMaxGMLevel(), entity->GetGMLevel(), GAME_MASTER_LEVEL_CIVILIAN); - GameMessages::SendChatModeUpdate(entity->GetObjectID(), GAME_MASTER_LEVEL_CIVILIAN); - entity->SetGMLevel(GAME_MASTER_LEVEL_CIVILIAN); + if ((entity->GetGMLevel() > user->GetMaxGMLevel()) || (entity->GetGMLevel() > eGameMasterLevel::CIVILIAN && user->GetMaxGMLevel() == eGameMasterLevel::JUNIOR_DEVELOPER)) { + WorldPackets::SendGMLevelChange(sysAddr, true, user->GetMaxGMLevel(), entity->GetGMLevel(), eGameMasterLevel::CIVILIAN); + GameMessages::SendChatModeUpdate(entity->GetObjectID(), eGameMasterLevel::CIVILIAN); + entity->SetGMLevel(eGameMasterLevel::CIVILIAN); GameMessages::SendToggleGMInvis(entity->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); @@ -155,6 +158,19 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } #endif + if (chatCommand == "togglenameplate" && (Game::config->GetValue("allow_nameplate_off") == "1" || entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER)) { + auto* character = entity->GetCharacter(); + + if (character && character->GetBillboardVisible()) { + character->SetBillboardVisible(false); + ChatPackets::SendSystemMessage(sysAddr, u"Your nameplate has been turned off and is not visible to players currently in this zone."); + } else { + character->SetBillboardVisible(true); + ChatPackets::SendSystemMessage(sysAddr, u"Your nameplate is now on and visible to all players."); + } + return; + } + //!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! //HANDLE ALL NON GM SLASH COMMANDS RIGHT HERE! //!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! @@ -168,7 +184,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } character->SetPvpEnabled(!character->GetPvpEnabled()); - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); std::stringstream message; message << character->GetName() << " changed their PVP flag to " << std::to_string(character->GetPvpEnabled()) << "!"; @@ -209,22 +225,6 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "skip-ags") { - auto* missionComponent = entity->GetComponent(); - - if (missionComponent != nullptr && missionComponent->HasMission(479)) { - missionComponent->CompleteMission(479); - } - } - - if (chatCommand == "skip-sg") { - auto* missionComponent = entity->GetComponent(); - - if (missionComponent != nullptr && missionComponent->HasMission(229)) { - missionComponent->CompleteMission(229); - } - } - if (chatCommand == "fix-stats") { // Reset skill component and buff component auto* skillComponent = entity->GetComponent(); @@ -247,40 +247,34 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } if (chatCommand == "credits" || chatCommand == "info") { - const auto& customText = chatCommand == "credits" ? VanityUtilities::ParseMarkdown("./vanity/CREDITS.md") : VanityUtilities::ParseMarkdown("./vanity/INFO.md"); + const auto& customText = chatCommand == "credits" ? VanityUtilities::ParseMarkdown((BinaryPathFinder::GetBinaryDir() / "vanity/CREDITS.md").string()) : VanityUtilities::ParseMarkdown((BinaryPathFinder::GetBinaryDir() / "vanity/INFO.md").string()); { AMFArrayValue args; - auto* state = new AMFStringValue(); - state->SetStringValue("Story"); + args.Insert("state", "Story"); - args.InsertValue("state", state); - - GameMessages::SendUIMessageServerToSingleClient(entity, entity->GetSystemAddress(), "pushGameState", &args); + GameMessages::SendUIMessageServerToSingleClient(entity, entity->GetSystemAddress(), "pushGameState", args); } entity->AddCallbackTimer(0.5f, [customText, entity]() { AMFArrayValue args; - auto* text = new AMFStringValue(); - text->SetStringValue(customText); - - args.InsertValue("visible", new AMFTrueValue()); - args.InsertValue("text", text); + args.Insert("visible", true); + args.Insert("text", customText); Game::logger->Log("SlashCommandHandler", "Sending %s", customText.c_str()); - GameMessages::SendUIMessageServerToSingleClient(entity, entity->GetSystemAddress(), "ToggleStoryBox", &args); + GameMessages::SendUIMessageServerToSingleClient(entity, entity->GetSystemAddress(), "ToggleStoryBox", args); }); return; } if (chatCommand == "leave-zone") { - const auto currentZone = dZoneManager::Instance()->GetZone()->GetZoneID().GetMapID(); + const auto currentZone = Game::zoneManager->GetZone()->GetZoneID().GetMapID(); - auto newZone = 0; + LWOMAPID newZone = 0; if (currentZone % 100 == 0) { ChatPackets::SendSystemMessage(sysAddr, u"You are not in an instanced zone."); return; @@ -288,14 +282,14 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit newZone = (currentZone / 100) * 100; } // If new zone would be inaccessible, then default to Avant Gardens. - if (!CheckIfAccessibleZone(newZone)) newZone = 1100; + if (!Game::zoneManager->CheckIfAccessibleZone(newZone)) newZone = 1100; ChatPackets::SendSystemMessage(sysAddr, u"Leaving zone..."); const auto objid = entity->GetObjectID(); ZoneInstanceManager::Instance()->RequestZoneTransfer(Game::server, newZone, 0, false, [objid](bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, std::string serverIP, uint16_t serverPort) { - auto* entity = EntityManager::Instance()->GetEntity(objid); + auto* entity = Game::entityManager->GetEntity(objid); if (entity == nullptr) { return; @@ -336,13 +330,13 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit }); } - if (user->GetMaxGMLevel() == 0 || entity->GetGMLevel() >= 0) { + if (user->GetMaxGMLevel() == eGameMasterLevel::CIVILIAN || entity->GetGMLevel() >= eGameMasterLevel::CIVILIAN) { if (chatCommand == "die") { entity->Smash(entity->GetObjectID()); } if (chatCommand == "resurrect") { - ScriptedActivityComponent* scriptedActivityComponent = dZoneManager::Instance()->GetZoneControlObject()->GetComponent(); + ScriptedActivityComponent* scriptedActivityComponent = Game::zoneManager->GetZoneControlObject()->GetComponent(); if (scriptedActivityComponent) { // check if user is in activity world and if so, they can't resurrect ChatPackets::SendSystemMessage(sysAddr, u"You cannot resurrect in an activity world."); @@ -357,12 +351,12 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } if (chatCommand == "instanceinfo") { - const auto zoneId = dZoneManager::Instance()->GetZone()->GetZoneID(); + const auto zoneId = Game::zoneManager->GetZone()->GetZoneID(); ChatPackets::SendSystemMessage(sysAddr, u"Map: " + (GeneralUtils::to_u16string(zoneId.GetMapID())) + u"\nClone: " + (GeneralUtils::to_u16string(zoneId.GetCloneID())) + u"\nInstance: " + (GeneralUtils::to_u16string(zoneId.GetInstanceID()))); } - if (entity->GetGMLevel() == 0) return; + if (entity->GetGMLevel() == eGameMasterLevel::CIVILIAN) return; } // Log command to database @@ -372,13 +366,13 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit stmt->execute(); delete stmt; - if (chatCommand == "setminifig" && args.size() == 2 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_FORUM_MODERATOR) { // could break characters so only allow if GM > 0 + if (chatCommand == "setminifig" && args.size() == 2 && entity->GetGMLevel() >= eGameMasterLevel::FORUM_MODERATOR) { // could break characters so only allow if GM > 0 int32_t minifigItemId; if (!GeneralUtils::TryParse(args[1], minifigItemId)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid Minifig Item Id ID."); return; } - EntityManager::Instance()->DestructEntity(entity, sysAddr); + Game::entityManager->DestructEntity(entity, sysAddr); auto* charComp = entity->GetComponent(); std::string lowerName = args[0]; if (lowerName.empty()) return; @@ -405,29 +399,29 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit charComp->m_Character->SetLeftHand(minifigItemId); charComp->m_Character->SetRightHand(minifigItemId); } else { - EntityManager::Instance()->ConstructEntity(entity); + Game::entityManager->ConstructEntity(entity); ChatPackets::SendSystemMessage(sysAddr, u"Invalid Minifig item to change, try one of the following: Eyebrows, Eyes, HairColor, HairStyle, Pants, LeftHand, Mouth, RightHand, Shirt, Hands"); return; } - EntityManager::Instance()->ConstructEntity(entity); + Game::entityManager->ConstructEntity(entity); ChatPackets::SendSystemMessage(sysAddr, GeneralUtils::ASCIIToUTF16(lowerName) + u" set to " + (GeneralUtils::to_u16string(minifigItemId))); GameMessages::SendToggleGMInvis(entity->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); // need to retoggle because it gets reenabled on creation of new character } - if ((chatCommand == "playanimation" || chatCommand == "playanim") && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if ((chatCommand == "playanimation" || chatCommand == "playanim") && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { std::u16string anim = GeneralUtils::ASCIIToUTF16(args[0], args[0].size()); - GameMessages::SendPlayAnimation(entity, anim); + RenderComponent::PlayAnimation(entity, anim); auto* possessorComponent = entity->GetComponent(); if (possessorComponent) { - auto* possessedComponent = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); - if (possessedComponent) GameMessages::SendPlayAnimation(possessedComponent, anim); + auto* possessedComponent = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); + if (possessedComponent) RenderComponent::PlayAnimation(possessedComponent, anim); } } - if (chatCommand == "list-spawns" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - for (const auto& pair : EntityManager::Instance()->GetSpawnPointEntities()) { + if (chatCommand == "list-spawns" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { + for (const auto& pair : Game::entityManager->GetSpawnPointEntities()) { ChatPackets::SendSystemMessage(sysAddr, GeneralUtils::ASCIIToUTF16(pair.first)); } @@ -436,7 +430,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "unlock-emote" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "unlock-emote" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { int32_t emoteID; if (!GeneralUtils::TryParse(args[0], emoteID)) { @@ -447,16 +441,16 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit entity->GetCharacter()->UnlockEmote(emoteID); } - if (chatCommand == "force-save" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "force-save" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { entity->GetCharacter()->SaveXMLToDatabase(); } - if (chatCommand == "kill" && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "kill" && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { ChatPackets::SendSystemMessage(sysAddr, u"Brutally murdering that player, if online on this server."); auto* user = UserManager::Instance()->GetUser(args[0]); if (user) { - auto* player = EntityManager::Instance()->GetEntity(user->GetLoggedInChar()); + auto* player = Game::entityManager->GetEntity(user->GetLoggedInChar()); player->Smash(entity->GetObjectID()); ChatPackets::SendSystemMessage(sysAddr, u"It has been done, do you feel good about yourself now?"); return; @@ -466,7 +460,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "speedboost" && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "speedboost" && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { float boost; if (!GeneralUtils::TryParse(args[0], boost)) { @@ -484,7 +478,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit if (possessor) { auto possessedID = possessor->GetPossessable(); if (possessedID != LWOOBJID_EMPTY) { - auto possessable = EntityManager::Instance()->GetEntity(possessedID); + auto possessable = Game::entityManager->GetEntity(possessedID); if (possessable) { auto* possessControllablePhysicsComponent = possessable->GetComponent(); if (possessControllablePhysicsComponent) { @@ -494,20 +488,20 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } } - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } - if (chatCommand == "freecam" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "freecam" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { const auto state = !entity->GetVar(u"freecam"); entity->SetVar(u"freecam", state); - GameMessages::SendSetPlayerControlScheme(entity, static_cast(state ? 9 : 1)); + GameMessages::SendSetPlayerControlScheme(entity, static_cast(state ? 9 : 1)); ChatPackets::SendSystemMessage(sysAddr, u"Toggled freecam."); return; } - if (chatCommand == "setcontrolscheme" && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "setcontrolscheme" && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { uint32_t scheme; if (!GeneralUtils::TryParse(args[0], scheme)) { @@ -515,13 +509,13 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - GameMessages::SendSetPlayerControlScheme(entity, static_cast(scheme)); + GameMessages::SendSetPlayerControlScheme(entity, static_cast(scheme)); ChatPackets::SendSystemMessage(sysAddr, u"Switched control scheme."); return; } - if (chatCommand == "approveproperty" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_LEAD_MODERATOR) { + if (chatCommand == "approveproperty" && entity->GetGMLevel() >= eGameMasterLevel::LEAD_MODERATOR) { if (PropertyManagementComponent::Instance() != nullptr) { PropertyManagementComponent::Instance()->UpdateApprovedStatus(true); @@ -530,59 +524,86 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "setuistate" && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - AMFStringValue* value = new AMFStringValue(); - value->SetStringValue(args[0]); + if (chatCommand == "setuistate" && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { + AMFArrayValue uiState; - AMFArrayValue args; - args.InsertValue("state", value); - GameMessages::SendUIMessageServerToSingleClient(entity, sysAddr, "pushGameState", &args); + uiState.Insert("state", args.at(0)); + + GameMessages::SendUIMessageServerToSingleClient(entity, sysAddr, "pushGameState", uiState); ChatPackets::SendSystemMessage(sysAddr, u"Switched UI state."); return; } - if (chatCommand == "toggle" && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - AMFTrueValue* value = new AMFTrueValue(); - + if (chatCommand == "toggle" && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { AMFArrayValue amfArgs; - amfArgs.InsertValue("visible", value); - GameMessages::SendUIMessageServerToSingleClient(entity, sysAddr, args[0], &amfArgs); + + amfArgs.Insert("visible", true); + + GameMessages::SendUIMessageServerToSingleClient(entity, sysAddr, args[0], amfArgs); ChatPackets::SendSystemMessage(sysAddr, u"Toggled UI state."); return; } - if ((chatCommand == "setinventorysize" || chatCommand == "setinvsize") && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - if (args.size() != 1) return; - + if ((chatCommand == "setinventorysize" || chatCommand == "setinvsize") && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { uint32_t size; - if (!GeneralUtils::TryParse(args[0], size)) { + if (!GeneralUtils::TryParse(args.at(0), size)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid size."); return; } - InventoryComponent* inventory = static_cast(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); - if (inventory) { - auto* items = inventory->GetInventory(ITEMS); + eInventoryType selectedInventory = eInventoryType::ITEMS; - items->SetSize(size); + // a possible inventory was provided if we got more than 1 argument + if (args.size() >= 2) { + selectedInventory = eInventoryType::INVALID; + if (!GeneralUtils::TryParse(args.at(1), selectedInventory)) { + // In this case, we treat the input as a string and try to find it in the reflection list + std::transform(args.at(1).begin(), args.at(1).end(), args.at(1).begin(), ::toupper); + for (uint32_t index = 0; index < NUMBER_OF_INVENTORIES; index++) { + if (std::string_view(args.at(1)) == std::string_view(InventoryType::InventoryTypeToString(static_cast(index)))) selectedInventory = static_cast(index); + } + } + if (selectedInventory == eInventoryType::INVALID) { + ChatPackets::SendSystemMessage(sysAddr, u"Invalid inventory."); + return; + } + + ChatPackets::SendSystemMessage(sysAddr, u"Setting inventory " + + GeneralUtils::ASCIIToUTF16(args.at(1)) + + u" to size " + + GeneralUtils::to_u16string(size)); + } else ChatPackets::SendSystemMessage(sysAddr, u"Setting inventory ITEMS to size " + GeneralUtils::to_u16string(size)); + + auto* inventoryComponent = entity->GetComponent(); + if (inventoryComponent) { + auto* inventory = inventoryComponent->GetInventory(selectedInventory); + + inventory->SetSize(size); } return; } - if (chatCommand == "runmacro" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "runmacro" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() != 1) return; // Only process if input does not contain separator charaters if (args[0].find("/") != std::string::npos) return; if (args[0].find("\\") != std::string::npos) return; - std::ifstream infile("./res/macros/" + args[0] + ".scm"); + auto buf = Game::assetManager->GetFileAsBuffer(("macros/" + args[0] + ".scm").c_str()); + + if (!buf.m_Success) { + ChatPackets::SendSystemMessage(sysAddr, u"Unknown macro! Is the filename right?"); + return; + } + + std::istream infile(&buf); if (infile.good()) { std::string line; @@ -593,10 +614,12 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit ChatPackets::SendSystemMessage(sysAddr, u"Unknown macro! Is the filename right?"); } + buf.close(); + return; } - if (chatCommand == "addmission" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "addmission" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() == 0) return; uint32_t missionID; @@ -606,12 +629,12 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - auto comp = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + auto comp = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (comp) comp->AcceptMission(missionID, true); return; } - if (chatCommand == "completemission" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "completemission" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() == 0) return; uint32_t missionID; @@ -621,13 +644,13 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - auto comp = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + auto comp = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (comp) comp->CompleteMission(missionID, true); return; } - if (chatCommand == "setflag" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() == 1) { - uint32_t flagId; + if (chatCommand == "setflag" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() == 1) { + int32_t flagId; if (!GeneralUtils::TryParse(args[0], flagId)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid flag id."); @@ -637,8 +660,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit entity->GetCharacter()->SetPlayerFlag(flagId, true); } - if (chatCommand == "setflag" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() == 2) { - uint32_t flagId; + if (chatCommand == "setflag" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() == 2) { + int32_t flagId; std::string onOffFlag = args[0]; if (!GeneralUtils::TryParse(args[1], flagId)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid flag id."); @@ -650,8 +673,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } entity->GetCharacter()->SetPlayerFlag(flagId, onOffFlag == "on"); } - if (chatCommand == "clearflag" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() == 1) { - uint32_t flagId; + if (chatCommand == "clearflag" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() == 1) { + int32_t flagId; if (!GeneralUtils::TryParse(args[0], flagId)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid flag id."); @@ -661,7 +684,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit entity->GetCharacter()->SetPlayerFlag(flagId, false); } - if (chatCommand == "resetmission" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "resetmission" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() == 0) return; uint32_t missionID; @@ -671,7 +694,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - auto* comp = static_cast(entity->GetComponent(COMPONENT_TYPE_MISSION)); + auto* comp = static_cast(entity->GetComponent(eReplicaComponentType::MISSION)); if (comp == nullptr) { return; @@ -683,12 +706,12 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - mission->SetMissionState(MissionState::MISSION_STATE_ACTIVE); + mission->SetMissionState(eMissionState::ACTIVE); return; } - if (chatCommand == "playeffect" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 3) { + if (chatCommand == "playeffect" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 3) { int32_t effectID = 0; if (!GeneralUtils::TryParse(args[0], effectID)) { @@ -699,11 +722,11 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit GameMessages::SendPlayFXEffect(entity->GetObjectID(), effectID, GeneralUtils::ASCIIToUTF16(args[1]), args[2]); } - if (chatCommand == "stopeffect" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 1) { + if (chatCommand == "stopeffect" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { GameMessages::SendStopFXEffect(entity, true, args[0]); } - if (chatCommand == "setanntitle" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "setanntitle" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() < 0) return; std::stringstream ss; @@ -714,7 +737,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "setannmsg" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "setannmsg" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() < 0) return; std::stringstream ss; @@ -725,7 +748,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "announce" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "announce" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (entity->GetCharacter()->GetAnnouncementTitle().size() == 0 || entity->GetCharacter()->GetAnnouncementMessage().size() == 0) { ChatPackets::SendSystemMessage(sysAddr, u"Use /setanntitle & /setannmsg <msg> first!"); return; @@ -735,10 +758,10 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "shutdownuniverse" && entity->GetGMLevel() == GAME_MASTER_LEVEL_OPERATOR) { + if (chatCommand == "shutdownuniverse" && entity->GetGMLevel() == eGameMasterLevel::OPERATOR) { //Tell the master server that we're going to be shutting down whole "universe": CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SHUTDOWN_UNIVERSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SHUTDOWN_UNIVERSE); Game::server->SendToMaster(&bitStream); ChatPackets::SendSystemMessage(sysAddr, u"Sent universe shutdown notification to master."); @@ -747,8 +770,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "getnavmeshheight" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - auto control = static_cast<ControllablePhysicsComponent*>(entity->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); + if (chatCommand == "getnavmeshheight" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { + auto control = static_cast<ControllablePhysicsComponent*>(entity->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); if (!control) return; float y = dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(control->GetPosition()); @@ -756,7 +779,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit ChatPackets::SendSystemMessage(sysAddr, msg); } - if (chatCommand == "gmadditem" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "gmadditem" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { if (args.size() == 1) { uint32_t itemLOT; @@ -765,9 +788,9 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - InventoryComponent* inventory = static_cast<InventoryComponent*>(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inventory = static_cast<InventoryComponent*>(entity->GetComponent(eReplicaComponentType::INVENTORY)); - inventory->AddItem(itemLOT, 1, eLootSourceType::LOOT_SOURCE_MODERATION); + inventory->AddItem(itemLOT, 1, eLootSourceType::MODERATION); } else if (args.size() == 2) { uint32_t itemLOT; @@ -783,15 +806,15 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - InventoryComponent* inventory = static_cast<InventoryComponent*>(entity->GetComponent(COMPONENT_TYPE_INVENTORY)); + InventoryComponent* inventory = static_cast<InventoryComponent*>(entity->GetComponent(eReplicaComponentType::INVENTORY)); - inventory->AddItem(itemLOT, count, eLootSourceType::LOOT_SOURCE_MODERATION); + inventory->AddItem(itemLOT, count, eLootSourceType::MODERATION); } else { ChatPackets::SendSystemMessage(sysAddr, u"Correct usage: /gmadditem <lot>"); } } - if (chatCommand == "mailitem" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_MODERATOR && args.size() >= 2) { + if (chatCommand == "mailitem" && entity->GetGMLevel() >= eGameMasterLevel::MODERATOR && args.size() >= 2) { const auto& playerName = args[0]; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT id from charinfo WHERE name=? LIMIT 1;"); @@ -840,7 +863,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "setname" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "setname" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { std::string name = ""; for (const auto& arg : args) { @@ -850,7 +873,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit GameMessages::SendSetName(entity->GetObjectID(), GeneralUtils::UTF8ToUTF16(name), UNASSIGNED_SYSTEM_ADDRESS); } - if (chatCommand == "title" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "title" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { std::string name = entity->GetCharacter()->GetName() + " - "; for (const auto& arg : args) { @@ -860,7 +883,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit GameMessages::SendSetName(entity->GetObjectID(), GeneralUtils::UTF8ToUTF16(name), UNASSIGNED_SYSTEM_ADDRESS); } - if ((chatCommand == "teleport" || chatCommand == "tele") && entity->GetGMLevel() >= GAME_MASTER_LEVEL_JUNIOR_MODERATOR) { + if ((chatCommand == "teleport" || chatCommand == "tele") && entity->GetGMLevel() >= eGameMasterLevel::JUNIOR_MODERATOR) { NiPoint3 pos{}; if (args.size() == 3) { @@ -914,22 +937,22 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit auto* possessorComponent = entity->GetComponent<PossessorComponent>(); if (possessorComponent) { - auto* possassableEntity = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); + auto* possassableEntity = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); if (possassableEntity != nullptr) { auto* vehiclePhysicsComponent = possassableEntity->GetComponent<VehiclePhysicsComponent>(); if (vehiclePhysicsComponent) { vehiclePhysicsComponent->SetPosition(pos); - EntityManager::Instance()->SerializeEntity(possassableEntity); + Game::entityManager->SerializeEntity(possassableEntity); } else GameMessages::SendTeleport(possassableEntity->GetObjectID(), pos, NiQuaternion(), sysAddr); } } } - if (chatCommand == "tpall" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "tpall" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { const auto pos = entity->GetPosition(); - const auto characters = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_CHARACTER); + const auto characters = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::CHARACTER); for (auto* character : characters) { GameMessages::SendTeleport(character->GetObjectID(), pos, NiQuaternion(), character->GetSystemAddress()); @@ -938,18 +961,18 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "dismount" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "dismount" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { auto* possessorComponent = entity->GetComponent<PossessorComponent>(); if (possessorComponent) { auto possessableId = possessorComponent->GetPossessable(); if (possessableId != LWOOBJID_EMPTY) { - auto* possessableEntity = EntityManager::Instance()->GetEntity(possessableId); + auto* possessableEntity = Game::entityManager->GetEntity(possessableId); if (possessableEntity) possessorComponent->Dismount(possessableEntity, true); } } } - if (chatCommand == "fly" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_JUNIOR_DEVELOPER) { + if (chatCommand == "fly" && entity->GetGMLevel() >= eGameMasterLevel::JUNIOR_DEVELOPER) { auto* character = entity->GetCharacter(); if (character) { @@ -985,7 +1008,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit //------- GM COMMANDS TO ACTUALLY MODERATE -------- - if (chatCommand == "mute" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_JUNIOR_DEVELOPER) { + if (chatCommand == "mute" && entity->GetGMLevel() >= eGameMasterLevel::JUNIOR_DEVELOPER) { if (args.size() >= 1) { auto* player = Player::GetPlayer(args[0]); @@ -1004,8 +1027,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit accountId = result->getUInt(1); characterId = result->getUInt64(2); - characterId = GeneralUtils::SetBit(characterId, OBJECT_BIT_CHARACTER); - characterId = GeneralUtils::SetBit(characterId, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(characterId, eObjectBits::CHARACTER); + GeneralUtils::SetBit(characterId, eObjectBits::PERSISTENT); } } @@ -1069,7 +1092,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit //Notify chat about it CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_MUTE_UPDATE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::MUTE_UPDATE); bitStream.Write(characterId); bitStream.Write(expire); @@ -1080,7 +1103,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } } - if (chatCommand == "kick" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_JUNIOR_MODERATOR) { + if (chatCommand == "kick" && entity->GetGMLevel() >= eGameMasterLevel::JUNIOR_MODERATOR) { if (args.size() == 1) { auto* player = Player::GetPlayer(args[0]); @@ -1090,7 +1113,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - Game::server->Disconnect(player->GetSystemAddress(), SERVER_DISCON_KICK); + Game::server->Disconnect(player->GetSystemAddress(), eServerDisconnectIdentifiers::KICK); ChatPackets::SendSystemMessage(sysAddr, u"Kicked: " + username); } else { @@ -1098,7 +1121,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } } - if (chatCommand == "ban" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_SENIOR_MODERATOR) { + if (chatCommand == "ban" && entity->GetGMLevel() >= eGameMasterLevel::SENIOR_MODERATOR) { if (args.size() == 1) { auto* player = Player::GetPlayer(args[0]); @@ -1136,7 +1159,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit delete userUpdate; if (player != nullptr) { - Game::server->Disconnect(player->GetSystemAddress(), SERVER_DISCON_KICK); + Game::server->Disconnect(player->GetSystemAddress(), eServerDisconnectIdentifiers::FREE_TRIAL_EXPIRED); } ChatPackets::SendSystemMessage(sysAddr, u"Banned: " + GeneralUtils::ASCIIToUTF16(args[0])); @@ -1147,8 +1170,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit //------------------------------------------------- - if (chatCommand == "buffme" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - auto dest = static_cast<DestroyableComponent*>(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + if (chatCommand == "buffme" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { + auto dest = static_cast<DestroyableComponent*>(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (dest) { dest->SetHealth(999); dest->SetMaxHealth(999.0f); @@ -1157,10 +1180,10 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit dest->SetImagination(999); dest->SetMaxImagination(999.0f); } - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } - if (chatCommand == "startcelebration" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() == 1) { + if (chatCommand == "startcelebration" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() == 1) { int32_t celebration; if (!GeneralUtils::TryParse(args[0], celebration)) { @@ -1171,8 +1194,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit GameMessages::SendStartCelebrationEffect(entity, entity->GetSystemAddress(), celebration); } - if (chatCommand == "buffmed" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { - auto dest = static_cast<DestroyableComponent*>(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + if (chatCommand == "buffmed" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { + auto dest = static_cast<DestroyableComponent*>(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (dest) { dest->SetHealth(9); dest->SetMaxHealth(9.0f); @@ -1181,26 +1204,32 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit dest->SetImagination(9); dest->SetMaxImagination(9.0f); } - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } - if (chatCommand == "refillstats" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "refillstats" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { - auto dest = static_cast<DestroyableComponent*>(entity->GetComponent(COMPONENT_TYPE_DESTROYABLE)); + auto dest = static_cast<DestroyableComponent*>(entity->GetComponent(eReplicaComponentType::DESTROYABLE)); if (dest) { dest->SetHealth((int)dest->GetMaxHealth()); dest->SetArmor((int)dest->GetMaxArmor()); dest->SetImagination((int)dest->GetMaxImagination()); } - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); } - if (chatCommand == "lookup" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() == 1) { + if (chatCommand == "lookup" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { auto query = CDClientDatabase::CreatePreppedStmt( "SELECT `id`, `name` FROM `Objects` WHERE `displayName` LIKE ?1 OR `name` LIKE ?1 OR `description` LIKE ?1 LIMIT 50"); + // Concatenate all of the arguments into a single query so a multi word query can be used properly. + std::string conditional = args[0]; + args.erase(args.begin()); + for (auto& argument : args) { + conditional += ' ' + argument; + } - const std::string query_text = "%" + args[0] + "%"; + const std::string query_text = "%" + conditional + "%"; query.bind(1, query_text.c_str()); auto tables = query.execQuery(); @@ -1212,8 +1241,8 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } } - if (chatCommand == "spawn" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 1) { - ControllablePhysicsComponent* comp = static_cast<ControllablePhysicsComponent*>(entity->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); + if (chatCommand == "spawn" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { + ControllablePhysicsComponent* comp = static_cast<ControllablePhysicsComponent*>(entity->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); if (!comp) return; uint32_t lot; @@ -1231,29 +1260,68 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit info.spawnerID = entity->GetObjectID(); info.spawnerNodeID = 0; - Entity* newEntity = EntityManager::Instance()->CreateEntity(info, nullptr); + Entity* newEntity = Game::entityManager->CreateEntity(info, nullptr); if (newEntity == nullptr) { ChatPackets::SendSystemMessage(sysAddr, u"Failed to spawn entity."); return; } - auto vehiclePhysicsComponent = newEntity->GetComponent<VehiclePhysicsComponent>(); - if (vehiclePhysicsComponent) { - auto newRot = newEntity->GetRotation(); - auto angles = newRot.GetEulerAngles(); - // make it right side up - angles.x -= PI; - // make it going in the direction of the player - angles.y -= PI; - newRot = NiQuaternion::FromEulerAngles(angles); - newEntity->SetRotation(newRot); - } - - EntityManager::Instance()->ConstructEntity(newEntity); + Game::entityManager->ConstructEntity(newEntity); } - if ((chatCommand == "giveuscore") && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "spawngroup" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 3) { + auto controllablePhysicsComponent = entity->GetComponent<ControllablePhysicsComponent>(); + if (!controllablePhysicsComponent) return; + + LOT lot{}; + uint32_t numberToSpawn{}; + float radiusToSpawnWithin{}; + + if (!GeneralUtils::TryParse(args[0], lot)) { + ChatPackets::SendSystemMessage(sysAddr, u"Invalid lot."); + return; + } + + if (!GeneralUtils::TryParse(args[1], numberToSpawn) && numberToSpawn > 0) { + ChatPackets::SendSystemMessage(sysAddr, u"Invalid number of enemies to spawn."); + return; + } + + // Must spawn within a radius of at least 0.0f + if (!GeneralUtils::TryParse(args[2], radiusToSpawnWithin) && radiusToSpawnWithin < 0.0f) { + ChatPackets::SendSystemMessage(sysAddr, u"Invalid radius to spawn within."); + return; + } + + EntityInfo info; + info.lot = lot; + info.spawner = nullptr; + info.spawnerID = entity->GetObjectID(); + info.spawnerNodeID = 0; + + auto playerPosition = controllablePhysicsComponent->GetPosition(); + while (numberToSpawn > 0) { + auto randomAngle = GeneralUtils::GenerateRandomNumber<float>(0.0f, 2 * PI); + auto randomRadius = GeneralUtils::GenerateRandomNumber<float>(0.0f, radiusToSpawnWithin); + + // Set the position to the generated random position plus the player position. This will + // spawn the entity in a circle around the player. As you get further from the player, the angle chosen will get less accurate. + info.pos = playerPosition + NiPoint3(cos(randomAngle) * randomRadius, 0.0f, sin(randomAngle) * randomRadius); + info.rot = NiQuaternion(); + + auto newEntity = Game::entityManager->CreateEntity(info); + if (newEntity == nullptr) { + ChatPackets::SendSystemMessage(sysAddr, u"Failed to spawn entity."); + return; + } + + Game::entityManager->ConstructEntity(newEntity); + numberToSpawn--; + } + } + + if ((chatCommand == "giveuscore") && args.size() >= 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { int32_t uscore; if (!GeneralUtils::TryParse(args[0], uscore)) { @@ -1263,11 +1331,19 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit CharacterComponent* character = entity->GetComponent<CharacterComponent>(); if (character) character->SetUScore(character->GetUScore() + uscore); - // LOOT_SOURCE_MODERATION should work but it doesn't. Relog to see uscore changes - GameMessages::SendModifyLEGOScore(entity, entity->GetSystemAddress(), uscore, eLootSourceType::LOOT_SOURCE_MODERATION); + // MODERATION should work but it doesn't. Relog to see uscore changes + + eLootSourceType lootType = eLootSourceType::MODERATION; + + int32_t type; + if (args.size() >= 2 && GeneralUtils::TryParse(args[1], type)) { + lootType = (eLootSourceType)type; + } + + GameMessages::SendModifyLEGOScore(entity, entity->GetSystemAddress(), uscore, lootType); } - if ((chatCommand == "setlevel") && args.size() >= 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if ((chatCommand == "setlevel") && args.size() >= 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { // We may be trying to set a specific players level to a level. If so override the entity with the requested players. std::string requestedPlayerToSetLevelOf = ""; if (args.size() > 1) { @@ -1335,7 +1411,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "pos" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "pos" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { const auto position = entity->GetPosition(); ChatPackets::SendSystemMessage(sysAddr, u"<" + (GeneralUtils::to_u16string(position.x)) + u", " + (GeneralUtils::to_u16string(position.y)) + u", " + (GeneralUtils::to_u16string(position.z)) + u">"); @@ -1343,7 +1419,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit std::cout << position.x << ", " << position.y << ", " << position.z << std::endl; } - if (chatCommand == "rot" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "rot" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { const auto rotation = entity->GetRotation(); ChatPackets::SendSystemMessage(sysAddr, u"<" + (GeneralUtils::to_u16string(rotation.w)) + u", " + (GeneralUtils::to_u16string(rotation.x)) + u", " + (GeneralUtils::to_u16string(rotation.y)) + u", " + (GeneralUtils::to_u16string(rotation.z)) + u">"); @@ -1351,23 +1427,23 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit std::cout << rotation.w << ", " << rotation.x << ", " << rotation.y << ", " << rotation.z << std::endl; } - if (chatCommand == "locrow" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "locrow" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { const auto position = entity->GetPosition(); const auto rotation = entity->GetRotation(); std::cout << "<location x=\"" << position.x << "\" y=\"" << position.y << "\" z=\"" << position.z << "\" rw=\"" << rotation.w << "\" rx=\"" << rotation.x << "\" ry=\"" << rotation.y << "\" rz=\"" << rotation.z << "\" />" << std::endl; } - if (chatCommand == "playlvlfx" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "playlvlfx" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { GameMessages::SendPlayFXEffect(entity, 7074, u"create", "7074", LWOOBJID_EMPTY, 1.0f, 1.0f, true); } - if (chatCommand == "playrebuildfx" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "playrebuildfx" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { GameMessages::SendPlayFXEffect(entity, 230, u"rebuild", "230", LWOOBJID_EMPTY, 1.0f, 1.0f, true); } - if ((chatCommand == "freemoney" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) && args.size() == 1) { - int32_t money; + if ((chatCommand == "freemoney" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) && args.size() == 1) { + int64_t money; if (!GeneralUtils::TryParse(args[0], money)) { ChatPackets::SendSystemMessage(sysAddr, u"Invalid money."); @@ -1375,10 +1451,10 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } auto* ch = entity->GetCharacter(); - ch->SetCoins(ch->GetCoins() + money, eLootSourceType::LOOT_SOURCE_MODERATION); + ch->SetCoins(ch->GetCoins() + money, eLootSourceType::MODERATION); } - if ((chatCommand == "setcurrency") && args.size() == 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if ((chatCommand == "setcurrency") && args.size() == 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { int32_t money; if (!GeneralUtils::TryParse(args[0], money)) { @@ -1387,17 +1463,17 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } auto* ch = entity->GetCharacter(); - ch->SetCoins(money, eLootSourceType::LOOT_SOURCE_MODERATION); + ch->SetCoins(money, eLootSourceType::MODERATION); } // Allow for this on even while not a GM, as it sometimes toggles incorrrectly. - if (chatCommand == "gminvis" && entity->GetParentUser()->GetMaxGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "gminvis" && entity->GetParentUser()->GetMaxGMLevel() >= eGameMasterLevel::DEVELOPER) { GameMessages::SendToggleGMInvis(entity->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); return; } - if (chatCommand == "gmimmune" && args.size() >= 1 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "gmimmune" && args.size() >= 1 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { auto* destroyableComponent = entity->GetComponent<DestroyableComponent>(); int32_t state = false; @@ -1414,7 +1490,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "buff" && args.size() >= 2 && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "buff" && args.size() >= 2 && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { auto* buffComponent = entity->GetComponent<BuffComponent>(); int32_t id = 0; @@ -1437,7 +1513,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if ((chatCommand == "testmap" && args.size() >= 1) && entity->GetGMLevel() >= GAME_MASTER_LEVEL_FORUM_MODERATOR) { + if ((chatCommand == "testmap" && args.size() >= 1) && entity->GetGMLevel() >= eGameMasterLevel::FORUM_MODERATOR) { ChatPackets::SendSystemMessage(sysAddr, u"Requesting map change..."); uint32_t reqZone; LWOCLONEID cloneId = 0; @@ -1465,11 +1541,11 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit const auto objid = entity->GetObjectID(); - if (force || CheckIfAccessibleZone(reqZone)) { // to prevent tomfoolery + if (force || Game::zoneManager->CheckIfAccessibleZone(reqZone)) { // to prevent tomfoolery ZoneInstanceManager::Instance()->RequestZoneTransfer(Game::server, reqZone, cloneId, false, [objid](bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, std::string serverIP, uint16_t serverPort) { - auto* entity = EntityManager::Instance()->GetEntity(objid); + auto* entity = Game::entityManager->GetEntity(objid); if (!entity) return; const auto sysAddr = entity->GetSystemAddress(); @@ -1496,7 +1572,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } } - if (chatCommand == "createprivate" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 3) { + if (chatCommand == "createprivate" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 3) { uint32_t zone; if (!GeneralUtils::TryParse(args[0], zone)) { @@ -1520,20 +1596,20 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if ((chatCommand == "debugui") && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if ((chatCommand == "debugui") && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { ChatPackets::SendSystemMessage(sysAddr, u"Opening UIDebugger..."); AMFArrayValue args; - GameMessages::SendUIMessageServerToSingleClient(entity, sysAddr, "ToggleUIDebugger;", nullptr); + GameMessages::SendUIMessageServerToSingleClient(entity, sysAddr, "ToggleUIDebugger;", args); } - if ((chatCommand == "boost") && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if ((chatCommand == "boost") && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { auto* possessorComponent = entity->GetComponent<PossessorComponent>(); if (possessorComponent == nullptr) { return; } - auto* vehicle = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); + auto* vehicle = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); if (vehicle == nullptr) { return; @@ -1558,44 +1634,44 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } - if ((chatCommand == "unboost") && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if ((chatCommand == "unboost") && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { auto* possessorComponent = entity->GetComponent<PossessorComponent>(); if (possessorComponent == nullptr) return; - auto* vehicle = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); + auto* vehicle = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); if (vehicle == nullptr) return; GameMessages::SendVehicleRemovePassiveBoostAction(vehicle->GetObjectID(), UNASSIGNED_SYSTEM_ADDRESS); } - if (chatCommand == "activatespawner" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 1) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(args[0]); + if (chatCommand == "activatespawner" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { + auto spawners = Game::zoneManager->GetSpawnersByName(args[0]); for (auto* spawner : spawners) { spawner->Activate(); } - spawners = dZoneManager::Instance()->GetSpawnersInGroup(args[0]); + spawners = Game::zoneManager->GetSpawnersInGroup(args[0]); for (auto* spawner : spawners) { spawner->Activate(); } } - if (chatCommand == "spawnphysicsverts" && entity->GetGMLevel() >= 6) { + if (chatCommand == "spawnphysicsverts" && entity->GetGMLevel() >= eGameMasterLevel::JUNIOR_DEVELOPER) { //Go tell physics to spawn all the vertices: - auto entities = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_PHANTOM_PHYSICS); + auto entities = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::PHANTOM_PHYSICS); for (auto en : entities) { - auto phys = static_cast<PhantomPhysicsComponent*>(en->GetComponent(COMPONENT_TYPE_PHANTOM_PHYSICS)); + auto phys = static_cast<PhantomPhysicsComponent*>(en->GetComponent(eReplicaComponentType::PHANTOM_PHYSICS)); if (phys) phys->SpawnVertices(); } } - if (chatCommand == "reportproxphys" && entity->GetGMLevel() >= 6) { - auto entities = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_PROXIMITY_MONITOR); + if (chatCommand == "reportproxphys" && entity->GetGMLevel() >= eGameMasterLevel::JUNIOR_DEVELOPER) { + auto entities = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::PROXIMITY_MONITOR); for (auto en : entities) { - auto phys = static_cast<ProximityMonitorComponent*>(en->GetComponent(COMPONENT_TYPE_PROXIMITY_MONITOR)); + auto phys = static_cast<ProximityMonitorComponent*>(en->GetComponent(eReplicaComponentType::PROXIMITY_MONITOR)); if (phys) { for (auto prox : phys->GetProximitiesData()) { if (!prox.second) continue; @@ -1608,21 +1684,21 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit } } - if (chatCommand == "triggerspawner" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 1) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(args[0]); + if (chatCommand == "triggerspawner" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { + auto spawners = Game::zoneManager->GetSpawnersByName(args[0]); for (auto* spawner : spawners) { spawner->Spawn(); } - spawners = dZoneManager::Instance()->GetSpawnersInGroup(args[0]); + spawners = Game::zoneManager->GetSpawnersInGroup(args[0]); for (auto* spawner : spawners) { spawner->Spawn(); } } - if (chatCommand == "reforge" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 2) { + if (chatCommand == "reforge" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 2) { LOT baseItem; LOT reforgedItem; @@ -1636,10 +1712,10 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit std::vector<LDFBaseData*> data{}; data.push_back(new LDFData<int32_t>(u"reforgedLOT", reforgedItem)); - inventoryComponent->AddItem(baseItem, 1, eLootSourceType::LOOT_SOURCE_MODERATION, eInventoryType::INVALID, data); + inventoryComponent->AddItem(baseItem, 1, eLootSourceType::MODERATION, eInventoryType::INVALID, data); } - if (chatCommand == "crash" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_OPERATOR) { + if (chatCommand == "crash" && entity->GetGMLevel() >= eGameMasterLevel::OPERATOR) { ChatPackets::SendSystemMessage(sysAddr, u"Crashing..."); int* badPtr = nullptr; @@ -1648,26 +1724,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "config-set" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 2) { - GameConfig::SetValue(args[0], args[1]); - - ChatPackets::SendSystemMessage( - sysAddr, u"Set config value: " + GeneralUtils::UTF8ToUTF16(args[0]) + u" to " + GeneralUtils::UTF8ToUTF16(args[1]) - ); - } - - if (chatCommand == "config-get" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 1) { - const auto& value = GameConfig::GetValue(args[0]); - - std::u16string u16key = GeneralUtils::UTF8ToUTF16(args[0]); - if (value.empty()) { - ChatPackets::SendSystemMessage(sysAddr, u"No value found for " + u16key); - } else { - ChatPackets::SendSystemMessage(sysAddr, u"Value for " + u16key + u": " + GeneralUtils::UTF8ToUTF16(value)); - } - } - - if (chatCommand == "metrics" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER) { + if (chatCommand == "metrics" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { for (const auto variable : Metrics::GetAllMetrics()) { auto* metric = Metrics::GetMetric(variable); @@ -1704,7 +1761,23 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - if (chatCommand == "rollloot" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_OPERATOR && args.size() >= 3) { + if (chatCommand == "reloadconfig" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER) { + Game::config->ReloadConfig(); + VanityUtilities::SpawnVanity(); + dpWorld::Instance().Reload(); + auto entities = Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::SCRIPTED_ACTIVITY); + for (auto entity : entities) { + auto* scriptedActivityComponent = entity->GetComponent<ScriptedActivityComponent>(); + if (!scriptedActivityComponent) continue; + + scriptedActivityComponent->ReloadConfig(); + } + Game::server->UpdateMaximumMtuSize(); + Game::server->UpdateBandwidthLimit(); + ChatPackets::SendSystemMessage(sysAddr, u"Successfully reloaded config for world!"); + } + + if (chatCommand == "rollloot" && entity->GetGMLevel() >= eGameMasterLevel::OPERATOR && args.size() >= 3) { uint32_t lootMatrixIndex = 0; uint32_t targetLot = 0; uint32_t loops = 1; @@ -1741,17 +1814,44 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit ChatPackets::SendSystemMessage(sysAddr, message); } - if (chatCommand == "inspect" && entity->GetGMLevel() >= GAME_MASTER_LEVEL_DEVELOPER && args.size() >= 1) { + if (chatCommand == "deleteinven" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { + eInventoryType inventoryType = eInventoryType::INVALID; + if (!GeneralUtils::TryParse(args[0], inventoryType)) { + // In this case, we treat the input as a string and try to find it in the reflection list + std::transform(args[0].begin(), args[0].end(), args[0].begin(), ::toupper); + Game::logger->Log("SlashCommandHandler", "looking for inventory %s", args[0].c_str()); + for (uint32_t index = 0; index < NUMBER_OF_INVENTORIES; index++) { + if (std::string_view(args[0]) == std::string_view(InventoryType::InventoryTypeToString(static_cast<eInventoryType>(index)))) inventoryType = static_cast<eInventoryType>(index); + } + } + + if (inventoryType == eInventoryType::INVALID || inventoryType >= NUMBER_OF_INVENTORIES) { + ChatPackets::SendSystemMessage(sysAddr, u"Invalid inventory provided."); + return; + } + + auto* inventoryComponent = entity->GetComponent<InventoryComponent>(); + if (!inventoryComponent) return; + + auto* inventoryToDelete = inventoryComponent->GetInventory(inventoryType); + if (!inventoryToDelete) return; + + inventoryToDelete->DeleteAllItems(); + Game::logger->Log("SlashCommandHandler", "Deleted inventory %s for user %llu", args[0].c_str(), entity->GetObjectID()); + ChatPackets::SendSystemMessage(sysAddr, u"Deleted inventory " + GeneralUtils::UTF8ToUTF16(args[0])); + } + + if (chatCommand == "inspect" && entity->GetGMLevel() >= eGameMasterLevel::DEVELOPER && args.size() >= 1) { Entity* closest = nullptr; - int32_t component; + eReplicaComponentType component; std::u16string ldf; bool isLDF = false; if (!GeneralUtils::TryParse(args[0], component)) { - component = -1; + component = eReplicaComponentType::INVALID; ldf = GeneralUtils::UTF8ToUTF16(args[0]); @@ -1762,7 +1862,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit auto closestDistance = 0.0f; - const auto candidates = EntityManager::Instance()->GetEntitiesByComponent(component); + const auto candidates = Game::entityManager->GetEntitiesByComponent(component); for (auto* candidate : candidates) { if (candidate->GetLOT() == 1 || candidate->GetLOT() == 8092) { @@ -1794,9 +1894,9 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit return; } - EntityManager::Instance()->SerializeEntity(closest); + Game::entityManager->SerializeEntity(closest); - auto* table = CDClientManager::Instance()->GetTable<CDObjectsTable>("Objects"); + auto* table = CDClientManager::Instance().GetTable<CDObjectsTable>(); const auto& info = table->GetByID(closest->GetLOT()); @@ -1811,7 +1911,7 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit std::stringstream stream; - stream << "Component [" << std::to_string(id) << "]"; + stream << "Component [" << std::to_string(static_cast<uint32_t>(id)) << "]"; ChatPackets::SendSystemMessage(sysAddr, GeneralUtils::ASCIIToUTF16(stream.str())); } @@ -1834,9 +1934,9 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit movingPlatformComponent->GotoWaypoint(value); } - EntityManager::Instance()->SerializeEntity(closest); + Game::entityManager->SerializeEntity(closest); } else if (args[1] == "-a" && args.size() >= 3) { - GameMessages::SendPlayAnimation(closest, GeneralUtils::UTF8ToUTF16(args[2])); + RenderComponent::PlayAnimation(closest, args.at(2)); } else if (args[1] == "-s") { for (auto* entry : closest->GetSettings()) { ChatPackets::SendSystemMessage(sysAddr, GeneralUtils::UTF8ToUTF16(entry->GetString())); @@ -1884,50 +1984,36 @@ void SlashCommandHandler::HandleChatCommand(const std::u16string& command, Entit auto* phantomPhysicsComponent = closest->GetComponent<PhantomPhysicsComponent>(); if (phantomPhysicsComponent != nullptr) { - ChatPackets::SendSystemMessage(sysAddr, u"Type: " + (GeneralUtils::to_u16string(phantomPhysicsComponent->GetEffectType()))); + ChatPackets::SendSystemMessage(sysAddr, u"Type: " + (GeneralUtils::to_u16string(static_cast<uint32_t>(phantomPhysicsComponent->GetEffectType())))); const auto dir = phantomPhysicsComponent->GetDirection(); ChatPackets::SendSystemMessage(sysAddr, u"Direction: <" + (GeneralUtils::to_u16string(dir.x)) + u", " + (GeneralUtils::to_u16string(dir.y)) + u", " + (GeneralUtils::to_u16string(dir.z)) + u">"); ChatPackets::SendSystemMessage(sysAddr, u"Multiplier: " + (GeneralUtils::to_u16string(phantomPhysicsComponent->GetDirectionalMultiplier()))); ChatPackets::SendSystemMessage(sysAddr, u"Active: " + (GeneralUtils::to_u16string(phantomPhysicsComponent->GetPhysicsEffectActive()))); } - if (closest->GetTrigger() != nullptr) { - ChatPackets::SendSystemMessage(sysAddr, u"Trigger: " + (GeneralUtils::to_u16string(closest->GetTrigger()->id))); + auto* triggerComponent = closest->GetComponent<TriggerComponent>(); + if (triggerComponent) { + auto trigger = triggerComponent->GetTrigger(); + if (trigger) { + ChatPackets::SendSystemMessage(sysAddr, u"Trigger: " + (GeneralUtils::to_u16string(trigger->id))); + } } } } } } -bool SlashCommandHandler::CheckIfAccessibleZone(const unsigned int zoneID) { - //We're gonna go ahead and presume we've got the db loaded already: - CDZoneTableTable* zoneTable = CDClientManager::Instance()->GetTable<CDZoneTableTable>("ZoneTable"); - const CDZoneTable* zone = zoneTable->Query(zoneID); - if (zone != nullptr) { - std::string zonePath = "./res/maps/" + zone->zoneName; - std::transform(zonePath.begin(), zonePath.end(), zonePath.begin(), ::tolower); - std::ifstream f(zonePath.c_str()); - return f.good(); - } else { - return false; - } -} - void SlashCommandHandler::SendAnnouncement(const std::string& title, const std::string& message) { AMFArrayValue args; - auto* titleValue = new AMFStringValue(); - titleValue->SetStringValue(title); - auto* messageValue = new AMFStringValue(); - messageValue->SetStringValue(message); - args.InsertValue("title", titleValue); - args.InsertValue("message", messageValue); + args.Insert("title", title); + args.Insert("message", message); - GameMessages::SendUIMessageServerToAllClients("ToggleAnnounce", &args); + GameMessages::SendUIMessageServerToAllClients("ToggleAnnounce", args); //Notify chat about it CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_ANNOUNCEMENT); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::ANNOUNCEMENT); bitStream.Write<uint32_t>(title.size()); for (auto character : title) { diff --git a/dGame/dUtilities/SlashCommandHandler.h b/dGame/dUtilities/SlashCommandHandler.h index 9ef09a2f..85b7c697 100644 --- a/dGame/dUtilities/SlashCommandHandler.h +++ b/dGame/dUtilities/SlashCommandHandler.h @@ -13,8 +13,6 @@ class Entity; namespace SlashCommandHandler { void HandleChatCommand(const std::u16string& command, Entity* entity, const SystemAddress& sysAddr); - bool CheckIfAccessibleZone(const unsigned int zoneID); - void SendAnnouncement(const std::string& title, const std::string& message); }; diff --git a/dGame/dUtilities/VanityUtilities.cpp b/dGame/dUtilities/VanityUtilities.cpp index 733d94c4..eb182e04 100644 --- a/dGame/dUtilities/VanityUtilities.cpp +++ b/dGame/dUtilities/VanityUtilities.cpp @@ -13,6 +13,8 @@ #include "tinyxml2.h" #include "Game.h" #include "dLogger.h" +#include "BinaryPathFinder.h" +#include "EntityInfo.h" #include <fstream> @@ -27,7 +29,7 @@ void VanityUtilities::SpawnVanity() { const uint32_t zoneID = Game::server->GetZoneID(); - ParseXML("./vanity/NPC.xml"); + ParseXML((BinaryPathFinder::GetBinaryDir() / "vanity/NPC.xml").string()); // Loop through all parties for (const auto& party : m_Parties) { @@ -63,32 +65,36 @@ void VanityUtilities::SpawnVanity() { npcIndex = GeneralUtils::GenerateRandomNumber<uint32_t>(0, npcList.size() - 1); } - const auto& npc = npcList[npcIndex]; + auto& npc = npcList[npcIndex]; taken.push_back(npcIndex); // Spawn the NPC - std::vector<LDFBaseData*> data = { new LDFData<std::vector<std::u16string>>( - u"syncLDF", { u"custom_script_client" }), - new LDFData<std::u16string>(u"custom_script_client", u"scripts\\ai\\SPEC\\MISSION_MINIGAME_CLIENT.lua") }; + Game::logger->Log("VanityUtilities", "ldf size is %i", npc.ldf.size()); + if (npc.ldf.empty()) { + npc.ldf = { + new LDFData<std::vector<std::u16string>>(u"syncLDF", { u"custom_script_client" }), + new LDFData<std::u16string>(u"custom_script_client", u"scripts\\ai\\SPEC\\MISSION_MINIGAME_CLIENT.lua") + }; + } // Spawn the NPC - auto* npcEntity = SpawnNPC(npc.m_LOT, npc.m_Name, location.m_Position, location.m_Rotation, npc.m_Equipment, data); - - npcEntity->SetVar<std::vector<std::string>>(u"chats", m_PartyPhrases); - - SetupNPCTalk(npcEntity); + auto* npcEntity = SpawnNPC(npc.m_LOT, npc.m_Name, location.m_Position, location.m_Rotation, npc.m_Equipment, npc.ldf); + if (!npc.m_Phrases.empty()) { + npcEntity->SetVar<std::vector<std::string>>(u"chats", m_PartyPhrases); + SetupNPCTalk(npcEntity); + } } return; } // Loop through all NPCs - for (const auto& pair : m_NPCs) { - if (pair.m_Locations.find(Game::server->GetZoneID()) == pair.m_Locations.end()) + for (auto& npc : m_NPCs) { + if (npc.m_Locations.find(Game::server->GetZoneID()) == npc.m_Locations.end()) continue; - const std::vector<VanityNPCLocation>& locations = pair.m_Locations.at(Game::server->GetZoneID()); + const std::vector<VanityNPCLocation>& locations = npc.m_Locations.at(Game::server->GetZoneID()); // Pick a random location const auto& location = locations[GeneralUtils::GenerateRandomNumber<int>( @@ -99,27 +105,30 @@ void VanityUtilities::SpawnVanity() { continue; } - std::vector<LDFBaseData*> data = { new LDFData<std::vector<std::u16string>>( - u"syncLDF", { u"custom_script_client" }), - new LDFData<std::u16string>(u"custom_script_client", u"scripts\\ai\\SPEC\\MISSION_MINIGAME_CLIENT.lua") }; - - // Spawn the NPC - auto* npc = SpawnNPC(pair.m_LOT, pair.m_Name, location.m_Position, location.m_Rotation, pair.m_Equipment, data); - - npc->SetVar<std::vector<std::string>>(u"chats", pair.m_Phrases); - - auto* scriptComponent = npc->GetComponent<ScriptComponent>(); - - if (scriptComponent != nullptr) { - scriptComponent->SetScript(pair.m_Script); - scriptComponent->SetSerialized(false); - - for (const auto& pair : pair.m_Flags) { - npc->SetVar<bool>(GeneralUtils::ASCIIToUTF16(pair.first), pair.second); - } + if (npc.ldf.empty()) { + npc.ldf = { + new LDFData<std::vector<std::u16string>>(u"syncLDF", { u"custom_script_client" }), + new LDFData<std::u16string>(u"custom_script_client", u"scripts\\ai\\SPEC\\MISSION_MINIGAME_CLIENT.lua") + }; } - SetupNPCTalk(npc); + // Spawn the NPC + auto* npcEntity = SpawnNPC(npc.m_LOT, npc.m_Name, location.m_Position, location.m_Rotation, npc.m_Equipment, npc.ldf); + if (!npc.m_Phrases.empty()){ + npcEntity->SetVar<std::vector<std::string>>(u"chats", npc.m_Phrases); + + auto* scriptComponent = npcEntity->GetComponent<ScriptComponent>(); + + if (scriptComponent && !npc.m_Script.empty()) { + scriptComponent->SetScript(npc.m_Script); + scriptComponent->SetSerialized(false); + + for (const auto& npc : npc.m_Flags) { + npcEntity->SetVar<bool>(GeneralUtils::ASCIIToUTF16(npc.first), npc.second); + } + } + SetupNPCTalk(npcEntity); + } } if (zoneID == 1200) { @@ -128,14 +137,14 @@ void VanityUtilities::SpawnVanity() { info.lot = 8139; info.pos = { 259.5f, 246.4f, -705.2f }; info.rot = { 0.0f, 0.0f, 1.0f, 0.0f }; - info.spawnerID = EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(); + info.spawnerID = Game::entityManager->GetZoneControlEntity()->GetObjectID(); info.settings = { new LDFData<bool>(u"hasCustomText", true), - new LDFData<std::string>(u"customText", ParseMarkdown("./vanity/TESTAMENT.md")) }; + new LDFData<std::string>(u"customText", ParseMarkdown((BinaryPathFinder::GetBinaryDir() / "vanity/TESTAMENT.md").string())) }; - auto* entity = EntityManager::Instance()->CreateEntity(info); + auto* entity = Game::entityManager->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(entity); + Game::entityManager->ConstructEntity(entity); } } } @@ -146,15 +155,16 @@ Entity* VanityUtilities::SpawnNPC(LOT lot, const std::string& name, const NiPoin info.lot = lot; info.pos = position; info.rot = rotation; - info.spawnerID = EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(); + info.spawnerID = Game::entityManager->GetZoneControlEntity()->GetObjectID(); info.settings = ldf; - auto* entity = EntityManager::Instance()->CreateEntity(info); + auto* entity = Game::entityManager->CreateEntity(info); entity->SetVar(u"npcName", name); + if (entity->GetVar<bool>(u"noGhosting")) entity->SetIsGhostingCandidate(false); auto* inventoryComponent = entity->GetComponent<InventoryComponent>(); - if (inventoryComponent != nullptr) { + if (inventoryComponent && !inventory.empty()) { inventoryComponent->SetNPCItems(inventory); } @@ -166,7 +176,7 @@ Entity* VanityUtilities::SpawnNPC(LOT lot, const std::string& name, const NiPoin destroyableComponent->SetHealth(0); } - EntityManager::Instance()->ConstructEntity(entity); + Game::entityManager->ConstructEntity(entity); return entity; } @@ -264,10 +274,7 @@ void VanityUtilities::ParseXML(const std::string& file) { // Get the NPC name auto* name = npc->Attribute("name"); - if (name == nullptr) { - Game::logger->Log("VanityUtilities", "Failed to parse NPC name"); - continue; - } + if (!name) name = ""; // Get the NPC lot auto* lot = npc->Attribute("lot"); @@ -279,71 +286,76 @@ void VanityUtilities::ParseXML(const std::string& file) { // Get the equipment auto* equipment = npc->FirstChildElement("equipment"); - - if (equipment == nullptr) { - Game::logger->Log("VanityUtilities", "Failed to parse NPC equipment"); - continue; - } - - auto* text = equipment->GetText(); - std::vector<LOT> inventory; - if (text != nullptr) { - std::string equipmentString(text); + if (equipment) { + auto* text = equipment->GetText(); - std::vector<std::string> splitEquipment = GeneralUtils::SplitString(equipmentString, ','); + if (text != nullptr) { + std::string equipmentString(text); - for (auto& item : splitEquipment) { - inventory.push_back(std::stoi(item)); + std::vector<std::string> splitEquipment = GeneralUtils::SplitString(equipmentString, ','); + + for (auto& item : splitEquipment) { + inventory.push_back(std::stoi(item)); + } } } + // Get the phrases auto* phrases = npc->FirstChildElement("phrases"); - if (phrases == nullptr) { - Game::logger->Log("VanityUtilities", "Failed to parse NPC phrases"); - continue; - } + std::vector<std::string> phraseList = {}; - std::vector<std::string> phraseList; - - for (auto* phrase = phrases->FirstChildElement("phrase"); phrase != nullptr; - phrase = phrase->NextSiblingElement("phrase")) { - // Get the phrase - auto* text = phrase->GetText(); - - if (text == nullptr) { - Game::logger->Log("VanityUtilities", "Failed to parse NPC phrase"); - continue; + if (phrases) { + for (auto* phrase = phrases->FirstChildElement("phrase"); phrase != nullptr; + phrase = phrase->NextSiblingElement("phrase")) { + // Get the phrase + auto* text = phrase->GetText(); + if (text == nullptr) { + Game::logger->Log("VanityUtilities", "Failed to parse NPC phrase"); + continue; + } + phraseList.push_back(text); } - - phraseList.push_back(text); } // Get the script auto* scriptElement = npc->FirstChildElement("script"); - std::string scriptName; + std::string scriptName = ""; if (scriptElement != nullptr) { auto* scriptNameAttribute = scriptElement->Attribute("name"); - - if (scriptNameAttribute == nullptr) { - Game::logger->Log("VanityUtilities", "Failed to parse NPC script name"); - continue; - } - - scriptName = scriptNameAttribute; + if (scriptNameAttribute) scriptName = scriptNameAttribute; } + auto* ldfElement = npc->FirstChildElement("ldf"); + std::vector<std::u16string> keys = {}; + + std::vector<LDFBaseData*> ldf = {}; + if(ldfElement) { + for (auto* entry = ldfElement->FirstChildElement("entry"); entry != nullptr; + entry = entry->NextSiblingElement("entry")) { + // Get the ldf data + auto* data = entry->Attribute("data"); + if (!data) continue; + + LDFBaseData* ldfData = LDFBaseData::DataFromString(data); + keys.push_back(ldfData->GetKey()); + ldf.push_back(ldfData); + } + } + if (!keys.empty()) ldf.push_back(new LDFData<std::vector<std::u16string>>(u"syncLDF", keys)); + VanityNPC npcData; npcData.m_Name = name; npcData.m_LOT = std::stoi(lot); npcData.m_Equipment = inventory; npcData.m_Phrases = phraseList; npcData.m_Script = scriptName; + npcData.ldf = ldf; // Get flags auto* flags = npc->FirstChildElement("flags"); @@ -524,7 +536,7 @@ void VanityUtilities::NPCTalk(Entity* npc) { npc->GetObjectID(), u"sendToclient_bubble", 0, 0, npc->GetObjectID(), selected, UNASSIGNED_SYSTEM_ADDRESS); } - EntityManager::Instance()->SerializeEntity(npc); + Game::entityManager->SerializeEntity(npc); const float nextTime = GeneralUtils::GenerateRandomNumber<float>(15, 60); diff --git a/dGame/dUtilities/VanityUtilities.h b/dGame/dUtilities/VanityUtilities.h index 2f1886ed..0cca0aba 100644 --- a/dGame/dUtilities/VanityUtilities.h +++ b/dGame/dUtilities/VanityUtilities.h @@ -20,6 +20,7 @@ struct VanityNPC std::string m_Script; std::map<std::string, bool> m_Flags; std::map<uint32_t, std::vector<VanityNPCLocation>> m_Locations; + std::vector<LDFBaseData*> ldf; }; struct VanityParty diff --git a/dGame/dUtilities/dLocale.cpp b/dGame/dUtilities/dLocale.cpp deleted file mode 100644 index 65ef3a58..00000000 --- a/dGame/dUtilities/dLocale.cpp +++ /dev/null @@ -1,81 +0,0 @@ -#include "dLocale.h" - -#include <clocale> -#include <fstream> -#include <sstream> -#include <vector> -#include <algorithm> - -#include "tinyxml2.h" -#include "Game.h" -#include "dConfig.h" - -dLocale::dLocale() { - if (Game::config->GetValue("locale_enabled") != "1") { - return; - } - - std::ifstream file(m_LocalePath); - - if (!file.good()) { - return; - } - - std::stringstream data; - data << file.rdbuf(); - - if (data.str().empty()) { - return; - } - - auto* doc = new tinyxml2::XMLDocument(); - - if (doc == nullptr) { - return; - } - - if (doc->Parse(data.str().c_str(), data.str().size()) != 0) { - return; - } - - std::hash<std::string> hash; - - auto* localization = doc->FirstChildElement("localization"); - auto* phrases = localization->FirstChildElement("phrases"); - - auto* phrase = phrases->FirstChildElement("phrase"); - - while (phrase != nullptr) { - // Add the phrase hash to the vector - m_Phrases.push_back(hash(phrase->Attribute("id"))); - phrase = phrase->NextSiblingElement("phrase"); - } - - file.close(); - - delete doc; -} - -dLocale::~dLocale() = default; - -std::string dLocale::GetTemplate(const std::string& phraseID) { - return "%[" + phraseID + "]"; -} - -bool dLocale::HasPhrase(const std::string& phraseID) { - if (Game::config->GetValue("locale_enabled") != "1") { - return true; - } - - // Compute the hash and see if it's in the vector - std::hash<std::string> hash; - std::size_t hashValue = hash(phraseID); - return std::find(m_Phrases.begin(), m_Phrases.end(), hashValue) != m_Phrases.end(); -} - -/*std::string dLocale::GetPhrase(const std::string& phraseID) { - if (m_Phrases.find(phraseID) == m_Phrases.end()) { - return ""; - } - return m_Phrases[phraseID]; -}*/ diff --git a/dGame/dUtilities/dLocale.h b/dGame/dUtilities/dLocale.h deleted file mode 100644 index db5d2a4f..00000000 --- a/dGame/dUtilities/dLocale.h +++ /dev/null @@ -1,19 +0,0 @@ -#pragma once -#include <map> -#include <string> -#include <vector> -#include <cstdint> - -class dLocale { -public: - dLocale(); - ~dLocale(); - static std::string GetTemplate(const std::string& phraseID); - bool HasPhrase(const std::string& phraseID); - //std::string GetPhrase(const std::string& phraseID); - -private: - std::string m_LocalePath = "./locale/locale.xml"; - std::string m_Locale = "en_US"; // TODO: add to config - std::vector<std::size_t> m_Phrases; -}; diff --git a/dMasterServer/CMakeLists.txt b/dMasterServer/CMakeLists.txt index 08fc63db..2161681f 100644 --- a/dMasterServer/CMakeLists.txt +++ b/dMasterServer/CMakeLists.txt @@ -1,9 +1,13 @@ -set(DMASTERSERVER_SOURCES "InstanceManager.cpp" - "MasterServer.cpp" - "ObjectIDManager.cpp") - -add_executable(MasterServer ${DMASTERSERVER_SOURCES}) -target_link_libraries(MasterServer ${COMMON_LIBRARIES}) +set(DMASTERSERVER_SOURCES + "InstanceManager.cpp" + "ObjectIDManager.cpp" +) + +add_library(dMasterServer ${DMASTERSERVER_SOURCES}) +add_executable(MasterServer "MasterServer.cpp") + +target_link_libraries(dMasterServer ${COMMON_LIBRARIES}) +target_link_libraries(MasterServer ${COMMON_LIBRARIES} dMasterServer) if(WIN32) add_dependencies(MasterServer WorldServer AuthServer ChatServer) diff --git a/dMasterServer/InstanceManager.cpp b/dMasterServer/InstanceManager.cpp index 0c605ed6..50f55b72 100644 --- a/dMasterServer/InstanceManager.cpp +++ b/dMasterServer/InstanceManager.cpp @@ -7,9 +7,11 @@ #include "CDClientDatabase.h" #include "CDClientManager.h" #include "CDZoneTableTable.h" -#include "dMessageIdentifiers.h" #include "MasterPackets.h" #include "PacketUtils.h" +#include "BinaryPathFinder.h" +#include "eConnectionType.h" +#include "eMasterMessageType.h" InstanceManager::InstanceManager(dLogger* logger, const std::string& externalIP) { mLogger = logger; @@ -30,6 +32,15 @@ Instance* InstanceManager::GetInstance(LWOMAPID mapID, bool isFriendTransfer, LW Instance* instance = FindInstance(mapID, isFriendTransfer, cloneID); if (instance) return instance; + // If we are shutting down, return a nullptr so a new instance is not created. + if (m_IsShuttingDown) { + Game::logger->Log("InstanceManager", + "Tried to create a new instance map/instance/clone %i/%i/%i, but Master is shutting down.", + mapID, + m_LastInstanceID + 1, + cloneID); + return nullptr; + } //TODO: Update this so that the IP is read from a configuration file instead int softCap = 8; @@ -48,13 +59,13 @@ Instance* InstanceManager::GetInstance(LWOMAPID mapID, bool isFriendTransfer, LW //Start the actual process: #ifdef _WIN32 - std::string cmd = "start ./WorldServer.exe -zone "; + std::string cmd = "start " + (BinaryPathFinder::GetBinaryDir() / "WorldServer.exe").string() + " -zone "; #else std::string cmd; if (std::atoi(Game::config->GetValue("use_sudo_world").c_str())) { - cmd = "sudo ./WorldServer -zone "; + cmd = "sudo " + (BinaryPathFinder::GetBinaryDir() / "WorldServer").string() + " -zone "; } else { - cmd = "./WorldServer -zone "; + cmd = (BinaryPathFinder::GetBinaryDir() / "WorldServer").string() + " -zone "; } #endif @@ -73,7 +84,7 @@ Instance* InstanceManager::GetInstance(LWOMAPID mapID, bool isFriendTransfer, LW cmd.append("&"); //Sends our next process to the background on Linux #endif - system(cmd.c_str()); + auto ret = system(cmd.c_str()); m_Instances.push_back(instance); @@ -149,7 +160,7 @@ void InstanceManager::RemoveInstance(Instance* instance) { if (m_Instances[i] == instance) { instance->SetShutdownComplete(true); - RedirectPendingRequests(instance); + if (!Game::shouldShutdown) RedirectPendingRequests(instance); delete m_Instances[i]; @@ -191,7 +202,7 @@ void InstanceManager::RequestAffirmation(Instance* instance, const PendingInstan CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_AFFIRM_TRANSFER_REQUEST); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::AFFIRM_TRANSFER_REQUEST); bitStream.Write(request.id); @@ -237,7 +248,7 @@ void InstanceManager::RedirectPendingRequests(Instance* instance) { for (const auto& request : instance->GetPendingAffirmations()) { auto* in = Game::im->GetInstance(zoneId.GetMapID(), false, zoneId.GetCloneID()); - if (!in->GetIsReady()) // Instance not ready, make a pending request + if (in && !in->GetIsReady()) // Instance not ready, make a pending request { in->GetPendingRequests().push_back(request); @@ -294,16 +305,25 @@ Instance* InstanceManager::CreatePrivateInstance(LWOMAPID mapID, LWOCLONEID clon return instance; } + if (m_IsShuttingDown) { + Game::logger->Log("InstanceManager", + "Tried to create a new private instance map/instance/clone %i/%i/%i, but Master is shutting down.", + mapID, + m_LastInstanceID + 1, + cloneID); + return nullptr; + } + int maxPlayers = 999; uint32_t port = GetFreePort(); instance = new Instance(mExternalIP, port, mapID, ++m_LastInstanceID, cloneID, maxPlayers, maxPlayers, true, password); //Start the actual process: - std::string cmd = "start ./WorldServer.exe -zone "; + std::string cmd = "start " + (BinaryPathFinder::GetBinaryDir() / "WorldServer").string() + " -zone "; #ifndef _WIN32 - cmd = "./WorldServer -zone "; + cmd = (BinaryPathFinder::GetBinaryDir() / "WorldServer").string() + " -zone "; #endif cmd.append(std::to_string(mapID)); @@ -321,7 +341,7 @@ Instance* InstanceManager::CreatePrivateInstance(LWOMAPID mapID, LWOCLONEID clon cmd.append("&"); //Sends our next process to the background on Linux #endif - system(cmd.c_str()); + auto ret = system(cmd.c_str()); m_Instances.push_back(instance); @@ -350,7 +370,7 @@ Instance* InstanceManager::FindPrivateInstance(const std::string& password) { } int InstanceManager::GetSoftCap(LWOMAPID mapID) { - CDZoneTableTable* zoneTable = CDClientManager::Instance()->GetTable<CDZoneTableTable>("ZoneTable"); + CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable<CDZoneTableTable>(); if (zoneTable) { const CDZoneTable* zone = zoneTable->Query(mapID); @@ -363,7 +383,7 @@ int InstanceManager::GetSoftCap(LWOMAPID mapID) { } int InstanceManager::GetHardCap(LWOMAPID mapID) { - CDZoneTableTable* zoneTable = CDClientManager::Instance()->GetTable<CDZoneTableTable>("ZoneTable"); + CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable<CDZoneTableTable>(); if (zoneTable) { const CDZoneTable* zone = zoneTable->Query(mapID); @@ -386,9 +406,9 @@ bool Instance::GetShutdownComplete() const { void Instance::Shutdown() { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SHUTDOWN); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SHUTDOWN); Game::server->Send(&bitStream, this->m_SysAddr, false); - Game::logger->Log("Instance", "Triggered world shutdown"); + Game::logger->Log("Instance", "Triggered world shutdown for zone/clone/instance %i/%i/%i", GetMapID(), GetCloneID(), GetInstanceID()); } diff --git a/dMasterServer/InstanceManager.h b/dMasterServer/InstanceManager.h index 2c5083d4..5f48f93b 100644 --- a/dMasterServer/InstanceManager.h +++ b/dMasterServer/InstanceManager.h @@ -128,6 +128,7 @@ public: Instance* CreatePrivateInstance(LWOMAPID mapID, LWOCLONEID cloneID, const std::string& password); Instance* FindPrivateInstance(const std::string& password); + void SetIsShuttingDown(bool value) { this->m_IsShuttingDown = value; }; private: dLogger* mLogger; @@ -136,6 +137,11 @@ private: unsigned short m_LastPort; LWOINSTANCEID m_LastInstanceID; + /** + * Whether or not the master server is currently shutting down. + */ + bool m_IsShuttingDown = false; + //Private functions: bool IsInstanceFull(Instance* instance, bool isFriendTransfer); int GetSoftCap(LWOMAPID mapID); diff --git a/dMasterServer/MasterServer.cpp b/dMasterServer/MasterServer.cpp index a692e826..f56cad7f 100644 --- a/dMasterServer/MasterServer.cpp +++ b/dMasterServer/MasterServer.cpp @@ -25,6 +25,10 @@ #include "dConfig.h" #include "dLogger.h" #include "dServer.h" +#include "AssetManager.h" +#include "BinaryPathFinder.h" +#include "eConnectionType.h" +#include "eMasterMessageType.h" //RakNet includes: #include "RakNetDefines.h" @@ -37,27 +41,32 @@ #include "MasterPackets.h" #include "ObjectIDManager.h" #include "PacketUtils.h" -#include "dMessageIdentifiers.h" +#include "FdbToSqlite.h" namespace Game { - dLogger* logger; - dServer* server; - InstanceManager* im; - dConfig* config; + dLogger* logger = nullptr; + dServer* server = nullptr; + InstanceManager* im = nullptr; + dConfig* config = nullptr; + AssetManager* assetManager = nullptr; + bool shouldShutdown = false; + std::mt19937 randomEngine; } //namespace Game bool shutdownSequenceStarted = false; -void ShutdownSequence(); -int FinalizeShutdown(); +void ShutdownSequence(int32_t signal = -1); +int32_t FinalizeShutdown(int32_t signal = -1); dLogger* SetupLogger(); void StartAuthServer(); void StartChatServer(); void HandlePacket(Packet* packet); std::map<uint32_t, std::string> activeSessions; -bool shouldShutdown = false; +SystemAddress authServerMasterPeerSysAddr; SystemAddress chatServerMasterPeerSysAddr; int main(int argc, char** argv) { + constexpr uint32_t masterFramerate = mediumFramerate; + constexpr uint32_t masterFrameDelta = mediumFrameDelta; Diagnostics::SetProcessName("Master"); Diagnostics::SetProcessFileName(argv[0]); Diagnostics::Initialize(); @@ -67,89 +76,144 @@ int main(int argc, char** argv) { #endif //Triggers the shutdown sequence at application exit - std::atexit(ShutdownSequence); - signal(SIGINT, [](int) { ShutdownSequence(); }); - signal(SIGTERM, [](int) { ShutdownSequence(); }); + std::atexit([]() { ShutdownSequence(); }); + signal(SIGINT, [](int32_t signal) { ShutdownSequence(EXIT_FAILURE); }); + signal(SIGTERM, [](int32_t signal) { ShutdownSequence(EXIT_FAILURE); }); //Create all the objects we need to run our service: Game::logger = SetupLogger(); if (!Game::logger) return EXIT_FAILURE; + if (!std::filesystem::exists(BinaryPathFinder::GetBinaryDir() / "authconfig.ini")) { + Game::logger->Log("MasterServer", "Couldnt find authconfig.ini"); + return EXIT_FAILURE; + } + + if (!std::filesystem::exists(BinaryPathFinder::GetBinaryDir() / "chatconfig.ini")) { + Game::logger->Log("MasterServer", "Couldnt find chatconfig.ini"); + return EXIT_FAILURE; + } + + if (!std::filesystem::exists(BinaryPathFinder::GetBinaryDir() / "masterconfig.ini")) { + Game::logger->Log("MasterServer", "Couldnt find masterconfig.ini"); + return EXIT_FAILURE; + } + + if (!std::filesystem::exists(BinaryPathFinder::GetBinaryDir() / "sharedconfig.ini")) { + Game::logger->Log("MasterServer", "Couldnt find sharedconfig.ini"); + return EXIT_FAILURE; + } + + if (!std::filesystem::exists(BinaryPathFinder::GetBinaryDir() / "worldconfig.ini")) { + Game::logger->Log("MasterServer", "Couldnt find worldconfig.ini"); + return EXIT_FAILURE; + } + + Game::config = new dConfig((BinaryPathFinder::GetBinaryDir() / "masterconfig.ini").string()); + Game::logger->SetLogToConsole(Game::config->GetValue("log_to_console") != "0"); + Game::logger->SetLogDebugStatements(Game::config->GetValue("log_debug_statements") == "1"); + + uint32_t clientNetVersion = 0; + if (!GeneralUtils::TryParse(Game::config->GetValue("client_net_version"), clientNetVersion)) { + Game::logger->Log("MasterServer", "Failed to parse (%s) as net version. Cannot start server as no clients could connect.",Game::config->GetValue("client_net_version").c_str()); + Game::logger->Log("MasterServer", "As of version 1.1.1, client_net_version is required to be defined in sharedconfig.ini as opposed to in CMakeVariables.txt as NET_VERSION."); + Game::logger->Log("MasterServer", "Rerun cmake to ensure all config values exist. If client_net_version already exists in sharedconfig.ini, please ensure it is a valid number."); + Game::logger->Log("MasterServer", "like 171022"); + return EXIT_FAILURE; + } + + Game::logger->Log("MasterServer", "Using net version %s", Game::config->GetValue("client_net_version").c_str()); + Game::logger->Log("MasterServer", "Starting Master server..."); Game::logger->Log("MasterServer", "Version: %i.%i", PROJECT_VERSION_MAJOR, PROJECT_VERSION_MINOR); Game::logger->Log("MasterServer", "Compiled on: %s", __TIMESTAMP__); - //Read our config: - dConfig config("masterconfig.ini"); - Game::config = &config; - Game::logger->SetLogToConsole(bool(std::stoi(config.GetValue("log_to_console")))); - Game::logger->SetLogDebugStatements(config.GetValue("log_debug_statements") == "1"); + //Connect to the MySQL Database + std::string mysql_host = Game::config->GetValue("mysql_host"); + std::string mysql_database = Game::config->GetValue("mysql_database"); + std::string mysql_username = Game::config->GetValue("mysql_username"); + std::string mysql_password = Game::config->GetValue("mysql_password"); - if (argc > 1 && (strcmp(argv[1], "-m") == 0 || strcmp(argv[1], "--migrations") == 0)) { - //Connect to the MySQL Database - std::string mysql_host = config.GetValue("mysql_host"); - std::string mysql_database = config.GetValue("mysql_database"); - std::string mysql_username = config.GetValue("mysql_username"); - std::string mysql_password = config.GetValue("mysql_password"); + try { + Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); + } catch (sql::SQLException& ex) { + Game::logger->Log("MasterServer", "Got an error while connecting to the database: %s", ex.what()); + Game::logger->Log("MigrationRunner", "Migrations not run"); + return EXIT_FAILURE; + } - try { - Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); - } catch (sql::SQLException& ex) { - Game::logger->Log("MasterServer", "Got an error while connecting to the database: %s", ex.what()); - Game::logger->Log("MigrationRunner", "Migrations not run"); - return EXIT_FAILURE; + try { + std::string clientPathStr = Game::config->GetValue("client_location"); + if (clientPathStr.empty()) clientPathStr = "./res"; + std::filesystem::path clientPath = std::filesystem::path(clientPathStr); + if (clientPath.is_relative()) { + clientPath = BinaryPathFinder::GetBinaryDir() / clientPath; } - MigrationRunner::RunMigrations(); - Game::logger->Log("MigrationRunner", "Finished running migrations"); + Game::assetManager = new AssetManager(clientPath); + } catch (std::runtime_error& ex) { + Game::logger->Log("MasterServer", "Got an error while setting up assets: %s", ex.what()); - return EXIT_SUCCESS; - } else { + return EXIT_FAILURE; + } - //Check CDClient exists - const std::string cdclient_path = "./res/CDServer.sqlite"; - std::ifstream cdclient_fd(cdclient_path); - if (!cdclient_fd.good()) { - Game::logger->Log("WorldServer", "%s could not be opened", cdclient_path.c_str()); - return EXIT_FAILURE; - } - cdclient_fd.close(); + MigrationRunner::RunMigrations(); - //Connect to CDClient - try { - CDClientDatabase::Connect(cdclient_path); - } catch (CppSQLite3Exception& e) { - Game::logger->Log("WorldServer", "Unable to connect to CDServer SQLite Database"); - Game::logger->Log("WorldServer", "Error: %s", e.errorMessage()); - Game::logger->Log("WorldServer", "Error Code: %i", e.errorCode()); - return EXIT_FAILURE; - } + const bool cdServerExists = std::filesystem::exists(BinaryPathFinder::GetBinaryDir() / "resServer" / "CDServer.sqlite"); + const bool oldCDServerExists = std::filesystem::exists(Game::assetManager->GetResPath() / "CDServer.sqlite"); + const bool fdbExists = std::filesystem::exists(Game::assetManager->GetResPath() / "cdclient.fdb"); - //Get CDClient initial information - try { - CDClientManager::Instance()->Initialize(); - } catch (CppSQLite3Exception& e) { - Game::logger->Log("WorldServer", "Failed to initialize CDServer SQLite Database"); - Game::logger->Log("WorldServer", "May be caused by corrupted file: %s", cdclient_path.c_str()); - Game::logger->Log("WorldServer", "Error: %s", e.errorMessage()); - Game::logger->Log("WorldServer", "Error Code: %i", e.errorCode()); - return EXIT_FAILURE; - } + if (!cdServerExists) { + if (oldCDServerExists) { + // If the file doesn't exist in the new CDServer location, copy it there. We copy because we may not have write permissions from the previous directory. + Game::logger->Log("MasterServer", "CDServer.sqlite is not located at resServer, but is located at res path. Copying file..."); + std::filesystem::copy_file(Game::assetManager->GetResPath() / "CDServer.sqlite", BinaryPathFinder::GetBinaryDir() / "resServer" / "CDServer.sqlite"); + } else { + Game::logger->Log("MasterServer", + "%s could not be found in resServer or res. Looking for %s to convert to sqlite.", + (BinaryPathFinder::GetBinaryDir() / "resServer" / "CDServer.sqlite").c_str(), + (Game::assetManager->GetResPath() / "cdclient.fdb").c_str()); - //Connect to the MySQL Database - std::string mysql_host = config.GetValue("mysql_host"); - std::string mysql_database = config.GetValue("mysql_database"); - std::string mysql_username = config.GetValue("mysql_username"); - std::string mysql_password = config.GetValue("mysql_password"); + AssetMemoryBuffer cdClientBuffer = Game::assetManager->GetFileAsBuffer("cdclient.fdb"); + if (!cdClientBuffer.m_Success) { + Game::logger->Log("MasterServer", "Failed to load %s", (Game::assetManager->GetResPath() / "cdclient.fdb").c_str()); + throw std::runtime_error("Aborting initialization due to missing cdclient.fdb."); + } - try { - Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); - } catch (sql::SQLException& ex) { - Game::logger->Log("MasterServer", "Got an error while connecting to the database: %s", ex.what()); - return EXIT_FAILURE; + Game::logger->Log("MasterServer", "Found %s. Converting to SQLite", (Game::assetManager->GetResPath() / "cdclient.fdb").c_str()); + Game::logger->Flush(); + + if (FdbToSqlite::Convert((BinaryPathFinder::GetBinaryDir() / "resServer").string()).ConvertDatabase(cdClientBuffer) == false) { + Game::logger->Log("MasterServer", "Failed to convert fdb to sqlite."); + return EXIT_FAILURE; + } + cdClientBuffer.close(); } } + //Connect to CDClient + try { + CDClientDatabase::Connect((BinaryPathFinder::GetBinaryDir() / "resServer" / "CDServer.sqlite").string()); + } catch (CppSQLite3Exception& e) { + Game::logger->Log("WorldServer", "Unable to connect to CDServer SQLite Database"); + Game::logger->Log("WorldServer", "Error: %s", e.errorMessage()); + Game::logger->Log("WorldServer", "Error Code: %i", e.errorCode()); + return EXIT_FAILURE; + } + + // Run migrations should any need to be run. + MigrationRunner::RunSQLiteMigrations(); + + //Get CDClient initial information + try { + CDClientManager::Instance(); + } catch (CppSQLite3Exception& e) { + Game::logger->Log("WorldServer", "Failed to initialize CDServer SQLite Database"); + Game::logger->Log("WorldServer", "May be caused by corrupted file: %s", (Game::assetManager->GetResPath() / "CDServer.sqlite").string().c_str()); + Game::logger->Log("WorldServer", "Error: %s", e.errorMessage()); + Game::logger->Log("WorldServer", "Error Code: %i", e.errorCode()); + return EXIT_FAILURE; + } //If the first command line argument is -a or --account then make the user //input a username and password, with the password being hidden. @@ -161,60 +225,97 @@ int main(int argc, char** argv) { std::cout << "Enter a username: "; std::cin >> username; + std::unique_ptr<sql::PreparedStatement> userLookupStatement(Database::CreatePreppedStmt("SELECT id FROM accounts WHERE name=? LIMIT 1;")); + userLookupStatement->setString(1, username.c_str()); + std::unique_ptr<sql::ResultSet> res(userLookupStatement->executeQuery()); + if (res->rowsCount() > 0) { + Game::logger->Log("MasterServer", "Account with name \"%s\" already exists", username.c_str()); + std::cout << "Do you want to change the password of that account? [y/n]?"; + std::string prompt = ""; + std::cin >> prompt; + if (prompt == "y" || prompt == "yes"){ + uint32_t accountId = 0; + res->next(); + accountId = res->getUInt(1); + if (accountId == 0) return EXIT_FAILURE; + + //Read the password from the console without echoing it. + #ifdef __linux__ + //This function is obsolete, but it only meant to be used by the + //sysadmin to create their first account. + password = getpass("Enter a password: "); + #else + std::cout << "Enter a password: "; + std::cin >> password; + #endif + + // Regenerate hash based on new password + char salt[BCRYPT_HASHSIZE]; + char hash[BCRYPT_HASHSIZE]; + int32_t bcryptState = ::bcrypt_gensalt(12, salt); + assert(bcryptState == 0); + bcryptState = ::bcrypt_hashpw(password.c_str(), salt, hash); + assert(bcryptState == 0); + + std::unique_ptr<sql::PreparedStatement> userUpdateStatement(Database::CreatePreppedStmt("UPDATE accounts SET password = ? WHERE id = ?;")); + userUpdateStatement->setString(1, std::string(hash, BCRYPT_HASHSIZE).c_str()); + userUpdateStatement->setUInt(2, accountId); + userUpdateStatement->execute(); + + Game::logger->Log("MasterServer", "Account \"%s\" password updated successfully!", username.c_str()); + } else { + Game::logger->Log("MasterServer", "Account \"%s\" was not updated.", username.c_str()); + } + return EXIT_SUCCESS; + } + //Read the password from the console without echoing it. -#ifdef __linux__ - //This function is obsolete, but it only meant to be used by the - //sysadmin to create their first account. - password = getpass("Enter a password: "); -#else - std::cout << "Enter a password: "; - std::cin >> password; -#endif + #ifdef __linux__ + //This function is obsolete, but it only meant to be used by the + //sysadmin to create their first account. + password = getpass("Enter a password: "); + #else + std::cout << "Enter a password: "; + std::cin >> password; + #endif //Generate new hash for bcrypt - char salt[BCRYPT_HASHSIZE]; char hash[BCRYPT_HASHSIZE]; - int32_t bcryptState = ::bcrypt_gensalt(12, salt); - assert(bcryptState == 0); - bcryptState = ::bcrypt_hashpw(password.c_str(), salt, hash); - assert(bcryptState == 0); //Create account + try { + std::unique_ptr<sql::PreparedStatement> statement(Database::CreatePreppedStmt("INSERT INTO accounts (name, password, gm_level) VALUES (?, ?, ?);")); + statement->setString(1, username.c_str()); + statement->setString(2, std::string(hash, BCRYPT_HASHSIZE).c_str()); + statement->setInt(3, 9); + statement->execute(); + } catch(sql::SQLException& e) { + Game::logger->Log("MasterServer", "A SQL error occurred!:\n %s", e.what()); + return EXIT_FAILURE; + } - auto* statement = Database::CreatePreppedStmt("INSERT INTO accounts (name, password, ""gm_level) VALUES (?, ?, ?);"); - statement->setString(1, username); - statement->setString(2, std::string(hash, BCRYPT_HASHSIZE).c_str()); - statement->setInt(3, 9); - - statement->execute(); - - delete statement; - - std::cout << "Account created successfully!\n"; - - Database::Destroy("MasterServer"); - delete Game::logger; - + Game::logger->Log("MasterServer", "Account created successfully!"); return EXIT_SUCCESS; } - int maxClients = 999; - int ourPort = 1000; - if (config.GetValue("max_clients") != "") maxClients = std::stoi(config.GetValue("max_clients")); - if (config.GetValue("port") != "") ourPort = std::stoi(config.GetValue("port")); + Game::randomEngine = std::mt19937(time(0)); + uint32_t maxClients = 999; + uint32_t ourPort = 1000; + if (Game::config->GetValue("max_clients") != "") maxClients = std::stoi(Game::config->GetValue("max_clients")); + if (Game::config->GetValue("port") != "") ourPort = std::stoi(Game::config->GetValue("port")); - Game::server = new dServer(config.GetValue("external_ip"), ourPort, 0, maxClients, true, false, Game::logger, "", 0, ServerType::Master); + Game::server = new dServer(Game::config->GetValue("external_ip"), ourPort, 0, maxClients, true, false, Game::logger, "", 0, ServerType::Master, Game::config, &Game::shouldShutdown); //Query for the database for a server labeled "master" auto* masterLookupStatement = Database::CreatePreppedStmt("SELECT id FROM `servers` WHERE `name` = 'master'"); auto* result = masterLookupStatement->executeQuery(); - auto master_server_ip = config.GetValue("master_ip"); + auto master_server_ip = Game::config->GetValue("master_ip"); if (master_server_ip == "") { master_server_ip = Game::server->GetIP(); @@ -223,7 +324,7 @@ int main(int argc, char** argv) { //If we found a server, update it's IP and port to the current one. if (result->next()) { auto* updateStatement = Database::CreatePreppedStmt("UPDATE `servers` SET `ip` = ?, `port` = ? WHERE `id` = ?"); - updateStatement->setString(1, master_server_ip); + updateStatement->setString(1, master_server_ip.c_str()); updateStatement->setInt(2, Game::server->GetPort()); updateStatement->setInt(3, result->getInt("id")); updateStatement->execute(); @@ -231,7 +332,7 @@ int main(int argc, char** argv) { } else { //If we didn't find a server, create one. auto* insertStatement = Database::CreatePreppedStmt("INSERT INTO `servers` (`name`, `ip`, `port`, `state`, `version`) VALUES ('master', ?, ?, 0, 171023)"); - insertStatement->setString(1, master_server_ip); + insertStatement->setString(1, master_server_ip.c_str()); insertStatement->setInt(2, Game::server->GetPort()); insertStatement->execute(); delete insertStatement; @@ -242,20 +343,23 @@ int main(int argc, char** argv) { Game::im = new InstanceManager(Game::logger, Game::server->GetIP()); //Depending on the config, start up servers: - if (config.GetValue("prestart_servers") != "" && config.GetValue("prestart_servers") == "1") { + if (Game::config->GetValue("prestart_servers") != "" && Game::config->GetValue("prestart_servers") == "1") { StartChatServer(); - Game::im->GetInstance(0, false, 0)->SetIsReady(true); - Game::im->GetInstance(1000, false, 0)->SetIsReady(true); - + Game::im->GetInstance(0, false, 0); + Game::im->GetInstance(1000, false, 0); StartAuthServer(); } auto t = std::chrono::high_resolution_clock::now(); Packet* packet = nullptr; - int framesSinceLastFlush = 0; - int framesSinceLastSQLPing = 0; - int framesSinceKillUniverseCommand = 0; + constexpr uint32_t logFlushTime = 15 * masterFramerate; + constexpr uint32_t sqlPingTime = 10 * 60 * masterFramerate; + constexpr uint32_t shutdownUniverseTime = 10 * 60 * masterFramerate; + constexpr uint32_t instanceReadyTimeout = 30 * masterFramerate; + uint32_t framesSinceLastFlush = 0; + uint32_t framesSinceLastSQLPing = 0; + uint32_t framesSinceKillUniverseCommand = 0; while (true) { //In world we'd update our other systems here. @@ -269,17 +373,17 @@ int main(int argc, char** argv) { } //Push our log every 15s: - if (framesSinceLastFlush >= 900) { + if (framesSinceLastFlush >= logFlushTime) { Game::logger->Flush(); framesSinceLastFlush = 0; } else framesSinceLastFlush++; //Every 10 min we ping our sql server to keep it alive hopefully: - if (framesSinceLastSQLPing >= 40000) { + if (framesSinceLastSQLPing >= sqlPingTime) { //Find out the master's IP for absolutely no reason: std::string masterIP; - int masterPort; + uint32_t masterPort; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -295,8 +399,8 @@ int main(int argc, char** argv) { framesSinceLastSQLPing++; //10m shutdown for universe kill command - if (shouldShutdown) { - if (framesSinceKillUniverseCommand >= 40000) { + if (Game::shouldShutdown) { + if (framesSinceKillUniverseCommand >= shutdownUniverseTime) { //Break main loop and exit break; } else @@ -320,7 +424,7 @@ int main(int argc, char** argv) { instance->SetAffirmationTimeout(affirmTimeout); - if (affirmTimeout == 1000) { + if (affirmTimeout == instanceReadyTimeout) { instance->Shutdown(); instance->SetIsShuttingDown(true); @@ -339,17 +443,15 @@ int main(int argc, char** argv) { } } - t += std::chrono::milliseconds(highFrameRate); + t += std::chrono::milliseconds(masterFrameDelta); std::this_thread::sleep_until(t); } - FinalizeShutdown(); - exit(EXIT_SUCCESS); - return EXIT_SUCCESS; + return FinalizeShutdown(EXIT_SUCCESS); } dLogger* SetupLogger() { std::string logPath = - "./logs/MasterServer_" + std::to_string(time(nullptr)) + ".log"; + (BinaryPathFinder::GetBinaryDir() / ("logs/MasterServer_" + std::to_string(time(nullptr)) + ".log")).string(); bool logToConsole = false; bool logDebugStatements = false; #ifdef _DEBUG @@ -373,9 +475,15 @@ void HandlePacket(Packet* packet) { Game::im->RemoveInstance(instance); //Delete the old } - if (packet->systemAddress == chatServerMasterPeerSysAddr && !shouldShutdown) { + if (packet->systemAddress == chatServerMasterPeerSysAddr) { + chatServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; StartChatServer(); } + + if (packet->systemAddress == authServerMasterPeerSysAddr) { + authServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; + StartAuthServer(); + } } if (packet->data[0] == ID_CONNECTION_LOST) { @@ -386,17 +494,24 @@ void HandlePacket(Packet* packet) { if (instance) { LWOZONEID zoneID = instance->GetZoneID(); //Get the zoneID so we can recreate a server Game::im->RemoveInstance(instance); //Delete the old - //Game::im->GetInstance(zoneID.GetMapID(), false, 0); //Create the new } - if (packet->systemAddress == chatServerMasterPeerSysAddr && !shouldShutdown) { + if (packet->systemAddress == chatServerMasterPeerSysAddr) { + chatServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; StartChatServer(); } + + if (packet->systemAddress == authServerMasterPeerSysAddr) { + authServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; + StartAuthServer(); + } } - if (packet->data[1] == MASTER) { - switch (packet->data[3]) { - case MSG_MASTER_REQUEST_PERSISTENT_ID: { + if (packet->length < 4) return; + + if (static_cast<eConnectionType>(packet->data[1]) == eConnectionType::MASTER) { + switch (static_cast<eMasterMessageType>(packet->data[3])) { + case eMasterMessageType::REQUEST_PERSISTENT_ID: { Game::logger->Log("MasterServer", "A persistent ID req"); RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -408,7 +523,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_REQUEST_ZONE_TRANSFER: { + case eMasterMessageType::REQUEST_ZONE_TRANSFER: { Game::logger->Log("MasterServer", "Received zone transfer req"); RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -421,14 +536,17 @@ void HandlePacket(Packet* packet) { inStream.Read(mythranShift); inStream.Read(zoneID); inStream.Read(zoneClone); - + if (shutdownSequenceStarted) { + Game::logger->Log("MasterServer", "Shutdown sequence has been started. Not creating a new zone."); + break; + } Instance* in = Game::im->GetInstance(zoneID, false, zoneClone); for (auto* instance : Game::im->GetInstances()) { Game::logger->Log("MasterServer", "Instance: %i/%i/%i -> %i", instance->GetMapID(), instance->GetCloneID(), instance->GetInstanceID(), instance == in); } - if (!in->GetIsReady()) //Instance not ready, make a pending request + if (in && !in->GetIsReady()) //Instance not ready, make a pending request { in->GetPendingRequests().push_back({ requestID, static_cast<bool>(mythranShift), packet->systemAddress }); Game::logger->Log("MasterServer", "Server not ready, adding pending request %llu %i %i", requestID, zoneID, zoneClone); @@ -441,7 +559,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_SERVER_INFO: { + case eMasterMessageType::SERVER_INFO: { //MasterPackets::HandleServerInfo(packet); //This is here because otherwise we'd have to include IM in @@ -487,12 +605,20 @@ void HandlePacket(Packet* packet) { chatServerMasterPeerSysAddr = copy; } + if (theirServerType == ServerType::Auth) { + SystemAddress copy; + copy.binaryAddress = packet->systemAddress.binaryAddress; + copy.port = packet->systemAddress.port; + + authServerMasterPeerSysAddr = copy; + } + Game::logger->Log("MasterServer", "Received server info, instance: %i port: %i", theirInstanceID, theirPort); break; } - case MSG_MASTER_SET_SESSION_KEY: { + case eMasterMessageType::SET_SESSION_KEY: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); uint32_t sessionKey = 0; @@ -506,7 +632,7 @@ void HandlePacket(Packet* packet) { activeSessions.erase(it.first); CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_NEW_SESSION_ALERT); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::NEW_SESSION_ALERT); bitStream.Write(sessionKey); bitStream.Write<uint32_t>(username.size()); for (auto character : username) { @@ -523,7 +649,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_REQUEST_SESSION_KEY: { + case eMasterMessageType::REQUEST_SESSION_KEY: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); std::string username = PacketUtils::ReadString(8, packet, false); @@ -531,7 +657,7 @@ void HandlePacket(Packet* packet) { for (auto key : activeSessions) { if (key.second == username) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SESSION_KEY_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SESSION_KEY_RESPONSE); bitStream.Write(key.first); PacketUtils::WriteString(bitStream, key.second, 64); Game::server->Send(&bitStream, packet->systemAddress, false); @@ -541,7 +667,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_PLAYER_ADDED: { + case eMasterMessageType::PLAYER_ADDED: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -561,7 +687,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_PLAYER_REMOVED: { + case eMasterMessageType::PLAYER_REMOVED: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -579,7 +705,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_CREATE_PRIVATE_ZONE: { + case eMasterMessageType::CREATE_PRIVATE_ZONE: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -592,7 +718,7 @@ void HandlePacket(Packet* packet) { uint32_t len; inStream.Read<uint32_t>(len); - for (int i = 0; len > i; i++) { + for (uint32_t i = 0; len > i; i++) { char character; inStream.Read<char>(character); password += character; @@ -603,7 +729,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_REQUEST_PRIVATE_ZONE: { + case eMasterMessageType::REQUEST_PRIVATE_ZONE: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -618,7 +744,7 @@ void HandlePacket(Packet* packet) { uint32_t len; inStream.Read<uint32_t>(len); - for (int i = 0; i < len; i++) { + for (uint32_t i = 0; i < len; i++) { char character; inStream.Read<char>(character); password += character; } @@ -638,7 +764,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_WORLD_READY: { + case eMasterMessageType::WORLD_READY: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -662,19 +788,23 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_PREP_ZONE: { + case eMasterMessageType::PREP_ZONE: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); - int zoneID; + int32_t zoneID; inStream.Read(zoneID); - - Game::logger->Log("MasterServer", "Prepping zone %i", zoneID); - Game::im->GetInstance(zoneID, false, 0); + if (shutdownSequenceStarted) { + Game::logger->Log("MasterServer", "Shutdown sequence has been started. Not prepping a new zone."); + break; + } else { + Game::logger->Log("MasterServer", "Prepping zone %i", zoneID); + Game::im->GetInstance(zoneID, false, 0); + } break; } - case MSG_MASTER_AFFIRM_TRANSFER_RESPONSE: { + case eMasterMessageType::AFFIRM_TRANSFER_RESPONSE: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -694,7 +824,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_SHUTDOWN_RESPONSE: { + case eMasterMessageType::SHUTDOWN_RESPONSE: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -709,9 +839,9 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_SHUTDOWN_UNIVERSE: { + case eMasterMessageType::SHUTDOWN_UNIVERSE: { Game::logger->Log("MasterServer", "Received shutdown universe command, shutting down in 10 minutes."); - shouldShutdown = true; + Game::shouldShutdown = true; break; } @@ -722,76 +852,95 @@ void HandlePacket(Packet* packet) { } void StartChatServer() { + if (Game::shouldShutdown) { + Game::logger->Log("MasterServer", "Currently shutting down. Chat will not be restarted."); + return; + } #ifdef __APPLE__ //macOS doesn't need sudo to run on ports < 1024 - system("./ChatServer&"); + auto result = system(((BinaryPathFinder::GetBinaryDir() / "ChatServer").string() + "&").c_str()); #elif _WIN32 - system("start ./ChatServer.exe"); + auto result = system(("start " + (BinaryPathFinder::GetBinaryDir() / "ChatServer.exe").string()).c_str()); #else if (std::atoi(Game::config->GetValue("use_sudo_chat").c_str())) { - system("sudo ./ChatServer&"); + auto result = system(("sudo " + (BinaryPathFinder::GetBinaryDir() / "ChatServer").string() + "&").c_str()); } else { - system("./ChatServer&"); - } + auto result = system(((BinaryPathFinder::GetBinaryDir() / "ChatServer").string() + "&").c_str()); +} #endif } void StartAuthServer() { + if (Game::shouldShutdown) { + Game::logger->Log("MasterServer", "Currently shutting down. Auth will not be restarted."); + return; + } #ifdef __APPLE__ - system("./AuthServer&"); + auto result = system(((BinaryPathFinder::GetBinaryDir() / "AuthServer").string() + "&").c_str()); #elif _WIN32 - system("start ./AuthServer.exe"); + auto result = system(("start " + (BinaryPathFinder::GetBinaryDir() / "AuthServer.exe").string()).c_str()); #else if (std::atoi(Game::config->GetValue("use_sudo_auth").c_str())) { - system("sudo ./AuthServer&"); + auto result = system(("sudo " + (BinaryPathFinder::GetBinaryDir() / "AuthServer").string() + "&").c_str()); } else { - system("./AuthServer&"); - } + auto result = system(((BinaryPathFinder::GetBinaryDir() / "AuthServer").string() + "&").c_str()); +} #endif } -void ShutdownSequence() { +void ShutdownSequence(int32_t signal) { if (shutdownSequenceStarted) { return; } + if (!Game::im) { + FinalizeShutdown(EXIT_FAILURE); + } + + Game::im->SetIsShuttingDown(true); shutdownSequenceStarted = true; + Game::shouldShutdown = true; - if (Game::im) { - for (auto* instance : Game::im->GetInstances()) { - if (instance == nullptr) { - continue; - } - - instance->Shutdown(); - } + { + CBITSTREAM; + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SHUTDOWN); + Game::server->Send(&bitStream, UNASSIGNED_SYSTEM_ADDRESS, true); + Game::logger->Log("MasterServer", "Triggered master shutdown"); } auto* objIdManager = ObjectIDManager::TryInstance(); - if (objIdManager != nullptr) { + if (objIdManager) { objIdManager->SaveToDatabase(); Game::logger->Log("MasterServer", "Saved ObjectIDTracker to DB"); } - auto t = std::chrono::high_resolution_clock::now(); - auto ticks = 0; + // A server might not be finished spinning up yet, remove all of those here. + for (auto* instance : Game::im->GetInstances()) { + if (!instance->GetIsReady()) { + Game::im->RemoveInstance(instance); + } + } - if (!Game::im) { - exit(EXIT_SUCCESS); + for (auto* instance : Game::im->GetInstances()) { + instance->SetIsShuttingDown(true); } Game::logger->Log("MasterServer", "Attempting to shutdown instances, max 60 seconds..."); + auto t = std::chrono::high_resolution_clock::now(); + uint32_t framesSinceShutdownStart = 0; + constexpr uint32_t maxShutdownTime = 60 * mediumFramerate; + bool allInstancesShutdown = false; + Packet* packet = nullptr; while (true) { - - auto packet = Game::server->Receive(); + packet = Game::server->Receive(); if (packet) { HandlePacket(packet); Game::server->DeallocatePacket(packet); packet = nullptr; } - auto done = true; + allInstancesShutdown = true; for (auto* instance : Game::im->GetInstances()) { if (instance == nullptr) { @@ -799,36 +948,37 @@ void ShutdownSequence() { } if (!instance->GetShutdownComplete()) { - done = false; + allInstancesShutdown = false; } } - if (done) { + if (allInstancesShutdown && authServerMasterPeerSysAddr == UNASSIGNED_SYSTEM_ADDRESS && chatServerMasterPeerSysAddr == UNASSIGNED_SYSTEM_ADDRESS) { Game::logger->Log("MasterServer", "Finished shutting down MasterServer!"); break; } - t += std::chrono::milliseconds(highFrameRate); + t += std::chrono::milliseconds(mediumFrameDelta); std::this_thread::sleep_until(t); - ticks++; + framesSinceShutdownStart++; - if (ticks == 600 * 6) { + if (framesSinceShutdownStart == maxShutdownTime) { Game::logger->Log("MasterServer", "Finished shutting down by timeout!"); break; } } - FinalizeShutdown(); + FinalizeShutdown(signal); } -int FinalizeShutdown() { +int32_t FinalizeShutdown(int32_t signal) { //Delete our objects here: Database::Destroy("MasterServer"); - delete Game::im; - delete Game::server; - delete Game::logger; + if (Game::config) delete Game::config; + if (Game::im) delete Game::im; + if (Game::server) delete Game::server; + if (Game::logger) delete Game::logger; - exit(EXIT_SUCCESS); - return EXIT_SUCCESS; + if (signal != EXIT_SUCCESS) exit(signal); + return signal; } diff --git a/dMasterServer/ObjectIDManager.cpp b/dMasterServer/ObjectIDManager.cpp index 0f3b98c9..83dde8dd 100644 --- a/dMasterServer/ObjectIDManager.cpp +++ b/dMasterServer/ObjectIDManager.cpp @@ -51,13 +51,13 @@ uint32_t ObjectIDManager::GeneratePersistentID(void) { uint32_t toReturn = ++this->currentPersistentID; // So we peroidically save our ObjID to the database: - if (toReturn % 25 == 0) { // TEMP: DISABLED FOR DEBUG / DEVELOPMENT! + // if (toReturn % 25 == 0) { // TEMP: DISABLED FOR DEBUG / DEVELOPMENT! sql::PreparedStatement* stmt = Database::CreatePreppedStmt( "UPDATE object_id_tracker SET last_object_id=?"); stmt->setUInt(1, toReturn); stmt->execute(); delete stmt; - } + // } return toReturn; } diff --git a/dNavigation/dNavMesh.cpp b/dNavigation/dNavMesh.cpp index b3c8a229..fdba4a2d 100644 --- a/dNavigation/dNavMesh.cpp +++ b/dNavigation/dNavMesh.cpp @@ -7,6 +7,7 @@ #include "dPlatforms.h" #include "NiPoint3.h" #include "BinaryIO.h" +#include "BinaryPathFinder.h" #include "dZoneManager.h" @@ -43,7 +44,7 @@ dNavMesh::~dNavMesh() { void dNavMesh::LoadNavmesh() { - std::string path = "./res/maps/navmeshes/" + std::to_string(m_ZoneId) + ".bin"; + std::string path = (BinaryPathFinder::GetBinaryDir() / "navmeshes/" / (std::to_string(m_ZoneId) + ".bin")).string(); if (!BinaryIO::DoesFileExist(path)) { return; diff --git a/dNavigation/dTerrain/RawFile.h b/dNavigation/dTerrain/RawFile.h index 84afae94..2a702c53 100644 --- a/dNavigation/dTerrain/RawFile.h +++ b/dNavigation/dTerrain/RawFile.h @@ -2,6 +2,7 @@ #include <string> #include <vector> +#include <cstdint> class RawChunk; struct RawMesh; diff --git a/dNet/AuthPackets.cpp b/dNet/AuthPackets.cpp index 60fe13c5..4d0d2fad 100644 --- a/dNet/AuthPackets.cpp +++ b/dNet/AuthPackets.cpp @@ -1,6 +1,5 @@ #include "AuthPackets.h" #include "PacketUtils.h" -#include "dMessageIdentifiers.h" #include "dNetCommon.h" #include "dServer.h" @@ -9,6 +8,7 @@ #include "ZoneInstanceManager.h" #include "MD5.h" #include "SHA512.h" +#include "GeneralUtils.h" #ifdef _WIN32 #include <bcrypt/BCrypt.hpp> @@ -21,6 +21,11 @@ #include "Game.h" #include "dConfig.h" +#include "eServerDisconnectIdentifiers.h" +#include "eLoginResponse.h" +#include "eConnectionType.h" +#include "eServerMessageType.h" +#include "eMasterMessageType.h" void AuthPackets::HandleHandshake(dServer* server, Packet* packet) { RakNet::BitStream inStream(packet->data, packet->length, false); @@ -29,16 +34,21 @@ void AuthPackets::HandleHandshake(dServer* server, Packet* packet) { inStream.Read(clientVersion); server->GetLogger()->Log("AuthPackets", "Received client version: %i", clientVersion); - SendHandshake(server, packet->systemAddress, server->GetIP(), server->GetPort()); + SendHandshake(server, packet->systemAddress, server->GetIP(), server->GetPort(), server->GetServerType()); } -void AuthPackets::SendHandshake(dServer* server, const SystemAddress& sysAddr, const std::string& nextServerIP, uint16_t nextServerPort) { +void AuthPackets::SendHandshake(dServer* server, const SystemAddress& sysAddr, const std::string& nextServerIP, uint16_t nextServerPort, const ServerType serverType) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, SERVER, MSG_SERVER_VERSION_CONFIRM); - bitStream.Write<unsigned int>(NET_VERSION); + PacketUtils::WriteHeader(bitStream, eConnectionType::SERVER, eServerMessageType::VERSION_CONFIRM); + uint32_t netVersion; + if (!GeneralUtils::TryParse(Game::config->GetValue("client_net_version"), netVersion)) { + Game::logger->Log("AuthPackets", "Failed to parse client_net_version. Cannot authenticate to %s:%i", nextServerIP.c_str(), nextServerPort); + return; + } + bitStream.Write<uint32_t>(netVersion); bitStream.Write(uint32_t(0x93)); - if (nextServerPort == 1001) bitStream.Write(uint32_t(1)); //Conn: auth + if (serverType == ServerType::Auth) bitStream.Write(uint32_t(1)); //Conn: auth else bitStream.Write(uint32_t(4)); //Conn: world bitStream.Write(uint32_t(0)); //Server process ID @@ -60,7 +70,7 @@ void AuthPackets::HandleLoginRequest(dServer* server, Packet* packet) { if (res->rowsCount() == 0) { server->GetLogger()->Log("AuthPackets", "No user found!"); - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_WRONG_PASS_OR_USER, "", "", 2001, username); + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::INVALID_USER, "", "", 2001, username); return; } @@ -84,14 +94,14 @@ void AuthPackets::HandleLoginRequest(dServer* server, Packet* packet) { //If we aren't running in live mode, then only GMs are allowed to enter: const auto& closedToNonDevs = Game::config->GetValue("closed_to_non_devs"); if (closedToNonDevs.size() > 0 && bool(std::stoi(closedToNonDevs)) && sqlGmLevel == 0) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::LOGIN_RESPONSE_PERMISSIONS_NOT_HIGH_ENOUGH, "The server is currently only open to developers.", "", 2001, username); + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::PERMISSIONS_NOT_HIGH_ENOUGH, "The server is currently only open to developers.", "", 2001, username); return; } if (Game::config->GetValue("dont_use_keys") != "1") { //Check to see if we have a play key: if (sqlPlayKey == 0 && sqlGmLevel == 0) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_PERMISSIONS_NOT_HIGH_ENOUGH, "Your account doesn't have a play key associated with it!", "", 2001, username); + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::PERMISSIONS_NOT_HIGH_ENOUGH, "Your account doesn't have a play key associated with it!", "", 2001, username); server->GetLogger()->Log("AuthPackets", "User %s tried to log in, but they don't have a play key.", username.c_str()); return; } @@ -103,7 +113,7 @@ void AuthPackets::HandleLoginRequest(dServer* server, Packet* packet) { bool isKeyActive = false; if (keyRes->rowsCount() == 0 && sqlGmLevel == 0) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_PERMISSIONS_NOT_HIGH_ENOUGH, "Your account doesn't have a play key associated with it!", "", 2001, username); + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::PERMISSIONS_NOT_HIGH_ENOUGH, "Your account doesn't have a play key associated with it!", "", 2001, username); return; } @@ -112,18 +122,18 @@ void AuthPackets::HandleLoginRequest(dServer* server, Packet* packet) { } if (!isKeyActive && sqlGmLevel == 0) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_PERMISSIONS_NOT_HIGH_ENOUGH, "Your play key has been disabled.", "", 2001, username); + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::PERMISSIONS_NOT_HIGH_ENOUGH, "Your play key has been disabled.", "", 2001, username); server->GetLogger()->Log("AuthPackets", "User %s tried to log in, but their play key was disabled", username.c_str()); return; } } if (sqlBanned) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_BANNED, "", "", 2001, username); return; + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::BANNED, "", "", 2001, username); return; } if (sqlLocked) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_ACCOUNT_LOCKED, "", "", 2001, username); return; + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::ACCOUNT_LOCKED, "", "", 2001, username); return; } /* @@ -169,25 +179,25 @@ void AuthPackets::HandleLoginRequest(dServer* server, Packet* packet) { } if (!loginSuccess) { - AuthPackets::SendLoginResponse(server, packet->systemAddress, LOGIN_RESPONSE_WRONG_PASS_OR_USER, "", "", 2001, username); + AuthPackets::SendLoginResponse(server, packet->systemAddress, eLoginResponse::WRONG_PASS, "", "", 2001, username); server->GetLogger()->Log("AuthPackets", "Wrong password used"); } else { SystemAddress system = packet->systemAddress; //Copy the sysAddr before the Packet gets destroyed from main if (!server->GetIsConnectedToMaster()) { - AuthPackets::SendLoginResponse(server, system, LOGIN_RESPONSE_GENERAL_FAILED, "", "", 0, username); + AuthPackets::SendLoginResponse(server, system, eLoginResponse::GENERAL_FAILED, "", "", 0, username); return; } ZoneInstanceManager::Instance()->RequestZoneTransfer(server, 0, 0, false, [system, server, username](bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, std::string zoneIP, uint16_t zonePort) { - AuthPackets::SendLoginResponse(server, system, LOGIN_RESPONSE_SUCCESS, "", zoneIP, zonePort, username); + AuthPackets::SendLoginResponse(server, system, eLoginResponse::SUCCESS, "", zoneIP, zonePort, username); }); } } void AuthPackets::SendLoginResponse(dServer* server, const SystemAddress& sysAddr, eLoginResponse responseCode, const std::string& errorMsg, const std::string& wServerIP, uint16_t wServerPort, std::string username) { RakNet::BitStream packet; - PacketUtils::WriteHeader(packet, CLIENT, MSG_CLIENT_LOGIN_RESPONSE); + PacketUtils::WriteHeader(packet, eConnectionType::CLIENT, eClientMessageType::LOGIN_RESPONSE); packet.Write(static_cast<uint8_t>(responseCode)); @@ -207,7 +217,7 @@ void AuthPackets::SendLoginResponse(dServer* server, const SystemAddress& sysAdd packet.Write(static_cast<uint16_t>(64)); // Version Minor // Writes the user key - uint32_t sessionKey = rand(); // not mt but whatever + uint32_t sessionKey = GeneralUtils::GenerateRandomNumber<uint32_t>(); std::string userHash = std::to_string(sessionKey); userHash = md5(userHash); PacketUtils::WritePacketWString(userHash, 33, &packet); @@ -253,7 +263,7 @@ void AuthPackets::SendLoginResponse(dServer* server, const SystemAddress& sysAdd //Inform the master server that we've created a session for this user: { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SET_SESSION_KEY); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SET_SESSION_KEY); bitStream.Write(sessionKey); PacketUtils::WriteString(bitStream, username, 66); server->SendToMaster(&bitStream); diff --git a/dNet/AuthPackets.h b/dNet/AuthPackets.h index e972d2f7..0f004ca4 100644 --- a/dNet/AuthPackets.h +++ b/dNet/AuthPackets.h @@ -5,11 +5,13 @@ #include "dCommonVars.h" #include "dNetCommon.h" +enum class ServerType : uint32_t; +enum class eLoginResponse : uint8_t; class dServer; namespace AuthPackets { void HandleHandshake(dServer* server, Packet* packet); - void SendHandshake(dServer* server, const SystemAddress& sysAddr, const std::string& nextServerIP, uint16_t nextServerPort); + void SendHandshake(dServer* server, const SystemAddress& sysAddr, const std::string& nextServerIP, uint16_t nextServerPort, const ServerType serverType); void HandleLoginRequest(dServer* server, Packet* packet); void SendLoginResponse(dServer* server, const SystemAddress& sysAddr, eLoginResponse responseCode, const std::string& errorMsg, const std::string& wServerIP, uint16_t wServerPort, std::string username); diff --git a/dNet/ChatPackets.cpp b/dNet/ChatPackets.cpp index 2d592c9b..41661523 100644 --- a/dNet/ChatPackets.cpp +++ b/dNet/ChatPackets.cpp @@ -8,12 +8,13 @@ #include "BitStream.h" #include "Game.h" #include "PacketUtils.h" -#include "dMessageIdentifiers.h" #include "dServer.h" +#include "eConnectionType.h" +#include "eChatMessageType.h" void ChatPackets::SendChatMessage(const SystemAddress& sysAddr, char chatChannel, const std::string& senderName, LWOOBJID playerObjectID, bool senderMythran, const std::u16string& message) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT, MSG_CHAT_GENERAL_CHAT_MESSAGE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT, eChatMessageType::GENERAL_CHAT_MESSAGE); bitStream.Write(static_cast<uint64_t>(0)); bitStream.Write(chatChannel); @@ -35,7 +36,7 @@ void ChatPackets::SendChatMessage(const SystemAddress& sysAddr, char chatChannel void ChatPackets::SendSystemMessage(const SystemAddress& sysAddr, const std::u16string& message, const bool broadcast) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT, MSG_CHAT_GENERAL_CHAT_MESSAGE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT, eChatMessageType::GENERAL_CHAT_MESSAGE); bitStream.Write(static_cast<uint64_t>(0)); bitStream.Write(static_cast<char>(4)); @@ -67,7 +68,7 @@ void ChatPackets::SendMessageFail(const SystemAddress& sysAddr) { //0x01 - "Upgrade to a full LEGO Universe Membership to chat with other players." CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_SEND_CANNED_TEXT); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::SEND_CANNED_TEXT); bitStream.Write<uint8_t>(0); //response type, options above ^ //docs say there's a wstring here-- no idea what it's for, or if it's even needed so leaving it as is for now. SEND_PACKET; diff --git a/dNet/ClientPackets.cpp b/dNet/ClientPackets.cpp index 39e835d2..8bebda93 100644 --- a/dNet/ClientPackets.cpp +++ b/dNet/ClientPackets.cpp @@ -31,8 +31,8 @@ #include "dConfig.h" #include "CharacterComponent.h" #include "Database.h" - - +#include "eGameMasterLevel.h" +#include "eReplicaComponentType.h" void ClientPackets::HandleChatMessage(const SystemAddress& sysAddr, Packet* packet) { User* user = UserManager::Instance()->GetUser(sysAddr); @@ -46,9 +46,7 @@ void ClientPackets::HandleChatMessage(const SystemAddress& sysAddr, Packet* pack return; } - CINSTREAM; - uint64_t header; - inStream.Read(header); + CINSTREAM_SKIP_HEADER; char chatChannel; uint16_t unknown; @@ -66,7 +64,7 @@ void ClientPackets::HandleChatMessage(const SystemAddress& sysAddr, Packet* pack } std::string playerName = user->GetLastUsedChar()->GetName(); - bool isMythran = user->GetLastUsedChar()->GetGMLevel() > 0; + bool isMythran = user->GetLastUsedChar()->GetGMLevel() > eGameMasterLevel::CIVILIAN; if (!user->GetLastChatMessageApproved() && !isMythran) return; @@ -82,14 +80,12 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac return; } - CINSTREAM; - uint64_t header; - inStream.Read(header); + CINSTREAM_SKIP_HEADER; - Entity* entity = EntityManager::Instance()->GetEntity(user->GetLastUsedChar()->GetObjectID()); + Entity* entity = Game::entityManager->GetEntity(user->GetLastUsedChar()->GetObjectID()); if (!entity) return; - ControllablePhysicsComponent* comp = static_cast<ControllablePhysicsComponent*>(entity->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); + ControllablePhysicsComponent* comp = static_cast<ControllablePhysicsComponent*>(entity->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); if (!comp) return; /* @@ -99,7 +95,7 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac comp->SetVelocity(zeroVel); comp->SetAngularVelocity(zeroVel); comp->SetIsOnGround(true); //probably8 - EntityManager::Instance()->SerializeEntity(entity); + Game::entityManager->SerializeEntity(entity); return; } */ @@ -140,10 +136,37 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac inStream.Read(angVelocity.z); } + // TODO figure out how to use these. Ignoring for now, but reading in if they exist. + bool hasLocalSpaceInfo{}; + LWOOBJID objectId{}; + NiPoint3 localSpacePosition{}; + bool hasLinearVelocity{}; + NiPoint3 linearVelocity{}; + if (inStream.Read(hasLocalSpaceInfo) && hasLocalSpaceInfo) { + inStream.Read(objectId); + inStream.Read(localSpacePosition.x); + inStream.Read(localSpacePosition.y); + inStream.Read(localSpacePosition.z); + if (inStream.Read(hasLinearVelocity) && hasLinearVelocity) { + inStream.Read(linearVelocity.x); + inStream.Read(linearVelocity.y); + inStream.Read(linearVelocity.z); + } + } + bool hasRemoteInputInfo{}; + RemoteInputInfo remoteInput{}; + + if (inStream.Read(hasRemoteInputInfo) && hasRemoteInputInfo) { + inStream.Read(remoteInput.m_RemoteInputX); + inStream.Read(remoteInput.m_RemoteInputY); + inStream.Read(remoteInput.m_IsPowersliding); + inStream.Read(remoteInput.m_IsModified); + } + bool updateChar = true; if (possessorComponent != nullptr) { - auto* possassableEntity = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); + auto* possassableEntity = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); if (possassableEntity != nullptr) { auto* possessableComponent = possassableEntity->GetComponent<PossessableComponent>(); @@ -154,9 +177,6 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac auto* vehiclePhysicsComponent = possassableEntity->GetComponent<VehiclePhysicsComponent>(); if (vehiclePhysicsComponent != nullptr) { - // This is flipped for whatever reason - rotation = NiQuaternion(rotation.z, rotation.y, rotation.x, rotation.w); - vehiclePhysicsComponent->SetPosition(position); vehiclePhysicsComponent->SetRotation(rotation); vehiclePhysicsComponent->SetIsOnGround(onGround); @@ -165,6 +185,7 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac vehiclePhysicsComponent->SetDirtyVelocity(velocityFlag); vehiclePhysicsComponent->SetAngularVelocity(angVelocity); vehiclePhysicsComponent->SetDirtyAngularVelocity(angVelocityFlag); + vehiclePhysicsComponent->SetRemoteInputInfo(remoteInput); } else { // Need to get the mount's controllable physics auto* controllablePhysicsComponent = possassableEntity->GetComponent<ControllablePhysicsComponent>(); @@ -178,7 +199,7 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac controllablePhysicsComponent->SetAngularVelocity(angVelocity); controllablePhysicsComponent->SetDirtyAngularVelocity(angVelocityFlag); } - EntityManager::Instance()->SerializeEntity(possassableEntity); + Game::entityManager->SerializeEntity(possassableEntity); } } @@ -206,9 +227,9 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac auto* player = static_cast<Player*>(entity); player->SetGhostReferencePoint(position); - EntityManager::Instance()->QueueGhostUpdate(player->GetObjectID()); + Game::entityManager->QueueGhostUpdate(player->GetObjectID()); - if (updateChar) EntityManager::Instance()->SerializeEntity(entity); + if (updateChar) Game::entityManager->SerializeEntity(entity); //TODO: add moving platform stuffs /*bool movingPlatformFlag; @@ -246,7 +267,7 @@ void ClientPackets::HandleClientPositionUpdate(const SystemAddress& sysAddr, Pac continue; } - EntityManager::Instance()->SerializeEntity(entity, player); + Game::entityManager->SerializeEntity(entity, player); } */ } @@ -268,7 +289,7 @@ void ClientPackets::HandleChatModerationRequest(const SystemAddress& sysAddr, Pa // Check if the player has restricted chat access auto* character = entity->GetCharacter(); - if (character->HasPermission(PermissionMap::RestrictedChatAccess)) { + if (character->HasPermission(ePermissionMap::RestrictedChatAccess)) { // Send a message to the player ChatPackets::SendSystemMessage( sysAddr, diff --git a/dNet/MasterPackets.cpp b/dNet/MasterPackets.cpp index e5b80e05..4233a37d 100644 --- a/dNet/MasterPackets.cpp +++ b/dNet/MasterPackets.cpp @@ -1,22 +1,23 @@ #include "MasterPackets.h" #include "BitStream.h" #include "PacketUtils.h" -#include "dMessageIdentifiers.h" #include "dCommonVars.h" #include "dServer.h" +#include "eConnectionType.h" +#include "eMasterMessageType.h" #include <string> void MasterPackets::SendPersistentIDRequest(dServer* server, uint64_t requestID) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_REQUEST_PERSISTENT_ID); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::REQUEST_PERSISTENT_ID); bitStream.Write(requestID); server->SendToMaster(&bitStream); } void MasterPackets::SendPersistentIDResponse(dServer* server, const SystemAddress& sysAddr, uint64_t requestID, uint32_t objID) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_REQUEST_PERSISTENT_ID_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::REQUEST_PERSISTENT_ID_RESPONSE); bitStream.Write(requestID); bitStream.Write(objID); @@ -26,7 +27,7 @@ void MasterPackets::SendPersistentIDResponse(dServer* server, const SystemAddres void MasterPackets::SendZoneTransferRequest(dServer* server, uint64_t requestID, bool mythranShift, uint32_t zoneID, uint32_t cloneID) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_REQUEST_ZONE_TRANSFER); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::REQUEST_ZONE_TRANSFER); bitStream.Write(requestID); bitStream.Write(static_cast<uint8_t>(mythranShift)); @@ -38,7 +39,7 @@ void MasterPackets::SendZoneTransferRequest(dServer* server, uint64_t requestID, void MasterPackets::SendZoneCreatePrivate(dServer* server, uint32_t zoneID, uint32_t cloneID, const std::string& password) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_CREATE_PRIVATE_ZONE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::CREATE_PRIVATE_ZONE); bitStream.Write(zoneID); bitStream.Write(cloneID); @@ -53,7 +54,7 @@ void MasterPackets::SendZoneCreatePrivate(dServer* server, uint32_t zoneID, uint void MasterPackets::SendZoneRequestPrivate(dServer* server, uint64_t requestID, bool mythranShift, const std::string& password) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_REQUEST_PRIVATE_ZONE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::REQUEST_PRIVATE_ZONE); bitStream.Write(requestID); bitStream.Write(static_cast<uint8_t>(mythranShift)); @@ -68,7 +69,7 @@ void MasterPackets::SendZoneRequestPrivate(dServer* server, uint64_t requestID, void MasterPackets::SendWorldReady(dServer* server, LWOMAPID zoneId, LWOINSTANCEID instanceId) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_WORLD_READY); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::WORLD_READY); bitStream.Write(zoneId); bitStream.Write(instanceId); @@ -78,7 +79,7 @@ void MasterPackets::SendWorldReady(dServer* server, LWOMAPID zoneId, LWOINSTANCE void MasterPackets::SendZoneTransferResponse(dServer* server, const SystemAddress& sysAddr, uint64_t requestID, bool mythranShift, uint32_t zoneID, uint32_t zoneInstance, uint32_t zoneClone, const std::string& serverIP, uint32_t serverPort) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_REQUEST_ZONE_TRANSFER_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::REQUEST_ZONE_TRANSFER_RESPONSE); bitStream.Write(requestID); bitStream.Write(static_cast<uint8_t>(mythranShift)); @@ -110,7 +111,7 @@ void MasterPackets::HandleServerInfo(Packet* packet) { void MasterPackets::SendServerInfo(dServer* server, Packet* packet) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SERVER_INFO); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SERVER_INFO); bitStream.Write(server->GetPort()); bitStream.Write(server->GetZoneID()); diff --git a/dNet/PacketUtils.cpp b/dNet/PacketUtils.cpp index 307ff633..77b314d6 100644 --- a/dNet/PacketUtils.cpp +++ b/dNet/PacketUtils.cpp @@ -1,17 +1,9 @@ #include "PacketUtils.h" -#include <MessageIdentifiers.h> #include <vector> #include <fstream> #include "dLogger.h" #include "Game.h" -void PacketUtils::WriteHeader(RakNet::BitStream& bitStream, uint16_t connectionType, uint32_t internalPacketID) { - bitStream.Write(MessageID(ID_USER_PACKET_ENUM)); - bitStream.Write(connectionType); - bitStream.Write(internalPacketID); - bitStream.Write(uint8_t(0)); -} - uint16_t PacketUtils::ReadPacketU16(uint32_t startLoc, Packet* packet) { if (startLoc + 2 > packet->length) return 0; diff --git a/dNet/PacketUtils.h b/dNet/PacketUtils.h index fbbd1839..d07759a0 100644 --- a/dNet/PacketUtils.h +++ b/dNet/PacketUtils.h @@ -1,11 +1,20 @@ #ifndef PACKETUTILS_H #define PACKETUTILS_H +#include <MessageIdentifiers.h> #include <BitStream.h> #include <string> +enum class eConnectionType : uint16_t; + namespace PacketUtils { - void WriteHeader(RakNet::BitStream& bitStream, uint16_t connectionType, uint32_t internalPacketID); + template<typename T> + void WriteHeader(RakNet::BitStream& bitStream, eConnectionType connectionType, T internalPacketID) { + bitStream.Write<uint8_t>(MessageID(ID_USER_PACKET_ENUM)); + bitStream.Write<eConnectionType>(connectionType); + bitStream.Write<uint32_t>(static_cast<uint32_t>(internalPacketID)); + bitStream.Write<uint8_t>(0); + } uint16_t ReadPacketU16(uint32_t startLoc, Packet* packet); uint32_t ReadPacketU32(uint32_t startLoc, Packet* packet); diff --git a/dNet/WorldPackets.cpp b/dNet/WorldPackets.cpp index 23d2c010..35c985eb 100644 --- a/dNet/WorldPackets.cpp +++ b/dNet/WorldPackets.cpp @@ -1,7 +1,6 @@ #include "dCommonVars.h" #include "WorldPackets.h" #include "BitStream.h" -#include "dMessageIdentifiers.h" #include "PacketUtils.h" #include "GeneralUtils.h" #include "User.h" @@ -14,12 +13,13 @@ #include "dZoneManager.h" #include "CharacterComponent.h" #include "ZCompression.h" +#include "eConnectionType.h" void WorldPackets::SendLoadStaticZone(const SystemAddress& sysAddr, float x, float y, float z, uint32_t checksum) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_LOAD_STATIC_ZONE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::LOAD_STATIC_ZONE); - auto zone = dZoneManager::Instance()->GetZone()->GetZoneID(); + auto zone = Game::zoneManager->GetZone()->GetZoneID(); bitStream.Write(static_cast<uint16_t>(zone.GetMapID())); bitStream.Write(static_cast<uint16_t>(zone.GetInstanceID())); //bitStream.Write(static_cast<uint32_t>(zone.GetCloneID())); @@ -41,7 +41,7 @@ void WorldPackets::SendCharacterList(const SystemAddress& sysAddr, User* user) { if (!user) return; RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_CHARACTER_LIST_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::CHARACTER_LIST_RESPONSE); std::vector<Character*> characters = user->GetCharacters(); bitStream.Write(static_cast<uint8_t>(characters.size())); @@ -88,30 +88,30 @@ void WorldPackets::SendCharacterList(const SystemAddress& sysAddr, User* user) { SEND_PACKET; } -void WorldPackets::SendCharacterCreationResponse(const SystemAddress& sysAddr, eCreationResponse response) { +void WorldPackets::SendCharacterCreationResponse(const SystemAddress& sysAddr, eCharacterCreationResponse response) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_CHARACTER_CREATE_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::CHARACTER_CREATE_RESPONSE); bitStream.Write(response); SEND_PACKET; } void WorldPackets::SendCharacterRenameResponse(const SystemAddress& sysAddr, eRenameResponse response) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_CHARACTER_RENAME_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::CHARACTER_RENAME_RESPONSE); bitStream.Write(response); SEND_PACKET; } void WorldPackets::SendCharacterDeleteResponse(const SystemAddress& sysAddr, bool response) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_DELETE_CHARACTER_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::DELETE_CHARACTER_RESPONSE); bitStream.Write(static_cast<uint8_t>(response)); SEND_PACKET; } void WorldPackets::SendTransferToWorld(const SystemAddress& sysAddr, const std::string& serverIP, uint32_t serverPort, bool mythranShift) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_TRANSFER_TO_WORLD); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::TRANSFER_TO_WORLD); PacketUtils::WriteString(bitStream, serverIP, 33); bitStream.Write(static_cast<uint16_t>(serverPort)); @@ -122,14 +122,14 @@ void WorldPackets::SendTransferToWorld(const SystemAddress& sysAddr, const std:: void WorldPackets::SendServerState(const SystemAddress& sysAddr) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_SERVER_STATES); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::SERVER_STATES); bitStream.Write(static_cast<uint8_t>(1)); //If the server is receiving this request, it probably is ready anyway. SEND_PACKET; } -void WorldPackets::SendCreateCharacter(const SystemAddress& sysAddr, Entity* entity, const std::string& xmlData, const std::u16string& username, int32_t gm) { +void WorldPackets::SendCreateCharacter(const SystemAddress& sysAddr, Entity* entity, const std::string& xmlData, const std::u16string& username, eGameMasterLevel gm) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_CREATE_CHARACTER); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::CREATE_CHARACTER); RakNet::BitStream data; data.Write<uint32_t>(7); //LDF key count @@ -144,8 +144,8 @@ void WorldPackets::SendCreateCharacter(const SystemAddress& sysAddr, Entity* ent LDFData<LOT>* lot = new LDFData<LOT>(u"template", 1); LDFData<std::string>* xmlConfigData = new LDFData<std::string>(u"xmlData", xmlData); LDFData<std::u16string>* name = new LDFData<std::u16string>(u"name", username); - LDFData<int32_t>* gmlevel = new LDFData<int32_t>(u"gmlevel", gm); - LDFData<int32_t>* chatmode = new LDFData<int32_t>(u"chatmode", gm); + LDFData<int32_t>* gmlevel = new LDFData<int32_t>(u"gmlevel", static_cast<int32_t>(gm)); + LDFData<int32_t>* chatmode = new LDFData<int32_t>(u"chatmode", static_cast<int32_t>(gm)); LDFData<int64_t>* reputation = new LDFData<int64_t>(u"reputation", character->GetReputation()); objid->WriteToPacket(&data); @@ -164,33 +164,41 @@ void WorldPackets::SendCreateCharacter(const SystemAddress& sysAddr, Entity* ent delete name; delete reputation; -#ifdef _WIN32 - bitStream.Write<uint32_t>(data.GetNumberOfBytesUsed() + 1); - bitStream.Write<uint8_t>(0); - bitStream.Write((char*)data.GetData(), data.GetNumberOfBytesUsed()); -#else //Compress the data before sending: - const int reservedSize = 5 * 1024 * 1024; - uint8_t compressedData[reservedSize]; + const uint32_t reservedSize = ZCompression::GetMaxCompressedLength(data.GetNumberOfBytesUsed()); + uint8_t* compressedData = new uint8_t[reservedSize]; + + // TODO There should be better handling here for not enough memory... + if (!compressedData) return; + size_t size = ZCompression::Compress(data.GetData(), data.GetNumberOfBytesUsed(), compressedData, reservedSize); + assert(size <= reservedSize); + bitStream.Write<uint32_t>(size + 9); //size of data + header bytes (8) bitStream.Write<uint8_t>(1); //compressed boolean, true bitStream.Write<uint32_t>(data.GetNumberOfBytesUsed()); bitStream.Write<uint32_t>(size); + /** + * In practice, this warning serves no purpose for us. We allocate the max memory needed on the heap + * and then compress the data. In the off chance that the compression actually increases the size, + * an assertion is done to prevent bad data from being saved or sent. + */ +#pragma warning(disable:6385) // C6385 Reading invalid data from 'compressedData'. for (size_t i = 0; i < size; i++) bitStream.Write(compressedData[i]); -#endif +#pragma warning(default:6385) - PacketUtils::SavePacket("chardata.bin", (const char*)bitStream.GetData(), static_cast<uint32_t>(bitStream.GetNumberOfBytesUsed())); + // PacketUtils::SavePacket("chardata.bin", (const char*)bitStream.GetData(), static_cast<uint32_t>(bitStream.GetNumberOfBytesUsed())); SEND_PACKET; + delete[] compressedData; Game::logger->Log("WorldPackets", "Sent CreateCharacter for ID: %llu", entity->GetObjectID()); } void WorldPackets::SendChatModerationResponse(const SystemAddress& sysAddr, bool requestAccepted, uint32_t requestID, const std::string& receiver, std::vector<std::pair<uint8_t, uint8_t>> unacceptedItems) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_CHAT_MODERATION_STRING); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::CHAT_MODERATION_STRING); bitStream.Write<uint8_t>(unacceptedItems.empty()); // Is sentence ok? bitStream.Write<uint16_t>(0x16); // Source ID, unknown @@ -212,14 +220,14 @@ void WorldPackets::SendChatModerationResponse(const SystemAddress& sysAddr, bool SEND_PACKET; } -void WorldPackets::SendGMLevelChange(const SystemAddress& sysAddr, bool success, uint8_t highestLevel, uint8_t prevLevel, uint8_t newLevel) { +void WorldPackets::SendGMLevelChange(const SystemAddress& sysAddr, bool success, eGameMasterLevel highestLevel, eGameMasterLevel prevLevel, eGameMasterLevel newLevel) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_MAKE_GM_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::MAKE_GM_RESPONSE); bitStream.Write<uint8_t>(success); - bitStream.Write<uint16_t>(highestLevel); - bitStream.Write<uint16_t>(prevLevel); - bitStream.Write<uint16_t>(newLevel); + bitStream.Write(static_cast<uint16_t>(highestLevel)); + bitStream.Write(static_cast<uint16_t>(prevLevel)); + bitStream.Write(static_cast<uint16_t>(newLevel)); SEND_PACKET; } diff --git a/dNet/WorldPackets.h b/dNet/WorldPackets.h index d9951941..ea8186c7 100644 --- a/dNet/WorldPackets.h +++ b/dNet/WorldPackets.h @@ -8,18 +8,21 @@ class User; struct SystemAddress; +enum class eGameMasterLevel : uint8_t; +enum class eCharacterCreationResponse : uint8_t; +enum class eRenameResponse : uint8_t; namespace WorldPackets { void SendLoadStaticZone(const SystemAddress& sysAddr, float x, float y, float z, uint32_t checksum); void SendCharacterList(const SystemAddress& sysAddr, User* user); - void SendCharacterCreationResponse(const SystemAddress& sysAddr, eCreationResponse response); + void SendCharacterCreationResponse(const SystemAddress& sysAddr, eCharacterCreationResponse response); void SendCharacterRenameResponse(const SystemAddress& sysAddr, eRenameResponse response); void SendCharacterDeleteResponse(const SystemAddress& sysAddr, bool response); void SendTransferToWorld(const SystemAddress& sysAddr, const std::string& serverIP, uint32_t serverPort, bool mythranShift); void SendServerState(const SystemAddress& sysAddr); - void SendCreateCharacter(const SystemAddress& sysAddr, Entity* entity, const std::string& xmlData, const std::u16string& username, int32_t gm); + void SendCreateCharacter(const SystemAddress& sysAddr, Entity* entity, const std::string& xmlData, const std::u16string& username, eGameMasterLevel gm); void SendChatModerationResponse(const SystemAddress& sysAddr, bool requestAccepted, uint32_t requestID, const std::string& receiver, std::vector<std::pair<uint8_t, uint8_t>> unacceptedItems); - void SendGMLevelChange(const SystemAddress& sysAddr, bool success, uint8_t highestLevel, uint8_t prevLevel, uint8_t newLevel); + void SendGMLevelChange(const SystemAddress& sysAddr, bool success, eGameMasterLevel highestLevel, eGameMasterLevel prevLevel, eGameMasterLevel newLevel); } #endif // WORLDPACKETS_H diff --git a/dNet/dMessageIdentifiers.h b/dNet/dMessageIdentifiers.h deleted file mode 100644 index 7dc08711..00000000 --- a/dNet/dMessageIdentifiers.h +++ /dev/null @@ -1,549 +0,0 @@ -#pragma once -#include "MessageIdentifiers.h" - -enum CONNECTION_TYPE { - SERVER = 0, //!< Means it is used throughout all servers - AUTH, //!< Means it is sent from the client authentication - CHAT, //!< Means it is sent from and to the chat server - CHAT_INTERNAL, //!< Unused - We can potentially use this in the future for various things - WORLD, //!< Means it is sent from the client world - CLIENT, //!< Means it is sent to the client from the world server - MASTER //!< Means it is sent to and from the master server -}; - -//! The Internal Server Packet Identifiers -enum SERVER { - MSG_SERVER_VERSION_CONFIRM = 0, /*!< Sent during a handshake to confirm the server/client version */ - MSG_SERVER_DISCONNECT_NOTIFY, /*!< Sent when a user disconnected */ - MSG_SERVER_GENERAL_NOTIFY /*!< A general notification */ -}; - -//! The Internal Authentication Packet Identifiers -enum AUTH { - MSG_AUTH_LOGIN_REQUEST = 0, /*!< Sent from the client when a user logs in */ - MSG_AUTH_LOGOUT_REQUEST, /*!< Sent from the client when a user logs out */ - MSG_AUTH_CREATE_NEW_ACCOUNT_REQUEST, /*!< Sent from the client when a user creates a new account */ - MSG_AUTH_LEGOINTERFACE_AUTH_RESPONSE, /*!< Unknown */ - MSG_AUTH_SESSIONKEY_RECEIVED_CONFIRM, /*!< Sent when the server recieved the session key (?) */ - MSG_AUTH_RUNTIME_CONFIG /*!< Unknown */ -}; - -//! The Internal Chat Packet Identifiers -enum CHAT { - MSG_CHAT_LOGIN_SESSION_NOTIFY = 0, /*!< When a user logs in */ - MSG_CHAT_GENERAL_CHAT_MESSAGE, /*!< Used for global chat messages */ - MSG_CHAT_PRIVATE_CHAT_MESSAGE, /*!< Used for private chat messages */ - MSG_CHAT_USER_CHANNEL_CHAT_MESSAGE, /*!< Unknown */ - MSG_CHAT_WORLD_DISCONNECT_REQUEST, /*!< Unknown */ - MSG_CHAT_WORLD_PROXIMITY_RESPONSE, /*!< Unknown */ - MSG_CHAT_WORLD_PARCEL_RESPONSE, /*!< Unknown */ - MSG_CHAT_ADD_FRIEND_REQUEST, /*!< When the client requests to add a friend */ - MSG_CHAT_ADD_FRIEND_RESPONSE, /*!< Sent from the server when the client adds a friend */ - MSG_CHAT_REMOVE_FRIEND, /*!< When the client removes a friend */ - MSG_CHAT_GET_FRIENDS_LIST, /*!< Sent when the client requests a user's friends list */ - MSG_CHAT_ADD_IGNORE, /*!< Sent when the client adds a friend to the "ignore" list */ - MSG_CHAT_REMOVE_IGNORE, /*!< Sent when the client removes a friend from the "ignore" list */ - MSG_CHAT_GET_IGNORE_LIST, /*!< Sent when the client requests a user's ignored list */ - MSG_CHAT_TEAM_MISSED_INVITE_CHECK, /*!< Unknown (Something with an unresponded-to friend request probably) */ - MSG_CHAT_TEAM_INVITE, /*!< When the client invites a user to a team */ - MSG_CHAT_TEAM_INVITE_RESPONSE, /*!< Sent from the server when the client invites someone to the team */ - MSG_CHAT_TEAM_KICK, /*!< Sent when the client kicks a member from a team */ - MSG_CHAT_TEAM_LEAVE, /*!< Sent when the client leaves a team */ - MSG_CHAT_TEAM_SET_LOOT, /*!< Unknown (Something to do with team loot) */ - MSG_CHAT_TEAM_SET_LEADER, /*!< Unknown (Probably sets the team leader or something) */ - MSG_CHAT_TEAM_GET_STATUS, /*!< Check to see if we are in a team or not, sent on world join */ - MSG_CHAT_GUILD_CREATE, /*!< Guild Creation */ - MSG_CHAT_GUILD_INVITE, /*!< Guild Invitation */ - MSG_CHAT_GUILD_INVITE_RESPONSE, /*!< Guild Invite Response */ - MSG_CHAT_GUILD_LEAVE, /*!< Guild Leave */ - MSG_CHAT_GUILD_KICK, /*!< Guild Kick */ - MSG_CHAT_GUILD_GET_STATUS, /*!< Guild Get Status */ - MSG_CHAT_GUILD_GET_ALL, /*!< Guild Get All */ - MSG_CHAT_SHOW_ALL, - MSG_CHAT_BLUEPRINT_MODERATED, - MSG_CHAT_BLUEPRINT_MODEL_READY, - MSG_CHAT_PROPERTY_READY_FOR_APPROVAL, - MSG_CHAT_PROPERTY_MODERATION_CHANGED, - MSG_CHAT_PROPERTY_BUILDMODE_CHANGED, - MSG_CHAT_PROPERTY_BUILDMODE_CHANGED_REPORT, - MSG_CHAT_MAIL, - MSG_CHAT_WORLD_INSTANCE_LOCATION_REQUEST, - MSG_CHAT_REPUTATION_UPDATE, - MSG_CHAT_SEND_CANNED_TEXT, - MSG_CHAT_GMLEVEL_UPDATE, - MSG_CHAT_CHARACTER_NAME_CHANGE_REQUEST, - MSG_CHAT_CSR_REQUEST, - MSG_CHAT_CSR_REPLY, - MSG_CHAT_GM_KICK, - MSG_CHAT_GM_ANNOUNCE, - MSG_CHAT_GM_MUTE, - MSG_CHAT_ACTIVITY_UPDATE, - MSG_CHAT_WORLD_ROUTE_PACKET, - MSG_CHAT_GET_ZONE_POPULATIONS, - MSG_CHAT_REQUEST_MINIMUM_CHAT_MODE, - MSG_CHAT_REQUEST_MINIMUM_CHAT_MODE_PRIVATE, - MSG_CHAT_MATCH_REQUEST, - MSG_CHAT_UGCMANIFEST_REPORT_MISSING_FILE, - MSG_CHAT_UGCMANIFEST_REPORT_DONE_FILE, - MSG_CHAT_UGCMANIFEST_REPORT_DONE_BLUEPRINT, - MSG_CHAT_UGCC_REQUEST, - MSG_CHAT_WHO, - MSG_CHAT_WORLD_PLAYERS_PET_MODERATED_ACKNOWLEDGE, - MSG_CHAT_ACHIEVEMENT_NOTIFY, - MSG_CHAT_GM_CLOSE_PRIVATE_CHAT_WINDOW, - MSG_CHAT_UNEXPECTED_DISCONNECT, - MSG_CHAT_PLAYER_READY, - MSG_CHAT_GET_DONATION_TOTAL, - MSG_CHAT_UPDATE_DONATION, - MSG_CHAT_PRG_CSR_COMMAND, - MSG_CHAT_HEARTBEAT_REQUEST_FROM_WORLD, - MSG_CHAT_UPDATE_FREE_TRIAL_STATUS -}; - -//! Used for packets related to chatting -enum CHAT_INTERNAL { - MSG_CHAT_INTERNAL_PLAYER_ADDED_NOTIFICATION = 0, - MSG_CHAT_INTERNAL_PLAYER_REMOVED_NOTIFICATION, - MSG_CHAT_INTERNAL_ADD_FRIEND, - MSG_CHAT_INTERNAL_ADD_BEST_FRIEND, - MSG_CHAT_INTERNAL_ADD_TO_TEAM, - MSG_CHAT_INTERNAL_ADD_BLOCK, - MSG_CHAT_INTERNAL_REMOVE_FRIEND, - MSG_CHAT_INTERNAL_REMOVE_BLOCK, - MSG_CHAT_INTERNAL_REMOVE_FROM_TEAM, - MSG_CHAT_INTERNAL_DELETE_TEAM, - MSG_CHAT_INTERNAL_REPORT, - MSG_CHAT_INTERNAL_PRIVATE_CHAT, - MSG_CHAT_INTERNAL_PRIVATE_CHAT_RESPONSE, - MSG_CHAT_INTERNAL_ANNOUNCEMENT, - MSG_CHAT_INTERNAL_MAIL_COUNT_UPDATE, - MSG_CHAT_INTERNAL_MAIL_SEND_NOTIFY, - MSG_CHAT_INTERNAL_REQUEST_USER_LIST, - MSG_CHAT_INTERNAL_FRIEND_LIST, - MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER, - MSG_CHAT_INTERNAL_TEAM_UPDATE, - MSG_CHAT_INTERNAL_MUTE_UPDATE, - MSG_CHAT_INTERNAL_CREATE_TEAM, -}; - -//! Used for packets send to the world -enum WORLD { - MSG_WORLD_CLIENT_VALIDATION = 1, // Session info - MSG_WORLD_CLIENT_CHARACTER_LIST_REQUEST, - MSG_WORLD_CLIENT_CHARACTER_CREATE_REQUEST, - MSG_WORLD_CLIENT_LOGIN_REQUEST, // Character selected - MSG_WORLD_CLIENT_GAME_MSG, - MSG_WORLD_CLIENT_CHARACTER_DELETE_REQUEST, - MSG_WORLD_CLIENT_CHARACTER_RENAME_REQUEST, - MSG_WORLD_CLIENT_HAPPY_FLOWER_MODE_NOTIFY, - MSG_WORLD_CLIENT_SLASH_RELOAD_MAP, // Reload map cmp - MSG_WORLD_CLIENT_SLASH_PUSH_MAP_REQUEST, // Push map req cmd - MSG_WORLD_CLIENT_SLASH_PUSH_MAP, // Push map cmd - MSG_WORLD_CLIENT_SLASH_PULL_MAP, // Pull map cmd - MSG_WORLD_CLIENT_LOCK_MAP_REQUEST, - MSG_WORLD_CLIENT_GENERAL_CHAT_MESSAGE, // General chat message - MSG_WORLD_CLIENT_HTTP_MONITOR_INFO_REQUEST, - MSG_WORLD_CLIENT_SLASH_DEBUG_SCRIPTS, // Debug scripts cmd - MSG_WORLD_CLIENT_MODELS_CLEAR, - MSG_WORLD_CLIENT_EXHIBIT_INSERT_MODEL, - MSG_WORLD_CLIENT_LEVEL_LOAD_COMPLETE, // Character data request - MSG_WORLD_CLIENT_TMP_GUILD_CREATE, - MSG_WORLD_CLIENT_ROUTE_PACKET, // Social? - MSG_WORLD_CLIENT_POSITION_UPDATE, - MSG_WORLD_CLIENT_MAIL, - MSG_WORLD_CLIENT_WORD_CHECK, // Whitelist word check - MSG_WORLD_CLIENT_STRING_CHECK, // Whitelist string check - MSG_WORLD_CLIENT_GET_PLAYERS_IN_ZONE, - MSG_WORLD_CLIENT_REQUEST_UGC_MANIFEST_INFO, - MSG_WORLD_CLIENT_BLUEPRINT_GET_ALL_DATA_REQUEST, - MSG_WORLD_CLIENT_CANCEL_MAP_QUEUE, - MSG_WORLD_CLIENT_HANDLE_FUNNESS, - MSG_WORLD_CLIENT_FAKE_PRG_CSR_MESSAGE, - MSG_WORLD_CLIENT_REQUEST_FREE_TRIAL_REFRESH, - MSG_WORLD_CLIENT_GM_SET_FREE_TRIAL_STATUS -}; - -//! An enum for packets sent to the client -enum CLIENT { - MSG_CLIENT_LOGIN_RESPONSE = 0, - MSG_CLIENT_LOGOUT_RESPONSE, - MSG_CLIENT_LOAD_STATIC_ZONE, - MSG_CLIENT_CREATE_OBJECT, - MSG_CLIENT_CREATE_CHARACTER, - MSG_CLIENT_CREATE_CHARACTER_EXTENDED, - MSG_CLIENT_CHARACTER_LIST_RESPONSE, - MSG_CLIENT_CHARACTER_CREATE_RESPONSE, - MSG_CLIENT_CHARACTER_RENAME_RESPONSE, - MSG_CLIENT_CHAT_CONNECT_RESPONSE, - MSG_CLIENT_AUTH_ACCOUNT_CREATE_RESPONSE, - MSG_CLIENT_DELETE_CHARACTER_RESPONSE, - MSG_CLIENT_GAME_MSG, - MSG_CLIENT_CONNECT_CHAT, - MSG_CLIENT_TRANSFER_TO_WORLD, - MSG_CLIENT_IMPENDING_RELOAD_NOTIFY, - MSG_CLIENT_MAKE_GM_RESPONSE, - MSG_CLIENT_HTTP_MONITOR_INFO_RESPONSE, - MSG_CLIENT_SLASH_PUSH_MAP_RESPONSE, - MSG_CLIENT_SLASH_PULL_MAP_RESPONSE, - MSG_CLIENT_SLASH_LOCK_MAP_RESPONSE, - MSG_CLIENT_BLUEPRINT_SAVE_RESPONSE, - MSG_CLIENT_BLUEPRINT_LUP_SAVE_RESPONSE, - MSG_CLIENT_BLUEPRINT_LOAD_RESPONSE_ITEMID, - MSG_CLIENT_BLUEPRINT_GET_ALL_DATA_RESPONSE, - MSG_CLIENT_MODEL_INSTANTIATE_RESPONSE, - MSG_CLIENT_DEBUG_OUTPUT, - MSG_CLIENT_ADD_FRIEND_REQUEST, - MSG_CLIENT_ADD_FRIEND_RESPONSE, - MSG_CLIENT_REMOVE_FRIEND_RESPONSE, - MSG_CLIENT_GET_FRIENDS_LIST_RESPONSE, - MSG_CLIENT_UPDATE_FRIEND_NOTIFY, - MSG_CLIENT_ADD_IGNORE_RESPONSE, - MSG_CLIENT_REMOVE_IGNORE_RESPONSE, - MSG_CLIENT_GET_IGNORE_LIST_RESPONSE, - MSG_CLIENT_TEAM_INVITE, - MSG_CLIENT_TEAM_INVITE_INITIAL_RESPONSE, - MSG_CLIENT_GUILD_CREATE_RESPONSE, - MSG_CLIENT_GUILD_GET_STATUS_RESPONSE, - MSG_CLIENT_GUILD_INVITE, - MSG_CLIENT_GUILD_INVITE_INITIAL_RESPONSE, - MSG_CLIENT_GUILD_INVITE_FINAL_RESPONSE, - MSG_CLIENT_GUILD_INVITE_CONFIRM, - MSG_CLIENT_GUILD_ADD_PLAYER, - MSG_CLIENT_GUILD_REMOVE_PLAYER, - MSG_CLIENT_GUILD_LOGIN_LOGOUT, - MSG_CLIENT_GUILD_RANK_CHANGE, - MSG_CLIENT_GUILD_DATA, - MSG_CLIENT_GUILD_STATUS, - MSG_CLIENT_MAIL, - MSG_CLIENT_DB_PROXY_RESULT, - MSG_CLIENT_SHOW_ALL_RESPONSE, - MSG_CLIENT_WHO_RESPONSE, - MSG_CLIENT_SEND_CANNED_TEXT, - MSG_CLIENT_UPDATE_CHARACTER_NAME, - MSG_CLIENT_SET_NETWORK_SIMULATOR, - MSG_CLIENT_INVALID_CHAT_MESSAGE, - MSG_CLIENT_MINIMUM_CHAT_MODE_RESPONSE, - MSG_CLIENT_MINIMUM_CHAT_MODE_RESPONSE_PRIVATE, - MSG_CLIENT_CHAT_MODERATION_STRING, - MSG_CLIENT_UGC_MANIFEST_RESPONSE, - MSG_CLIENT_IN_LOGIN_QUEUE, - MSG_CLIENT_SERVER_STATES, - MSG_CLIENT_GM_CLOSE_TARGET_CHAT_WINDOW, - MSG_CLIENT_GENERAL_TEXT_FOR_LOCALIZATION, - MSG_CLIENT_UPDATE_FREE_TRIAL_STATUS, - MSG_CLIENT_UGC_DOWNLOAD_FAILED = 120 -}; - -//! Used for packets sent to the master server -enum MASTER { - MSG_MASTER_REQUEST_PERSISTENT_ID = 1, - MSG_MASTER_REQUEST_PERSISTENT_ID_RESPONSE, - MSG_MASTER_REQUEST_ZONE_TRANSFER, - MSG_MASTER_REQUEST_ZONE_TRANSFER_RESPONSE, - MSG_MASTER_SERVER_INFO, - MSG_MASTER_REQUEST_SESSION_KEY, - MSG_MASTER_SET_SESSION_KEY, - MSG_MASTER_SESSION_KEY_RESPONSE, - MSG_MASTER_PLAYER_ADDED, - MSG_MASTER_PLAYER_REMOVED, - - MSG_MASTER_CREATE_PRIVATE_ZONE, - MSG_MASTER_REQUEST_PRIVATE_ZONE, - - MSG_MASTER_WORLD_READY, - MSG_MASTER_PREP_ZONE, - - MSG_MASTER_SHUTDOWN, - MSG_MASTER_SHUTDOWN_RESPONSE, - MSG_MASTER_SHUTDOWN_IMMEDIATE, - - MSG_MASTER_SHUTDOWN_UNIVERSE, - - MSG_MASTER_AFFIRM_TRANSFER_REQUEST, - MSG_MASTER_AFFIRM_TRANSFER_RESPONSE, - - MSG_MASTER_NEW_SESSION_ALERT -}; - -//! The Game messages -enum GAME_MSG : unsigned short { - GAME_MSG_TELEPORT = 19, - GAME_MSG_SET_PLAYER_CONTROL_SCHEME = 26, - GAME_MSG_DROP_CLIENT_LOOT = 30, - GAME_MSG_DIE = 37, - GAME_MSG_REQUEST_DIE = 38, - GAME_MSG_PLAY_EMOTE = 41, - GAME_MSG_PLAY_ANIMATION = 43, - GAME_MSG_CONTROL_BEHAVIOR = 48, - GAME_MSG_SET_NAME = 72, - GAME_MSG_ECHO_START_SKILL = 118, - GAME_MSG_START_SKILL = 119, - GAME_MSG_VERIFY_ACK = 121, - GAME_MSG_ADD_SKILL = 127, - GAME_MSG_REMOVE_SKILL = 128, - GAME_MSG_SET_CURRENCY = 133, - GAME_MSG_PICKUP_CURRENCY = 137, - GAME_MSG_PICKUP_ITEM = 139, - GAME_MSG_TEAM_PICKUP_ITEM = 140, - GAME_MSG_PLAY_FX_EFFECT = 154, - GAME_MSG_STOP_FX_EFFECT = 155, - GAME_MSG_REQUEST_RESURRECT = 159, - GAME_MSG_RESURRECT = 160, - GAME_MSG_PUSH_EQUIPPED_ITEMS_STATE = 191, - GAME_MSG_POP_EQUIPPED_ITEMS_STATE = 192, - GAME_MSG_SET_GM_LEVEL = 193, - GAME_MSG_SET_STUNNED = 198, - GAME_MSG_KNOCKBACK = 202, - GAME_MSG_REBUILD_CANCEL = 209, - GAME_MSG_ENABLE_REBUILD = 213, - GAME_MSG_MOVE_ITEM_IN_INVENTORY = 224, - GAME_MSG_ADD_ITEM_TO_INVENTORY_CLIENT_SYNC = 227, - GAME_MSG_REMOVE_ITEM_FROM_INVENTORY = 230, - GAME_MSG_EQUIP_ITEM = 231, - GAME_MSG_UN_EQUIP_ITEM = 233, - GAME_MSG_OFFER_MISSION = 248, - GAME_MSG_RESPOND_TO_MISSION = 249, - GAME_MSG_NOTIFY_MISSION = 254, - GAME_MSG_NOTIFY_MISSION_TASK = 255, - GAME_MSG_REBUILD_NOTIFY_STATE = 336, - GAME_MSG_TERMINATE_INTERACTION = 357, - GAME_MSG_SERVER_TERMINATE_INTERACTION = 358, - GAME_MSG_REQUEST_USE = 364, - GAME_MSG_VENDOR_OPEN_WINDOW = 369, - GAME_MSG_BUY_FROM_VENDOR = 373, - GAME_MSG_SELL_TO_VENDOR = 374, - GAME_MSG_TEAM_SET_OFF_WORLD_FLAG = 383, - GAME_MSG_SET_INVENTORY_SIZE = 389, - GAME_MSG_ACKNOWLEDGE_POSSESSION = 391, - GAME_MSG_SET_SHOOTING_GALLERY_PARAMS = 400, - GAME_MSG_REQUEST_ACTIVITY_START_STOP = 402, - GAME_MSG_REQUEST_ACTIVITY_ENTER = 403, - GAME_MSG_REQUEST_ACTIVITY_EXIT = 404, - GAME_MSG_ACTIVITY_ENTER = 405, - GAME_MSG_ACTIVITY_EXIT = 406, - GAME_MSG_ACTIVITY_START = 407, - GAME_MSG_ACTIVITY_STOP = 408, - GAME_MSG_SHOOTING_GALLERY_CLIENT_AIM_UPDATE = 409, - GAME_MSG_SHOOTING_GALLERY_FIRE = 411, - GAME_MSG_REQUEST_VENDOR_STATUS_UPDATE = 416, - GAME_MSG_VENDOR_STATUS_UPDATE = 417, - GAME_MSG_NOTIFY_CLIENT_SHOOTING_GALLERY_SCORE = 425, - GAME_MSG_CONSUME_CLIENT_ITEM = 427, - GAME_MSG_CLIENT_ITEM_CONSUMED = 428, - GAME_MSG_UPDATE_SHOOTING_GALLERY_ROTATION = 448, - GAME_MSG_SET_FLAG = 471, - GAME_MSG_NOTIFY_CLIENT_FLAG_CHANGE = 472, - GAME_MSG_VENDOR_TRANSACTION_RESULT = 476, - GAME_MSG_HAS_BEEN_COLLECTED = 486, - GAME_MSG_DISPLAY_CHAT_BUBBLE = 495, - GAME_MSG_SPAWN_PET = 498, - GAME_MSG_DESPAWN_PET = 499, - GAME_MSG_PLAYER_LOADED = 505, - GAME_MSG_PLAYER_READY = 509, - GAME_MSG_REQUEST_LINKED_MISSION = 515, - GAME_MSG_INVALID_ZONE_TRANSFER_LIST = 519, - GAME_MSG_MISSION_DIALOGUE_OK = 520, - GAME_MSG_DISPLAY_MESSAGE_BOX = 529, - GAME_MSG_MESSAGE_BOX_RESPOND = 530, - GAME_MSG_CHOICE_BOX_RESPOND = 531, - GAME_MSG_SMASH = 537, - GAME_MSG_UNSMASH = 538, - GAME_MSG_SET_SHOOTING_GALLERY_RETICULE_EFFECT = 548, - GAME_MSG_PLACE_MODEL_RESPONSE = 0x223, - GAME_MSG_SET_JET_PACK_MODE = 561, - GAME_MSG_REGISTER_PET_ID = 565, - GAME_MSG_REGISTER_PET_DBID = 566, - GAME_MSG_SHOW_ACTIVITY_COUNTDOWN = 568, - GAME_MSG_START_ACTIVITY_TIME = 576, - GAME_MSG_ACTIVITY_PAUSE = 602, - GAME_MSG_USE_NON_EQUIPMENT_ITEM = 603, - GAME_MSG_USE_ITEM_RESULT = 607, - GAME_MSG_COMMAND_PET = 640, - GAME_MSG_PET_RESPONSE = 641, - GAME_MSG_REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA = 648, - GAME_MSG_SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA = 649, - GAME_MSG_NOTIFY_OBJECT = 656, - GAME_MSG_CLIENT_NOTIFY_PET = 659, - GAME_MSG_NOTIFY_PET = 660, - GAME_MSG_NOTIFY_PET_TAMING_MINIGAME = 661, - GAME_MSG_START_SERVER_PET_MINIGAME_TIMER = 662, - GAME_MSG_CLIENT_EXIT_TAMING_MINIGAME = 663, - GAME_MSG_PET_NAME_CHANGED = 686, - GAME_MSG_PET_TAMING_MINIGAME_RESULT = 667, - GAME_MSG_PET_TAMING_TRY_BUILD_RESULT = 668, - GAME_MSG_NOTIFY_TAMING_BUILD_SUCCESS = 673, - GAME_MSG_NOTIFY_TAMING_MODEL_LOADED_ON_SERVER = 674, - GAME_MSG_ADD_PET_TO_PLAYER = 681, - GAME_MSG_REQUEST_SET_PET_NAME = 683, - GAME_MSG_SET_PET_NAME = 684, - GAME_MSG_NOTIFY_PET_TAMING_PUZZLE_SELECTED = 675, - GAME_MSG_SHOW_PET_ACTION_BUTTON = 692, - GAME_MSG_SET_EMOTE_LOCK_STATE = 693, - GAME_MSG_USE_ITEM_REQUIREMENTS_RESPONSE = 703, - GAME_MSG_PLAY_EMBEDDED_EFFECT_ON_ALL_CLIENTS_NEAR_OBJECT = 713, - GAME_MSG_DOWNLOAD_PROPERTY_DATA = 716, - GAME_MSG_QUERY_PROPERTY_DATA = 717, - GAME_MSG_PROPERTY_EDITOR_BEGIN = 724, - GAME_MSG_PROPERTY_EDITOR_END = 725, - GAME_MSG_START_PATHING = 735, - GAME_MSG_NOTIFY_CLIENT_ZONE_OBJECT = 737, - GAME_MSG_UPDATE_REPUTATION = 746, - GAME_MSG_PROPERTY_RENTAL_RESPONSE = 750, - GAME_MSG_REQUEST_PLATFORM_RESYNC = 760, - GAME_MSG_PLATFORM_RESYNC = 761, - GAME_MSG_PLAY_CINEMATIC = 762, - GAME_MSG_END_CINEMATIC = 763, - GAME_MSG_CINEMATIC_UPDATE = 764, - GAME_MSG_TOGGLE_GHOST_REFERENCE_OVERRIDE = 767, - GAME_MSG_SET_GHOST_REFERENCE_POSITION = 768, - GAME_MSG_FIRE_EVENT_SERVER_SIDE = 770, - GAME_MSG_SET_NETWORK_SCRIPT_VAR = 781, - GAME_MSG_UPDATE_MODEL_FROM_CLIENT = 793, - GAME_MSG_DELETE_MODEL_FROM_CLIENT = 794, - GAME_MSG_PLAY_ND_AUDIO_EMITTER = 821, - GAME_MSG_PLAY2_DAMBIENT_SOUND = 831, - GAME_MSG_ENTER_PROPERTY1 = 840, - GAME_MSG_ENTER_PROPERTY2 = 841, - GAME_MSG_PROPERTY_ENTRANCE_SYNC = 842, - GAME_MSG_PROPERTY_SELECT_QUERY = 845, - GAME_MSG_PARSE_CHAT_MESSAGE = 850, - GAME_MSG_BROADCAST_TEXT_TO_CHATBOX = 858, - GAME_MSG_OPEN_PROPERTY_MANAGEMENT = 860, - GAME_MSG_OPEN_PROPERTY_VENDOR = 861, - GAME_MSG_UPDATE_PROPERTY_OR_MODEL_FOR_FILTER_CHECK = 863, - GAME_MSG_CLIENT_TRADE_REQUEST = 868, - GAME_MSG_SERVER_TRADE_REQUEST = 869, - GAME_MSG_SERVER_TRADE_INVITE = 870, - GAME_MSG_CLIENT_TRADE_REPLY = 871, - GAME_MSG_SERVER_TRADE_REPLY = 872, - GAME_MSG_SERVER_TRADE_INITIAL_REPLY = 873, - GAME_MSG_SERVER_TRADE_FINAL_REPLY = 874, - GAME_MSG_CLIENT_TRADE_UPDATE = 875, - GAME_MSG_SERVER_SIDE_TRADE_UPDATE = 876, - GAME_MSG_SERVER_TRADE_UPDATE = 877, - GAME_MSG_CLIENT_TRADE_CANCEL = 878, - GAME_MSG_CLIENT_SIDE_TRADE_CANCEL = 879, - GAME_MSG_CLIENT_TRADE_ACCEPT = 880, - GAME_MSG_SERVER_SIDE_TRADE_ACCEPT = 881, - GAME_MSG_SERVER_SIDE_TRADE_CANCEL = 882, - GAME_MSG_SERVER_TRADE_CANCEL = 883, - GAME_MSG_SERVER_TRADE_ACCEPT = 884, - GAME_MSG_READY_FOR_UPDATES = 888, - GAME_MSG_ORIENT_TO_OBJECT = 905, - GAME_MSG_ORIENT_TO_POSITION = 906, - GAME_MSG_ORIENT_TO_ANGLE = 907, - GAME_MSG_BOUNCER_ACTIVE_STATUS = 942, - GAME_MSG_BBB_LOAD_ITEM_REQUEST = 1000, - GAME_MSG_BBB_SAVE_REQUEST = 1001, - GAME_MSG_BBB_SAVE_RESPONSE = 1006, - GAME_MSG_NOTIFY_CLIENT_OBJECT = 1042, - GAME_MSG_DISPLAY_ZONE_SUMMARY = 1043, - GAME_MSG_ACTIVITY_STATE_CHANGE_REQUEST = 1053, - GAME_MSG_MODIFY_PLAYER_ZONE_STATISTIC = 1046, - GAME_MSG_START_BUILDING_WITH_ITEM = 1057, - GAME_MSG_START_ARRANGING_WITH_ITEM = 1061, - GAME_MSG_FINISH_ARRANGING_WITH_ITEM = 1062, - GAME_MSG_DONE_ARRANGING_WITH_ITEM = 1063, - GAME_MSG_SET_BUILD_MODE = 1068, - GAME_MSG_BUILD_MODE_SET = 1069, - GAME_MSG_SET_BUILD_MODE_CONFIRMED = 1073, - GAME_MSG_NOTIFY_CLIENT_FAILED_PRECONDITION = 1081, - GAME_MSG_MOVE_ITEM_BETWEEN_INVENTORY_TYPES = 1093, - GAME_MSG_MODULAR_BUILD_BEGIN = 1094, - GAME_MSG_MODULAR_BUILD_END = 1095, - GAME_MSG_MODULAR_BUILD_MOVE_AND_EQUIP = 1096, - GAME_MSG_MODULAR_BUILD_FINISH = 1097, - GAME_MSG_REPORT_BUG = 1198, - GAME_MSG_MISSION_DIALOGUE_CANCELLED = 1129, - GAME_MSG_ECHO_SYNC_SKILL = 1144, - GAME_MSG_SYNC_SKILL = 1145, - GAME_MSG_REQUEST_SERVER_PROJECTILE_IMPACT = 1148, - GAME_MSG_DO_CLIENT_PROJECTILE_IMPACT = 1151, - GAME_MSG_MODULAR_BUILD_CONVERT_MODEL = 1155, - GAME_MSG_SET_PLAYER_ALLOWED_RESPAWN = 1165, - GAME_MSG_UI_MESSAGE_SERVER_TO_SINGLE_CLIENT = 1184, - GAME_MSG_UI_MESSAGE_SERVER_TO_ALL_CLIENTS = 1185, - GAME_MSG_PET_TAMING_TRY_BUILD = 1197, - GAME_MSG_REQUEST_SMASH_PLAYER = 1202, - GAME_MSG_FIRE_EVENT_CLIENT_SIDE = 1213, - GAME_MSG_TOGGLE_GM_INVIS = 1218, - GAME_MSG_CHANGE_OBJECT_WORLD_STATE = 1223, - GAME_MSG_VEHICLE_LOCK_INPUT = 1230, - GAME_MSG_VEHICLE_UNLOCK_INPUT = 1231, - GAME_MSG_RACING_RESET_PLAYER_TO_LAST_RESET = 1252, - GAME_MSG_RACING_SERVER_SET_PLAYER_LAP_AND_PLANE = 1253, - GAME_MSG_RACING_SET_PLAYER_RESET_INFO = 1254, - GAME_MSG_RACING_PLAYER_INFO_RESET_FINISHED = 1255, - GAME_MSG_LOCK_NODE_ROTATION = 1260, - GAME_MSG_VEHICLE_SET_WHEEL_LOCK_STATE = 1273, - GAME_MSG_NOTIFY_VEHICLE_OF_RACING_OBJECT = 1276, - GAME_MSG_PLAYER_REACHED_RESPAWN_CHECKPOINT = 1296, - GAME_MSG_HANDLE_UGC_EQUIP_POST_DELETE_BASED_ON_EDIT_MODE = 1300, - GAME_MSG_HANDLE_UGC_EQUIP_PRE_CREATE_BASED_ON_EDIT_MODE = 1301, - GAME_MSG_PROPERTY_CONTENTS_FROM_CLIENT = 1305, - GAME_MSG_GET_MODELS_ON_PROPERTY = 1306, - GAME_MSG_MATCH_REQUEST = 1308, - GAME_MSG_MATCH_RESPONSE = 1309, - GAME_MSG_MATCH_UPDATE = 1310, - GAME_MSG_MODULE_ASSEMBLY_DB_DATA_FOR_CLIENT = 1131, - GAME_MSG_MODULE_ASSEMBLY_QUERY_DATA = 1132, - GAME_MSG_VEHICLE_ADD_PASSIVE_BOOST_ACTION = 1340, - GAME_MSG_VEHICLE_REMOVE_PASSIVE_BOOST_ACTION = 1341, - GAME_MSG_VEHICLE_NOTIFY_SERVER_ADD_PASSIVE_BOOST_ACTION = 1342, - GAME_MSG_VEHICLE_NOTIFY_SERVER_REMOVE_PASSIVE_BOOST_ACTION = 1343, - GAME_MSG_VEHICLE_ADD_SLOWDOWN_ACTION = 1344, - GAME_MSG_VEHICLE_REMOVE_SLOWDOWN_ACTION = 1345, - GAME_MSG_VEHICLE_NOTIFY_SERVER_ADD_SLOWDOWN_ACTION = 1346, - GAME_MSG_VEHICLE_NOTIFY_SERVER_REMOVE_SLOWDOWN_ACTION = 1347, - GAME_MSG_BUYBACK_FROM_VENDOR = 1350, - GAME_MSG_SET_PROPERTY_ACCESS = 1366, - GAME_MSG_ZONE_PROPERTY_MODEL_PLACED = 1369, - GAME_MSG_ZONE_PROPERTY_MODEL_ROTATED = 1370, - GAME_MSG_ZONE_PROPERTY_MODEL_REMOVED_WHILE_EQUIPPED = 1371, - GAME_MSG_ZONE_PROPERTY_MODEL_EQUIPPED = 1372, - GAME_MSG_ZONE_PROPERTY_MODEL_PICKED_UP = 1373, - GAME_MSG_ZONE_PROPERTY_MODEL_REMOVED = 1374, - GAME_MSG_NOTIFY_RACING_CLIENT = 1390, - GAME_MSG_RACING_PLAYER_HACK_CAR = 1391, - GAME_MSG_RACING_PLAYER_LOADED = 1392, - GAME_MSG_RACING_CLIENT_READY = 1393, - GAME_MSG_UPDATE_CHAT_MODE = 1395, - GAME_MSG_VEHICLE_NOTIFY_FINISHED_RACE = 1396, - GAME_MSG_SET_CONSUMABLE_ITEM = 1409, - GAME_MSG_SET_PET_NAME_MODERATED = 1448, - GAME_MSG_MODIFY_LEGO_SCORE = 1459, - GAME_MSG_RESTORE_TO_POST_LOAD_STATS = 1468, - GAME_MSG_SET_RAIL_MOVEMENT = 1471, - GAME_MSG_START_RAIL_MOVEMENT = 1472, - GAME_MSG_CANCEL_RAIL_MOVEMENT = 1474, - GAME_MSG_CLIENT_RAIL_MOVEMENT_READY = 1476, - GAME_MSG_PLAYER_RAIL_ARRIVED_NOTIFICATION = 1477, - GAME_MSG_UPDATE_PLAYER_STATISTIC = 1481, - GAME_MSG_MODULAR_ASSEMBLY_NIF_COMPLETED = 1498, - GAME_MSG_NOTIFY_NOT_ENOUGH_INV_SPACE = 1516, - GAME_MSG_TEAM_SET_LEADER = 0x0615, - GAME_MSG_TEAM_INVITE_CONFIRM = 0x0616, - GAME_MSG_TEAM_GET_STATUS_RESPONSE = 0x0617, - GAME_MSG_TEAM_ADD_PLAYER = 0x061a, - GAME_MSG_TEAM_REMOVE_PLAYER = 0x061b, - GAME_MSG_START_CELEBRATION_EFFECT = 1618, - GAME_MSG_ADD_BUFF = 1647, - GAME_MSG_SERVER_DONE_LOADING_ALL_OBJECTS = 1642, - GAME_MSG_PLACE_PROPERTY_MODEL = 1170, - GAME_MSG_VEHICLE_NOTIFY_HIT_IMAGINATION_SERVER = 1606, - GAME_MSG_ADD_RUN_SPEED_MODIFIER = 1505, - GAME_MSG_HANDLE_HOT_PROPERTY_DATA = 1511, - GAME_MSG_SEND_HOT_PROPERTY_DATA = 1510, - GAME_MSG_REMOVE_RUN_SPEED_MODIFIER = 1506, - GAME_MSG_UPDATE_PROPERTY_PERFORMANCE_COST = 1547, - GAME_MSG_PROPERTY_ENTRANCE_BEGIN = 1553, - GAME_MSG_REQUEST_MOVE_ITEM_BETWEEN_INVENTORY_TYPES = 1666, - GAME_MSG_RESPONSE_MOVE_ITEM_BETWEEN_INVENTORY_TYPES = 1667, - GAME_MSG_PLAYER_SET_CAMERA_CYCLING_MODE = 1676, - GAME_MSG_SET_MOUNT_INVENTORY_ID = 1726, - GAME_MSG_NOTIFY_SERVER_LEVEL_PROCESSING_COMPLETE = 1734, - GAME_MSG_NOTIFY_LEVEL_REWARDS = 1735, - GAME_MSG_DISMOUNT_COMPLETE = 1756, - GAME_MSG_MARK_INVENTORY_ITEM_AS_ACTIVE = 1767, - END -}; diff --git a/dNet/dServer.cpp b/dNet/dServer.cpp index 4c032f33..610f06a5 100644 --- a/dNet/dServer.cpp +++ b/dNet/dServer.cpp @@ -2,12 +2,15 @@ #include "dServer.h" #include "dNetCommon.h" #include "dLogger.h" +#include "dConfig.h" #include "RakNetworkFactory.h" #include "MessageIdentifiers.h" +#include "eConnectionType.h" +#include "eServerMessageType.h" +#include "eMasterMessageType.h" #include "PacketUtils.h" -#include "dMessageIdentifiers.h" #include "MasterPackets.h" #include "ZoneInstanceManager.h" @@ -35,7 +38,7 @@ public: } } ReceiveDownloadCompleteCB; -dServer::dServer(const std::string& ip, int port, int instanceID, int maxConnections, bool isInternal, bool useEncryption, dLogger* logger, const std::string masterIP, int masterPort, ServerType serverType, unsigned int zoneID) { +dServer::dServer(const std::string& ip, int port, int instanceID, int maxConnections, bool isInternal, bool useEncryption, dLogger* logger, const std::string masterIP, int masterPort, ServerType serverType, dConfig* config, bool* shouldShutdown, unsigned int zoneID) { mIP = ip; mPort = port; mZoneID = zoneID; @@ -50,7 +53,8 @@ dServer::dServer(const std::string& ip, int port, int instanceID, int maxConnect mNetIDManager = nullptr; mReplicaManager = nullptr; mServerType = serverType; - + mConfig = config; + mShouldShutdown = shouldShutdown; //Attempt to start our server here: mIsOkay = Startup(); @@ -116,15 +120,18 @@ Packet* dServer::ReceiveFromMaster() { } if (packet->data[0] == ID_USER_PACKET_ENUM) { - if (packet->data[1] == MASTER) { - switch (packet->data[3]) { - case MSG_MASTER_REQUEST_ZONE_TRANSFER_RESPONSE: { + if (static_cast<eConnectionType>(packet->data[1]) == eConnectionType::MASTER) { + switch (static_cast<eMasterMessageType>(packet->data[3])) { + case eMasterMessageType::REQUEST_ZONE_TRANSFER_RESPONSE: { uint64_t requestID = PacketUtils::ReadPacketU64(8, packet); ZoneInstanceManager::Instance()->HandleRequestZoneTransferResponse(requestID, packet); break; } + case eMasterMessageType::SHUTDOWN: + *mShouldShutdown = true; + break; - //When we handle these packets in World instead dServer, we just return the packet's pointer. + //When we handle these packets in World instead dServer, we just return the packet's pointer. default: return packet; @@ -159,9 +166,9 @@ void dServer::SendToMaster(RakNet::BitStream* bitStream) { mMasterPeer->Send(bitStream, SYSTEM_PRIORITY, RELIABLE_ORDERED, 0, mMasterSystemAddress, false); } -void dServer::Disconnect(const SystemAddress& sysAddr, uint32_t disconNotifyID) { +void dServer::Disconnect(const SystemAddress& sysAddr, eServerDisconnectIdentifiers disconNotifyID) { RakNet::BitStream bitStream; - PacketUtils::WriteHeader(bitStream, SERVER, MSG_SERVER_DISCONNECT_NOTIFY); + PacketUtils::WriteHeader(bitStream, eConnectionType::SERVER, eServerMessageType::DISCONNECT_NOTIFY); bitStream.Write(disconNotifyID); mPeer->Send(&bitStream, SYSTEM_PRIORITY, RELIABLE_ORDERED, 0, sysAddr, false); @@ -182,7 +189,8 @@ bool dServer::Startup() { if (mIsInternal) { mPeer->SetIncomingPassword("3.25 DARKFLAME1", 15); } else { - //mPeer->SetPerConnectionOutgoingBandwidthLimit(800000); //100Kb/s + UpdateBandwidthLimit(); + UpdateMaximumMtuSize(); mPeer->SetIncomingPassword("3.25 ND1", 8); } @@ -192,8 +200,21 @@ bool dServer::Startup() { return true; } +void dServer::UpdateMaximumMtuSize() { + auto maxMtuSize = mConfig->GetValue("maximum_mtu_size"); + mPeer->SetMTUSize(maxMtuSize.empty() ? 1228 : std::stoi(maxMtuSize)); +} + +void dServer::UpdateBandwidthLimit() { + auto newBandwidth = mConfig->GetValue("maximum_outgoing_bandwidth"); + mPeer->SetPerConnectionOutgoingBandwidthLimit(!newBandwidth.empty() ? std::stoi(newBandwidth) : 0); +} + void dServer::Shutdown() { - mPeer->Shutdown(1000); + if (mPeer) { + mPeer->Shutdown(1000); + RakNetworkFactory::DestroyRakPeerInterface(mPeer); + } if (mNetIDManager) { delete mNetIDManager; @@ -205,10 +226,9 @@ void dServer::Shutdown() { mReplicaManager = nullptr; } - //RakNetworkFactory::DestroyRakPeerInterface(mPeer); //Not needed, we already called Shutdown ourselves. - if (mServerType != ServerType::Master) { + if (mServerType != ServerType::Master && mMasterPeer) { mMasterPeer->Shutdown(1000); - //RakNetworkFactory::DestroyRakPeerInterface(mMasterPeer); + RakNetworkFactory::DestroyRakPeerInterface(mMasterPeer); } } diff --git a/dNet/dServer.h b/dNet/dServer.h index bd052f86..797647b6 100644 --- a/dNet/dServer.h +++ b/dNet/dServer.h @@ -5,6 +5,8 @@ #include "NetworkIDManager.h" class dLogger; +class dConfig; +enum class eServerDisconnectIdentifiers : uint32_t; enum class ServerType : uint32_t { Master, @@ -15,17 +17,32 @@ enum class ServerType : uint32_t { class dServer { public: - dServer(const std::string& ip, int port, int instanceID, int maxConnections, bool isInternal, bool useEncryption, dLogger* logger, const std::string masterIP, int masterPort, ServerType serverType, unsigned int zoneID = 0); + // Default constructor should only used for testing! + dServer() {}; + dServer( + const std::string& ip, + int port, + int instanceID, + int maxConnections, + bool isInternal, + bool useEncryption, + dLogger* logger, + const std::string masterIP, + int masterPort, + ServerType serverType, + dConfig* config, + bool* shouldShutdown, + unsigned int zoneID = 0); ~dServer(); Packet* ReceiveFromMaster(); Packet* Receive(); void DeallocatePacket(Packet* packet); void DeallocateMasterPacket(Packet* packet); - void Send(RakNet::BitStream* bitStream, const SystemAddress& sysAddr, bool broadcast); + virtual void Send(RakNet::BitStream* bitStream, const SystemAddress& sysAddr, bool broadcast); void SendToMaster(RakNet::BitStream* bitStream); - void Disconnect(const SystemAddress& sysAddr, uint32_t disconNotifyID); + void Disconnect(const SystemAddress& sysAddr, eServerDisconnectIdentifiers disconNotifyID); bool IsConnected(const SystemAddress& sysAddr); const std::string& GetIP() const { return mIP; } @@ -40,6 +57,8 @@ public: const int GetInstanceID() const { return mInstanceID; } ReplicaManager* GetReplicaManager() { return mReplicaManager; } void UpdateReplica(); + void UpdateBandwidthLimit(); + void UpdateMaximumMtuSize(); int GetPing(const SystemAddress& sysAddr) const; int GetLatestPing(const SystemAddress& sysAddr) const; @@ -55,10 +74,16 @@ private: bool ConnectToMaster(); private: - dLogger* mLogger; - RakPeerInterface* mPeer; - ReplicaManager* mReplicaManager; - NetworkIDManager* mNetIDManager; + dLogger* mLogger = nullptr; + dConfig* mConfig = nullptr; + RakPeerInterface* mPeer = nullptr; + ReplicaManager* mReplicaManager = nullptr; + NetworkIDManager* mNetIDManager = nullptr; + + /** + * Whether or not to shut down the server. Pointer to Game::shouldShutdown. + */ + bool* mShouldShutdown = nullptr; SocketDescriptor mSocketDescriptor; std::string mIP; int mPort; @@ -71,7 +96,7 @@ private: bool mMasterConnectionActive; ServerType mServerType; - RakPeerInterface* mMasterPeer; + RakPeerInterface* mMasterPeer = nullptr; SocketDescriptor mMasterSocketDescriptor; SystemAddress mMasterSystemAddress; std::string mMasterIP; diff --git a/dPhysics/dpEntity.cpp b/dPhysics/dpEntity.cpp index 42c195f2..c7ed56f8 100644 --- a/dPhysics/dpEntity.cpp +++ b/dPhysics/dpEntity.cpp @@ -34,7 +34,6 @@ dpEntity::dpEntity(const LWOOBJID& objectID, NiPoint3 boxDimensions, bool isStat m_CollisionGroup = COLLISION_GROUP_ALL; m_CollisionShape = new dpShapeBox(this, boxDimensions.x, boxDimensions.y, boxDimensions.z); - if (boxDimensions.x > 100.0f) m_IsGargantuan = true; } dpEntity::dpEntity(const LWOOBJID& objectID, float width, float height, float depth, bool isStatic) { @@ -45,7 +44,6 @@ dpEntity::dpEntity(const LWOOBJID& objectID, float width, float height, float de m_CollisionGroup = COLLISION_GROUP_ALL; m_CollisionShape = new dpShapeBox(this, width, height, depth); - if (width > 100.0f) m_IsGargantuan = true; } dpEntity::dpEntity(const LWOOBJID& objectID, float radius, bool isStatic) { @@ -56,7 +54,6 @@ dpEntity::dpEntity(const LWOOBJID& objectID, float radius, bool isStatic) { m_CollisionGroup = COLLISION_GROUP_ALL; m_CollisionShape = new dpShapeSphere(this, radius); - if (radius > 200.0f) m_IsGargantuan = true; } dpEntity::~dpEntity() { @@ -146,5 +143,12 @@ void dpEntity::SetAngularVelocity(const NiPoint3& newAngularVelocity) { void dpEntity::SetGrid(dpGrid* grid) { m_Grid = grid; + + if (m_CollisionShape->GetShapeType() == dpShapeType::Sphere && static_cast<dpShapeSphere*>(m_CollisionShape)->GetRadius() * 2.0f > static_cast<float>(m_Grid->CELL_SIZE)) { + m_IsGargantuan = true; + } else if (m_CollisionShape->GetShapeType() == dpShapeType::Box && static_cast<dpShapeBox*>(m_CollisionShape)->GetWidth() > static_cast<float>(m_Grid->CELL_SIZE)) { + m_IsGargantuan = true; + } + m_Grid->Add(this); } diff --git a/dPhysics/dpGrid.cpp b/dPhysics/dpGrid.cpp index 6e44ade9..c3259b51 100644 --- a/dPhysics/dpGrid.cpp +++ b/dPhysics/dpGrid.cpp @@ -6,6 +6,7 @@ dpGrid::dpGrid(int numCells, int cellSize) { NUM_CELLS = numCells; CELL_SIZE = cellSize; + m_DeleteGrid = true; //dumb method but i can't be bothered @@ -23,6 +24,7 @@ dpGrid::dpGrid(int numCells, int cellSize) { } dpGrid::~dpGrid() { + if (!this->m_DeleteGrid) return; for (auto& x : m_Cells) { //x for (auto& y : x) { //y for (auto en : y) { @@ -41,8 +43,8 @@ void dpGrid::Add(dpEntity* entity) { if (cellX < 0) cellX = 0; if (cellZ < 0) cellZ = 0; - if (cellX > NUM_CELLS) cellX = NUM_CELLS; - if (cellZ > NUM_CELLS) cellZ = NUM_CELLS; + if (cellX >= NUM_CELLS) cellX = NUM_CELLS - 1; + if (cellZ >= NUM_CELLS) cellZ = NUM_CELLS - 1; //Add to cell: m_Cells[cellX][cellZ].push_front(entity); @@ -61,13 +63,13 @@ void dpGrid::Move(dpEntity* entity, float x, float z) { if (cellX < 0) cellX = 0; if (cellZ < 0) cellZ = 0; - if (cellX > NUM_CELLS) cellX = NUM_CELLS; - if (cellZ > NUM_CELLS) cellZ = NUM_CELLS; + if (cellX >= NUM_CELLS) cellX = NUM_CELLS - 1; + if (cellZ >= NUM_CELLS) cellZ = NUM_CELLS - 1; if (oldCellX < 0) oldCellX = 0; if (oldCellZ < 0) oldCellZ = 0; - if (oldCellX > NUM_CELLS) oldCellX = NUM_CELLS; - if (oldCellZ > NUM_CELLS) oldCellZ = NUM_CELLS; + if (oldCellX >= NUM_CELLS) oldCellX = NUM_CELLS - 1; + if (oldCellZ >= NUM_CELLS) oldCellZ = NUM_CELLS - 1; if (oldCellX == cellX && oldCellZ == cellZ) return; @@ -85,8 +87,8 @@ void dpGrid::Delete(dpEntity* entity) { if (oldCellX < 0) oldCellX = 0; if (oldCellZ < 0) oldCellZ = 0; - if (oldCellX > NUM_CELLS) oldCellX = NUM_CELLS; - if (oldCellZ > NUM_CELLS) oldCellZ = NUM_CELLS; + if (oldCellX >= NUM_CELLS) oldCellX = NUM_CELLS - 1; + if (oldCellZ >= NUM_CELLS) oldCellZ = NUM_CELLS - 1; m_Cells[oldCellX][oldCellZ].remove(entity); diff --git a/dPhysics/dpGrid.h b/dPhysics/dpGrid.h index a10f165e..229e7449 100644 --- a/dPhysics/dpGrid.h +++ b/dPhysics/dpGrid.h @@ -23,6 +23,15 @@ public: void Update(float deltaTime); + /** + * Sets the delete grid parameter to value. When false, the grid will not clean up memory. + * + * @param value Whether or not to delete entities on deletion of the grid. + */ + void SetDeleteGrid(bool value) { this->m_DeleteGrid = value; }; + + std::vector<std::vector<std::forward_list<dpEntity*>>> GetCells() { return this->m_Cells; }; + private: void HandleEntity(dpEntity* entity, dpEntity* other); void HandleCell(int x, int z, float deltaTime); @@ -31,4 +40,5 @@ private: //cells on X, cells on Y for that X, then another vector that contains the entities within that cell. std::vector<std::vector<std::forward_list<dpEntity*>>> m_Cells; std::map<LWOOBJID, dpEntity*> m_GargantuanObjects; + bool m_DeleteGrid = true; }; diff --git a/dPhysics/dpWorld.cpp b/dPhysics/dpWorld.cpp index 510da518..70fbfa3a 100644 --- a/dPhysics/dpWorld.cpp +++ b/dPhysics/dpWorld.cpp @@ -9,7 +9,7 @@ #include "dLogger.h" #include "dConfig.h" -void dpWorld::Initialize(unsigned int zoneID) { +void dpWorld::Initialize(unsigned int zoneID, bool generateNewNavMesh) { phys_sp_tilecount = std::atoi(Game::config->GetValue("phys_sp_tilecount").c_str()); phys_sp_tilesize = std::atoi(Game::config->GetValue("phys_sp_tilesize").c_str()); @@ -21,13 +21,37 @@ void dpWorld::Initialize(unsigned int zoneID) { m_Grid = new dpGrid(phys_sp_tilecount, phys_sp_tilesize); } - m_NavMesh = new dNavMesh(zoneID); + if (generateNewNavMesh) m_NavMesh = new dNavMesh(zoneID); Game::logger->Log("dpWorld", "Physics world initialized!"); + m_ZoneID = zoneID; +} + +void dpWorld::Reload() { + if (m_Grid) { + m_Grid->SetDeleteGrid(false); + auto oldGridCells = m_Grid->GetCells(); + delete m_Grid; + m_Grid = nullptr; + + Initialize(m_ZoneID, false); + for (auto column : oldGridCells) { + for (auto row : column) { + for (auto entity : row) { + AddEntity(entity); + } + } + } + Game::logger->Log("dpWorld", "Successfully reloaded physics world!"); + } else { + Game::logger->Log("dpWorld", "No physics world to reload!"); + } } dpWorld::~dpWorld() { if (m_Grid) { + // Triple check this is true + m_Grid->SetDeleteGrid(true); delete m_Grid; m_Grid = nullptr; } @@ -103,14 +127,14 @@ bool dpWorld::ShouldUseSP(unsigned int zoneID) { // Only large maps should be added as tiling likely makes little difference on small maps. switch (zoneID) { - case 1100: // Avant Gardens - case 1200: // Nimbus Station - case 1300: // Gnarled Forest - case 1400: // Forbidden Valley - case 1800: // Crux Prime - case 1900: // Nexus Tower - case 2000: // Ninjago - return true; + case 1100: // Avant Gardens + case 1200: // Nimbus Station + case 1300: // Gnarled Forest + case 1400: // Forbidden Valley + case 1800: // Crux Prime + case 1900: // Nexus Tower + case 2000: // Ninjago + return true; } return false; diff --git a/dPhysics/dpWorld.h b/dPhysics/dpWorld.h index 45e550cb..d48435d0 100644 --- a/dPhysics/dpWorld.h +++ b/dPhysics/dpWorld.h @@ -19,7 +19,8 @@ class dpGrid; class dpWorld : public Singleton<dpWorld> { public: - void Initialize(unsigned int zoneID); + void Initialize(unsigned int zoneID, bool generateNewNavMesh = true); + void Reload(); ~dpWorld(); @@ -43,4 +44,5 @@ private: std::vector<dpEntity*> m_DynamicEntites; dNavMesh* m_NavMesh = nullptr; + uint32_t m_ZoneID = 0; }; diff --git a/dScripts/02_server/CMakeLists.txt b/dScripts/02_server/CMakeLists.txt new file mode 100644 index 00000000..1e38386f --- /dev/null +++ b/dScripts/02_server/CMakeLists.txt @@ -0,0 +1,45 @@ +set(DSCRIPTS_SOURCES_02_SERVER) + +add_subdirectory(DLU) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_DLU}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "DLU/${file}") +endforeach() + +add_subdirectory(Enemy) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "Enemy/${file}") +endforeach() + +add_subdirectory(Equipment) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_EQUIPMENT}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "Equipment/${file}") +endforeach() + +add_subdirectory(Map) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "Map/${file}") +endforeach() + +add_subdirectory(Minigame) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MINIGAME}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "Minigame/${file}") +endforeach() + +add_subdirectory(Objects) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_OBJECTS}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "Objects/${file}") +endforeach() + +add_subdirectory(Pets) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_PETS}) + set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} "Pets/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER ${DSCRIPTS_SOURCES_02_SERVER} PARENT_SCOPE) diff --git a/dScripts/02_server/DLU/CMakeLists.txt b/dScripts/02_server/DLU/CMakeLists.txt new file mode 100644 index 00000000..64d4cbbd --- /dev/null +++ b/dScripts/02_server/DLU/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_DLU + "DLUVanityNPC.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/DLU/DLUVanityNPC.cpp b/dScripts/02_server/DLU/DLUVanityNPC.cpp new file mode 100644 index 00000000..ba2c6604 --- /dev/null +++ b/dScripts/02_server/DLU/DLUVanityNPC.cpp @@ -0,0 +1,47 @@ +#include "DLUVanityNPC.h" +#include "GameMessages.h" +#include "dServer.h" +#include "VanityUtilities.h" +#include "RenderComponent.h" + +void DLUVanityNPC::OnStartup(Entity* self) { + m_NPC = VanityUtilities::GetNPC("averysumner - Destroyer of Worlds"); + + if (m_NPC == nullptr) { + return; + } + + if (self->GetVar<bool>(u"teleport")) { + self->AddTimer("setupTeleport", 15.0f); + } +} + +void DLUVanityNPC::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "setupTeleport") { + RenderComponent::PlayAnimation(self, u"interact"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportBeam", "teleportBeam"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportRings", "teleportRings"); + + self->AddTimer("teleport", 2.0f); + self->AddTimer("stopFX", 2.0f); + } else if (timerName == "stopFX") { + GameMessages::SendStopFXEffect(self, true, "teleportBeam"); + GameMessages::SendStopFXEffect(self, true, "teleportRings"); + } else if (timerName == "teleport") { + std::vector<VanityNPCLocation>& locations = m_NPC->m_Locations[Game::server->GetZoneID()]; + + selectLocation: + VanityNPCLocation& newLocation = locations[GeneralUtils::GenerateRandomNumber<size_t>(0, locations.size() - 1)]; + + if (self->GetPosition() == newLocation.m_Position) { + goto selectLocation; // cry about it + } + + self->SetPosition(newLocation.m_Position); + self->SetRotation(newLocation.m_Rotation); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportBeam", "teleportBeam"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportRings", "teleportRings"); + self->AddTimer("stopFX", 2.0f); + self->AddTimer("setupTeleport", 15.0f); + } +} diff --git a/dScripts/DLUVanityNPC.h b/dScripts/02_server/DLU/DLUVanityNPC.h similarity index 100% rename from dScripts/DLUVanityNPC.h rename to dScripts/02_server/DLU/DLUVanityNPC.h diff --git a/dScripts/02_server/Enemy/AG/BossSpiderQueenEnemyServer.cpp b/dScripts/02_server/Enemy/AG/BossSpiderQueenEnemyServer.cpp new file mode 100644 index 00000000..8d092078 --- /dev/null +++ b/dScripts/02_server/Enemy/AG/BossSpiderQueenEnemyServer.cpp @@ -0,0 +1,633 @@ +#include "BossSpiderQueenEnemyServer.h" + +#include "GeneralUtils.h" + +#include "MissionComponent.h" +#include "EntityManager.h" +#include "Entity.h" +#include "dZoneManager.h" + +#include "DestroyableComponent.h" +#include "ControllablePhysicsComponent.h" +#include "BaseCombatAIComponent.h" + +#include "GameMessages.h" +#include "SkillComponent.h" +#include "eReplicaComponentType.h" +#include "RenderComponent.h" + +#include <vector> + +//---------------------------------------------------------------- +//--On Startup, process necessary AI events +//---------------------------------------------------------------- +void BossSpiderQueenEnemyServer::OnStartup(Entity* self) { + // Make immune to stuns + //self:SetStunImmunity{ StateChangeType = "PUSH", bImmuneToStunAttack = true, bImmuneToStunMove = true, bImmuneToStunTurn = true, bImmuneToStunUseItem = true, bImmuneToStunEquip = true, bImmuneToStunInteract = true, bImmuneToStunJump = true } + + // Make immune to knockbacks and pulls + //self:SetStatusImmunity{ StateChangeType = "PUSH", bImmuneToPullToPoint = true, bImmuneToKnockback = true, bImmuneToInterrupt = true } + + //Get our components: + destroyable = static_cast<DestroyableComponent*>(self->GetComponent(eReplicaComponentType::DESTROYABLE)); + controllable = static_cast<ControllablePhysicsComponent*>(self->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); + combat = static_cast<BaseCombatAIComponent*>(self->GetComponent(eReplicaComponentType::BASE_COMBAT_AI)); + + if (!destroyable || !controllable) return; + + // Determine Spider Boss health transition thresholds + int spiderBossHealth = destroyable->GetMaxHealth(); + int transitionTickHealth = spiderBossHealth / 3; + + int Stage2HealthThreshold = spiderBossHealth - transitionTickHealth; + int Stage3HealthThreshold = spiderBossHealth - (2 * transitionTickHealth); + ThresholdTable = { Stage2HealthThreshold, Stage3HealthThreshold }; + + originRotation = controllable->GetRotation(); + combat->SetStunImmune(true); + + m_CurrentBossStage = 1; + + // Obtain faction and collision group to save for subsequent resets +} + +void BossSpiderQueenEnemyServer::OnDie(Entity* self, Entity* killer) { + if (Game::zoneManager->GetZoneID().GetMapID() == instanceZoneID) { + auto* missionComponent = killer->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) + return; + + missionComponent->CompleteMission(instanceMissionID); + } + + // There is suppose to be a 0.1 second delay here but that may be admitted? + auto* controller = Game::entityManager->GetZoneControlEntity(); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", 10, 0, 0, "", UNASSIGNED_SYSTEM_ADDRESS); + + self->SetPosition({ 10000, 0, 10000 }); + + Game::entityManager->SerializeEntity(self); + + controller->OnFireEventServerSide(self, "ClearProperty"); +} + +void BossSpiderQueenEnemyServer::WithdrawSpider(Entity* self, const bool withdraw) { + const auto withdrawn = self->GetBoolean(u"isWithdrawn"); + + if (withdrawn == withdraw) { + return; + } + + if (withdraw) { + //Move spider away from battle zone + // Disabled because we cant option the reset collition group right now + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", 10, 0, 0, "", UNASSIGNED_SYSTEM_ADDRESS); + + //First rotate for anim + NiQuaternion rot = NiQuaternion::IDENTITY; + + controllable->SetStatic(false); + + controllable->SetRotation(rot); + + controllable->SetStatic(true); + + controllable->SetDirtyPosition(true); + + rot = controllable->GetRotation(); + + Game::entityManager->SerializeEntity(self); + + auto* baseCombatAi = self->GetComponent<BaseCombatAIComponent>(); + + baseCombatAi->SetDisabled(true); + + float animTime = PlayAnimAndReturnTime(self, spiderWithdrawAnim); + float withdrawTime = animTime - 0.25f; + + combat->SetStunImmune(false); + combat->Stun(withdrawTime + 6.0f); + combat->SetStunImmune(true); + + //TODO: Set faction to -1 and set immunity + destroyable->SetFaction(-1); + destroyable->SetIsImmune(true); + Game::entityManager->SerializeEntity(self); + + self->AddTimer("WithdrawComplete", withdrawTime + 1.0f); + waitForIdle = true; + } else { + controllable->SetStatic(false); + + //Cancel all remaining timers for say idle anims: + self->CancelAllTimers(); + + auto* baseCombatAi = self->GetComponent<BaseCombatAIComponent>(); + + baseCombatAi->SetDisabled(false); + + // Move the Spider to its ground location + // preparing its stage attacks, and removing invulnerability + //destroyable->SetIsImmune(false); + + // Run the advance animation and prepare a timer for resuming AI + float animTime = PlayAnimAndReturnTime(self, spiderAdvanceAnim); + animTime += 1.f; + + float attackPause = animTime - 0.4f; + + destroyable->SetFaction(4); + destroyable->SetIsImmune(false); + + //Advance stage + m_CurrentBossStage++; + + //Reset the current wave death counter + m_DeathCounter = 0; + + Game::entityManager->SerializeEntity(self); + + // Prepare a timer for post leap attack + self->AddTimer("AdvanceAttack", attackPause); + + // Prepare a timer for post leap + self->AddTimer("AdvanceComplete", animTime); + } + + self->SetBoolean(u"isWithdrawn", withdraw); +} + +void BossSpiderQueenEnemyServer::SpawnSpiderWave(Entity* self, int spiderCount) { + // The Spider Queen Boss is withdrawing and requesting the spawn + // of a hatchling wave + + // Clamp invalid Spiderling number requests to the maximum amount of eggs available + if ((spiderCount > maxSpiderEggCnt) || (spiderCount < 0)) + spiderCount = maxSpiderEggCnt; + + // Reset our wave manager reference variables + hatchCounter = spiderCount; + hatchList = {}; + + // Run the wave manager + SpiderWaveManager(self); +} + +void BossSpiderQueenEnemyServer::SpiderWaveManager(Entity* self) { + auto SpiderEggNetworkID = self->GetI64(u"SpiderEggNetworkID"); + + std::vector<LWOOBJID> spiderEggs{}; + + auto spooders = Game::entityManager->GetEntitiesInGroup("EGG"); + for (auto spodder : spooders) { + spiderEggs.push_back(spodder->GetObjectID()); + } + + // Select a number of random spider eggs from the list equal to the + // current number needed to complete the current wave + if (!spiderEggs.empty()) { + for (int i = 0; i < hatchCounter; i++) { + // Select a random spider egg + auto randomEggLoc = GeneralUtils::GenerateRandomNumber<int>(0, spiderEggs.size() - 1); + auto randomEgg = spiderEggs[randomEggLoc]; + + //Just a quick check to try and prevent dupes: + for (auto en : hatchList) { + if (en == randomEgg) { + randomEggLoc++; + randomEgg = spiderEggs[randomEggLoc]; + } + } + + if (randomEgg) { + auto* eggEntity = Game::entityManager->GetEntity(randomEgg); + + if (eggEntity == nullptr) { + continue; + } + + // Prep the selected spider egg + eggEntity->OnFireEventServerSide(self, "prepEgg"); + + // Add the prepped egg to our hatchList + hatchList.push_back(eggEntity->GetObjectID()); + + // Decrement the hatchCounter + hatchCounter = hatchCounter - 1; + } + + // Remove it from our spider egg list + spiderEggs[randomEggLoc] = LWOOBJID_EMPTY; + + if (spiderEggs.size() <= 0 || (hatchCounter <= 0)) { + break; + } + } + } + + if (hatchCounter > 0) { + // We still have more eggs to hatch, poll the SpiderWaveManager again + self->AddTimer("PollSpiderWaveManager", 1.0f); + + } else { + // We have successfully readied a full wave + // initiate hatching! + for (auto egg : hatchList) { + auto* eggEntity = Game::entityManager->GetEntity(egg); + + if (eggEntity == nullptr) { + continue; + } + + eggEntity->OnFireEventServerSide(self, "hatchEgg"); + + auto time = PlayAnimAndReturnTime(self, spiderWithdrawIdle); + combat->SetStunImmune(false); + combat->Stun(time += 6.0f); + combat->SetStunImmune(true); + + self->AddTimer("checkForSpiders", 6.0f); + + } + + hatchList.clear(); + + } + +} + +void BossSpiderQueenEnemyServer::ToggleForSpecial(Entity* self, const bool state) { + self->SetBoolean(u"stoppedFlag", state); + + combat->SetDisabled(state); +} + +void BossSpiderQueenEnemyServer::RunRainOfFire(Entity* self) { + if (self->GetBoolean(u"stoppedFlag")) { + self->AddTimer("ROF", GeneralUtils::GenerateRandomNumber<float>(10, 20)); + + return; + } + + ToggleForSpecial(self, true); + + impactList.clear(); + + auto index = 0u; + for (const auto& rofGroup : ROFTargetGroupIDTable) { + const auto spawners = Game::zoneManager->GetSpawnersInGroup(rofGroup); + + std::vector<LWOOBJID> spawned; + + for (auto* spawner : spawners) { + for (const auto* node : spawner->m_Info.nodes) { + spawned.insert(spawned.end(), node->entities.begin(), node->entities.end()); + } + } + + if (index == 0) { + impactList.insert(impactList.end(), spawned.begin(), spawned.end()); + } else { + const auto randomIndex = GeneralUtils::GenerateRandomNumber<int32_t>(0, spawned.size() - 1); + + impactList.push_back(spawned[randomIndex]); + } + + index++; + } + + const auto animTime = PlayAnimAndReturnTime(self, spiderROFAnim); + + self->AddTimer("StartROF", animTime); +} + +void BossSpiderQueenEnemyServer::RainOfFireManager(Entity* self) { + if (!impactList.empty()) { + auto* entity = Game::entityManager->GetEntity(impactList[0]); + + impactList.erase(impactList.begin()); + + if (entity == nullptr) { + Game::logger->Log("BossSpiderQueenEnemyServer", "Failed to find impact!"); + + return; + } + + auto* skillComponent = entity->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + Game::logger->Log("BossSpiderQueenEnemyServer", "Failed to find impact skill component!"); + + return; + } + + skillComponent->CalculateBehavior(1376, 32168, LWOOBJID_EMPTY, true); + + self->AddTimer("PollROFManager", 0.5f); + + return; + } + + ToggleForSpecial(self, false); + + self->AddTimer("ROF", GeneralUtils::GenerateRandomNumber<float>(20, 40)); +} + +void BossSpiderQueenEnemyServer::RapidFireShooterManager(Entity* self) { + if (attackTargetTable.empty()) { + const auto animationTime = PlayAnimAndReturnTime(self, spiderJeerAnim); + + self->AddTimer("RFSTauntComplete", animationTime); + + ToggleForSpecial(self, false); + + return; + } + + const auto target = attackTargetTable[0]; + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + skillComponent->CalculateBehavior(1394, 32612, target, true); + + attackTargetTable.erase(attackTargetTable.begin()); + + self->AddTimer("PollRFSManager", 0.3f); +} + +void BossSpiderQueenEnemyServer::RunRapidFireShooter(Entity* self) { + const auto targets = self->GetTargetsInPhantom(); + + if (self->GetBoolean(u"stoppedFlag")) { + self->AddTimer("RFS", GeneralUtils::GenerateRandomNumber<float>(5, 10)); + + return; + } + + if (targets.empty()) { + Game::logger->Log("BossSpiderQueenEnemyServer", "Failed to find RFS targets"); + + self->AddTimer("RFS", GeneralUtils::GenerateRandomNumber<float>(5, 10)); + + return; + } + + ToggleForSpecial(self, true); + + const auto randomTarget = GeneralUtils::GenerateRandomNumber<int32_t>(0, targets.size() - 1); + + auto attackFocus = targets[randomTarget]; + + attackTargetTable.push_back(attackFocus); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + skillComponent->CalculateBehavior(1480, 36652, attackFocus, true); + + RapidFireShooterManager(self); + + PlayAnimAndReturnTime(self, spiderSingleShot); + + self->AddTimer("RFS", GeneralUtils::GenerateRandomNumber<float>(10, 15)); +} + +void BossSpiderQueenEnemyServer::OnTimerDone(Entity* self, const std::string timerName) { + if (timerName == "PollSpiderWaveManager") { + //Call the manager again to attempt to finish prepping a Spiderling wave + //Run the wave manager + SpiderWaveManager(self); + } else if (timerName == "disableWaitForIdle") { waitForIdle = false; } else if (timerName == "checkForSpiders") { + //Don't do anything if we ain't withdrawn: + const auto withdrawn = self->GetBoolean(u"isWithdrawn"); + if (!withdrawn) return; + + NiQuaternion rot = NiQuaternion::IDENTITY; + + //First rotate for anim + controllable->SetStatic(false); + controllable->SetRotation(rot); + controllable->SetStatic(true); + Game::entityManager->SerializeEntity(self); + + //Play the Spider Boss' mountain idle anim + auto time = PlayAnimAndReturnTime(self, spiderWithdrawIdle); + combat->SetStunImmune(false); + combat->Stun(time); + combat->SetStunImmune(true); + + rot = controllable->GetRotation(); + + //If there are still baby spiders, don't do anyhting either + const auto spiders = Game::entityManager->GetEntitiesInGroup("BabySpider"); + if (spiders.size() > 0) + self->AddTimer("checkForSpiders", time); + else + WithdrawSpider(self, false); + } else if (timerName == "PollROFManager") { + //Call the manager again to attempt to initiate an impact on another random location + //Run the ROF Manager + RainOfFireManager(self); + + } else if (timerName == "PollRFSManager") { + //Call the manager again to attempt to initiate a rapid fire shot at the next sequential target + //Run the ROF Manager + RapidFireShooterManager(self); + + } else if (timerName == "StartROF") { + //Re-enable Spider Boss + //ToggleForSpecial(self, false); + + RainOfFireManager(self); + + } else if (timerName == "PollSpiderSkillManager") { + //Call the skill manager again to attempt to run the current Spider Boss + //stage's special attack again + //SpiderSkillManager(self, true); + PlayAnimAndReturnTime(self, spiderJeerAnim); + + } else if (timerName == "RFS") { + RunRapidFireShooter(self); + } else if (timerName == "ROF") { + RunRainOfFire(self); + } else if (timerName == "RFSTauntComplete") { + //Determine an appropriate random time to check our manager again + // local spiderCooldownDelay = math.random(s1DelayMin, s1DelayMax) + + //Set a timer based on our random cooldown determination + //to pulse the SpiderSkillManager again + + //GAMEOBJ:GetTimer():AddTimerWithCancel(spiderCooldownDelay, "PollSpiderSkillManager", self) + + //Re-enable Spider Boss + //ToggleForSpecial(self, false); + + } else if (timerName == "WithdrawComplete") { + //Play the Spider Boss' mountain idle anim + PlayAnimAndReturnTime(self, spiderWithdrawIdle); + + //The Spider Boss has retreated, hatch a wave! + int currentStage = m_CurrentBossStage; + + //Prepare a Spiderling wave and initiate egg hatch events + //self->SetVar(u"SpiderWaveCount", ) + + //TODO: Actually spawn the spiders here + hatchCounter = 2; + if (currentStage > 1) hatchCounter++; + + SpawnSpiderWave(self, spiderWaveCntTable[currentStage - 1]); + + } else if (timerName == "AdvanceAttack") { + //TODO: Can we even do knockbacks yet? @Wincent01 + // Yes ^ + + //Fire the melee smash skill to throw players back + /*local landingTarget = self:GetVar("LandingTarget") or false + + if((landingTarget) and (landingTarget:Exists())) { + local advSmashFlag = landingTarget:CastSkill{skillID = bossLandingSkill} + landingTarget:PlayEmbeddedEffectOnAllClientsNearObject{radius = 100, fromObjectID = landingTarget, effectName = "camshake-bridge"} + }*/ + + auto landingTarget = self->GetI64(u"LandingTarget"); + auto landingEntity = Game::entityManager->GetEntity(landingTarget); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(bossLandingSkill, 37739, LWOOBJID_EMPTY); + } + + if (landingEntity) { + auto* landingSkill = landingEntity->GetComponent<SkillComponent>(); + + if (landingSkill != nullptr) { + landingSkill->CalculateBehavior(bossLandingSkill, 37739, LWOOBJID_EMPTY, true); + } + } + + GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(self, u"camshake-bridge", self->GetObjectID(), 100.0f); + + } else if (timerName == "AdvanceComplete") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", 11, 0, 0, "", UNASSIGNED_SYSTEM_ADDRESS); + + //Wind up, telegraphing next round + float animTime = PlayAnimAndReturnTime(self, spiderJeerAnim); + self->AddTimer("AdvanceTauntComplete", animTime); + + } else if (timerName == "AdvanceTauntComplete") { + + //Declare a default special Spider Boss skill cooldown + int spiderCooldownDelay = 10; + + if (m_CurrentBossStage == 2) { + spiderCooldownDelay = GeneralUtils::GenerateRandomNumber<int>(s1DelayMin, s1DelayMax); + } else if (m_CurrentBossStage == 3) { + spiderCooldownDelay = GeneralUtils::GenerateRandomNumber<int>(s2DelayMin, s2DelayMax); + } + + //Set a timer based on our random cooldown determination + //to pulse the SpiderSkillManager + self->AddTimer("PollSpiderSkillManager", spiderCooldownDelay); + + //Remove current status immunity + /*self:SetStatusImmunity{ StateChangeType = "POP", bImmuneToSpeed = true, bImmuneToBasicAttack = true, bImmuneToDOT = true} + + self:SetStunned{StateChangeType = "POP", + bCantMove = true, + bCantJump = true, + bCantTurn = true, + bCantAttack = true, + bCantUseItem = true, + bCantEquip = true, + bCantInteract = true, + bIgnoreImmunity = true}*/ + + destroyable->SetIsImmune(false); + destroyable->SetFaction(4); + + Game::entityManager->SerializeEntity(self); + + } else if (timerName == "Clear") { + Game::entityManager->FireEventServerSide(self, "ClearProperty"); + self->CancelAllTimers(); + } else if (timerName == "UnlockSpecials") { + //We no longer need to lock specials + self->SetBoolean(u"bSpecialLock", false); + + //Did we queue a spcial attack? + if (self->GetBoolean(u"bSpecialQueued")) { + self->SetBoolean(u"bSpecialQueued", false); + } + } +} + +void BossSpiderQueenEnemyServer::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + if (m_CurrentBossStage > 0 && !self->HasTimer("RFS")) { + self->AddTimer("RFS", 5.0f); + } + + if (m_CurrentBossStage > 0 && !self->HasTimer("ROF")) { + self->AddTimer("ROF", 10.0f); + } + + if (m_CurrentBossStage > ThresholdTable.size()) { + return; + } + + int currentThreshold = ThresholdTable[m_CurrentBossStage - 1]; + + if (destroyable->GetHealth() <= currentThreshold) { + auto isWithdrawn = self->GetBoolean(u"isWithdrawn"); + + if (!isWithdrawn) { + self->CancelAllTimers(); + + self->SetBoolean(u"isSpecialAttacking", false); + self->SetBoolean(u"bSpecialLock", false); + + WithdrawSpider(self, true); + } + } +} + +void BossSpiderQueenEnemyServer::OnUpdate(Entity* self) { + auto isWithdrawn = self->GetBoolean(u"isWithdrawn"); + + if (!isWithdrawn) return; + + if (controllable->GetRotation() == NiQuaternion::IDENTITY) { + return; + } + + controllable->SetStatic(false); + controllable->SetRotation(NiQuaternion::IDENTITY); + controllable->SetStatic(true); + + Game::entityManager->SerializeEntity(self); +} + +//---------------------------------------------- +//--Utility function capable of playing a priority +//-- animation on a targetand returning either the +//-- anim time, or a desired default +//---------------------------------------------- +float BossSpiderQueenEnemyServer::PlayAnimAndReturnTime(Entity* self, const std::u16string& animID) { + //TODO: Get the actual animation time + + // Get the anim time + float animTimer = RenderComponent::GetAnimationTime(self, animID); + + // If we have an animation play it + if (animTimer > 0) { + animTimer = RenderComponent::PlayAnimation(self, animID); + } + + // If the anim time is less than the the default time use default + if (animTimer < defaultAnimPause) { + animTimer = defaultAnimPause; + } + + return animTimer; +} diff --git a/dScripts/BossSpiderQueenEnemyServer.h b/dScripts/02_server/Enemy/AG/BossSpiderQueenEnemyServer.h similarity index 100% rename from dScripts/BossSpiderQueenEnemyServer.h rename to dScripts/02_server/Enemy/AG/BossSpiderQueenEnemyServer.h diff --git a/dScripts/02_server/Enemy/AG/CMakeLists.txt b/dScripts/02_server/Enemy/AG/CMakeLists.txt new file mode 100644 index 00000000..865d4c3c --- /dev/null +++ b/dScripts/02_server/Enemy/AG/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_AG + "BossSpiderQueenEnemyServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Enemy/AM/AmDarklingDragon.cpp b/dScripts/02_server/Enemy/AM/AmDarklingDragon.cpp new file mode 100644 index 00000000..9895395c --- /dev/null +++ b/dScripts/02_server/Enemy/AM/AmDarklingDragon.cpp @@ -0,0 +1,158 @@ +#include "AmDarklingDragon.h" +#include "BaseCombatAIComponent.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "SkillComponent.h" +#include "BaseCombatAIComponent.h" +#include "EntityInfo.h" +#include "eAninmationFlags.h" +#include "RenderComponent.h" + +void AmDarklingDragon::OnStartup(Entity* self) { + self->SetVar<int32_t>(u"weakspot", 0); + + auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + + if (baseCombatAIComponent != nullptr) { + baseCombatAIComponent->SetStunImmune(true); + } +} + +void AmDarklingDragon::OnDie(Entity* self, Entity* killer) { + if (self->GetVar<bool>(u"bDied")) { + return; + } + + self->SetVar<bool>(u"bDied", true); + + auto golemId = self->GetVar<LWOOBJID>(u"Golem"); + + auto* golem = Game::entityManager->GetEntity(golemId); + + if (golem != nullptr) { + golem->Smash(self->GetObjectID()); + } +} + +void AmDarklingDragon::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + GameMessages::SendPlayFXEffect(self, -1, u"gothit", "", LWOOBJID_EMPTY, 1, 1, true); + + if (true) { + auto weakpoint = self->GetVar<int32_t>(u"weakspot"); + + if (weakpoint == 1) { + self->Smash(attacker->GetObjectID()); + } + } + + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + + if (destroyableComponent != nullptr) { + if (destroyableComponent->GetArmor() > 0) return; + + auto weakpoint = self->GetVar<int32_t>(u"weakpoint"); + + if (weakpoint == 0) { + self->AddTimer("ReviveTimer", 12); + + auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (baseCombatAIComponent != nullptr) { + baseCombatAIComponent->SetDisabled(true); + baseCombatAIComponent->SetStunned(true); + } + + if (skillComponent != nullptr) { + skillComponent->Interrupt(); + skillComponent->Reset(); + } + + self->SetVar<int32_t>(u"weakpoint", 2); + GameMessages::SendChangeIdleFlags(self->GetObjectID(), eAnimationFlags::IDLE_NONE, eAnimationFlags::IDLE_COMBAT, UNASSIGNED_SYSTEM_ADDRESS); + float animationTime = RenderComponent::PlayAnimation(self, u"stunstart", 1.7f); + + self->AddTimer("timeToStunLoop", 1.0f); + + auto position = self->GetPosition(); + auto forward = self->GetRotation().GetForwardVector(); + auto backwards = forward * -1; + + forward.x *= 10; + forward.z *= 10; + + auto rotation = self->GetRotation(); + + auto objectPosition = NiPoint3(); + + objectPosition.y = position.y; + objectPosition.x = position.x - (backwards.x * 8); + objectPosition.z = position.z - (backwards.z * 8); + + auto golem = self->GetVar<int32_t>(u"DragonSmashingGolem"); + + EntityInfo info{}; + info.lot = golem != 0 ? golem : 8340; + info.pos = objectPosition; + info.rot = rotation; + info.spawnerID = self->GetObjectID(); + info.settings = { + new LDFData<std::string>(u"rebuild_activators", + std::to_string(objectPosition.x + forward.x) + "\x1f" + + std::to_string(objectPosition.y) + "\x1f" + + std::to_string(objectPosition.z + forward.z) + ), + new LDFData<int32_t>(u"respawn", 100000), + new LDFData<float>(u"rebuild_reset_time", 15), + new LDFData<bool>(u"no_timed_spawn", true), + new LDFData<LWOOBJID>(u"Dragon", self->GetObjectID()) + }; + + auto* golemObject = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(golemObject); + } + } +} + +void AmDarklingDragon::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "ReviveHeldTimer") { + self->AddTimer("backToAttack", 2.5); + } else if (timerName == "ExposeWeakSpotTimer") { + self->SetVar<int32_t>(u"weakspot", 1); + } else if (timerName == "timeToStunLoop") { + RenderComponent::PlayAnimation(self, u"stunloop", 1.8f); + } else if (timerName == "ReviveTimer") { + RenderComponent::PlayAnimation(self, u"stunend", 2.0f); + self->AddTimer("backToAttack", 1); + } else if (timerName == "backToAttack") { + auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (baseCombatAIComponent != nullptr) { + baseCombatAIComponent->SetDisabled(false); + baseCombatAIComponent->SetStunned(false); + } + if (skillComponent != nullptr) { + skillComponent->Interrupt(); + skillComponent->Reset(); + } + GameMessages::SendChangeIdleFlags(self->GetObjectID(), eAnimationFlags::IDLE_COMBAT, eAnimationFlags::IDLE_NONE, UNASSIGNED_SYSTEM_ADDRESS); + self->SetVar<int32_t>(u"weakspot", -1); + GameMessages::SendNotifyObject(self->GetObjectID(), self->GetObjectID(), u"DragonRevive", UNASSIGNED_SYSTEM_ADDRESS); + } +} + +void AmDarklingDragon::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + if (args != "rebuildDone") return; + + self->AddTimer("ExposeWeakSpotTimer", 3.8f); + + self->CancelTimer("ReviveTimer"); + + self->AddTimer("ReviveHeldTimer", 10.5f); + + self->SetVar<LWOOBJID>(u"Golem", sender->GetObjectID()); + + RenderComponent::PlayAnimation(self, u"quickbuildhold", 1.9f); +} diff --git a/dScripts/AmDarklingDragon.h b/dScripts/02_server/Enemy/AM/AmDarklingDragon.h similarity index 100% rename from dScripts/AmDarklingDragon.h rename to dScripts/02_server/Enemy/AM/AmDarklingDragon.h diff --git a/dScripts/AmDarklingMech.cpp b/dScripts/02_server/Enemy/AM/AmDarklingMech.cpp similarity index 100% rename from dScripts/AmDarklingMech.cpp rename to dScripts/02_server/Enemy/AM/AmDarklingMech.cpp diff --git a/dScripts/AmDarklingMech.h b/dScripts/02_server/Enemy/AM/AmDarklingMech.h similarity index 100% rename from dScripts/AmDarklingMech.h rename to dScripts/02_server/Enemy/AM/AmDarklingMech.h diff --git a/dScripts/AmSkeletonEngineer.cpp b/dScripts/02_server/Enemy/AM/AmSkeletonEngineer.cpp similarity index 100% rename from dScripts/AmSkeletonEngineer.cpp rename to dScripts/02_server/Enemy/AM/AmSkeletonEngineer.cpp diff --git a/dScripts/AmSkeletonEngineer.h b/dScripts/02_server/Enemy/AM/AmSkeletonEngineer.h similarity index 100% rename from dScripts/AmSkeletonEngineer.h rename to dScripts/02_server/Enemy/AM/AmSkeletonEngineer.h diff --git a/dScripts/02_server/Enemy/AM/CMakeLists.txt b/dScripts/02_server/Enemy/AM/CMakeLists.txt new file mode 100644 index 00000000..dd20edb0 --- /dev/null +++ b/dScripts/02_server/Enemy/AM/CMakeLists.txt @@ -0,0 +1,5 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_AM + "AmDarklingMech.cpp" + "AmSkeletonEngineer.cpp" + "AmDarklingDragon.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Enemy/CMakeLists.txt b/dScripts/02_server/Enemy/CMakeLists.txt new file mode 100644 index 00000000..408ff733 --- /dev/null +++ b/dScripts/02_server/Enemy/CMakeLists.txt @@ -0,0 +1,45 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY) + +add_subdirectory(AG) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_AG}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "AG/${file}") +endforeach() + +add_subdirectory(AM) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_AM}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "AM/${file}") +endforeach() + +add_subdirectory(FV) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_FV}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "FV/${file}") +endforeach() + +add_subdirectory(General) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_GENERAL}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "General/${file}") +endforeach() + +add_subdirectory(Survival) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_SURVIVAL}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "Survival/${file}") +endforeach() + +add_subdirectory(VE) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_VE}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "VE/${file}") +endforeach() + +add_subdirectory(Waves) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_ENEMY_WAVES}) + set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} "Waves/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY ${DSCRIPTS_SOURCES_02_SERVER_ENEMY} PARENT_SCOPE) diff --git a/dScripts/02_server/Enemy/FV/CMakeLists.txt b/dScripts/02_server/Enemy/FV/CMakeLists.txt new file mode 100644 index 00000000..5146d8d8 --- /dev/null +++ b/dScripts/02_server/Enemy/FV/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_FV + "FvMaelstromCavalry.cpp" + "FvMaelstromDragon.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Enemy/FV/FvMaelstromCavalry.cpp b/dScripts/02_server/Enemy/FV/FvMaelstromCavalry.cpp new file mode 100644 index 00000000..84eb70e7 --- /dev/null +++ b/dScripts/02_server/Enemy/FV/FvMaelstromCavalry.cpp @@ -0,0 +1,30 @@ +#include "FvMaelstromCavalry.h" +#include "EntityManager.h" + +void FvMaelstromCavalry::OnStartup(Entity* self) { + for (const auto& group : self->GetGroups()) { + const auto& objects = Game::entityManager->GetEntitiesInGroup(group); + + for (auto* obj : objects) { + if (obj->GetLOT() != 8551) continue; + + obj->OnFireEventServerSide(self, "ISpawned"); + } + } +} + +void FvMaelstromCavalry::OnDie(Entity* self, Entity* killer) { + if (killer == nullptr) { + return; + } + + if (killer->GetLOT() != 8665) { + return; + } + + const auto& triggers = Game::entityManager->GetEntitiesInGroup("HorsemenTrigger"); + + for (auto* trigger : triggers) { + trigger->OnFireEventServerSide(self, "HorsemenDeath"); + } +} diff --git a/dScripts/FvMaelstromCavalry.h b/dScripts/02_server/Enemy/FV/FvMaelstromCavalry.h similarity index 100% rename from dScripts/FvMaelstromCavalry.h rename to dScripts/02_server/Enemy/FV/FvMaelstromCavalry.h diff --git a/dScripts/02_server/Enemy/FV/FvMaelstromDragon.cpp b/dScripts/02_server/Enemy/FV/FvMaelstromDragon.cpp new file mode 100644 index 00000000..ff7e7a51 --- /dev/null +++ b/dScripts/02_server/Enemy/FV/FvMaelstromDragon.cpp @@ -0,0 +1,179 @@ +#include "FvMaelstromDragon.h" +#include "EntityManager.h" +#include "SkillComponent.h" +#include "BaseCombatAIComponent.h" +#include "DestroyableComponent.h" +#include "eAninmationFlags.h" +#include "EntityInfo.h" +#include "RenderComponent.h" + +void FvMaelstromDragon::OnStartup(Entity* self) { + self->SetVar<int32_t>(u"weakspot", 0); + + auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + + if (baseCombatAIComponent != nullptr) { + baseCombatAIComponent->SetStunImmune(true); + } +} + +void FvMaelstromDragon::OnDie(Entity* self, Entity* killer) { + if (self->GetVar<bool>(u"bDied")) { + return; + } + + self->SetVar<bool>(u"bDied", true); + + auto position = self->GetPosition(); + auto rotation = self->GetRotation(); + + auto chestObject = 11229; + + EntityInfo info{}; + info.lot = chestObject; + info.pos = position; + info.rot = rotation; + info.spawnerID = self->GetObjectID(); + + auto* chest = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(chest); + + auto golemId = self->GetVar<LWOOBJID>(u"Golem"); + + auto* golem = Game::entityManager->GetEntity(golemId); + + if (golem != nullptr) { + golem->Smash(self->GetObjectID()); + } +} + +void FvMaelstromDragon::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + GameMessages::SendPlayFXEffect(self, -1, u"gothit", "", LWOOBJID_EMPTY, 1, 1, true); + + if (true) { + auto weakpoint = self->GetVar<int32_t>(u"weakspot"); + + if (weakpoint == 1) { + self->Smash(attacker->GetObjectID()); + } + } + + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + + if (destroyableComponent != nullptr) { + if (destroyableComponent->GetArmor() > 0) return; + + auto weakpoint = self->GetVar<int32_t>(u"weakpoint"); + + if (weakpoint == 0) { + Game::logger->Log("FvMaelstromDragon", "Activating weakpoint"); + + self->AddTimer("ReviveTimer", 12); + + auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (baseCombatAIComponent != nullptr) { + baseCombatAIComponent->SetDisabled(true); + baseCombatAIComponent->SetStunned(true); + } + + if (skillComponent != nullptr) { + skillComponent->Interrupt(); + skillComponent->Reset(); + } + + self->SetVar<int32_t>(u"weakpoint", 2); + + GameMessages::SendChangeIdleFlags(self->GetObjectID(), eAnimationFlags::IDLE_NONE, eAnimationFlags::IDLE_COMBAT, UNASSIGNED_SYSTEM_ADDRESS); + RenderComponent::PlayAnimation(self, u"stunstart", 1.7f); + + self->AddTimer("timeToStunLoop", 1.0f); + + auto position = self->GetPosition(); + auto forward = self->GetRotation().GetForwardVector(); + auto backwards = forward * -1; + + forward.x *= 10; + forward.z *= 10; + + auto rotation = self->GetRotation(); + + auto objectPosition = NiPoint3(); + + objectPosition.y = position.y; + objectPosition.x = position.x - (backwards.x * 8); + objectPosition.z = position.z - (backwards.z * 8); + + auto golem = self->GetVar<int32_t>(u"DragonSmashingGolem"); + + EntityInfo info{}; + info.lot = golem != 0 ? golem : 8340; + info.pos = objectPosition; + info.rot = rotation; + info.spawnerID = self->GetObjectID(); + info.settings = { + new LDFData<std::string>(u"rebuild_activators", + std::to_string(objectPosition.x + forward.x) + "\x1f" + + std::to_string(objectPosition.y) + "\x1f" + + std::to_string(objectPosition.z + forward.z) + ), + new LDFData<int32_t>(u"respawn", 100000), + new LDFData<float>(u"rebuild_reset_time", 15), + new LDFData<bool>(u"no_timed_spawn", true), + new LDFData<LWOOBJID>(u"Dragon", self->GetObjectID()) + }; + + auto* golemObject = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(golemObject); + } + } +} + +void FvMaelstromDragon::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "ReviveHeldTimer") { + self->AddTimer("backToAttack", 2.5); + } else if (timerName == "ExposeWeakSpotTimer") { + self->SetVar<int32_t>(u"weakspot", 1); + } else if (timerName == "timeToStunLoop") { + RenderComponent::PlayAnimation(self, u"stunloop", 1.8f); + } else if (timerName == "ReviveTimer") { + RenderComponent::PlayAnimation(self, u"stunend", 2.0f); + self->AddTimer("backToAttack", 1.0f); + } else if (timerName == "backToAttack") { + auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (baseCombatAIComponent != nullptr) { + baseCombatAIComponent->SetDisabled(false); + baseCombatAIComponent->SetStunned(false); + } + + if (skillComponent != nullptr) { + skillComponent->Interrupt(); + skillComponent->Reset(); + } + GameMessages::SendChangeIdleFlags(self->GetObjectID(), eAnimationFlags::IDLE_COMBAT, eAnimationFlags::IDLE_NONE, UNASSIGNED_SYSTEM_ADDRESS); + self->SetVar<int32_t>(u"weakspot", -1); + + GameMessages::SendNotifyObject(self->GetObjectID(), self->GetObjectID(), u"DragonRevive", UNASSIGNED_SYSTEM_ADDRESS); + } +} + +void +FvMaelstromDragon::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args != "rebuildDone") return; + + self->AddTimer("ExposeWeakSpotTimer", 3.8f); + + self->CancelTimer("ReviveTimer"); + + self->AddTimer("ReviveHeldTimer", 10.5f); + + self->SetVar<LWOOBJID>(u"Golem", sender->GetObjectID()); + + RenderComponent::PlayAnimation(self, u"quickbuildhold", 1.9f); +} diff --git a/dScripts/FvMaelstromDragon.h b/dScripts/02_server/Enemy/FV/FvMaelstromDragon.h similarity index 100% rename from dScripts/FvMaelstromDragon.h rename to dScripts/02_server/Enemy/FV/FvMaelstromDragon.h diff --git a/dScripts/02_server/Enemy/General/BaseEnemyApe.cpp b/dScripts/02_server/Enemy/General/BaseEnemyApe.cpp new file mode 100644 index 00000000..07221248 --- /dev/null +++ b/dScripts/02_server/Enemy/General/BaseEnemyApe.cpp @@ -0,0 +1,142 @@ +#include "BaseEnemyApe.h" +#include "BaseCombatAIComponent.h" +#include "DestroyableComponent.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "SkillComponent.h" +#include "eAninmationFlags.h" +#include "RenderComponent.h" +#include "eStateChangeType.h" + +void BaseEnemyApe::OnStartup(Entity* self) { + self->SetVar<uint32_t>(u"timesStunned", 2); + self->SetVar<bool>(u"knockedOut", false); +} + +void BaseEnemyApe::OnDie(Entity* self, Entity* killer) { + auto* anchor = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"QB")); + if (anchor != nullptr && !anchor->GetIsDead()) { + anchor->Smash(self->GetObjectID(), eKillType::SILENT); + } +} + +void BaseEnemyApe::OnSkillCast(Entity* self, uint32_t skillID) { + const auto groundPoundSkill = self->GetVar<uint32_t>(u"GroundPoundSkill") != 0 ? self->GetVar<uint32_t>(u"GroundPoundSkill") : 725; + const auto spawnQuickBuildTime = self->GetVar<float_t>(u"spawnQBTime") != 0.0f ? self->GetVar<float_t>(u"spawnQBTime") : 5.0f; + + if (skillID == groundPoundSkill && self->GetVar<LWOOBJID>(u"QB") == LWOOBJID_EMPTY) { + self->AddTimer("spawnQBTime", spawnQuickBuildTime); + } +} + +void BaseEnemyApe::OnHit(Entity* self, Entity* attacker) { + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr && destroyableComponent->GetArmor() < 1 && !self->GetBoolean(u"knockedOut")) { + StunApe(self, true); + self->CancelTimer("spawnQBTime"); + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent) { + skillComponent->Reset(); + } + RenderComponent::PlayAnimation(self, u"disable", 1.7f); + GameMessages::SendChangeIdleFlags(self->GetObjectID(), eAnimationFlags::IDLE_NONE, eAnimationFlags::IDLE_COMBAT, UNASSIGNED_SYSTEM_ADDRESS); + const auto reviveTime = self->GetVar<float_t>(u"reviveTime") != 0.0f + ? self->GetVar<float_t>(u"reviveTime") : 12.0f; + self->AddTimer("reviveTime", reviveTime); + } +} + +void BaseEnemyApe::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "reviveTime") { + + // Revives the ape, giving it back some armor + const auto timesStunned = self->GetVar<uint32_t>(u"timesStunned"); + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->SetArmor(destroyableComponent->GetMaxArmor() / timesStunned); + } + Game::entityManager->SerializeEntity(self); + GameMessages::SendChangeIdleFlags(self->GetObjectID(), eAnimationFlags::IDLE_COMBAT, eAnimationFlags::IDLE_NONE, UNASSIGNED_SYSTEM_ADDRESS); + self->SetVar<uint32_t>(u"timesStunned", timesStunned + 1); + StunApe(self, false); + + } else if (timerName == "spawnQBTime" && self->GetVar<LWOOBJID>(u"QB") == LWOOBJID_EMPTY) { + // Spawn QB in front of ape. + const auto position = self->GetPosition(); + const auto rotation = self->GetRotation(); + + const auto backwardVector = rotation.GetForwardVector() * -1; + const auto objectPosition = NiPoint3( + position.GetX() - (backwardVector.GetX() * 8), + position.GetY(), + position.GetZ() - (backwardVector.GetZ() * 8) + ); + + EntityInfo entityInfo{}; + + entityInfo.pos = position; + entityInfo.rot = rotation; + entityInfo.pos.SetY(entityInfo.pos.GetY() + 13.0f); + + entityInfo.spawnerID = self->GetObjectID(); + entityInfo.lot = self->GetVar<LOT>(u"QuickbuildAnchorLOT") != 0 + ? self->GetVar<LOT>(u"QuickbuildAnchorLOT") : 7549; + entityInfo.settings = { + new LDFData<std::string>(u"rebuild_activators", + std::to_string(objectPosition.GetX()) + "\x1f" + + std::to_string(objectPosition.GetY()) + "\x1f" + + std::to_string(objectPosition.GetZ()) + ), + new LDFData<bool>(u"no_timed_spawn", true), + new LDFData<LWOOBJID>(u"ape", self->GetObjectID()) + }; + + auto* anchor = Game::entityManager->CreateEntity(entityInfo); + Game::entityManager->ConstructEntity(anchor); + self->SetVar<LWOOBJID>(u"QB", anchor->GetObjectID()); + + } else if (timerName == "anchorDamageTimer") { + + // Attacks the ape with some god skill + const auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"smasher")); + if (player == nullptr) { + return; + } + + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(1273, 29446, self->GetObjectID(), true, false, player->GetObjectID()); + } + + self->SetVar<LWOOBJID>(u"QB", LWOOBJID_EMPTY); + } +} + +void BaseEnemyApe::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args == "rebuildDone" && sender != nullptr) { + self->SetVar<LWOOBJID>(u"smasher", sender->GetObjectID()); + const auto anchorDamageDelayTime = self->GetVar<float_t>(u"AnchorDamageDelayTime") != 0.0f ? self->GetVar<float_t>(u"AnchorDamageDelayTime") : 0.5f; + self->AddTimer("anchorDamageTimer", anchorDamageDelayTime); + } +} + +void BaseEnemyApe::StunApe(Entity* self, bool stunState) { + auto* combatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + if (combatAIComponent != nullptr) { + combatAIComponent->SetDisabled(stunState); + combatAIComponent->SetStunned(stunState); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent != nullptr) { + skillComponent->Interrupt(); + } + + GameMessages::SendSetStunned(self->GetObjectID(), stunState ? eStateChangeType::PUSH : eStateChangeType::POP, UNASSIGNED_SYSTEM_ADDRESS, self->GetObjectID(), + true, true, true, true, true, + true, true, true, true); + + self->SetBoolean(u"knockedOut", stunState); + } +} diff --git a/dScripts/BaseEnemyApe.h b/dScripts/02_server/Enemy/General/BaseEnemyApe.h similarity index 100% rename from dScripts/BaseEnemyApe.h rename to dScripts/02_server/Enemy/General/BaseEnemyApe.h diff --git a/dScripts/02_server/Enemy/General/BaseEnemyMech.cpp b/dScripts/02_server/Enemy/General/BaseEnemyMech.cpp new file mode 100644 index 00000000..c652dab0 --- /dev/null +++ b/dScripts/02_server/Enemy/General/BaseEnemyMech.cpp @@ -0,0 +1,45 @@ +#include "BaseEnemyMech.h" +#include "Entity.h" +#include "ControllablePhysicsComponent.h" +#include "EntityManager.h" +#include "dpWorld.h" +#include "EntityInfo.h" +#include "GeneralUtils.h" +#include "DestroyableComponent.h" +#include "eReplicaComponentType.h" + +void BaseEnemyMech::OnStartup(Entity* self) { + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->SetFaction(4); + } +} + +void BaseEnemyMech::OnDie(Entity* self, Entity* killer) { + ControllablePhysicsComponent* controlPhys = static_cast<ControllablePhysicsComponent*>(self->GetComponent(eReplicaComponentType::CONTROLLABLE_PHYSICS)); + if (!controlPhys) return; + + NiPoint3 newLoc = { controlPhys->GetPosition().x, dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(controlPhys->GetPosition()), controlPhys->GetPosition().z }; + + EntityInfo info = EntityInfo(); + std::vector<LDFBaseData*> cfg; + std::u16string activatorPosStr; + activatorPosStr += (GeneralUtils::to_u16string(controlPhys->GetPosition().x)); + activatorPosStr.push_back(0x1f); + activatorPosStr += (GeneralUtils::to_u16string(controlPhys->GetPosition().y)); + activatorPosStr.push_back(0x1f); + activatorPosStr += (GeneralUtils::to_u16string(controlPhys->GetPosition().z)); + + LDFBaseData* activatorPos = new LDFData<std::u16string>(u"rebuild_activators", activatorPosStr); + cfg.push_back(activatorPos); + info.lot = qbTurretLOT; + info.pos = newLoc; + info.rot = controlPhys->GetRotation(); + info.spawnerID = self->GetObjectID(); + info.settings = cfg; + + Entity* turret = Game::entityManager->CreateEntity(info, nullptr); + if (turret) { + Game::entityManager->ConstructEntity(turret); + } +} diff --git a/dScripts/BaseEnemyMech.h b/dScripts/02_server/Enemy/General/BaseEnemyMech.h similarity index 100% rename from dScripts/BaseEnemyMech.h rename to dScripts/02_server/Enemy/General/BaseEnemyMech.h diff --git a/dScripts/02_server/Enemy/General/CMakeLists.txt b/dScripts/02_server/Enemy/General/CMakeLists.txt new file mode 100644 index 00000000..5486d2b0 --- /dev/null +++ b/dScripts/02_server/Enemy/General/CMakeLists.txt @@ -0,0 +1,7 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_GENERAL + "BaseEnemyMech.cpp" + "BaseEnemyApe.cpp" + "GfApeSmashingQB.cpp" + "TreasureChestDragonServer.cpp" + "EnemyNjBuff.cpp" + PARENT_SCOPE) diff --git a/dScripts/EnemyNjBuff.cpp b/dScripts/02_server/Enemy/General/EnemyNjBuff.cpp similarity index 100% rename from dScripts/EnemyNjBuff.cpp rename to dScripts/02_server/Enemy/General/EnemyNjBuff.cpp diff --git a/dScripts/EnemyNjBuff.h b/dScripts/02_server/Enemy/General/EnemyNjBuff.h similarity index 100% rename from dScripts/EnemyNjBuff.h rename to dScripts/02_server/Enemy/General/EnemyNjBuff.h diff --git a/dScripts/02_server/Enemy/General/GfApeSmashingQB.cpp b/dScripts/02_server/Enemy/General/GfApeSmashingQB.cpp new file mode 100644 index 00000000..aa4a56a7 --- /dev/null +++ b/dScripts/02_server/Enemy/General/GfApeSmashingQB.cpp @@ -0,0 +1,24 @@ +#include "GfApeSmashingQB.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "Entity.h" +#include "RenderComponent.h" + +void GfApeSmashingQB::OnStartup(Entity* self) { + self->SetNetworkVar<LWOOBJID>(u"lootTagOwner", self->GetVar<LWOOBJID>(u"lootTagOwner")); +} + +void GfApeSmashingQB::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "anchorBreakTime") { + self->Smash(); + } +} + +void GfApeSmashingQB::OnRebuildComplete(Entity* self, Entity* target) { + auto* ape = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"ape")); + if (ape != nullptr) { + ape->OnFireEventServerSide(target, "rebuildDone"); + RenderComponent::PlayAnimation(self, u"smash", 1.7f); + self->AddTimer("anchorBreakTime", 1.0f); + } +} diff --git a/dScripts/GfApeSmashingQB.h b/dScripts/02_server/Enemy/General/GfApeSmashingQB.h similarity index 100% rename from dScripts/GfApeSmashingQB.h rename to dScripts/02_server/Enemy/General/GfApeSmashingQB.h diff --git a/dScripts/02_server/Enemy/General/TreasureChestDragonServer.cpp b/dScripts/02_server/Enemy/General/TreasureChestDragonServer.cpp new file mode 100644 index 00000000..87416b4d --- /dev/null +++ b/dScripts/02_server/Enemy/General/TreasureChestDragonServer.cpp @@ -0,0 +1,43 @@ +#include "TreasureChestDragonServer.h" +#include "ScriptedActivityComponent.h" +#include "TeamManager.h" +#include "EntityManager.h" +#include "Loot.h" + +void TreasureChestDragonServer::OnStartup(Entity* self) { + +} + +void TreasureChestDragonServer::OnUse(Entity* self, Entity* user) { + if (self->GetVar<bool>(u"bUsed")) { + return; + } + + self->SetVar<bool>(u"bUsed", true); + + auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); + + if (scriptedActivityComponent == nullptr) { + return; + } + + auto rating = 1; + + auto* team = TeamManager::Instance()->GetTeam(user->GetObjectID()); + + if (team != nullptr) { + rating = team->members.size(); + + for (const auto member : team->members) { + auto* memberObject = Game::entityManager->GetEntity(member); + + if (memberObject == nullptr) continue; + + LootGenerator::Instance().DropActivityLoot(memberObject, self, scriptedActivityComponent->GetActivityID(), rating); + } + } else { + LootGenerator::Instance().DropActivityLoot(user, self, scriptedActivityComponent->GetActivityID(), rating); + } + + self->Smash(self->GetObjectID()); +} diff --git a/dScripts/TreasureChestDragonServer.h b/dScripts/02_server/Enemy/General/TreasureChestDragonServer.h similarity index 100% rename from dScripts/TreasureChestDragonServer.h rename to dScripts/02_server/Enemy/General/TreasureChestDragonServer.h diff --git a/dScripts/AgSurvivalMech.cpp b/dScripts/02_server/Enemy/Survival/AgSurvivalMech.cpp similarity index 100% rename from dScripts/AgSurvivalMech.cpp rename to dScripts/02_server/Enemy/Survival/AgSurvivalMech.cpp diff --git a/dScripts/AgSurvivalMech.h b/dScripts/02_server/Enemy/Survival/AgSurvivalMech.h similarity index 100% rename from dScripts/AgSurvivalMech.h rename to dScripts/02_server/Enemy/Survival/AgSurvivalMech.h diff --git a/dScripts/AgSurvivalSpiderling.cpp b/dScripts/02_server/Enemy/Survival/AgSurvivalSpiderling.cpp similarity index 100% rename from dScripts/AgSurvivalSpiderling.cpp rename to dScripts/02_server/Enemy/Survival/AgSurvivalSpiderling.cpp diff --git a/dScripts/AgSurvivalSpiderling.h b/dScripts/02_server/Enemy/Survival/AgSurvivalSpiderling.h similarity index 100% rename from dScripts/AgSurvivalSpiderling.h rename to dScripts/02_server/Enemy/Survival/AgSurvivalSpiderling.h diff --git a/dScripts/AgSurvivalStromling.cpp b/dScripts/02_server/Enemy/Survival/AgSurvivalStromling.cpp similarity index 100% rename from dScripts/AgSurvivalStromling.cpp rename to dScripts/02_server/Enemy/Survival/AgSurvivalStromling.cpp diff --git a/dScripts/AgSurvivalStromling.h b/dScripts/02_server/Enemy/Survival/AgSurvivalStromling.h similarity index 100% rename from dScripts/AgSurvivalStromling.h rename to dScripts/02_server/Enemy/Survival/AgSurvivalStromling.h diff --git a/dScripts/02_server/Enemy/Survival/CMakeLists.txt b/dScripts/02_server/Enemy/Survival/CMakeLists.txt new file mode 100644 index 00000000..55236240 --- /dev/null +++ b/dScripts/02_server/Enemy/Survival/CMakeLists.txt @@ -0,0 +1,5 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_SURVIVAL + "AgSurvivalStromling.cpp" + "AgSurvivalMech.cpp" + "AgSurvivalSpiderling.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Enemy/VE/CMakeLists.txt b/dScripts/02_server/Enemy/VE/CMakeLists.txt new file mode 100644 index 00000000..a1cc39ad --- /dev/null +++ b/dScripts/02_server/Enemy/VE/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_VE + "VeMech.cpp" + PARENT_SCOPE) diff --git a/dScripts/VeMech.cpp b/dScripts/02_server/Enemy/VE/VeMech.cpp similarity index 100% rename from dScripts/VeMech.cpp rename to dScripts/02_server/Enemy/VE/VeMech.cpp diff --git a/dScripts/VeMech.h b/dScripts/02_server/Enemy/VE/VeMech.h similarity index 100% rename from dScripts/VeMech.h rename to dScripts/02_server/Enemy/VE/VeMech.h diff --git a/dScripts/02_server/Enemy/Waves/CMakeLists.txt b/dScripts/02_server/Enemy/Waves/CMakeLists.txt new file mode 100644 index 00000000..a25aca38 --- /dev/null +++ b/dScripts/02_server/Enemy/Waves/CMakeLists.txt @@ -0,0 +1,6 @@ +set(DSCRIPTS_SOURCES_02_SERVER_ENEMY_WAVES + "WaveBossHammerling.cpp" + "WaveBossApe.cpp" + "WaveBossSpiderling.cpp" + "WaveBossHorsemen.cpp" + PARENT_SCOPE) diff --git a/dScripts/WaveBossApe.cpp b/dScripts/02_server/Enemy/Waves/WaveBossApe.cpp similarity index 100% rename from dScripts/WaveBossApe.cpp rename to dScripts/02_server/Enemy/Waves/WaveBossApe.cpp diff --git a/dScripts/WaveBossApe.h b/dScripts/02_server/Enemy/Waves/WaveBossApe.h similarity index 100% rename from dScripts/WaveBossApe.h rename to dScripts/02_server/Enemy/Waves/WaveBossApe.h diff --git a/dScripts/WaveBossHammerling.cpp b/dScripts/02_server/Enemy/Waves/WaveBossHammerling.cpp similarity index 100% rename from dScripts/WaveBossHammerling.cpp rename to dScripts/02_server/Enemy/Waves/WaveBossHammerling.cpp diff --git a/dScripts/WaveBossHammerling.h b/dScripts/02_server/Enemy/Waves/WaveBossHammerling.h similarity index 100% rename from dScripts/WaveBossHammerling.h rename to dScripts/02_server/Enemy/Waves/WaveBossHammerling.h diff --git a/dScripts/WaveBossHorsemen.cpp b/dScripts/02_server/Enemy/Waves/WaveBossHorsemen.cpp similarity index 100% rename from dScripts/WaveBossHorsemen.cpp rename to dScripts/02_server/Enemy/Waves/WaveBossHorsemen.cpp diff --git a/dScripts/WaveBossHorsemen.h b/dScripts/02_server/Enemy/Waves/WaveBossHorsemen.h similarity index 100% rename from dScripts/WaveBossHorsemen.h rename to dScripts/02_server/Enemy/Waves/WaveBossHorsemen.h diff --git a/dScripts/WaveBossSpiderling.cpp b/dScripts/02_server/Enemy/Waves/WaveBossSpiderling.cpp similarity index 100% rename from dScripts/WaveBossSpiderling.cpp rename to dScripts/02_server/Enemy/Waves/WaveBossSpiderling.cpp diff --git a/dScripts/WaveBossSpiderling.h b/dScripts/02_server/Enemy/Waves/WaveBossSpiderling.h similarity index 100% rename from dScripts/WaveBossSpiderling.h rename to dScripts/02_server/Enemy/Waves/WaveBossSpiderling.h diff --git a/dScripts/02_server/Equipment/BootyDigServer.cpp b/dScripts/02_server/Equipment/BootyDigServer.cpp new file mode 100644 index 00000000..d27d1faa --- /dev/null +++ b/dScripts/02_server/Equipment/BootyDigServer.cpp @@ -0,0 +1,54 @@ +#include "BootyDigServer.h" +#include "EntityManager.h" +#include "RenderComponent.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" +#include "Loot.h" + +void BootyDigServer::OnStartup(Entity* self) { + auto* zoneControlObject = Game::entityManager->GetZoneControlEntity(); + if (zoneControlObject != nullptr) { + zoneControlObject->OnFireEventServerSide(self, "CheckForPropertyOwner"); + } +} + +void BootyDigServer::OnPlayerLoaded(Entity* self, Entity* player) { + auto* zoneControlObject = Game::entityManager->GetZoneControlEntity(); + if (zoneControlObject != nullptr) { + zoneControlObject->OnFireEventServerSide(self, "CheckForPropertyOwner"); + } +} + +void +BootyDigServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + + auto propertyOwner = self->GetNetworkVar<std::string>(u"PropertyOwnerID"); + auto* player = self->GetParentEntity(); + if (player == nullptr) + return; + + if (args == "ChestReady" && (propertyOwner == std::to_string(LWOOBJID_EMPTY) || player->GetVar<bool>(u"bootyDug"))) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } else if (args == "ChestOpened") { + // Make sure players only dig up one booty per instance + player->SetVar<bool>(u"bootyDug", true); + + auto* missionComponent = player->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + auto* mission = missionComponent->GetMission(1881); + if (mission != nullptr && (mission->GetMissionState() == eMissionState::ACTIVE || mission->GetMissionState() == eMissionState::COMPLETE_ACTIVE)) { + mission->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) + renderComponent->PlayEffect(7730, u"cast", "bootyshine"); + + LootGenerator::Instance().DropLoot(player, self, 231, 75, 75); + } + } + } else if (args == "ChestDead") { + self->Smash(player->GetObjectID(), eKillType::SILENT); + } +} diff --git a/dScripts/BootyDigServer.h b/dScripts/02_server/Equipment/BootyDigServer.h similarity index 100% rename from dScripts/BootyDigServer.h rename to dScripts/02_server/Equipment/BootyDigServer.h diff --git a/dScripts/02_server/Equipment/CMakeLists.txt b/dScripts/02_server/Equipment/CMakeLists.txt new file mode 100644 index 00000000..d29e7ca7 --- /dev/null +++ b/dScripts/02_server/Equipment/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_02_SERVER_EQUIPMENT + "MaestromExtracticatorServer.cpp" + "BootyDigServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Equipment/MaestromExtracticatorServer.cpp b/dScripts/02_server/Equipment/MaestromExtracticatorServer.cpp new file mode 100644 index 00000000..689eaac3 --- /dev/null +++ b/dScripts/02_server/Equipment/MaestromExtracticatorServer.cpp @@ -0,0 +1,47 @@ +#include "MaestromExtracticatorServer.h" +#include "GameMessages.h" +#include "GeneralUtils.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "RenderComponent.h" + +void MaestromExtracticatorServer::OnStartup(Entity* self) { + float animTime = RenderComponent::PlayAnimation(self, failAnim); + if (animTime == 0.0f) animTime = defaultTime; + + self->AddTimer("PlayFail", animTime); + self->AddTimer("RemoveSample", destroyAfterNoSampleTime); +} + +void MaestromExtracticatorServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + if (sender == nullptr) return; + + if (args == "attemptCollection") { + Entity* player = Game::entityManager->GetEntity(self->GetSpawnerID()); + if (!player) return; + + auto missionComponent = player->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) return; + + missionComponent->Progress(eMissionTaskType::SMASH, 14718); + CollectSample(self, sender->GetObjectID()); + sender->ScheduleKillAfterUpdate(); + } +} + +void MaestromExtracticatorServer::CollectSample(Entity* self, LWOOBJID sampleObj) { + self->AddTimer("RemoveSample", PlayAnimAndReturnTime(self, collectAnim)); +} + +float MaestromExtracticatorServer::PlayAnimAndReturnTime(Entity* self, std::string animID) { + return RenderComponent::PlayAnimation(self, animID); +} + +void MaestromExtracticatorServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "RemoveSample") { + self->ScheduleKillAfterUpdate(); + } else if (timerName == "PlayFail") { + RenderComponent::PlayAnimation(self, failAnim); + } +} diff --git a/dScripts/02_server/Equipment/MaestromExtracticatorServer.h b/dScripts/02_server/Equipment/MaestromExtracticatorServer.h new file mode 100644 index 00000000..f0e976f3 --- /dev/null +++ b/dScripts/02_server/Equipment/MaestromExtracticatorServer.h @@ -0,0 +1,18 @@ +#pragma once +#include "CppScripts.h" + +class MaestromExtracticatorServer : public CppScripts::Script { +public: + void OnStartup(Entity* self); + void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3); + void CollectSample(Entity* self, LWOOBJID sampleObj); + float PlayAnimAndReturnTime(Entity* self, std::string animID); + void OnTimerDone(Entity* self, std::string timerName); + +private: + const std::string failAnim = "idle_maelstrom"; + const std::string collectAnim = "collect_maelstrom"; + const float defaultTime = 4.0f; + const float destroyAfterNoSampleTime = 8.0f; +}; diff --git a/dScripts/AgBugsprayer.cpp b/dScripts/02_server/Map/AG/AgBugsprayer.cpp similarity index 100% rename from dScripts/AgBugsprayer.cpp rename to dScripts/02_server/Map/AG/AgBugsprayer.cpp diff --git a/dScripts/AgBugsprayer.h b/dScripts/02_server/Map/AG/AgBugsprayer.h similarity index 100% rename from dScripts/AgBugsprayer.h rename to dScripts/02_server/Map/AG/AgBugsprayer.h diff --git a/dScripts/02_server/Map/AG/AgCagedBricksServer.cpp b/dScripts/02_server/Map/AG/AgCagedBricksServer.cpp new file mode 100644 index 00000000..4d7e8a64 --- /dev/null +++ b/dScripts/02_server/Map/AG/AgCagedBricksServer.cpp @@ -0,0 +1,29 @@ +#include "AgCagedBricksServer.h" +#include "InventoryComponent.h" +#include "GameMessages.h" +#include "Character.h" +#include "EntityManager.h" +#include "eReplicaComponentType.h" +#include "ePlayerFlag.h" + +void AgCagedBricksServer::OnUse(Entity* self, Entity* user) { + //Tell the client to spawn the baby spiderling: + auto spooders = Game::entityManager->GetEntitiesInGroup("cagedSpider"); + for (auto spodder : spooders) { + GameMessages::SendFireEventClientSide(spodder->GetObjectID(), user->GetSystemAddress(), u"toggle", LWOOBJID_EMPTY, 0, 0, user->GetObjectID()); + } + + //Set the flag & mission status: + auto character = user->GetCharacter(); + + if (!character) return; + + character->SetPlayerFlag(ePlayerFlag::CAGED_SPIDER, true); + + //Remove the maelstrom cube: + auto inv = static_cast<InventoryComponent*>(user->GetComponent(eReplicaComponentType::INVENTORY)); + + if (inv) { + inv->RemoveItem(14553, 1); + } +} diff --git a/dScripts/AgCagedBricksServer.h b/dScripts/02_server/Map/AG/AgCagedBricksServer.h similarity index 100% rename from dScripts/AgCagedBricksServer.h rename to dScripts/02_server/Map/AG/AgCagedBricksServer.h diff --git a/dScripts/02_server/Map/AG/AgLaserSensorServer.cpp b/dScripts/02_server/Map/AG/AgLaserSensorServer.cpp new file mode 100644 index 00000000..7fcea9fa --- /dev/null +++ b/dScripts/02_server/Map/AG/AgLaserSensorServer.cpp @@ -0,0 +1,28 @@ +#include "AgLaserSensorServer.h" + +#include "PhantomPhysicsComponent.h" +#include "SkillComponent.h" +#include "ePhysicsEffectType.h" + +void AgLaserSensorServer::OnStartup(Entity* self) { + self->SetBoolean(u"active", true); + auto repelForce = self->GetVarAs<float>(u"repelForce"); + if (!repelForce) repelForce = m_RepelForce; + auto* phantomPhysicsComponent = self->GetComponent<PhantomPhysicsComponent>(); + if (!phantomPhysicsComponent) return; + phantomPhysicsComponent->SetPhysicsEffectActive(true); + phantomPhysicsComponent->SetEffectType(ePhysicsEffectType::REPULSE); + phantomPhysicsComponent->SetDirectionalMultiplier(repelForce); + phantomPhysicsComponent->SetDirection(NiPoint3::UNIT_Y); +} + + +void AgLaserSensorServer::OnCollisionPhantom(Entity* self, Entity* target) { + auto active = self->GetVar<bool>(u"active"); + if (!active) return; + auto skillCastID = self->GetVarAs<float>(u"skillCastID"); + if (skillCastID == 0) skillCastID = m_SkillCastID; + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (!skillComponent) return; + skillComponent->CastSkill(m_SkillCastID, target->GetObjectID()); +} diff --git a/dScripts/02_server/Map/AG/AgLaserSensorServer.h b/dScripts/02_server/Map/AG/AgLaserSensorServer.h new file mode 100644 index 00000000..ee7fb6ec --- /dev/null +++ b/dScripts/02_server/Map/AG/AgLaserSensorServer.h @@ -0,0 +1,14 @@ +#pragma once +#include "CppScripts.h" + +class SkillComponent; + +class AgLaserSensorServer : public CppScripts::Script { +public: + void OnStartup(Entity* self); + void OnCollisionPhantom(Entity* self, Entity* target); +private: + float m_RepelForce = -25.0f; + int m_SkillCastID = 163; +}; + diff --git a/dScripts/02_server/Map/AG/AgMonumentBirds.cpp b/dScripts/02_server/Map/AG/AgMonumentBirds.cpp new file mode 100644 index 00000000..c9df4dc1 --- /dev/null +++ b/dScripts/02_server/Map/AG/AgMonumentBirds.cpp @@ -0,0 +1,36 @@ +#include "AgMonumentBirds.h" +#include "GameMessages.h" +#include "Entity.h" +#include "RenderComponent.h" +#include "EntityManager.h" + +//-------------------------------------------------------------- +//Makes the ag birds fly away when you get close and smashes them. +//Created mrb... 6 / 3 / 11 +//Ported Max 20/07/2020 +//-------------------------------------------------------------- + +void AgMonumentBirds::OnStartup(Entity* self) { + self->SetProximityRadius(flyRadius, "MonumentBirds"); +} + +void AgMonumentBirds::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (self->GetVar<bool>(u"IsFlying")) return; + + if (name == "MonumentBirds" && status == "ENTER") { + self->AddTimer("killBird", 1.0f); + RenderComponent::PlayAnimation(self, sOnProximityAnim); + self->SetVar<bool>(u"IsFlying", true); + self->SetVar<LWOOBJID>(u"PlayerID", entering->GetObjectID()); + } +} + +void AgMonumentBirds::OnTimerDone(Entity* self, std::string timerName) { + if (timerName != "killBird") return; + + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"PlayerID")); + + if (player == nullptr) return; + + self->ScheduleKillAfterUpdate(player); +} diff --git a/dScripts/AgMonumentBirds.h b/dScripts/02_server/Map/AG/AgMonumentBirds.h similarity index 100% rename from dScripts/AgMonumentBirds.h rename to dScripts/02_server/Map/AG/AgMonumentBirds.h diff --git a/dScripts/02_server/Map/AG/AgMonumentLaserServer.cpp b/dScripts/02_server/Map/AG/AgMonumentLaserServer.cpp new file mode 100644 index 00000000..34c5c9a6 --- /dev/null +++ b/dScripts/02_server/Map/AG/AgMonumentLaserServer.cpp @@ -0,0 +1,17 @@ +#include "AgMonumentLaserServer.h" +#include "EntityManager.h" + +void AgMonumentLaserServer::OnStartup(Entity* self) { + auto lasers = Game::entityManager->GetEntitiesInGroup(self->GetVarAsString(u"volGroup")); + for (auto laser : lasers) { + if (laser) laser->SetBoolean(u"active", true); + } +} + +void AgMonumentLaserServer::OnDie(Entity* self, Entity* killer) { + auto lasers = Game::entityManager->GetEntitiesInGroup(self->GetVarAsString(u"volGroup")); + for (auto laser : lasers) { + if (laser) laser->SetBoolean(u"active", false); + } +} + diff --git a/dScripts/02_server/Map/AG/AgMonumentLaserServer.h b/dScripts/02_server/Map/AG/AgMonumentLaserServer.h new file mode 100644 index 00000000..8163948b --- /dev/null +++ b/dScripts/02_server/Map/AG/AgMonumentLaserServer.h @@ -0,0 +1,8 @@ +#pragma once +#include "CppScripts.h" + +class AgMonumentLaserServer : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnDie(Entity* self, Entity* killer) override; +}; diff --git a/dScripts/02_server/Map/AG/AgMonumentRaceCancel.cpp b/dScripts/02_server/Map/AG/AgMonumentRaceCancel.cpp new file mode 100644 index 00000000..1f0db946 --- /dev/null +++ b/dScripts/02_server/Map/AG/AgMonumentRaceCancel.cpp @@ -0,0 +1,9 @@ +#include "AgMonumentRaceCancel.h" +#include "EntityManager.h" + +void AgMonumentRaceCancel::OnCollisionPhantom(Entity* self, Entity* target) { + auto managers = Game::entityManager->GetEntitiesInGroup("race_manager"); + if (!managers.empty()) { + managers[0]->OnFireEventServerSide(target, "course_cancel"); + } +} diff --git a/dScripts/AgMonumentRaceCancel.h b/dScripts/02_server/Map/AG/AgMonumentRaceCancel.h similarity index 100% rename from dScripts/AgMonumentRaceCancel.h rename to dScripts/02_server/Map/AG/AgMonumentRaceCancel.h diff --git a/dScripts/02_server/Map/AG/AgMonumentRaceGoal.cpp b/dScripts/02_server/Map/AG/AgMonumentRaceGoal.cpp new file mode 100644 index 00000000..6d305d4b --- /dev/null +++ b/dScripts/02_server/Map/AG/AgMonumentRaceGoal.cpp @@ -0,0 +1,15 @@ +#include "AgMonumentRaceGoal.h" +#include "EntityManager.h" + + +void AgMonumentRaceGoal::OnStartup(Entity* self) { + self->SetProximityRadius(15, "RaceGoal"); +} + +void AgMonumentRaceGoal::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name == "RaceGoal" && entering && entering->IsPlayer() && status == "ENTER") { + auto managers = Game::entityManager->GetEntitiesInGroup("race_manager"); + if (managers.empty() || !managers.at(0)) return; + managers.at(0)->OnFireEventServerSide(entering, "course_finish"); + } +} diff --git a/dScripts/AgMonumentRaceGoal.h b/dScripts/02_server/Map/AG/AgMonumentRaceGoal.h similarity index 100% rename from dScripts/AgMonumentRaceGoal.h rename to dScripts/02_server/Map/AG/AgMonumentRaceGoal.h diff --git a/dScripts/02_server/Map/AG/CMakeLists.txt b/dScripts/02_server/Map/AG/CMakeLists.txt new file mode 100644 index 00000000..df26dee4 --- /dev/null +++ b/dScripts/02_server/Map/AG/CMakeLists.txt @@ -0,0 +1,16 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_AG + "AgCagedBricksServer.cpp" + "NpcWispServer.cpp" + "NpcEpsilonServer.cpp" + "AgLaserSensorServer.cpp" + "AgMonumentLaserServer.cpp" + "AgMonumentBirds.cpp" + "RemoveRentalGear.cpp" + "NpcNjAssistantServer.cpp" + "AgBugsprayer.cpp" + "NpcAgCourseStarter.cpp" + "AgMonumentRaceGoal.cpp" + "AgMonumentRaceCancel.cpp" + "NpcCowboyServer.cpp" + "NpcPirateServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/AG/NpcAgCourseStarter.cpp b/dScripts/02_server/Map/AG/NpcAgCourseStarter.cpp new file mode 100644 index 00000000..b47260f9 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcAgCourseStarter.cpp @@ -0,0 +1,106 @@ +#include "NpcAgCourseStarter.h" +#include "EntityManager.h" +#include "ScriptedActivityComponent.h" +#include "GameMessages.h" +#include "LeaderboardManager.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" +#include "MissionComponent.h" +#include <ctime> + +void NpcAgCourseStarter::OnStartup(Entity* self) { + +} + +void NpcAgCourseStarter::OnUse(Entity* self, Entity* user) { + auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); + + if (scriptedActivityComponent == nullptr) { + return; + } + + if (scriptedActivityComponent->GetActivityPlayerData(user->GetObjectID()) != nullptr) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"exit", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); + } else { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"start", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); + } +} + +void NpcAgCourseStarter::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); + + if (scriptedActivityComponent == nullptr) { + return; + } + + if (identifier == u"player_dialog_cancel_course" && button == 1) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stop_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"cancel_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + + scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); + + Game::entityManager->SerializeEntity(self); + } else if (identifier == u"player_dialog_start_course" && button == 1) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"start_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + + GameMessages::SendActivityStart(self->GetObjectID(), sender->GetSystemAddress()); + + auto* data = scriptedActivityComponent->AddActivityPlayerData(sender->GetObjectID()); + + if (data->values[1] != 0) return; + + time_t startTime = std::time(0) + 4; // Offset for starting timer + + data->values[1] = *(float*)&startTime; + + Game::entityManager->SerializeEntity(self); + } else if (identifier == u"FootRaceCancel") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stop_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + + if (scriptedActivityComponent->GetActivityPlayerData(sender->GetObjectID()) != nullptr) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"exit", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + } else { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"start", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + } + + scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); + } +} + +void NpcAgCourseStarter::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); + if (scriptedActivityComponent == nullptr) return; + + auto* data = scriptedActivityComponent->GetActivityPlayerData(sender->GetObjectID()); + if (data == nullptr) return; + + if (args == "course_cancel") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"cancel_timer", 0, 0, + LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); + } else if (args == "course_finish") { + time_t endTime = std::time(0); + time_t finish = (endTime - *(time_t*)&data->values[1]); + + data->values[2] = *(float*)&finish; + + auto* missionComponent = sender->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + missionComponent->ForceProgressTaskType(1884, 1, 1, false); + missionComponent->Progress(eMissionTaskType::PERFORM_ACTIVITY, -finish, self->GetObjectID(), + "performact_time"); + } + + Game::entityManager->SerializeEntity(self); + LeaderboardManager::SaveScore(sender->GetObjectID(), scriptedActivityComponent->GetActivityID(), static_cast<float>(finish)); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"ToggleLeaderBoard", + scriptedActivityComponent->GetActivityID(), 0, sender->GetObjectID(), + "", sender->GetSystemAddress()); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stop_timer", 1, finish, LWOOBJID_EMPTY, "", + sender->GetSystemAddress()); + + scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); + } +} diff --git a/dScripts/NpcAgCourseStarter.h b/dScripts/02_server/Map/AG/NpcAgCourseStarter.h similarity index 100% rename from dScripts/NpcAgCourseStarter.h rename to dScripts/02_server/Map/AG/NpcAgCourseStarter.h diff --git a/dScripts/02_server/Map/AG/NpcCowboyServer.cpp b/dScripts/02_server/Map/AG/NpcCowboyServer.cpp new file mode 100644 index 00000000..6dd212a4 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcCowboyServer.cpp @@ -0,0 +1,26 @@ +#include "NpcCowboyServer.h" +#include "eMissionState.h" +#include "InventoryComponent.h" + +void NpcCowboyServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID != 1880) { + return; + } + + auto* inventoryComponent = target->GetComponent<InventoryComponent>(); + + if (inventoryComponent == nullptr) { + return; + } + + if (missionState == eMissionState::COMPLETE_ACTIVE || + missionState == eMissionState::ACTIVE || + missionState == eMissionState::AVAILABLE || + missionState == eMissionState::COMPLETE_AVAILABLE) { + if (inventoryComponent->GetLotCount(14378) == 0) { + inventoryComponent->AddItem(14378, 1, eLootSourceType::NONE); + } + } else if (missionState == eMissionState::READY_TO_COMPLETE || missionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { + inventoryComponent->RemoveItem(14378, 1); + } +} diff --git a/dScripts/02_server/Map/AG/NpcCowboyServer.h b/dScripts/02_server/Map/AG/NpcCowboyServer.h new file mode 100644 index 00000000..4493315c --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcCowboyServer.h @@ -0,0 +1,7 @@ +#pragma once +#include "CppScripts.h" + +class NpcCowboyServer : public CppScripts::Script +{ + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +}; diff --git a/dScripts/02_server/Map/AG/NpcEpsilonServer.cpp b/dScripts/02_server/Map/AG/NpcEpsilonServer.cpp new file mode 100644 index 00000000..a928e9d9 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcEpsilonServer.cpp @@ -0,0 +1,10 @@ +#include "NpcEpsilonServer.h" +#include "GameMessages.h" + +void NpcEpsilonServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + + //If we are completing the Nexus Force join mission, play the celebration for it: + if (missionID == 1851) { + GameMessages::SendStartCelebrationEffect(target, target->GetSystemAddress(), 22); + } +} diff --git a/dScripts/02_server/Map/AG/NpcEpsilonServer.h b/dScripts/02_server/Map/AG/NpcEpsilonServer.h new file mode 100644 index 00000000..4de76bcd --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcEpsilonServer.h @@ -0,0 +1,7 @@ +#pragma once +#include "CppScripts.h" + +class NpcEpsilonServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState); +}; + diff --git a/dScripts/02_server/Map/AG/NpcNjAssistantServer.cpp b/dScripts/02_server/Map/AG/NpcNjAssistantServer.cpp new file mode 100644 index 00000000..8ee2e988 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcNjAssistantServer.cpp @@ -0,0 +1,29 @@ +#include "NpcNjAssistantServer.h" +#include "GameMessages.h" +#include "InventoryComponent.h" +#include "MissionComponent.h" +#include "Item.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" + +void NpcNjAssistantServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID != mailMission) return; + + if (missionState == eMissionState::COMPLETE || missionState == eMissionState::READY_TO_COMPLETE) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"switch", 0, 0, LWOOBJID_EMPTY, "", target->GetSystemAddress()); + + auto* inv = static_cast<InventoryComponent*>(target->GetComponent(eReplicaComponentType::INVENTORY)); + + // If we are ready to complete our missions, we take the kit from you: + if (inv && missionState == eMissionState::READY_TO_COMPLETE) { + auto* id = inv->FindItemByLot(14397); //the kit's lot + + if (id != nullptr) { + inv->RemoveItem(id->GetLot(), id->GetCount()); + } + } + } else if (missionState == eMissionState::AVAILABLE) { + auto* missionComponent = static_cast<MissionComponent*>(target->GetComponent(eReplicaComponentType::MISSION)); + missionComponent->CompleteMission(mailAchievement, true); + } +} diff --git a/dScripts/02_server/Map/AG/NpcNjAssistantServer.h b/dScripts/02_server/Map/AG/NpcNjAssistantServer.h new file mode 100644 index 00000000..1f932752 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcNjAssistantServer.h @@ -0,0 +1,10 @@ +#pragma once +#include "CppScripts.h" + +class NpcNjAssistantServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState); + +private: + int mailMission = 1728; //mission to get the item out of your mailbox + int mailAchievement = 1729; // fun fact: spelled "Achivement" in the actual script +}; diff --git a/dScripts/02_server/Map/AG/NpcPirateServer.cpp b/dScripts/02_server/Map/AG/NpcPirateServer.cpp new file mode 100644 index 00000000..cad0d64c --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcPirateServer.cpp @@ -0,0 +1,18 @@ +#include "NpcPirateServer.h" +#include "eMissionState.h" +#include "InventoryComponent.h" + +void NpcPirateServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + auto* inventory = target->GetComponent<InventoryComponent>(); + if (inventory != nullptr && missionID == 1881) { + auto* luckyShovel = inventory->FindItemByLot(14591); + + // Add or remove the lucky shovel based on whether the mission was completed or started + if ((missionState == eMissionState::AVAILABLE || missionState == eMissionState::COMPLETE_AVAILABLE) + && luckyShovel == nullptr) { + inventory->AddItem(14591, 1, eLootSourceType::NONE); + } else if (missionState == eMissionState::READY_TO_COMPLETE || missionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { + inventory->RemoveItem(14591, 1); + } + } +} diff --git a/dScripts/02_server/Map/AG/NpcPirateServer.h b/dScripts/02_server/Map/AG/NpcPirateServer.h new file mode 100644 index 00000000..118aec89 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcPirateServer.h @@ -0,0 +1,6 @@ +#pragma once +#include "CppScripts.h" + +class NpcPirateServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +}; diff --git a/dScripts/02_server/Map/AG/NpcWispServer.cpp b/dScripts/02_server/Map/AG/NpcWispServer.cpp new file mode 100644 index 00000000..e3b5398d --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcWispServer.cpp @@ -0,0 +1,44 @@ +#include "NpcWispServer.h" +#include "InventoryComponent.h" +#include "EntityManager.h" +#include "Entity.h" +#include "GameMessages.h" +#include "eMissionState.h" + +void NpcWispServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID != 1849 && missionID != 1883) + return; + + auto* inventory = target->GetComponent<InventoryComponent>(); + if (inventory == nullptr) + return; + + LOT maelstromVacuumLot = 14592; + auto* maelstromVacuum = inventory->FindItemByLot(maelstromVacuumLot); + + // For the daily we add the maelstrom vacuum if the player doesn't have it yet + if (missionID == 1883 && (missionState == eMissionState::AVAILABLE || missionState == eMissionState::COMPLETE_AVAILABLE) + && maelstromVacuum == nullptr) { + inventory->AddItem(maelstromVacuumLot, 1, eLootSourceType::NONE); + } else if (missionState == eMissionState::READY_TO_COMPLETE || missionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { + inventory->RemoveItem(maelstromVacuumLot, 1); + } + + // Next up hide or show the samples based on the mission state + auto visible = 1; + if (missionState == eMissionState::READY_TO_COMPLETE || missionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { + visible = 0; + } + + auto groups = missionID == 1849 + ? std::vector<std::string> { "MaelstromSamples" } + : std::vector<std::string>{ "MaelstromSamples", "MaelstromSamples2ndary1", "MaelstromSamples2ndary2" }; + + for (const auto& group : groups) { + auto samples = Game::entityManager->GetEntitiesInGroup(group); + for (auto* sample : samples) { + GameMessages::SendNotifyClientObject(sample->GetObjectID(), u"SetVisibility", visible, 0, + target->GetObjectID(), "", target->GetSystemAddress()); + } + } +} diff --git a/dScripts/02_server/Map/AG/NpcWispServer.h b/dScripts/02_server/Map/AG/NpcWispServer.h new file mode 100644 index 00000000..6eaa09e9 --- /dev/null +++ b/dScripts/02_server/Map/AG/NpcWispServer.h @@ -0,0 +1,6 @@ +#pragma once +#include "CppScripts.h" + +class NpcWispServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState); +}; diff --git a/dScripts/02_server/Map/AG/RemoveRentalGear.cpp b/dScripts/02_server/Map/AG/RemoveRentalGear.cpp new file mode 100644 index 00000000..f9bdf1ce --- /dev/null +++ b/dScripts/02_server/Map/AG/RemoveRentalGear.cpp @@ -0,0 +1,42 @@ +#include "RemoveRentalGear.h" +#include "InventoryComponent.h" +#include "Item.h" +#include "eMissionState.h" +#include "Character.h" +#include "eReplicaComponentType.h" +#include "ePlayerFlag.h" + +/* +-------------------------------------------------------------- +--Removes the rental gear from the player on mission turn in +-- +--created mrb ... 5 / 25 / 11 +--updated abeechler 6 / 27 / 11 ... Add session flag resetting for set equips +--ported Max 21/07/2020 +-------------------------------------------------------------- +--add missionID configData to the object in HF to remove this +--gear what the specified mission is completed +-------------------------------------------------------------- +*/ + +void RemoveRentalGear::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID != defaultMission && missionID != 313) return; + + if (missionState == eMissionState::COMPLETE || missionState == eMissionState::READY_TO_COMPLETE) { + auto inv = static_cast<InventoryComponent*>(target->GetComponent(eReplicaComponentType::INVENTORY)); + if (!inv) return; + + //remove the inventory items + for (int item : gearSets) { + auto* id = inv->FindItemByLot(item); + if (id) { + inv->UnEquipItem(id); + inv->RemoveItem(id->GetLot(), id->GetCount()); + } + } + + //reset the equipment flag + auto character = target->GetCharacter(); + if (character) character->SetPlayerFlag(ePlayerFlag::EQUPPED_TRIAL_FACTION_GEAR, false); + } +} diff --git a/dScripts/02_server/Map/AG/RemoveRentalGear.h b/dScripts/02_server/Map/AG/RemoveRentalGear.h new file mode 100644 index 00000000..cf9002d7 --- /dev/null +++ b/dScripts/02_server/Map/AG/RemoveRentalGear.h @@ -0,0 +1,11 @@ +#pragma once +#include "CppScripts.h" + +class RemoveRentalGear : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState); + +private: + int defaultMission = 768; //mission to remove gearSets on completion + std::vector<int> gearSets = { 14359,14321,14353,14315 }; //inventory items to remove +}; + diff --git a/dScripts/02_server/Map/AG_Spider_Queen/CMakeLists.txt b/dScripts/02_server/Map/AG_Spider_Queen/CMakeLists.txt new file mode 100644 index 00000000..f4204c13 --- /dev/null +++ b/dScripts/02_server/Map/AG_Spider_Queen/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_AG_SPIDER_QUEEN + "ZoneAgSpiderQueen.cpp" + "SpiderBossTreasureChestServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/SpiderBossTreasureChestServer.cpp b/dScripts/02_server/Map/AG_Spider_Queen/SpiderBossTreasureChestServer.cpp similarity index 100% rename from dScripts/SpiderBossTreasureChestServer.cpp rename to dScripts/02_server/Map/AG_Spider_Queen/SpiderBossTreasureChestServer.cpp diff --git a/dScripts/SpiderBossTreasureChestServer.h b/dScripts/02_server/Map/AG_Spider_Queen/SpiderBossTreasureChestServer.h similarity index 100% rename from dScripts/SpiderBossTreasureChestServer.h rename to dScripts/02_server/Map/AG_Spider_Queen/SpiderBossTreasureChestServer.h diff --git a/dScripts/02_server/Map/AG_Spider_Queen/ZoneAgSpiderQueen.cpp b/dScripts/02_server/Map/AG_Spider_Queen/ZoneAgSpiderQueen.cpp new file mode 100644 index 00000000..2091041b --- /dev/null +++ b/dScripts/02_server/Map/AG_Spider_Queen/ZoneAgSpiderQueen.cpp @@ -0,0 +1,84 @@ +#include "ZoneAgSpiderQueen.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "ZoneAgProperty.h" +#include "DestroyableComponent.h" +#include "EntityInfo.h" + +void ZoneAgSpiderQueen::SetGameVariables(Entity* self) { + ZoneAgProperty::SetGameVariables(self); + + // Disable property flags + self->SetVar<int32_t>(defeatedProperyFlag, 0); + self->SetVar<int32_t>(placedModelFlag, 0); + self->SetVar<uint32_t>(guardFirstMissionFlag, 0); + self->SetVar<uint32_t>(guardMissionFlag, 0); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 0); +} + +void ZoneAgSpiderQueen::OnStartup(Entity* self) { + LoadInstance(self); + + SpawnSpots(self); + StartMaelstrom(self, nullptr); +} + +void ZoneAgSpiderQueen::BasePlayerLoaded(Entity* self, Entity* player) { + ActivityManager::UpdatePlayer(self, player->GetObjectID()); + ActivityManager::TakeActivityCost(self, player->GetObjectID()); + + // Make sure the player has full stats when they join + auto* playerDestroyableComponent = player->GetComponent<DestroyableComponent>(); + if (playerDestroyableComponent != nullptr) { + playerDestroyableComponent->SetImagination(playerDestroyableComponent->GetMaxImagination()); + playerDestroyableComponent->SetArmor(playerDestroyableComponent->GetMaxArmor()); + playerDestroyableComponent->SetHealth(playerDestroyableComponent->GetMaxHealth()); + } + + self->SetNetworkVar(u"unclaimed", true); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"maelstromSkyOn", 0, 0, LWOOBJID_EMPTY, + "", player->GetSystemAddress()); +} + +void +ZoneAgSpiderQueen::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args == "ClearProperty") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayCinematic", 0, 0, + LWOOBJID_EMPTY, destroyedCinematic, UNASSIGNED_SYSTEM_ADDRESS); + self->AddTimer("tornadoOff", 0.5f); + } else { + ZoneAgProperty::BaseOnFireEventServerSide(self, sender, args); + } +} + +void ZoneAgSpiderQueen::OnPlayerExit(Entity* self, Entity* player) { + UpdatePlayer(self, player->GetObjectID(), true); +} + +void ZoneAgSpiderQueen::OnTimerDone(Entity* self, std::string timerName) { + + // Disable some stuff from the regular property + if (timerName == "BoundsVisOn" || timerName == "GuardFlyAway" || timerName == "ShowVendor") + return; + + if (timerName == "killSpider") { + auto spawnTargets = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(LandTargetGroup)); + for (auto* spawnTarget : spawnTargets) { + EntityInfo info{}; + + info.spawnerID = spawnTarget->GetObjectID(); + info.pos = spawnTarget->GetPosition(); + info.rot = spawnTarget->GetRotation(); + info.lot = chestObject; + info.settings = { + new LDFData<LWOOBJID>(u"parent_tag", self->GetObjectID()) + }; + + auto* chest = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(chest); + } + } + + ZoneAgProperty::BaseTimerDone(self, timerName); +} diff --git a/dScripts/ZoneAgSpiderQueen.h b/dScripts/02_server/Map/AG_Spider_Queen/ZoneAgSpiderQueen.h similarity index 100% rename from dScripts/ZoneAgSpiderQueen.h rename to dScripts/02_server/Map/AG_Spider_Queen/ZoneAgSpiderQueen.h diff --git a/dScripts/02_server/Map/AM/AmBlueX.cpp b/dScripts/02_server/Map/AM/AmBlueX.cpp new file mode 100644 index 00000000..180d4c54 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmBlueX.cpp @@ -0,0 +1,56 @@ +#include "AmBlueX.h" +#include "SkillComponent.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "Character.h" + +void AmBlueX::OnUse(Entity* self, Entity* user) { + auto* skillComponent = user->GetComponent<SkillComponent>(); + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(m_SwordSkill, m_SwordBehavior, self->GetObjectID()); + } +} + +void AmBlueX::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (message == "FireDukesStrike") { + self->SetNetworkVar<bool>(m_XUsedVariable, true); + self->SetNetworkVar<bool>(m_StartEffectVariable, true); + + auto* character = caster->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(self->GetVar<int32_t>(m_FlagVariable), true); + } + + EntityInfo info{}; + info.lot = m_FXObject; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.spawnerID = self->GetObjectID(); + + auto* fxObject = Game::entityManager->CreateEntity(info, nullptr, self); + Game::entityManager->ConstructEntity(fxObject); + + auto fxObjectID = fxObject->GetObjectID(); + auto playerID = caster->GetObjectID(); + + // Add a callback for the bomb to explode + self->AddCallbackTimer(m_BombTime, [this, self, fxObjectID, playerID]() { + auto* fxObject = Game::entityManager->GetEntity(fxObjectID); + auto* player = Game::entityManager->GetEntity(playerID); + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) + return; + + // Cast the skill that destroys the object + if (player != nullptr) { + skillComponent->CalculateBehavior(m_AOESkill, m_AOEBehavior, LWOOBJID_EMPTY, false, false, playerID); + } else { + skillComponent->CalculateBehavior(m_AOESkill, m_AOEBehavior, LWOOBJID_EMPTY); + } + + fxObject->Smash(); + self->Smash(); + }); + } +} diff --git a/dScripts/AmBlueX.h b/dScripts/02_server/Map/AM/AmBlueX.h similarity index 100% rename from dScripts/AmBlueX.h rename to dScripts/02_server/Map/AM/AmBlueX.h diff --git a/dScripts/02_server/Map/AM/AmBridge.cpp b/dScripts/02_server/Map/AM/AmBridge.cpp new file mode 100644 index 00000000..2fce627d --- /dev/null +++ b/dScripts/02_server/Map/AM/AmBridge.cpp @@ -0,0 +1,28 @@ +#include "AmBridge.h" +#include "EntityManager.h" + +void AmBridge::OnStartup(Entity* self) { + +} + +void AmBridge::OnRebuildComplete(Entity* self, Entity* target) { + const auto consoles = Game::entityManager->GetEntitiesInGroup("Console" + GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"bridge"))); + + if (consoles.empty()) { + return; + } + + auto* console = consoles[0]; + + console->NotifyObject(self, "BridgeBuilt"); + + self->AddTimer("SmashBridge", 50); +} + +void AmBridge::OnTimerDone(Entity* self, std::string timerName) { + if (timerName != "SmashBridge") { + return; + } + + self->Smash(self->GetObjectID(), eKillType::VIOLENT); +} diff --git a/dScripts/AmBridge.h b/dScripts/02_server/Map/AM/AmBridge.h similarity index 100% rename from dScripts/AmBridge.h rename to dScripts/02_server/Map/AM/AmBridge.h diff --git a/dScripts/02_server/Map/AM/AmConsoleTeleportServer.cpp b/dScripts/02_server/Map/AM/AmConsoleTeleportServer.cpp new file mode 100644 index 00000000..0627b27a --- /dev/null +++ b/dScripts/02_server/Map/AM/AmConsoleTeleportServer.cpp @@ -0,0 +1,30 @@ +#include "AmConsoleTeleportServer.h" +#include "ChooseYourDestinationNsToNt.h" +#include "Amf3.h" + +void AmConsoleTeleportServer::OnStartup(Entity* self) { + self->SetVar(u"teleportAnim", m_TeleportAnim); + self->SetVar(u"teleportString", m_TeleportString); + self->SetVar(u"teleportEffectID", m_TeleportEffectID); + self->SetVar(u"teleportEffectTypes", m_TeleportEffectTypes); +} + +void AmConsoleTeleportServer::OnUse(Entity* self, Entity* user) { + BaseOnUse(self, user); +} + +void AmConsoleTeleportServer::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + BaseOnMessageBoxResponse(self, sender, button, identifier, userData); +} + +void AmConsoleTeleportServer::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { + +} + +void AmConsoleTeleportServer::OnTimerDone(Entity* self, std::string timerName) { + BaseOnTimerDone(self, timerName); +} + +void AmConsoleTeleportServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); +} diff --git a/dScripts/AmConsoleTeleportServer.h b/dScripts/02_server/Map/AM/AmConsoleTeleportServer.h similarity index 100% rename from dScripts/AmConsoleTeleportServer.h rename to dScripts/02_server/Map/AM/AmConsoleTeleportServer.h diff --git a/dScripts/02_server/Map/AM/AmDrawBridge.cpp b/dScripts/02_server/Map/AM/AmDrawBridge.cpp new file mode 100644 index 00000000..e6b80764 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmDrawBridge.cpp @@ -0,0 +1,122 @@ +#include "AmDrawBridge.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "SimplePhysicsComponent.h" +#include "eTerminateType.h" + +void AmDrawBridge::OnStartup(Entity* self) { + self->SetNetworkVar(u"InUse", false); + self->SetVar(u"BridgeDown", false); +} + +void AmDrawBridge::OnUse(Entity* self, Entity* user) { + auto* bridge = GetBridge(self); + + if (bridge == nullptr) { + return; + } + + if (!self->GetNetworkVar<bool>(u"InUse")) { + self->SetNetworkVar(u"startEffect", 5); + + self->AddTimer("ChangeBridge", 5); + + self->SetNetworkVar(u"InUse", true); + } + + auto* player = user; + + GameMessages::SendTerminateInteraction(player->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} + +void AmDrawBridge::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "ChangeBridge") { + auto* bridge = GetBridge(self); + + if (bridge == nullptr) { + return; + } + + if (!self->GetVar<bool>(u"BridgeDown")) { + self->SetVar(u"BridgeDown", true); + + MoveBridgeDown(self, bridge, true); + } else { + self->SetVar(u"BridgeDown", false); + + MoveBridgeDown(self, bridge, false); + } + + self->SetNetworkVar(u"BridgeLeaving", true); + self->SetVar(u"BridgeDown", false); + } else if (timerName == "SmashEffectBridge") { + self->SetNetworkVar(u"SmashBridge", 5); + } else if (timerName == "rotateBridgeDown") { + auto* bridge = GetBridge(self); + + if (bridge == nullptr) { + return; + } + + self->SetNetworkVar(u"BridgeLeaving", false); + + auto* simplePhysicsComponent = bridge->GetComponent<SimplePhysicsComponent>(); + + if (simplePhysicsComponent == nullptr) { + return; + } + + simplePhysicsComponent->SetAngularVelocity(NiPoint3::ZERO); + + Game::entityManager->SerializeEntity(bridge); + } +} + +void AmDrawBridge::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { + if (name == "BridgeBuilt") { + self->SetVar(u"BridgeID", sender->GetObjectID()); + + self->AddTimer("SmashEffectBridge", 45); + + self->SetNetworkVar(u"BridgeDead", true); + + sender->AddDieCallback([this, self, sender]() { + NotifyDie(self, sender); + }); + } +} + +void AmDrawBridge::MoveBridgeDown(Entity* self, Entity* bridge, bool down) { + auto* simplePhysicsComponent = bridge->GetComponent<SimplePhysicsComponent>(); + + if (simplePhysicsComponent == nullptr) { + return; + } + + auto forwardVect = simplePhysicsComponent->GetRotation().GetForwardVector(); + + auto degrees = down ? 90.0f : -90.0f; + + const auto travelTime = 2.0f; + + forwardVect = forwardVect * (float)((degrees / travelTime) * (3.14f / 180.0f)); + + simplePhysicsComponent->SetAngularVelocity(forwardVect); + + Game::entityManager->SerializeEntity(bridge); + + self->AddTimer("rotateBridgeDown", travelTime); +} + +void AmDrawBridge::NotifyDie(Entity* self, Entity* other) { + self->SetNetworkVar(u"InUse", false); + self->SetVar(u"BridgeDown", false); + + self->CancelAllTimers(); +} + +Entity* AmDrawBridge::GetBridge(Entity* self) { + const auto bridgeID = self->GetVar<LWOOBJID>(u"BridgeID"); + + return Game::entityManager->GetEntity(bridgeID); +} diff --git a/dScripts/AmDrawBridge.h b/dScripts/02_server/Map/AM/AmDrawBridge.h similarity index 100% rename from dScripts/AmDrawBridge.h rename to dScripts/02_server/Map/AM/AmDrawBridge.h diff --git a/dScripts/02_server/Map/AM/AmDropshipComputer.cpp b/dScripts/02_server/Map/AM/AmDropshipComputer.cpp new file mode 100644 index 00000000..1ee2aba3 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmDropshipComputer.cpp @@ -0,0 +1,82 @@ +#include "AmDropshipComputer.h" +#include "MissionComponent.h" +#include "RebuildComponent.h" +#include "InventoryComponent.h" +#include "dZoneManager.h" +#include "eMissionState.h" + +void AmDropshipComputer::OnStartup(Entity* self) { + self->AddTimer("reset", 45.0f); +} + +void AmDropshipComputer::OnUse(Entity* self, Entity* user) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent == nullptr || rebuildComponent->GetState() != eRebuildState::COMPLETED) { + return; + } + + auto* missionComponent = user->GetComponent<MissionComponent>(); + auto* inventoryComponent = user->GetComponent<InventoryComponent>(); + + if (missionComponent == nullptr || inventoryComponent == nullptr) { + return; + } + + if (inventoryComponent->GetLotCount(m_NexusTalonDataCard) != 0 || missionComponent->GetMission(979)->GetMissionState() == eMissionState::COMPLETE) { + return; + } + + inventoryComponent->AddItem(m_NexusTalonDataCard, 1, eLootSourceType::NONE); +} + +void AmDropshipComputer::OnDie(Entity* self, Entity* killer) { + const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); + + int32_t pipeNum = 0; + if (!GeneralUtils::TryParse<int32_t>(myGroup.substr(10, 1), pipeNum)) { + return; + } + + const auto pipeGroup = myGroup.substr(0, 10); + + const auto nextPipeNum = pipeNum + 1; + + const auto samePipeSpawners = Game::zoneManager->GetSpawnersByName(myGroup); + + if (!samePipeSpawners.empty()) { + samePipeSpawners[0]->SoftReset(); + + samePipeSpawners[0]->Deactivate(); + } + + if (killer != nullptr && killer->IsPlayer()) { + const auto nextPipe = pipeGroup + std::to_string(nextPipeNum); + + const auto nextPipeSpawners = Game::zoneManager->GetSpawnersByName(nextPipe); + + if (!nextPipeSpawners.empty()) { + nextPipeSpawners[0]->Activate(); + } + } else { + const auto nextPipe = pipeGroup + "1"; + + const auto firstPipeSpawners = Game::zoneManager->GetSpawnersByName(nextPipe); + + if (!firstPipeSpawners.empty()) { + firstPipeSpawners[0]->Activate(); + } + } +} + +void AmDropshipComputer::OnTimerDone(Entity* self, std::string timerName) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent == nullptr) { + return; + } + + if (timerName == "reset" && rebuildComponent->GetState() == eRebuildState::OPEN) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } +} diff --git a/dScripts/AmDropshipComputer.h b/dScripts/02_server/Map/AM/AmDropshipComputer.h similarity index 100% rename from dScripts/AmDropshipComputer.h rename to dScripts/02_server/Map/AM/AmDropshipComputer.h diff --git a/dScripts/AmScrollReaderServer.cpp b/dScripts/02_server/Map/AM/AmScrollReaderServer.cpp similarity index 100% rename from dScripts/AmScrollReaderServer.cpp rename to dScripts/02_server/Map/AM/AmScrollReaderServer.cpp diff --git a/dScripts/AmScrollReaderServer.h b/dScripts/02_server/Map/AM/AmScrollReaderServer.h similarity index 100% rename from dScripts/AmScrollReaderServer.h rename to dScripts/02_server/Map/AM/AmScrollReaderServer.h diff --git a/dScripts/02_server/Map/AM/AmShieldGenerator.cpp b/dScripts/02_server/Map/AM/AmShieldGenerator.cpp new file mode 100644 index 00000000..7c785c41 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmShieldGenerator.cpp @@ -0,0 +1,145 @@ +#include "AmShieldGenerator.h" +#include "EntityManager.h" +#include "DestroyableComponent.h" +#include "GameMessages.h" +#include "EntityInfo.h" +#include "MovementAIComponent.h" +#include "BaseCombatAIComponent.h" +#include "SkillComponent.h" + +void AmShieldGenerator::OnStartup(Entity* self) { + self->SetProximityRadius(20, "shield"); + self->SetProximityRadius(21, "buffer"); + + StartShield(self); +} + +void AmShieldGenerator::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + auto* destroyableComponent = entering->GetComponent<DestroyableComponent>(); + + if (status == "ENTER" && name == "shield") { + if (destroyableComponent->HasFaction(4)) { + EnemyEnteredShield(self, entering); + } + } + + if (name != "buffer" || !entering->IsPlayer()) { + return; + } + + auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + if (status == "ENTER") { + const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); + + if (iter == entitiesInProximity.end()) { + entitiesInProximity.push_back(entering->GetObjectID()); + } + } else if (status == "LEAVE") { + const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); + + if (iter != entitiesInProximity.end()) { + entitiesInProximity.erase(iter); + } + } + + self->SetVar<std::vector<LWOOBJID>>(u"Players", entitiesInProximity); +} + +void AmShieldGenerator::OnDie(Entity* self, Entity* killer) { + self->CancelAllTimers(); + + auto* child = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Child")); + + if (child != nullptr) { + child->Kill(); + } +} + +void AmShieldGenerator::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "BuffPlayers") { + BuffPlayers(self); + + self->AddTimer("BuffPlayers", 3.0f); + } else if (timerName == "PlayFX") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 5351, u"generatorOn", "generatorOn"); + + self->AddTimer("PlayFX", 1.5f); + } else if (timerName == "RefreshEnemies") { + auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); + + for (const auto enemyID : enemiesInProximity) { + auto* enemy = Game::entityManager->GetEntity(enemyID); + + if (enemy != nullptr) { + EnemyEnteredShield(self, enemy); + } + } + + self->AddTimer("RefreshEnemies", 1.5f); + } +} + +void AmShieldGenerator::StartShield(Entity* self) { + self->AddTimer("PlayFX", 1.5f); + self->AddTimer("BuffPlayers", 3.0f); + self->AddTimer("RefreshEnemies", 1.5f); + + const auto myPos = self->GetPosition(); + const auto myRot = self->GetRotation(); + + EntityInfo info{}; + info.lot = 13111; + info.pos = myPos; + info.rot = myRot; + info.spawnerID = self->GetObjectID(); + + auto* child = Game::entityManager->CreateEntity(info); + + self->SetVar(u"Child", child->GetObjectID()); + + BuffPlayers(self); +} + +void AmShieldGenerator::BuffPlayers(Entity* self) { + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + return; + } + + auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + for (const auto playerID : entitiesInProximity) { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + skillComponent->CalculateBehavior(1200, 27024, playerID, true); + } +} + +void AmShieldGenerator::EnemyEnteredShield(Entity* self, Entity* intruder) { + auto* baseCombatAIComponent = intruder->GetComponent<BaseCombatAIComponent>(); + auto* movementAIComponent = intruder->GetComponent<MovementAIComponent>(); + + if (baseCombatAIComponent == nullptr || movementAIComponent == nullptr) { + return; + } + + auto dir = intruder->GetRotation().GetForwardVector() * -1; + dir.y += 15; + dir.x *= 50; + dir.z *= 50; + + // TODO: Figure out how todo knockback, I'll stun them for now + + if (NiPoint3::DistanceSquared(self->GetPosition(), intruder->GetPosition()) < 20 * 20) { + baseCombatAIComponent->Stun(2.0f); + movementAIComponent->SetDestination(baseCombatAIComponent->GetStartPosition()); + } + + baseCombatAIComponent->ClearThreat(); +} diff --git a/dScripts/AmShieldGenerator.h b/dScripts/02_server/Map/AM/AmShieldGenerator.h similarity index 100% rename from dScripts/AmShieldGenerator.h rename to dScripts/02_server/Map/AM/AmShieldGenerator.h diff --git a/dScripts/02_server/Map/AM/AmShieldGeneratorQuickbuild.cpp b/dScripts/02_server/Map/AM/AmShieldGeneratorQuickbuild.cpp new file mode 100644 index 00000000..0c09a5da --- /dev/null +++ b/dScripts/02_server/Map/AM/AmShieldGeneratorQuickbuild.cpp @@ -0,0 +1,203 @@ +#include "AmShieldGeneratorQuickbuild.h" +#include "EntityManager.h" +#include "DestroyableComponent.h" +#include "GameMessages.h" +#include "MovementAIComponent.h" +#include "BaseCombatAIComponent.h" +#include "SkillComponent.h" +#include "EntityInfo.h" +#include "RebuildComponent.h" +#include "MissionComponent.h" + +void AmShieldGeneratorQuickbuild::OnStartup(Entity* self) { + self->SetProximityRadius(20, "shield"); + self->SetProximityRadius(21, "buffer"); +} + +void AmShieldGeneratorQuickbuild::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + auto* destroyableComponent = entering->GetComponent<DestroyableComponent>(); + + if (name == "shield") { + if (!destroyableComponent->HasFaction(4) || entering->IsPlayer()) { + return; + } + + auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); + + if (status == "ENTER") { + EnemyEnteredShield(self, entering); + + const auto& iter = std::find(enemiesInProximity.begin(), enemiesInProximity.end(), entering->GetObjectID()); + + if (iter == enemiesInProximity.end()) { + enemiesInProximity.push_back(entering->GetObjectID()); + } + } else if (status == "LEAVE") { + const auto& iter = std::find(enemiesInProximity.begin(), enemiesInProximity.end(), entering->GetObjectID()); + + if (iter != enemiesInProximity.end()) { + enemiesInProximity.erase(iter); + } + } + + self->SetVar<std::vector<LWOOBJID>>(u"Enemies", enemiesInProximity); + } + + if (name != "buffer" || !entering->IsPlayer()) { + return; + } + + auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + if (status == "ENTER") { + const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); + + if (iter == entitiesInProximity.end()) { + entitiesInProximity.push_back(entering->GetObjectID()); + } + } else if (status == "LEAVE") { + const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); + + if (iter != entitiesInProximity.end()) { + entitiesInProximity.erase(iter); + } + } + + self->SetVar<std::vector<LWOOBJID>>(u"Players", entitiesInProximity); +} + +void AmShieldGeneratorQuickbuild::OnDie(Entity* self, Entity* killer) { + self->CancelAllTimers(); + + auto* child = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Child")); + + if (child != nullptr) { + child->Kill(); + } +} + +void AmShieldGeneratorQuickbuild::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "BuffPlayers") { + BuffPlayers(self); + + self->AddTimer("BuffPlayers", 3.0f); + } else if (timerName == "PlayFX") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 5351, u"generatorOn", "generatorOn"); + + self->AddTimer("PlayFX", 1.5f); + } else if (timerName == "RefreshEnemies") { + auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); + + for (const auto enemyID : enemiesInProximity) { + auto* enemy = Game::entityManager->GetEntity(enemyID); + + if (enemy != nullptr) { + EnemyEnteredShield(self, enemy); + } + } + + self->AddTimer("RefreshEnemies", 1.5f); + } +} + +void AmShieldGeneratorQuickbuild::OnRebuildComplete(Entity* self, Entity* target) { + StartShield(self); + + auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); + + for (const auto enemyID : enemiesInProximity) { + auto* enemy = Game::entityManager->GetEntity(enemyID); + + if (enemy != nullptr) { + enemy->Smash(); + } + } + + auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + for (const auto playerID : entitiesInProximity) { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + continue; + } + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent == nullptr) { + return; + } + + missionComponent->ForceProgressTaskType(987, 1, 1, false); + } +} + +void AmShieldGeneratorQuickbuild::StartShield(Entity* self) { + self->AddTimer("PlayFX", 1.5f); + self->AddTimer("BuffPlayers", 3.0f); + self->AddTimer("RefreshEnemies", 1.5f); + + const auto myPos = self->GetPosition(); + const auto myRot = self->GetRotation(); + + EntityInfo info{}; + info.lot = 13111; + info.pos = myPos; + info.rot = myRot; + info.spawnerID = self->GetObjectID(); + + auto* child = Game::entityManager->CreateEntity(info); + + self->SetVar(u"Child", child->GetObjectID()); + + BuffPlayers(self); +} + +void AmShieldGeneratorQuickbuild::BuffPlayers(Entity* self) { + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + return; + } + + auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + for (const auto playerID : entitiesInProximity) { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + skillComponent->CalculateBehavior(1200, 27024, playerID, true); + } +} + +void AmShieldGeneratorQuickbuild::EnemyEnteredShield(Entity* self, Entity* intruder) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent == nullptr || rebuildComponent->GetState() != eRebuildState::COMPLETED) { + return; + } + + auto* baseCombatAIComponent = intruder->GetComponent<BaseCombatAIComponent>(); + auto* movementAIComponent = intruder->GetComponent<MovementAIComponent>(); + + if (baseCombatAIComponent == nullptr || movementAIComponent == nullptr) { + return; + } + + auto dir = intruder->GetRotation().GetForwardVector() * -1; + dir.y += 15; + dir.x *= 50; + dir.z *= 50; + + // TODO: Figure out how todo knockback, I'll stun them for now + + if (NiPoint3::DistanceSquared(self->GetPosition(), intruder->GetPosition()) < 20 * 20) { + baseCombatAIComponent->Stun(2.0f); + movementAIComponent->SetDestination(baseCombatAIComponent->GetStartPosition()); + } + + baseCombatAIComponent->ClearThreat(); +} diff --git a/dScripts/AmShieldGeneratorQuickbuild.h b/dScripts/02_server/Map/AM/AmShieldGeneratorQuickbuild.h similarity index 100% rename from dScripts/AmShieldGeneratorQuickbuild.h rename to dScripts/02_server/Map/AM/AmShieldGeneratorQuickbuild.h diff --git a/dScripts/02_server/Map/AM/AmSkullkinDrill.cpp b/dScripts/02_server/Map/AM/AmSkullkinDrill.cpp new file mode 100644 index 00000000..576688e7 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmSkullkinDrill.cpp @@ -0,0 +1,326 @@ +#include "AmSkullkinDrill.h" +#include "GameMessages.h" +#include "MovingPlatformComponent.h" +#include "DestroyableComponent.h" +#include "ProximityMonitorComponent.h" +#include "MissionComponent.h" +#include "EntityInfo.h" +#include "RenderComponent.h" +#include "eStateChangeType.h" + +void AmSkullkinDrill::OnStartup(Entity* self) { + self->SetNetworkVar(u"bIsInUse", false); + self->SetVar(u"bActive", true); + + GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"spin", "active"); + + auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); + + if (movingPlatformComponent == nullptr) { + return; + } + + movingPlatformComponent->SetSerialized(true); + + movingPlatformComponent->GotoWaypoint(0); + + auto* standObj = GetStandObj(self); + + if (standObj != nullptr) { + standObj->SetVar(u"bActive", true); + } + + self->SetProximityRadius(5, "spin_distance"); +} + +Entity* AmSkullkinDrill::GetStandObj(Entity* self) { + const auto& myGroup = self->GetGroups(); + + if (myGroup.empty()) { + return nullptr; + } + + std::string groupName = "Drill_Stand_"; + + groupName.push_back(myGroup[0][myGroup[0].size() - 1]); + + const auto standObjs = Game::entityManager->GetEntitiesInGroup(groupName); + + if (standObjs.empty()) { + return nullptr; + } + + return standObjs[0]; +} + +void AmSkullkinDrill::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (message != "NinjagoSpinEvent" || self->GetNetworkVar<bool>(u"bIsInUse")) { + return; + } + + auto* proximityMonitorComponent = self->GetComponent<ProximityMonitorComponent>(); + + if (proximityMonitorComponent == nullptr || !proximityMonitorComponent->IsInProximity("spin_distance", caster->GetObjectID())) { + return; + } + + self->SetVar(u"activaterID", caster->GetObjectID()); + + self->SetNetworkVar(u"bIsInUse", true); + + TriggerDrill(self); +} + +void AmSkullkinDrill::TriggerDrill(Entity* self) { + RenderComponent::PlayAnimation(self, u"slowdown"); + + self->AddTimer("killDrill", 10.0f); + + auto* standObj = GetStandObj(self); + + if (standObj != nullptr) { + standObj->SetVar(u"bActive", false); + } + + auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); + + if (movingPlatformComponent == nullptr) { + return; + } + + movingPlatformComponent->GotoWaypoint(1); +} + +void AmSkullkinDrill::OnWaypointReached(Entity* self, uint32_t waypointIndex) { + if (waypointIndex == 1) { + auto myPos = self->GetPosition(); + auto myRot = self->GetRotation(); + + myPos.y -= 21; + + EntityInfo info = {}; + info.lot = 12346; + info.pos = myPos; + info.rot = myRot; + info.scale = 3; // Needs the scale, otherwise attacks fail + info.spawnerID = self->GetObjectID(); + + auto* child = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(child); + + self->SetVar(u"ChildSmash", child->GetObjectID()); + + child->AddDieCallback([this, self]() { + const auto& userID = self->GetVar<LWOOBJID>(u"activaterID"); + + auto* player = Game::entityManager->GetEntity(userID); + + if (player == nullptr) { + return; + } + + OnHitOrHealResult(self, player, 1); + }); + } + + OnArrived(self, waypointIndex); +} + +void AmSkullkinDrill::OnUse(Entity* self, Entity* user) { + if (self->GetNetworkVar<bool>(u"bIsInUse")) { + return; + } + + self->SetNetworkVar(u"bIsInUse", true); + + GameMessages::SendPlayFXEffect(user->GetObjectID(), 5499, u"on-anim", "tornado"); + GameMessages::SendPlayFXEffect(user->GetObjectID(), 5502, u"on-anim", "staff"); + + const auto userID = user->GetObjectID(); + + self->SetVar(u"userID", userID); + self->SetVar(u"activaterID", userID); + + PlayAnim(self, user, "spinjitzu-staff-windup"); + PlayCinematic(self); + + FreezePlayer(self, user, true); +} + +void AmSkullkinDrill::FreezePlayer(Entity* self, Entity* player, bool bFreeze) { + auto StateChangeType = eStateChangeType::POP; + + if (bFreeze) { + if (player->GetIsDead()) { + return; + } + + StateChangeType = eStateChangeType::PUSH; + } else { + if (player->GetIsDead()) { + // + } + } + + GameMessages::SendSetStunned(player->GetObjectID(), StateChangeType, player->GetSystemAddress(), self->GetObjectID(), + true, false, true, false, true, false, true + ); +} + +void AmSkullkinDrill::OnArrived(Entity* self, uint32_t waypointIndex) { + auto* standObj = GetStandObj(self); + + if (waypointIndex == 1) { + RenderComponent::PlayAnimation(self, u"no-spin"); + GameMessages::SendStopFXEffect(self, true, "active"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"indicator", "indicator"); + + self->SetVar(u"bActive", false); + + const auto playerID = self->GetVar<LWOOBJID>(u"userID"); + + auto* player = Game::entityManager->GetEntity(playerID); + + if (player != nullptr) { + PlayAnim(self, player, "spinjitzu-staff-end"); + } + + if (standObj != nullptr) { + standObj->SetVar(u"bActive", false); + } + + return; + } else { + RenderComponent::PlayAnimation(self, u"idle"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"spin", "active"); + GameMessages::SendStopFXEffect(self, true, "indicator"); + } +} + +void AmSkullkinDrill::PlayCinematic(Entity* self) { + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"userID")); + + if (player == nullptr) { + return; + } + + const auto& cine = self->GetVar<std::u16string>(u"cinematic"); + + if (cine.empty()) { + return; + } + + GameMessages::SendPlayCinematic(player->GetObjectID(), cine, player->GetSystemAddress()); +} + +void AmSkullkinDrill::PlayAnim(Entity* self, Entity* player, const std::string& animName) { + const auto animTime = animName == "spinjitzu-staff-end" ? 0.5f : 1.0f; + + RenderComponent::PlayAnimation(player, animName); + + self->AddTimer("AnimDone_" + animName, animTime); +} + +void AmSkullkinDrill::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + + if (destroyableComponent == nullptr || !attacker->IsPlayer()) { + return; + } + + if (self->GetVar<bool>(u"bActive")) { + return; + } + + const auto activaterID = self->GetVar<LWOOBJID>(u"activaterID"); + + auto* activator = Game::entityManager->GetEntity(activaterID); + + // TODO: Missions + if (activator != nullptr) { + auto* missionComponent = activator->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + for (const auto missionID : m_MissionsToUpdate) { + missionComponent->ForceProgressValue(missionID, 1, self->GetLOT()); + } + } + } + + self->Smash(attacker->GetObjectID(), eKillType::SILENT); + + self->CancelAllTimers(); + + auto* standObj = GetStandObj(self); + + if (standObj != nullptr) { + GameMessages::SendPlayFXEffect(standObj->GetObjectID(), 4946, u"explode", "explode"); + } +} + +void AmSkullkinDrill::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "killDrill") { + const auto childID = self->GetVar<LWOOBJID>(u"ChildSmash"); + + auto* child = Game::entityManager->GetEntity(childID); + + if (child != nullptr) { + child->Smash(self->GetObjectID(), eKillType::SILENT); + } + + self->SetNetworkVar(u"bIsInUse", false); + self->SetVar(u"bActive", true); + self->SetVar(u"activaterID", LWOOBJID_EMPTY); + + auto* standObj = GetStandObj(self); + + if (standObj != nullptr) { + standObj->SetVar(u"bActive", true); + } + + auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); + + if (movingPlatformComponent == nullptr) { + return; + } + + movingPlatformComponent->GotoWaypoint(0); + + return; + } + + const auto& data = GeneralUtils::SplitString(timerName, '_'); + + if (data.empty()) { + return; + } + + if (data[0] == "AnimDone") { + const auto& animName = data[1]; + + const auto playerID = self->GetVar<LWOOBJID>(u"userID"); + + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + if (animName == "spinjitzu-staff-windup") { + TriggerDrill(self); + + RenderComponent::PlayAnimation(player, u"spinjitzu-staff-loop"); + } else if (animName == "spinjitzu-staff-end") { + FreezePlayer(self, player, false); + + self->SetVar(u"userID", LWOOBJID_EMPTY); + + GameMessages::SendStopFXEffect(player, true, "tornado"); + GameMessages::SendStopFXEffect(player, true, "staff"); + } + + } else if (data[0] == "TryUnFreezeAgain") { + + } +} diff --git a/dScripts/AmSkullkinDrill.h b/dScripts/02_server/Map/AM/AmSkullkinDrill.h similarity index 100% rename from dScripts/AmSkullkinDrill.h rename to dScripts/02_server/Map/AM/AmSkullkinDrill.h diff --git a/dScripts/02_server/Map/AM/AmSkullkinDrillStand.cpp b/dScripts/02_server/Map/AM/AmSkullkinDrillStand.cpp new file mode 100644 index 00000000..5493dc24 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmSkullkinDrillStand.cpp @@ -0,0 +1,37 @@ +#include "AmSkullkinDrillStand.h" +#include "GameMessages.h" +#include "dpEntity.h" +#include "Entity.h" +#include "RenderComponent.h" + +void AmSkullkinDrillStand::OnStartup(Entity* self) { + self->SetVar(u"bActive", true); + + self->SetProximityRadius(new dpEntity(self->GetObjectID(), { 6, 14, 6 }), "knockback"); +} + +void AmSkullkinDrillStand::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { + +} + +void AmSkullkinDrillStand::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (!self->GetVar<bool>(u"bActive")) { + return; + } + + if (!entering->IsPlayer() || status != "ENTER" || name != "knockback") { + return; + } + + auto myPos = self->GetPosition(); + + auto objPos = entering->GetPosition(); + + NiPoint3 newVec = { (objPos.x - myPos.x) * 4.5f, 15, (objPos.z - myPos.z) * 4.5f }; + + GameMessages::SendKnockback(entering->GetObjectID(), self->GetObjectID(), self->GetObjectID(), 0, newVec); + + GameMessages::SendPlayFXEffect(entering->GetObjectID(), 1378, u"create", "pushBack"); + + RenderComponent::PlayAnimation(entering, u"knockback-recovery"); +} diff --git a/dScripts/AmSkullkinDrillStand.h b/dScripts/02_server/Map/AM/AmSkullkinDrillStand.h similarity index 100% rename from dScripts/AmSkullkinDrillStand.h rename to dScripts/02_server/Map/AM/AmSkullkinDrillStand.h diff --git a/dScripts/02_server/Map/AM/AmSkullkinTower.cpp b/dScripts/02_server/Map/AM/AmSkullkinTower.cpp new file mode 100644 index 00000000..0c0f7515 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmSkullkinTower.cpp @@ -0,0 +1,245 @@ +#include "AmSkullkinTower.h" +#include "EntityManager.h" +#include "DestroyableComponent.h" +#include "MovingPlatformComponent.h" +#include "EntityInfo.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "RenderComponent.h" + +void AmSkullkinTower::OnStartup(Entity* self) { + self->SetProximityRadius(20, "Tower"); + + // onPhysicsComponentReady + + auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); + + if (movingPlatformComponent != nullptr) { + movingPlatformComponent->StopPathing(); + } + + SpawnLegs(self, "Left"); + SpawnLegs(self, "Right"); + SpawnLegs(self, "Rear"); +} + +void AmSkullkinTower::SpawnLegs(Entity* self, const std::string& loc) { + auto pos = self->GetPosition(); + auto rot = self->GetRotation(); + pos.y += self->GetVarAs<float>(u"vert_offset"); + + auto newRot = rot; + auto offset = self->GetVarAs<float>(u"hort_offset"); + + auto legLOT = self->GetVar<LOT>(u"legLOT"); + + if (legLOT == 0) { + return; + } + + std::vector<LDFBaseData*> config = { new LDFData<std::string>(u"Leg", loc) }; + + EntityInfo info{}; + info.lot = legLOT; + info.spawnerID = self->GetObjectID(); + info.settings = config; + info.rot = newRot; + + if (loc == "Right") { + const auto dir = rot.GetForwardVector(); + pos.x += dir.x * offset; + pos.z += dir.z * offset; + info.pos = pos; + } else if (loc == "Rear") { + const auto dir = rot.GetRightVector(); + pos.x += dir.x * offset; + pos.z += dir.z * offset; + info.pos = pos; + } else if (loc == "Left") { + const auto dir = rot.GetForwardVector() * -1; + pos.x += dir.x * offset; + pos.z += dir.z * offset; + info.pos = pos; + } + + info.rot = NiQuaternion::LookAt(info.pos, self->GetPosition()); + + auto* entity = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(entity); + + OnChildLoaded(self, entity); +} + +void AmSkullkinTower::OnChildLoaded(Entity* self, Entity* child) { + auto legTable = self->GetVar<std::vector<LWOOBJID>>(u"legTable"); + + legTable.push_back(child->GetObjectID()); + + self->SetVar(u"legTable", legTable); + + const auto selfID = self->GetObjectID(); + + child->AddDieCallback([this, selfID, child]() { + auto* self = Game::entityManager->GetEntity(selfID); + auto* destroyableComponent = child->GetComponent<DestroyableComponent>(); + + if (destroyableComponent == nullptr || self == nullptr) { + return; + } + + NotifyDie(self, child, destroyableComponent->GetKiller()); + }); +} + +void AmSkullkinTower::NotifyDie(Entity* self, Entity* other, Entity* killer) { + auto players = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + const auto& iter = std::find(players.begin(), players.end(), killer->GetObjectID()); + + if (iter == players.end()) { + players.push_back(killer->GetObjectID()); + } + + self->SetVar(u"Players", players); + + OnChildRemoved(self, other); +} + +void AmSkullkinTower::OnChildRemoved(Entity* self, Entity* child) { + auto legTable = self->GetVar<std::vector<LWOOBJID>>(u"legTable"); + + const auto& iter = std::find(legTable.begin(), legTable.end(), child->GetObjectID()); + + if (iter != legTable.end()) { + legTable.erase(iter); + } + + self->SetVar(u"legTable", legTable); + + if (legTable.size() == 2) { + RenderComponent::PlayAnimation(self, u"wobble-1"); + } else if (legTable.size() == 1) { + RenderComponent::PlayAnimation(self, u"wobble-2"); + } else if (legTable.empty()) { + const auto animTime = 2.5f; + + RenderComponent::PlayAnimation(self, u"fall"); + + self->AddTimer("spawnGuys", animTime - 0.2f); + + self->CancelTimer("RespawnLeg"); + self->CancelTimer("RespawnLeg"); + self->CancelTimer("RespawnLeg"); + + std::vector<int32_t> missionIDs; + + auto missionsString = self->GetVar<std::u16string>(u"missions"); + + if (!missionsString.empty()) { + // Split the missions string by '_' + const auto missions = GeneralUtils::SplitString( + GeneralUtils::UTF16ToWTF8(missionsString), + '_' + ); + + for (const auto& mission : missions) { + int32_t missionID = 0; + + if (!GeneralUtils::TryParse(mission, missionID)) { + continue; + } + + missionIDs.push_back(missionID); + } + } + + const auto& players = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + for (const auto& playerID : players) { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + continue; + } + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent == nullptr) { + continue; + } + + for (const auto missionID : missionIDs) { + missionComponent->ForceProgressValue(missionID, 1, self->GetLOT()); + } + + //missionComponent->ForceProgressValue(1305, 1, self->GetLOT()); + } + } + + auto deadLegs = self->GetVar<std::vector<std::string>>(u"DeadLegs"); + + const auto& leg = child->GetVar<std::string>(u"Leg"); + + const auto& legIter = std::find(deadLegs.begin(), deadLegs.end(), leg); + + if (legIter == deadLegs.end()) { + deadLegs.push_back(leg); + } + + self->SetVar(u"DeadLegs", deadLegs); + + self->AddTimer("RespawnLeg", 20); +} + +void AmSkullkinTower::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (status != "LEAVE") { + return; + } + + auto players = self->GetVar<std::vector<LWOOBJID>>(u"Players"); + + const auto& iter = std::find(players.begin(), players.end(), entering->GetObjectID()); + + if (iter != players.end()) { + players.erase(iter); + } + + self->SetVar(u"Players", players); +} + +void AmSkullkinTower::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "RespawnLeg") { + auto deadLegs = self->GetVar<std::vector<std::string>>(u"DeadLegs"); + + if (deadLegs.empty()) { + return; + } + + SpawnLegs(self, deadLegs[0]); + + deadLegs.erase(deadLegs.begin()); + + self->SetVar<std::vector<std::string>>(u"DeadLegs", deadLegs); + } else if (timerName == "spawnGuys") { + EntityInfo info{}; + info.lot = self->GetVar<LOT>(u"enemyToSpawn"); + auto pos = self->GetPosition(); + pos.y += 7; + info.pos = pos; + info.rot = self->GetRotation(); + info.spawnerID = self->GetObjectID(); + + for (size_t i = 0; i < 2; i++) { + info.pos.x += i * 2; // Just to set the apart a bit + + auto* entity = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(entity); + } + + self->AddTimer("killTower", 0.7f); + } else if (timerName == "killTower") { + self->Smash(self->GetObjectID()); + } +} diff --git a/dScripts/AmSkullkinTower.h b/dScripts/02_server/Map/AM/AmSkullkinTower.h similarity index 100% rename from dScripts/AmSkullkinTower.h rename to dScripts/02_server/Map/AM/AmSkullkinTower.h diff --git a/dScripts/02_server/Map/AM/AmTeapotServer.cpp b/dScripts/02_server/Map/AM/AmTeapotServer.cpp new file mode 100644 index 00000000..93f05326 --- /dev/null +++ b/dScripts/02_server/Map/AM/AmTeapotServer.cpp @@ -0,0 +1,23 @@ +#include "AmTeapotServer.h" +#include "InventoryComponent.h" +#include "GameMessages.h" +#include "Item.h" +#include "eTerminateType.h" + +void AmTeapotServer::OnUse(Entity* self, Entity* user) { + auto* inventoryComponent = user->GetComponent<InventoryComponent>(); + if (!inventoryComponent) return; + + auto* blueFlowerItem = inventoryComponent->FindItemByLot(BLUE_FLOWER_LEAVES, eInventoryType::ITEMS); + if (!blueFlowerItem) { + blueFlowerItem = inventoryComponent->FindItemByLot(BLUE_FLOWER_LEAVES, eInventoryType::VAULT_ITEMS); + if (!blueFlowerItem) return; + } + + // The client allows you to use the teapot only if you have a stack of 10 leaves in some inventory somewhere. + if (blueFlowerItem->GetCount() >= 10) { + blueFlowerItem->SetCount(blueFlowerItem->GetCount() - 10); + inventoryComponent->AddItem(WU_S_IMAGINATION_TEA, 1); + } + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} diff --git a/dScripts/AmTeapotServer.h b/dScripts/02_server/Map/AM/AmTeapotServer.h similarity index 100% rename from dScripts/AmTeapotServer.h rename to dScripts/02_server/Map/AM/AmTeapotServer.h diff --git a/dScripts/AmTemplateSkillVolume.cpp b/dScripts/02_server/Map/AM/AmTemplateSkillVolume.cpp similarity index 100% rename from dScripts/AmTemplateSkillVolume.cpp rename to dScripts/02_server/Map/AM/AmTemplateSkillVolume.cpp diff --git a/dScripts/AmTemplateSkillVolume.h b/dScripts/02_server/Map/AM/AmTemplateSkillVolume.h similarity index 100% rename from dScripts/AmTemplateSkillVolume.h rename to dScripts/02_server/Map/AM/AmTemplateSkillVolume.h diff --git a/dScripts/02_server/Map/AM/CMakeLists.txt b/dScripts/02_server/Map/AM/CMakeLists.txt new file mode 100644 index 00000000..d3d13b73 --- /dev/null +++ b/dScripts/02_server/Map/AM/CMakeLists.txt @@ -0,0 +1,19 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_AM + "AmConsoleTeleportServer.cpp" + "RandomSpawnerFin.cpp" + "RandomSpawnerPit.cpp" + "RandomSpawnerStr.cpp" + "RandomSpawnerZip.cpp" + "AmBridge.cpp" + "AmDrawBridge.cpp" + "AmShieldGenerator.cpp" + "AmShieldGeneratorQuickbuild.cpp" + "AmDropshipComputer.cpp" + "AmScrollReaderServer.cpp" + "AmTemplateSkillVolume.cpp" + "AmSkullkinDrill.cpp" + "AmSkullkinDrillStand.cpp" + "AmSkullkinTower.cpp" + "AmBlueX.cpp" + "AmTeapotServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/RandomSpawnerFin.cpp b/dScripts/02_server/Map/AM/RandomSpawnerFin.cpp similarity index 100% rename from dScripts/RandomSpawnerFin.cpp rename to dScripts/02_server/Map/AM/RandomSpawnerFin.cpp diff --git a/dScripts/RandomSpawnerFin.h b/dScripts/02_server/Map/AM/RandomSpawnerFin.h similarity index 100% rename from dScripts/RandomSpawnerFin.h rename to dScripts/02_server/Map/AM/RandomSpawnerFin.h diff --git a/dScripts/RandomSpawnerPit.cpp b/dScripts/02_server/Map/AM/RandomSpawnerPit.cpp similarity index 100% rename from dScripts/RandomSpawnerPit.cpp rename to dScripts/02_server/Map/AM/RandomSpawnerPit.cpp diff --git a/dScripts/RandomSpawnerPit.h b/dScripts/02_server/Map/AM/RandomSpawnerPit.h similarity index 100% rename from dScripts/RandomSpawnerPit.h rename to dScripts/02_server/Map/AM/RandomSpawnerPit.h diff --git a/dScripts/RandomSpawnerStr.cpp b/dScripts/02_server/Map/AM/RandomSpawnerStr.cpp similarity index 100% rename from dScripts/RandomSpawnerStr.cpp rename to dScripts/02_server/Map/AM/RandomSpawnerStr.cpp diff --git a/dScripts/RandomSpawnerStr.h b/dScripts/02_server/Map/AM/RandomSpawnerStr.h similarity index 100% rename from dScripts/RandomSpawnerStr.h rename to dScripts/02_server/Map/AM/RandomSpawnerStr.h diff --git a/dScripts/RandomSpawnerZip.cpp b/dScripts/02_server/Map/AM/RandomSpawnerZip.cpp similarity index 100% rename from dScripts/RandomSpawnerZip.cpp rename to dScripts/02_server/Map/AM/RandomSpawnerZip.cpp diff --git a/dScripts/RandomSpawnerZip.h b/dScripts/02_server/Map/AM/RandomSpawnerZip.h similarity index 100% rename from dScripts/RandomSpawnerZip.h rename to dScripts/02_server/Map/AM/RandomSpawnerZip.h diff --git a/dScripts/02_server/Map/CMakeLists.txt b/dScripts/02_server/Map/CMakeLists.txt new file mode 100644 index 00000000..feed8a97 --- /dev/null +++ b/dScripts/02_server/Map/CMakeLists.txt @@ -0,0 +1,81 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP) + +add_subdirectory(AG) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_AG}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "AG/${file}") +endforeach() + +add_subdirectory(AG_Spider_Queen) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_AG_SPIDER_QUEEN}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "AG_Spider_Queen/${file}") +endforeach() + +add_subdirectory(AM) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_AM}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "AM/${file}") +endforeach() + +add_subdirectory(FV) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_FV}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "FV/${file}") +endforeach() + +add_subdirectory(General) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "General/${file}") +endforeach() + +add_subdirectory(GF) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_GF}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "GF/${file}") +endforeach() + +add_subdirectory(njhub) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "njhub/${file}") +endforeach() + +add_subdirectory(NS) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_NS}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "NS/${file}") +endforeach() + +add_subdirectory(NT) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_NT}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "NT/${file}") +endforeach() + +add_subdirectory(PR) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_PR}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "PR/${file}") +endforeach() + +add_subdirectory(Property) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "Property/${file}") +endforeach() + +add_subdirectory(SS) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_SS}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "SS/${file}") +endforeach() + +add_subdirectory(VE) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_VE}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} "VE/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MAP ${DSCRIPTS_SOURCES_02_SERVER_MAP} PARENT_SCOPE) diff --git a/dScripts/02_server/Map/FV/CMakeLists.txt b/dScripts/02_server/Map/FV/CMakeLists.txt new file mode 100644 index 00000000..505d97c4 --- /dev/null +++ b/dScripts/02_server/Map/FV/CMakeLists.txt @@ -0,0 +1,14 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_FV + "EnemyRoninSpawner.cpp" + "FvCandle.cpp" + "FvFong.cpp" + "FvHorsemenTrigger.cpp" + "ImgBrickConsoleQB.cpp") + +add_subdirectory(Racing) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_FV_RACING}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_FV ${DSCRIPTS_SOURCES_02_SERVER_MAP_FV} "Racing/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MAP_FV ${DSCRIPTS_SOURCES_02_SERVER_MAP_FV} PARENT_SCOPE) diff --git a/dScripts/02_server/Map/FV/EnemyRoninSpawner.cpp b/dScripts/02_server/Map/FV/EnemyRoninSpawner.cpp new file mode 100644 index 00000000..ddaafc65 --- /dev/null +++ b/dScripts/02_server/Map/FV/EnemyRoninSpawner.cpp @@ -0,0 +1,69 @@ +#include "EnemyRoninSpawner.h" +#include "SkillComponent.h" +#include "RenderComponent.h" +#include "EntityInfo.h" +#include "EntityManager.h" + +void EnemyRoninSpawner::OnStartup(Entity* self) { + self->SetProximityRadius(15, "ronin"); +} + +void EnemyRoninSpawner::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "hatchTime") { + auto* renderComponent = self->GetComponent<RenderComponent>(); + + if (renderComponent != nullptr) { + renderComponent->PlayEffect(644, u"create", "BurstFX1"); + } + + EntityInfo info{}; + info.lot = 7815; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.spawnerID = self->GetObjectID(); + + auto* spawnedEntity = Game::entityManager->CreateEntity(info); + + if (spawnedEntity == nullptr) { + return; + } + + Game::entityManager->ConstructEntity(spawnedEntity); + + spawnedEntity->AddCallbackTimer(60, [spawnedEntity]() { + spawnedEntity->Smash(spawnedEntity->GetObjectID()); + }); + + self->Smash(self->GetObjectID()); + } +} + +void EnemyRoninSpawner::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (entering->IsPlayer() && name == "ronin" && status == "ENTER" && !self->GetVar<bool>(u"hatching")) { + StartHatching(self); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(305, 3568, LWOOBJID_EMPTY); + } + } +} + +void EnemyRoninSpawner::OnHit(Entity* self, Entity* attacker) { + if (!self->GetVar<bool>(u"hatching")) { + StartHatching(self); + } +} + +void EnemyRoninSpawner::StartHatching(Entity* self) { + self->SetVar(u"hatching", true); + + auto* renderComponent = self->GetComponent<RenderComponent>(); + + if (renderComponent != nullptr) { + renderComponent->PlayEffect(2260, u"rebuild_medium", "WakeUpFX1"); + } + + self->AddTimer("hatchTime", 2); +} diff --git a/dScripts/EnemyRoninSpawner.h b/dScripts/02_server/Map/FV/EnemyRoninSpawner.h similarity index 100% rename from dScripts/EnemyRoninSpawner.h rename to dScripts/02_server/Map/FV/EnemyRoninSpawner.h diff --git a/dScripts/02_server/Map/FV/FvCandle.cpp b/dScripts/02_server/Map/FV/FvCandle.cpp new file mode 100644 index 00000000..0c4344d0 --- /dev/null +++ b/dScripts/02_server/Map/FV/FvCandle.cpp @@ -0,0 +1,56 @@ +#include "FvCandle.h" +#include "MissionComponent.h" +#include "RenderComponent.h" +#include "eReplicaComponentType.h" + +std::vector<int32_t> FvCandle::m_Missions = { 850, 1431, 1529, 1566, 1603 }; + +void FvCandle::OnStartup(Entity* self) { + auto* render = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + if (render == nullptr) + return; + + render->PlayEffect(2108, u"create", "candle_light", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + self->SetI32(u"Smoke", static_cast<int32_t>(5)); + self->SetBoolean(u"AmHit", false); +} + +void FvCandle::OnHit(Entity* self, Entity* attacker) { + BlowOutCandle(self, attacker); +} + +void FvCandle::BlowOutCandle(Entity* self, Entity* blower) { + if (self->GetBoolean(u"AmHit")) + return; + + auto* render = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + if (render == nullptr) + return; + + auto* missionComponent = blower->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + for (const auto mission : m_Missions) { + missionComponent->ForceProgressTaskType(mission, 1, 1); + } + } + + //Update mission tasks here + self->SetBoolean(u"AmHit", true); + + render->StopEffect("candle_light", false); + render->PlayEffect(2109, u"create", "candle_smoke", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + + self->AddTimer("SmokeTime", self->GetI32(u"Smoke")); +} + +void FvCandle::OnTimerDone(Entity* self, std::string timerName) { + self->SetBoolean(u"AmHit", false); + + auto* render = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + if (render == nullptr) + return; + + render->StopEffect("candle_smoke", false); + render->PlayEffect(2108, u"create", "candle_light", LWOOBJID_EMPTY, 1.0f, 1.0f, true); +} diff --git a/dScripts/FvCandle.h b/dScripts/02_server/Map/FV/FvCandle.h similarity index 100% rename from dScripts/FvCandle.h rename to dScripts/02_server/Map/FV/FvCandle.h diff --git a/dScripts/02_server/Map/FV/FvFong.cpp b/dScripts/02_server/Map/FV/FvFong.cpp new file mode 100644 index 00000000..13c637e7 --- /dev/null +++ b/dScripts/02_server/Map/FV/FvFong.cpp @@ -0,0 +1,10 @@ +#include "FvFong.h" +#include "Darkitect.h" +#include "eMissionState.h" + +void FvFong::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID == 734 && missionState == eMissionState::READY_TO_COMPLETE) { + Darkitect Baron; + Baron.Reveal(self, target); + } +} diff --git a/dScripts/02_server/Map/FV/FvFong.h b/dScripts/02_server/Map/FV/FvFong.h new file mode 100644 index 00000000..c0b6d48b --- /dev/null +++ b/dScripts/02_server/Map/FV/FvFong.h @@ -0,0 +1,8 @@ +#pragma once +#include "CppScripts.h" + +class FvFong : public CppScripts::Script +{ +public: + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +}; diff --git a/dScripts/02_server/Map/FV/FvHorsemenTrigger.cpp b/dScripts/02_server/Map/FV/FvHorsemenTrigger.cpp new file mode 100644 index 00000000..04bf94c5 --- /dev/null +++ b/dScripts/02_server/Map/FV/FvHorsemenTrigger.cpp @@ -0,0 +1,53 @@ +#include "FvHorsemenTrigger.h" +#include "EntityManager.h" +#include "MissionComponent.h" + +void FvHorsemenTrigger::OnStartup(Entity* self) { + self->SetProximityRadius(40, "horsemenTrigger"); + + self->SetVar<std::vector<LWOOBJID>>(u"players", {}); +} + +void FvHorsemenTrigger::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name != "horsemenTrigger" || !entering->IsPlayer()) { + return; + } + + auto players = self->GetVar<std::vector<LWOOBJID>>(u"players"); + + const auto& iter = std::find(players.begin(), players.end(), entering->GetObjectID()); + + if (status == "ENTER" && iter == players.end()) { + players.push_back(entering->GetObjectID()); + } else if (status == "LEAVE" && iter != players.end()) { + players.erase(iter); + } + + self->SetVar<std::vector<LWOOBJID>>(u"players", players); +} + +void +FvHorsemenTrigger::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + auto players = self->GetVar<std::vector<LWOOBJID>>(u"players"); + + if (args == "HorsemenDeath") { + for (const auto& playerId : self->GetVar<std::vector<LWOOBJID>>(u"players")) { + auto* player = Game::entityManager->GetEntity(playerId); + + if (player == nullptr) { + continue; + } + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent == nullptr) { + continue; + } + + for (const auto missionId : m_Missions) { + missionComponent->ForceProgressTaskType(missionId, 1, 1); + } + } + } +} diff --git a/dScripts/FvHorsemenTrigger.h b/dScripts/02_server/Map/FV/FvHorsemenTrigger.h similarity index 100% rename from dScripts/FvHorsemenTrigger.h rename to dScripts/02_server/Map/FV/FvHorsemenTrigger.h diff --git a/dScripts/02_server/Map/FV/ImgBrickConsoleQB.cpp b/dScripts/02_server/Map/FV/ImgBrickConsoleQB.cpp new file mode 100644 index 00000000..cf065ad6 --- /dev/null +++ b/dScripts/02_server/Map/FV/ImgBrickConsoleQB.cpp @@ -0,0 +1,242 @@ +#include "ImgBrickConsoleQB.h" +#include "RebuildComponent.h" +#include "dZoneManager.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "InventoryComponent.h" +#include "eTerminateType.h" + +int32_t ImgBrickConsoleQB::ResetBricks = 30; +int32_t ImgBrickConsoleQB::ResetConsole = 60; +int32_t ImgBrickConsoleQB::ResetInteract = 45; + +void ImgBrickConsoleQB::OnStartup(Entity* self) { + self->SetNetworkVar(u"used", false); + + self->AddTimer("reset", ResetBricks); +} + +void ImgBrickConsoleQB::OnUse(Entity* self, Entity* user) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent->GetState() == eRebuildState::COMPLETED) { + if (!self->GetNetworkVar<bool>(u"used")) { + const auto consoles = Game::entityManager->GetEntitiesInGroup("Console"); + + auto bothBuilt = false; + + for (auto* console : consoles) { + auto* consoleRebuildComponent = console->GetComponent<RebuildComponent>(); + + if (consoleRebuildComponent->GetState() != eRebuildState::COMPLETED) { + continue; + } + + console->CancelAllTimers(); + + if (console->GetNetworkVar<bool>(u"used")) { + bothBuilt = true; + } + } + + if (bothBuilt) { + SmashCanister(self); + } else { + SpawnBrick(self); + } + + self->AddTimer("Die", ResetInteract); + + auto onFX = 0; + + const auto location = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"console")); + + if (location == "Left") { + onFX = 2776; + } else { + onFX = 2779; + } + + const auto& facility = Game::entityManager->GetEntitiesInGroup("FacilityPipes"); + + if (!facility.empty()) { + GameMessages::SendStopFXEffect(facility[0], true, location + "PipeEnergy"); + GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), onFX, u"create", location + "PipeOn"); + } + } + + auto* player = user; + + auto* missionComponent = player->GetComponent<MissionComponent>(); + auto* inventoryComponent = player->GetComponent<InventoryComponent>(); + + if (missionComponent != nullptr && inventoryComponent != nullptr) { + if (missionComponent->GetMissionState(1302) == eMissionState::ACTIVE) { + inventoryComponent->RemoveItem(13074, 1); + + missionComponent->ForceProgressTaskType(1302, 1, 1); + } + + if (missionComponent->GetMissionState(1926) == eMissionState::ACTIVE) { + inventoryComponent->RemoveItem(14472, 1); + + missionComponent->ForceProgressTaskType(1926, 1, 1); + } + } + + self->SetNetworkVar(u"used", true); + + GameMessages::SendTerminateInteraction(player->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); + } +} + +void ImgBrickConsoleQB::SpawnBrick(Entity* self) { + const auto netDevil = Game::zoneManager->GetSpawnersByName("MaelstromBug"); + if (!netDevil.empty()) { + netDevil[0]->Reset(); + netDevil[0]->Deactivate(); + } + + const auto brick = Game::zoneManager->GetSpawnersByName("Imagination"); + if (!brick.empty()) { + brick[0]->Activate(); + } +} + +void ImgBrickConsoleQB::SmashCanister(Entity* self) { + const auto brick = Game::entityManager->GetEntitiesInGroup("Imagination"); + if (!brick.empty()) { + GameMessages::SendPlayFXEffect(brick[0]->GetObjectID(), 122, u"create", "bluebrick"); + GameMessages::SendPlayFXEffect(brick[0]->GetObjectID(), 1034, u"cast", "imaginationexplosion"); + } + + const auto canisters = Game::entityManager->GetEntitiesInGroup("Canister"); + for (auto* canister : canisters) { + canister->Smash(canister->GetObjectID(), eKillType::VIOLENT); + } + + const auto canister = Game::zoneManager->GetSpawnersByName("BrickCanister"); + if (!canister.empty()) { + canister[0]->Reset(); + canister[0]->Deactivate(); + } +} + +void ImgBrickConsoleQB::OnRebuildComplete(Entity* self, Entity* target) { + auto energyFX = 0; + + const auto location = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"console")); + + if (location == "Left") { + energyFX = 2775; + } else { + energyFX = 2778; + } + + const auto& facility = Game::entityManager->GetEntitiesInGroup("FacilityPipes"); + + if (!facility.empty()) { + GameMessages::SendStopFXEffect(facility[0], true, location + "PipeOff"); + GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), energyFX, u"create", location + "PipeEnergy"); + } + + const auto consoles = Game::entityManager->GetEntitiesInGroup("Console"); + + for (auto* console : consoles) { + auto* consoleRebuildComponent = console->GetComponent<RebuildComponent>(); + + if (consoleRebuildComponent->GetState() != eRebuildState::COMPLETED) { + continue; + } + + console->CancelAllTimers(); + } + + self->AddTimer("Die", ResetConsole); +} + +void ImgBrickConsoleQB::OnDie(Entity* self, Entity* killer) { + if (self->GetVar<bool>(u"Died")) { + return; + } + + self->CancelAllTimers(); + + self->SetVar(u"Died", true); + + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent->GetState() == eRebuildState::COMPLETED) { + auto offFX = 0; + + const auto location = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"console")); + + if (location == "Left") { + offFX = 2774; + } else { + offFX = 2777; + } + + const auto& facility = Game::entityManager->GetEntitiesInGroup("FacilityPipes"); + + if (!facility.empty()) { + GameMessages::SendStopFXEffect(facility[0], true, location + "PipeEnergy"); + GameMessages::SendStopFXEffect(facility[0], true, location + "PipeOn"); + GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), offFX, u"create", location + "PipeOff"); + GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), 2750, u"create", location + "imagination_canister"); + } + } + + const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); + + const auto pipeGroup = myGroup.substr(0, 10); + + const auto firstPipe = pipeGroup + "1"; + + const auto samePipeSpawner = Game::zoneManager->GetSpawnersByName(myGroup); + if (!samePipeSpawner.empty()) { + samePipeSpawner[0]->Reset(); + samePipeSpawner[0]->Deactivate(); + } + + const auto firstPipeSpawner = Game::zoneManager->GetSpawnersByName(firstPipe); + if (!firstPipeSpawner.empty()) { + firstPipeSpawner[0]->Activate(); + } + + const auto netdevil = Game::zoneManager->GetSpawnersByName("Imagination"); + if (!netdevil.empty()) { + netdevil[0]->Reset(); + netdevil[0]->Deactivate(); + } + + const auto brick = Game::zoneManager->GetSpawnersByName("MaelstromBug"); + if (!brick.empty()) { + brick[0]->Activate(); + } + + const auto canister = Game::zoneManager->GetSpawnersByName("BrickCanister"); + if (!canister.empty()) { + canister[0]->Activate(); + } + + self->SetNetworkVar(u"used", false); +} + +void ImgBrickConsoleQB::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "reset") { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent->GetState() == eRebuildState::OPEN) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } + } else if (timerName == "Die") { + const auto consoles = Game::entityManager->GetEntitiesInGroup("Console"); + + for (auto* console : consoles) { + console->Smash(console->GetObjectID(), eKillType::VIOLENT); + } + } +} diff --git a/dScripts/ImgBrickConsoleQB.h b/dScripts/02_server/Map/FV/ImgBrickConsoleQB.h similarity index 100% rename from dScripts/ImgBrickConsoleQB.h rename to dScripts/02_server/Map/FV/ImgBrickConsoleQB.h diff --git a/dScripts/02_server/Map/FV/Racing/CMakeLists.txt b/dScripts/02_server/Map/FV/Racing/CMakeLists.txt new file mode 100644 index 00000000..89536b67 --- /dev/null +++ b/dScripts/02_server/Map/FV/Racing/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_FV_RACING + "RaceMaelstromGeiser.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/FV/Racing/RaceMaelstromGeiser.cpp b/dScripts/02_server/Map/FV/Racing/RaceMaelstromGeiser.cpp new file mode 100644 index 00000000..48d1b255 --- /dev/null +++ b/dScripts/02_server/Map/FV/Racing/RaceMaelstromGeiser.cpp @@ -0,0 +1,85 @@ +#include "RaceMaelstromGeiser.h" +#include "GameMessages.h" +#include "PossessableComponent.h" +#include "PossessorComponent.h" +#include "EntityManager.h" +#include "RacingControlComponent.h" +#include "dZoneManager.h" + +void RaceMaelstromGeiser::OnStartup(Entity* self) { + self->SetVar(u"AmFiring", false); + + self->AddTimer("downTime", self->GetVar<int32_t>(u"startTime")); + + self->SetProximityRadius(15, "deathZone"); +} + +void RaceMaelstromGeiser::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (!entering->IsPlayer() || name != "deathZone" || status != "ENTER") { + return; + } + + if (!self->GetVar<bool>(u"AmFiring")) { + return; + } + + auto* possessableComponent = entering->GetComponent<PossessableComponent>(); + + Entity* vehicle; + Entity* player; + + if (possessableComponent != nullptr) { + player = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); + + if (player == nullptr) { + return; + } + + vehicle = entering; + } else if (entering->IsPlayer()) { + auto* possessorComponent = entering->GetComponent<PossessorComponent>(); + + if (possessorComponent == nullptr) { + return; + } + + vehicle = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); + + if (vehicle == nullptr) { + return; + } + + player = entering; + } else { + return; + } + + + GameMessages::SendDie(vehicle, self->GetObjectID(), LWOOBJID_EMPTY, true, eKillType::VIOLENT, u"", 0, 0, 0, true, false, 0); + + auto* zoneController = Game::zoneManager->GetZoneControlObject(); + + auto* racingControlComponent = zoneController->GetComponent<RacingControlComponent>(); + + if (racingControlComponent != nullptr) { + racingControlComponent->OnRequestDie(player); + } +} + +void RaceMaelstromGeiser::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "downTime") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 4048, u"rebuild_medium", "geiser", LWOOBJID_EMPTY, 1, 1, true); + + self->AddTimer("buildUpTime", 1); + } else if (timerName == "buildUpTime") { + self->SetVar(u"AmFiring", true); + + self->AddTimer("killTime", 1.5f); + } else if (timerName == "killTime") { + GameMessages::SendStopFXEffect(self, true, "geiser"); + + self->SetVar(u"AmFiring", false); + + self->AddTimer("downTime", 3.0); + } +} diff --git a/dScripts/RaceMaelstromGeiser.h b/dScripts/02_server/Map/FV/Racing/RaceMaelstromGeiser.h similarity index 100% rename from dScripts/RaceMaelstromGeiser.h rename to dScripts/02_server/Map/FV/Racing/RaceMaelstromGeiser.h diff --git a/dScripts/02_server/Map/GF/CMakeLists.txt b/dScripts/02_server/Map/GF/CMakeLists.txt new file mode 100644 index 00000000..90cc38a5 --- /dev/null +++ b/dScripts/02_server/Map/GF/CMakeLists.txt @@ -0,0 +1,6 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_GF + "GfTikiTorch.cpp" + "GfCaptainsCannon.cpp" + "MastTeleport.cpp" + "SpawnLionServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/GF/GfCaptainsCannon.cpp b/dScripts/02_server/Map/GF/GfCaptainsCannon.cpp new file mode 100644 index 00000000..242bfe4f --- /dev/null +++ b/dScripts/02_server/Map/GF/GfCaptainsCannon.cpp @@ -0,0 +1,86 @@ +#include "GfCaptainsCannon.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "RenderComponent.h" +#include "eTerminateType.h" +#include "eStateChangeType.h" + +void GfCaptainsCannon::OnUse(Entity* self, Entity* user) { + if (self->GetVar<bool>(u"bIsInUse")) { + return; + } + + self->SetVar<LWOOBJID>(u"userID", user->GetObjectID()); + + self->SetVar<bool>(u"bIsInUse", true); + self->SetNetworkVar<bool>(u"bIsInUse", true); + + GameMessages::SendSetStunned(user->GetObjectID(), eStateChangeType::PUSH, user->GetSystemAddress(), + LWOOBJID_EMPTY, true, true, true, true, true, true, true, true + ); + + auto position = self->GetPosition(); + auto forward = self->GetRotation().GetForwardVector(); + + position.x += forward.x * -3; + position.z += forward.z * -3; + + auto rotation = self->GetRotation(); + + GameMessages::SendTeleport(user->GetObjectID(), position, rotation, user->GetSystemAddress()); + + RenderComponent::PlayAnimation(user, u"cannon-strike-no-equip"); + + GameMessages::SendPlayFXEffect(user->GetObjectID(), 6039, u"hook", "hook", LWOOBJID_EMPTY, 1, 1, true); + + self->AddTimer("FireCannon", 1.667f); +} + +void GfCaptainsCannon::OnTimerDone(Entity* self, std::string timerName) { + const auto playerId = self->GetVar<LWOOBJID>(u"userID"); + + auto* player = Game::entityManager->GetEntity(playerId); + + if (player == nullptr) { + self->SetVar<bool>(u"bIsInUse", false); + self->SetNetworkVar<bool>(u"bIsInUse", false); + + return; + } + + if (timerName == "FireCannon") { + float cinematicTime = 6.3f; + + GameMessages::SendPlayCinematic(playerId, u"Cannon_Cam", player->GetSystemAddress()); + + self->AddTimer("cinematicTimer", cinematicTime); + + const auto sharkObjects = Game::entityManager->GetEntitiesInGroup("SharkCannon"); + + for (auto* shark : sharkObjects) { + if (shark->GetLOT() != m_SharkItemID) continue; + + RenderComponent::PlayAnimation(shark, u"cannon"); + } + + GameMessages::SendPlay2DAmbientSound(player, "{7457d85c-4537-4317-ac9d-2f549219ea87}"); + } else if (timerName == "cinematicTimer") { + GameMessages::SendSetStunned(playerId, eStateChangeType::POP, player->GetSystemAddress(), + LWOOBJID_EMPTY, true, true, true, true, true, true, true, true + ); + + self->SetVar<bool>(u"bIsInUse", false); + self->SetNetworkVar<bool>(u"bIsInUse", false); + + GameMessages::SendStopFXEffect(player, true, "hook"); + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->ForceProgress(601, 910, 1); + } + + GameMessages::SendTerminateInteraction(playerId, eTerminateType::FROM_INTERACTION, self->GetObjectID()); + } +} diff --git a/dScripts/GfCaptainsCannon.h b/dScripts/02_server/Map/GF/GfCaptainsCannon.h similarity index 100% rename from dScripts/GfCaptainsCannon.h rename to dScripts/02_server/Map/GF/GfCaptainsCannon.h diff --git a/dScripts/02_server/Map/GF/GfTikiTorch.cpp b/dScripts/02_server/Map/GF/GfTikiTorch.cpp new file mode 100644 index 00000000..e61abd76 --- /dev/null +++ b/dScripts/02_server/Map/GF/GfTikiTorch.cpp @@ -0,0 +1,79 @@ +#include "GfTikiTorch.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "RenderComponent.h" +#include "eMissionTaskType.h" +#include "eReplicaComponentType.h" +#include "RenderComponent.h" +#include "eTerminateType.h" + +void GfTikiTorch::OnStartup(Entity* self) { + LightTorch(self); +} + +void GfTikiTorch::OnUse(Entity* self, Entity* killer) { + if (self->GetBoolean(u"isInUse")) { + self->SetBoolean(u"isInUse", false); + return; + } + + RenderComponent::PlayAnimation(self, u"interact"); + self->SetI64(u"userID", killer->GetObjectID()); + + for (int i = 0; i < m_numspawn; i++) { + GameMessages::SendDropClientLoot(killer, self->GetObjectID(), 935, 0, self->GetPosition()); + } + + self->AddTimer("InteractionCooldown", 4); +} + +void GfTikiTorch::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "Relight") { + LightTorch(self); + } else if (timerName == "InteractionCooldown") { + Entity* player = Game::entityManager->GetEntity(self->GetI64(u"userID")); + + if (player != nullptr && player->GetCharacter()) { + GameMessages::SendTerminateInteraction(player->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); + } + + self->SetBoolean(u"isInUse", false); + + self->SetI64(u"userID", 0); + } +} + +void GfTikiTorch::LightTorch(Entity* self) { + auto* renderComponent = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + if (renderComponent == nullptr) + return; + + self->SetBoolean(u"isInUse", false); + + renderComponent->PlayEffect(611, u"fire", "tikitorch"); + self->SetBoolean(u"isBurning", true); +} + +void GfTikiTorch::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (self->GetBoolean(u"isBurning") && message == "waterspray") { + RenderComponent::PlayAnimation(self, u"water"); + + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + renderComponent->StopEffect("tikitorch"); + renderComponent->PlayEffect(611, u"water", "water"); + renderComponent->PlayEffect(611, u"steam", "steam"); + } + + auto* casterMissionComponent = caster->GetComponent<MissionComponent>(); + if (casterMissionComponent != nullptr) { + for (const auto missionID : m_missions) { + casterMissionComponent->ForceProgressTaskType(missionID, static_cast<uint32_t>(eMissionTaskType::SCRIPT), 1); + } + } + + self->AddTimer("Relight", 7.0f); + self->SetBoolean(u"isBurning", false); + } +} diff --git a/dScripts/GfTikiTorch.h b/dScripts/02_server/Map/GF/GfTikiTorch.h similarity index 100% rename from dScripts/GfTikiTorch.h rename to dScripts/02_server/Map/GF/GfTikiTorch.h diff --git a/dScripts/02_server/Map/GF/MastTeleport.cpp b/dScripts/02_server/Map/GF/MastTeleport.cpp new file mode 100644 index 00000000..29df4bf5 --- /dev/null +++ b/dScripts/02_server/Map/GF/MastTeleport.cpp @@ -0,0 +1,93 @@ +#include "MastTeleport.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "Preconditions.h" +#include "eEndBehavior.h" +#include "DestroyableComponent.h" +#include "eStateChangeType.h" + +#ifdef _WIN32 +#define _USE_MATH_DEFINES +#include <math.h> +#endif +#include "RenderComponent.h" + +void MastTeleport::OnStartup(Entity* self) { + self->SetNetworkVar<std::string>(u"hookPreconditions", "154;44", UNASSIGNED_SYSTEM_ADDRESS); +} + +void MastTeleport::OnRebuildComplete(Entity* self, Entity* target) { + if (Preconditions::Check(target, 154) && Preconditions::Check(target, 44)) { + self->SetVar<LWOOBJID>(u"userID", target->GetObjectID()); + + GameMessages::SendSetStunned(target->GetObjectID(), eStateChangeType::PUSH, target->GetSystemAddress(), + LWOOBJID_EMPTY, true, true, true, true, true, true, true + ); + auto* destroyableComponent = target->GetComponent<DestroyableComponent>(); + if (destroyableComponent) destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, true, true, true, true, true, false, false, true, true); + + self->AddTimer("Start", 3); + } +} + +void MastTeleport::OnTimerDone(Entity* self, std::string timerName) { + const auto playerId = self->GetVar<LWOOBJID>(u"userID"); + + auto* player = Game::entityManager->GetEntity(playerId); + + if (player == nullptr) return; + + if (timerName == "Start") { + auto position = self->GetPosition(); + auto rotation = self->GetRotation(); + + GameMessages::SendTeleport(playerId, position, rotation, player->GetSystemAddress(), true); + + GameMessages::SendTeleport(playerId, position, rotation, player->GetSystemAddress(), true); + + const auto cinematic = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"Cinematic")); + const auto leanIn = self->GetVar<float>(u"LeanIn"); + + if (!cinematic.empty()) { + GameMessages::SendPlayCinematic(playerId, GeneralUtils::ASCIIToUTF16(cinematic), player->GetSystemAddress(), + true, true, false, false, eEndBehavior::RETURN, false, leanIn + ); + } + + GameMessages::SendPlayFXEffect(playerId, 6039, u"hook", "hook", LWOOBJID_EMPTY, 1, 1, true); + + float animationTime = 6.25f; + animationTime = RenderComponent::PlayAnimation(player, "crow-swing-no-equip", 4.0f); + + RenderComponent::PlayAnimation(self, u"swing"); + + self->AddTimer("PlayerAnimDone", animationTime); + } else if (timerName == "PlayerAnimDone") { + GameMessages::SendStopFXEffect(player, true, "hook"); + + auto forward = self->GetRotation().GetForwardVector(); + + const auto degrees = -25.0f; + + const auto rads = degrees * (static_cast<float>(M_PI) / 180.0f); + + const Vector3 newPlayerRot = { 0, rads, 0 }; + + auto position = self->GetPosition(); + + position.x += (forward.x * 20.5f); + position.y += 12; + position.z += (forward.z * 20.5f); + + GameMessages::SendOrientToAngle(playerId, true, rads, player->GetSystemAddress()); + + GameMessages::SendTeleport(playerId, position, NiQuaternion::IDENTITY, player->GetSystemAddress()); + + GameMessages::SendSetStunned(playerId, eStateChangeType::POP, player->GetSystemAddress(), + LWOOBJID_EMPTY, true, true, true, true, true, true, true + ); + auto* destroyableComponent = player->GetComponent<DestroyableComponent>(); + if (destroyableComponent) destroyableComponent->SetStatusImmunity(eStateChangeType::POP, true, true, true, true, true, false, false, true, true); + Game::entityManager->SerializeEntity(player); + } +} diff --git a/dScripts/MastTeleport.h b/dScripts/02_server/Map/GF/MastTeleport.h similarity index 100% rename from dScripts/MastTeleport.h rename to dScripts/02_server/Map/GF/MastTeleport.h diff --git a/dScripts/SpawnLionServer.cpp b/dScripts/02_server/Map/GF/SpawnLionServer.cpp similarity index 100% rename from dScripts/SpawnLionServer.cpp rename to dScripts/02_server/Map/GF/SpawnLionServer.cpp diff --git a/dScripts/SpawnLionServer.h b/dScripts/02_server/Map/GF/SpawnLionServer.h similarity index 100% rename from dScripts/SpawnLionServer.h rename to dScripts/02_server/Map/GF/SpawnLionServer.h diff --git a/dScripts/02_server/Map/General/BankInteractServer.cpp b/dScripts/02_server/Map/General/BankInteractServer.cpp new file mode 100644 index 00000000..9b563491 --- /dev/null +++ b/dScripts/02_server/Map/General/BankInteractServer.cpp @@ -0,0 +1,25 @@ +#include "BankInteractServer.h" +#include "GameMessages.h" +#include "Entity.h" +#include "Amf3.h" + +void BankInteractServer::OnUse(Entity* self, Entity* user) { + AMFArrayValue args; + + args.Insert("state", "bank"); + + GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", args); +} + +void BankInteractServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + if (args == "ToggleBank") { + AMFArrayValue args; + + args.Insert("visible", false); + + GameMessages::SendUIMessageServerToSingleClient(sender, sender->GetSystemAddress(), "ToggleBank", args); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"CloseBank", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); + } +} diff --git a/dScripts/BankInteractServer.h b/dScripts/02_server/Map/General/BankInteractServer.h similarity index 100% rename from dScripts/BankInteractServer.h rename to dScripts/02_server/Map/General/BankInteractServer.h diff --git a/dScripts/BaseInteractDropLootServer.cpp b/dScripts/02_server/Map/General/BaseInteractDropLootServer.cpp similarity index 100% rename from dScripts/BaseInteractDropLootServer.cpp rename to dScripts/02_server/Map/General/BaseInteractDropLootServer.cpp diff --git a/dScripts/BaseInteractDropLootServer.h b/dScripts/02_server/Map/General/BaseInteractDropLootServer.h similarity index 100% rename from dScripts/BaseInteractDropLootServer.h rename to dScripts/02_server/Map/General/BaseInteractDropLootServer.h diff --git a/dScripts/Binoculars.cpp b/dScripts/02_server/Map/General/Binoculars.cpp similarity index 100% rename from dScripts/Binoculars.cpp rename to dScripts/02_server/Map/General/Binoculars.cpp diff --git a/dScripts/Binoculars.h b/dScripts/02_server/Map/General/Binoculars.h similarity index 100% rename from dScripts/Binoculars.h rename to dScripts/02_server/Map/General/Binoculars.h diff --git a/dScripts/02_server/Map/General/CMakeLists.txt b/dScripts/02_server/Map/General/CMakeLists.txt new file mode 100644 index 00000000..1fe3d785 --- /dev/null +++ b/dScripts/02_server/Map/General/CMakeLists.txt @@ -0,0 +1,28 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL + "BankInteractServer.cpp" + "BaseInteractDropLootServer.cpp" + "Binoculars.cpp" + "ExplodingAsset.cpp" + "ForceVolumeServer.cpp" + "GrowingFlower.cpp" + "ImaginationBackpackHealServer.cpp" + "InvalidScript.cpp" + "MailBoxServer.cpp" + "NjRailSwitch.cpp" + "PetDigServer.cpp" + "PropertyDevice.cpp" + "PropertyPlatform.cpp" + "QbEnemyStunner.cpp" + "QbSpawner.cpp" + "StoryBoxInteractServer.cpp" + "TokenConsoleServer.cpp" + "TouchMissionUpdateServer.cpp" + "WishingWellServer.cpp") + +add_subdirectory(Ninjago) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL_NINJAGO}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL ${DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL} "Ninjago/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL ${DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL} PARENT_SCOPE) diff --git a/dScripts/02_server/Map/General/ExplodingAsset.cpp b/dScripts/02_server/Map/General/ExplodingAsset.cpp new file mode 100644 index 00000000..ee8f8e68 --- /dev/null +++ b/dScripts/02_server/Map/General/ExplodingAsset.cpp @@ -0,0 +1,94 @@ +#include "ExplodingAsset.h" +#include "DestroyableComponent.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "SkillComponent.h" +#include "eMissionTaskType.h" +#include "RenderComponent.h" + +//TODO: this has to be updated so that you only get killed if you're in a certain radius. +//And so that all entities in a certain radius are killed, not just the attacker. + +void ExplodingAsset::OnStartup(Entity* self) { + self->SetProximityRadius(20.0f, "outRadius"); + self->SetVar<int32_t>(u"playersNearChest", 0); + self->SetProximityRadius(10.0f, "crateHitters"); +} + +void ExplodingAsset::OnHit(Entity* self, Entity* attacker) { + std::vector<Entity*> entities; + entities.push_back(attacker); + + if (!self->GetBoolean(u"bIsHit")) { + for (Entity* en : entities) { + if (en->GetObjectID() == attacker->GetObjectID()) { + if (Vector3::DistanceSquared(en->GetPosition(), self->GetPosition()) > 10 * 10) continue; + + auto* destroyable = en->GetComponent<DestroyableComponent>(); + if (destroyable == nullptr) { + continue; + } + + destroyable->Smash(attacker->GetObjectID()); + } + } + } + + attacker = attacker->GetOwner(); + self->SetBoolean(u"bIsHit", true); + self->SetOwnerOverride(attacker->GetObjectID()); + + GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(self, u"camshake", self->GetObjectID(), 16); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(147, 4721, LWOOBJID_EMPTY, true); + } + + const auto missionID = self->GetVar<int32_t>(u"missionID"); + auto achievementIDs = self->GetVar<std::u16string>(u"achieveID"); + + // Progress all scripted missions related to this asset + auto* missionComponent = attacker->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + if (missionID != 0) { + missionComponent->ForceProgressValue(missionID, + static_cast<uint32_t>(eMissionTaskType::SCRIPT), + self->GetLOT(), false); + } + + if (!achievementIDs.empty()) { + for (const auto& achievementID : GeneralUtils::SplitString(achievementIDs, u'_')) { + missionComponent->ForceProgressValue(std::stoi(GeneralUtils::UTF16ToWTF8(achievementID)), + static_cast<uint32_t>(eMissionTaskType::SCRIPT), + self->GetLOT()); + } + } + } + + self->ScheduleKillAfterUpdate(); +} + +void ExplodingAsset::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + auto* destuctableComponent = entering->GetComponent<DestroyableComponent>(); + + if (destuctableComponent == nullptr) return; + + const auto& factions = destuctableComponent->GetFactionIDs(); + + if (!std::count(factions.begin(), factions.end(), 1)) return; + + if (status == "ENTER") { + RenderComponent::PlayAnimation(self, u"bounce"); + GameMessages::SendPlayFXEffect(self, -1, u"anim", "bouncin", LWOOBJID_EMPTY, 1, 1, true); + self->SetVar(u"playersNearChest", self->GetVar<int32_t>(u"playersNearChest") + 1); + } else if (status == "LEAVE") { + self->SetVar(u"playersNearChest", self->GetVar<int32_t>(u"playersNearChest") - 1); + + if (self->GetVar<int32_t>(u"playersNearChest") < 1) { + RenderComponent::PlayAnimation(self, u"idle"); + GameMessages::SendStopFXEffect(self, true, "bouncin"); + self->SetVar<int32_t>(u"playersNearChest", 0); + } + } +} diff --git a/dScripts/ExplodingAsset.h b/dScripts/02_server/Map/General/ExplodingAsset.h similarity index 100% rename from dScripts/ExplodingAsset.h rename to dScripts/02_server/Map/General/ExplodingAsset.h diff --git a/dScripts/02_server/Map/General/ForceVolumeServer.cpp b/dScripts/02_server/Map/General/ForceVolumeServer.cpp new file mode 100644 index 00000000..7febe87e --- /dev/null +++ b/dScripts/02_server/Map/General/ForceVolumeServer.cpp @@ -0,0 +1,22 @@ +#include "ForceVolumeServer.h" +#include "PhantomPhysicsComponent.h" +#include "EntityManager.h" +#include "ePhysicsEffectType.h" + +void ForceVolumeServer::OnStartup(Entity* self) { + auto* phantomPhysicsComponent = self->GetComponent<PhantomPhysicsComponent>(); + + if (phantomPhysicsComponent == nullptr) return; + + const auto forceAmount = self->GetVar<float>(u"ForceAmt"); + const auto forceX = self->GetVar<float>(u"ForceX"); + const auto forceY = self->GetVar<float>(u"ForceY"); + const auto forceZ = self->GetVar<float>(u"ForceZ"); + + phantomPhysicsComponent->SetEffectType(ePhysicsEffectType::PUSH); + phantomPhysicsComponent->SetDirectionalMultiplier(forceAmount); + phantomPhysicsComponent->SetDirection({ forceX, forceY, forceZ }); + phantomPhysicsComponent->SetPhysicsEffectActive(true); + + Game::entityManager->SerializeEntity(self); +} diff --git a/dScripts/ForceVolumeServer.h b/dScripts/02_server/Map/General/ForceVolumeServer.h similarity index 100% rename from dScripts/ForceVolumeServer.h rename to dScripts/02_server/Map/General/ForceVolumeServer.h diff --git a/dScripts/02_server/Map/General/GrowingFlower.cpp b/dScripts/02_server/Map/General/GrowingFlower.cpp new file mode 100644 index 00000000..ad88528f --- /dev/null +++ b/dScripts/02_server/Map/General/GrowingFlower.cpp @@ -0,0 +1,38 @@ +#include "GrowingFlower.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" +#include "Loot.h" + +void GrowingFlower::OnSkillEventFired(Entity* self, Entity* target, const std::string& message) { + if (!self->GetVar<bool>(u"blooming") && (message == "waterspray" || message == "shovelgrow")) { + self->SetVar<bool>(u"blooming", true); + self->SetNetworkVar(u"blooming", true); + self->AddTimer("FlowerDie", GrowingFlower::aliveTime); + + const auto mission1 = self->GetVar<int32_t>(u"missionID"); + const auto mission2 = self->GetVar<int32_t>(u"missionID2"); + + LootGenerator::Instance().DropActivityLoot(target, self, self->GetLOT(), 0); + + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + for (const auto mission : achievementIDs) + missionComponent->ForceProgressTaskType(mission, static_cast<uint32_t>(eMissionTaskType::SCRIPT), 1); + + if (mission1 && missionComponent->GetMissionState(mission1) == eMissionState::ACTIVE) + missionComponent->ForceProgressTaskType(mission1, static_cast<uint32_t>(eMissionTaskType::SCRIPT), 1); + + if (mission2 && missionComponent->GetMissionState(mission2) == eMissionState::ACTIVE) + missionComponent->ForceProgressTaskType(mission2, static_cast<uint32_t>(eMissionTaskType::SCRIPT), 1); + } + } +} + +void GrowingFlower::OnTimerDone(Entity* self, std::string message) { + if (message == "FlowerDie") { + self->Smash(); + } +} + +const std::vector<uint32_t> GrowingFlower::achievementIDs = { 143, 152, 153, 1409, 1507, 1544, 1581, 1845 }; diff --git a/dScripts/GrowingFlower.h b/dScripts/02_server/Map/General/GrowingFlower.h similarity index 100% rename from dScripts/GrowingFlower.h rename to dScripts/02_server/Map/General/GrowingFlower.h diff --git a/dScripts/02_server/Map/General/ImaginationBackpackHealServer.cpp b/dScripts/02_server/Map/General/ImaginationBackpackHealServer.cpp new file mode 100644 index 00000000..8b3da9fa --- /dev/null +++ b/dScripts/02_server/Map/General/ImaginationBackpackHealServer.cpp @@ -0,0 +1,22 @@ +#include "ImaginationBackpackHealServer.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" + +void ImaginationBackpackHealServer::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (message == "CastImaginationBackpack") { + auto healMission = self->GetVar<int32_t>(u"FXOffMis"); + if (healMission == 0) + healMission = self->GetVar<int32_t>(u"FXOnMis"); + if (healMission == 0) + return; + + auto* missionComponent = caster->GetComponent<MissionComponent>(); + if (missionComponent != nullptr && missionComponent->GetMissionState(healMission) == eMissionState::ACTIVE) { + missionComponent->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"ClearMaelstrom", 0, 0, + caster->GetObjectID(), "", caster->GetSystemAddress()); + } + } +} diff --git a/dScripts/ImaginationBackpackHealServer.h b/dScripts/02_server/Map/General/ImaginationBackpackHealServer.h similarity index 100% rename from dScripts/ImaginationBackpackHealServer.h rename to dScripts/02_server/Map/General/ImaginationBackpackHealServer.h diff --git a/dScripts/InvalidScript.cpp b/dScripts/02_server/Map/General/InvalidScript.cpp similarity index 100% rename from dScripts/InvalidScript.cpp rename to dScripts/02_server/Map/General/InvalidScript.cpp diff --git a/dScripts/InvalidScript.h b/dScripts/02_server/Map/General/InvalidScript.h similarity index 100% rename from dScripts/InvalidScript.h rename to dScripts/02_server/Map/General/InvalidScript.h diff --git a/dScripts/02_server/Map/General/MailBoxServer.cpp b/dScripts/02_server/Map/General/MailBoxServer.cpp new file mode 100644 index 00000000..53f3b3a2 --- /dev/null +++ b/dScripts/02_server/Map/General/MailBoxServer.cpp @@ -0,0 +1,20 @@ +#include "MailBoxServer.h" +#include "Amf3.h" +#include "GameMessages.h" +#include "Entity.h" + +void MailBoxServer::OnUse(Entity* self, Entity* user) { + AMFArrayValue args; + + args.Insert("state", "Mail"); + + GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", args); +} + +void MailBoxServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + if (args == "toggleMail") { + AMFArrayValue args; + args.Insert("visible", false); + GameMessages::SendUIMessageServerToSingleClient(sender, sender->GetSystemAddress(), "ToggleMail", args); + } +} diff --git a/dScripts/MailBoxServer.h b/dScripts/02_server/Map/General/MailBoxServer.h similarity index 100% rename from dScripts/MailBoxServer.h rename to dScripts/02_server/Map/General/MailBoxServer.h diff --git a/dScripts/02_server/Map/General/Ninjago/CMakeLists.txt b/dScripts/02_server/Map/General/Ninjago/CMakeLists.txt new file mode 100644 index 00000000..2b74e921 --- /dev/null +++ b/dScripts/02_server/Map/General/Ninjago/CMakeLists.txt @@ -0,0 +1,6 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_GENERAL_NINJAGO + "NjRailActivatorsServer.cpp" + "NjRailPostServer.cpp" + "NjIceRailActivator.cpp" + "NjhubLavaPlayerDeathTrigger.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/General/Ninjago/NjIceRailActivator.cpp b/dScripts/02_server/Map/General/Ninjago/NjIceRailActivator.cpp new file mode 100644 index 00000000..f7f6a117 --- /dev/null +++ b/dScripts/02_server/Map/General/Ninjago/NjIceRailActivator.cpp @@ -0,0 +1,27 @@ +#include "NjIceRailActivator.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "Entity.h" +#include "RenderComponent.h" + +void NjIceRailActivator::OnPlayerRailArrived(Entity* self, Entity* sender, const std::u16string& pathName, + int32_t waypoint) { + const auto breakPoint = self->GetVar<int32_t>(BreakpointVariable); + if (breakPoint == waypoint) { + const auto& blockGroup = self->GetVar<std::u16string>(BlockGroupVariable); + + for (auto* block : Game::entityManager->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(blockGroup))) { + RenderComponent::PlayAnimation(block, u"explode"); + + const auto blockID = block->GetObjectID(); + + self->AddCallbackTimer(1.0f, [self, blockID]() { + auto* block = Game::entityManager->GetEntity(blockID); + + if (block != nullptr) { + block->Kill(self); + } + }); + } + } +} diff --git a/dScripts/NjIceRailActivator.h b/dScripts/02_server/Map/General/Ninjago/NjIceRailActivator.h similarity index 100% rename from dScripts/NjIceRailActivator.h rename to dScripts/02_server/Map/General/Ninjago/NjIceRailActivator.h diff --git a/dScripts/02_server/Map/General/Ninjago/NjRailActivatorsServer.cpp b/dScripts/02_server/Map/General/Ninjago/NjRailActivatorsServer.cpp new file mode 100644 index 00000000..5ca726af --- /dev/null +++ b/dScripts/02_server/Map/General/Ninjago/NjRailActivatorsServer.cpp @@ -0,0 +1,16 @@ +#include "NjRailActivatorsServer.h" +#include "RebuildComponent.h" +#include "Character.h" + +void NjRailActivatorsServer::OnUse(Entity* self, Entity* user) { + const auto flag = self->GetVar<int32_t>(u"RailFlagNum"); + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + // Only allow use if this is not a quick build or the quick build is built + if (rebuildComponent == nullptr || rebuildComponent->GetState() == eRebuildState::COMPLETED) { + auto* character = user->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(flag, true); + } + } +} diff --git a/dScripts/NjRailActivatorsServer.h b/dScripts/02_server/Map/General/Ninjago/NjRailActivatorsServer.h similarity index 100% rename from dScripts/NjRailActivatorsServer.h rename to dScripts/02_server/Map/General/Ninjago/NjRailActivatorsServer.h diff --git a/dScripts/02_server/Map/General/Ninjago/NjRailPostServer.cpp b/dScripts/02_server/Map/General/Ninjago/NjRailPostServer.cpp new file mode 100644 index 00000000..f47f7e7b --- /dev/null +++ b/dScripts/02_server/Map/General/Ninjago/NjRailPostServer.cpp @@ -0,0 +1,51 @@ +#include "NjRailPostServer.h" +#include "RebuildComponent.h" +#include "EntityManager.h" + +void NjRailPostServer::OnStartup(Entity* self) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + if (rebuildComponent != nullptr) { + self->SetNetworkVar<bool>(NetworkNotActiveVariable, true); + } +} + +void NjRailPostServer::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, + int32_t param2) { + if (name == "PostRebuilt") { + self->SetNetworkVar<bool>(NetworkNotActiveVariable, false); + } else if (name == "PostDied") { + self->SetNetworkVar<bool>(NetworkNotActiveVariable, true); + } +} + +void NjRailPostServer::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::COMPLETED) { + auto* relatedRail = GetRelatedRail(self); + if (relatedRail == nullptr) + return; + + relatedRail->NotifyObject(self, "PostRebuilt"); + + if (self->GetVar<bool>(NotActiveVariable)) + return; + + self->SetNetworkVar(NetworkNotActiveVariable, false); + } else if (state == eRebuildState::RESETTING) { + auto* relatedRail = GetRelatedRail(self); + if (relatedRail == nullptr) + return; + + relatedRail->NotifyObject(self, "PostDied"); + } +} + +Entity* NjRailPostServer::GetRelatedRail(Entity* self) { + const auto& railGroup = self->GetVar<std::u16string>(RailGroupVariable); + if (!railGroup.empty()) { + for (auto* entity : Game::entityManager->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(railGroup))) { + return entity; + } + } + + return nullptr; +} diff --git a/dScripts/NjRailPostServer.h b/dScripts/02_server/Map/General/Ninjago/NjRailPostServer.h similarity index 100% rename from dScripts/NjRailPostServer.h rename to dScripts/02_server/Map/General/Ninjago/NjRailPostServer.h diff --git a/dScripts/02_server/Map/General/Ninjago/NjhubLavaPlayerDeathTrigger.cpp b/dScripts/02_server/Map/General/Ninjago/NjhubLavaPlayerDeathTrigger.cpp new file mode 100644 index 00000000..e2b889ef --- /dev/null +++ b/dScripts/02_server/Map/General/Ninjago/NjhubLavaPlayerDeathTrigger.cpp @@ -0,0 +1,9 @@ +#include "NjhubLavaPlayerDeathTrigger.h" +#include "Entity.h" + +void NjhubLavaPlayerDeathTrigger::OnCollisionPhantom(Entity* self, Entity* target) { + if (!target->IsPlayer()) + return; + + target->Smash(self->GetObjectID(), eKillType::VIOLENT, u"drown"); +} diff --git a/dScripts/NjhubLavaPlayerDeathTrigger.h b/dScripts/02_server/Map/General/Ninjago/NjhubLavaPlayerDeathTrigger.h similarity index 100% rename from dScripts/NjhubLavaPlayerDeathTrigger.h rename to dScripts/02_server/Map/General/Ninjago/NjhubLavaPlayerDeathTrigger.h diff --git a/dScripts/NjRailSwitch.cpp b/dScripts/02_server/Map/General/NjRailSwitch.cpp similarity index 100% rename from dScripts/NjRailSwitch.cpp rename to dScripts/02_server/Map/General/NjRailSwitch.cpp diff --git a/dScripts/NjRailSwitch.h b/dScripts/02_server/Map/General/NjRailSwitch.h similarity index 100% rename from dScripts/NjRailSwitch.h rename to dScripts/02_server/Map/General/NjRailSwitch.h diff --git a/dScripts/02_server/Map/General/PetDigServer.cpp b/dScripts/02_server/Map/General/PetDigServer.cpp new file mode 100644 index 00000000..0ea78e4f --- /dev/null +++ b/dScripts/02_server/Map/General/PetDigServer.cpp @@ -0,0 +1,236 @@ +#include "dZoneManager.h" +#include "PetDigServer.h" +#include "MissionComponent.h" +#include "EntityManager.h" +#include "Character.h" +#include "PetComponent.h" +#include "User.h" +#include "eMissionState.h" + +std::vector<LWOOBJID> PetDigServer::treasures{}; + +const DigInfo PetDigServer::defaultDigInfo = DigInfo{ 3495, -1, -1, false, false, false, false }; + +/** + * Summary of all the special treasure behaviors, indexed by their lot + */ +const std::map<LOT, DigInfo> PetDigServer::digInfoMap{ + // Regular treasures + {3495, defaultDigInfo}, + + // Pet cove treasure + {7612, DigInfo { 7612, -1, -1, false, false, false, false }}, + + // Gnarled Forest flag treasure + {7410, DigInfo { 7410, -1, -1, false, true, false, false }}, + + // Gnarled Forest crab treasure + {9308, DigInfo { 9308, 7694, -1, false, false, false, false }}, + + // Avant Gardens mission treasure + {9307, DigInfo { 9307, -1, -1, false, true, false, true }}, + + // Avant Gardens bouncer treasure + {7559, DigInfo { 7559, -1, -1, false, false, true, false }}, + + // Crux Prime dragon treasure + {13098, DigInfo { 13098, 13067, 1298, false, false, false, false }}, + + // Bone treasure (can only be digged using the dragon) + {12192, DigInfo { 12192, -1, -1, true, false, false, false }}, +}; + +void PetDigServer::OnStartup(Entity* self) { + treasures.push_back(self->GetObjectID()); + const auto digInfoIterator = digInfoMap.find(self->GetLOT()); + const auto digInfo = digInfoIterator != digInfoMap.end() ? digInfoIterator->second : defaultDigInfo; + + // Reset any bouncers that might've been created by the previous dig + if (digInfo.bouncer) { + auto bounceNumber = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"BouncerNumber")); + auto bouncerSpawners = Game::zoneManager->GetSpawnersByName("PetBouncer" + bounceNumber); + auto switchSpawners = Game::zoneManager->GetSpawnersByName("PetBouncerSwitch" + bounceNumber); + + for (auto* bouncerSpawner : bouncerSpawners) { + for (auto* bouncer : bouncerSpawner->m_Info.nodes) + bouncerSpawner->Deactivate(); + bouncerSpawner->Reset(); + } + + for (auto* switchSpawner : switchSpawners) { + switchSpawner->Deactivate(); + switchSpawner->Reset(); + } + } +} + +void PetDigServer::OnDie(Entity* self, Entity* killer) { + const auto iterator = std::find(treasures.begin(), treasures.end(), self->GetObjectID()); + if (iterator != treasures.end()) { + treasures.erase(iterator); + } + + auto* owner = killer->GetOwner(); + const auto digInfoIterator = digInfoMap.find(self->GetLOT()); + const auto digInfo = digInfoIterator != digInfoMap.end() ? digInfoIterator->second : defaultDigInfo; + + if (digInfo.spawnLot >= 0) { + PetDigServer::SpawnPet(self, owner, digInfo); + } else if (digInfo.builderOnly) { + + // Some treasures may only be retrieved by the player that built the diggable + auto builder = self->GetVar<LWOOBJID>(u"builder"); // Set by the pet dig build script + if (builder != owner->GetObjectID()) + return; + } else if (digInfo.xBuild) { + PetDigServer::HandleXBuildDig(self, owner, killer); + return; + } else if (digInfo.bouncer) { + PetDigServer::HandleBouncerDig(self, owner); + } + + PetDigServer::ProgressPetDigMissions(owner, self); + + self->SetNetworkVar<bool>(u"treasure_dug", true); + // TODO: Reset other pets + + // Handles smashing leftovers (edge case for the AG X) + auto* xObject = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"X")); + if (xObject != nullptr) { + xObject->Smash(xObject->GetObjectID(), eKillType::VIOLENT); + } +} + +void PetDigServer::HandleXBuildDig(const Entity* self, Entity* owner, Entity* pet) { + auto playerID = self->GetVar<LWOOBJID>(u"builder"); + if (playerID == LWOOBJID_EMPTY || playerID != owner->GetObjectID()) + return; + + auto* playerEntity = Game::entityManager->GetEntity(playerID); + if (!playerEntity || !playerEntity->GetParentUser() || !playerEntity->GetParentUser()->GetLastUsedChar()) + return; + + auto* player = playerEntity->GetCharacter(); + const auto groupID = self->GetVar<std::u16string>(u"groupID"); + int32_t playerFlag = 0; + + // The flag that the player dug up + if (groupID == u"Flag1") { + playerFlag = 61; + } else if (groupID == u"Flag2") { + playerFlag = 62; + } else if (groupID == u"Flag3") { + playerFlag = 63; + } + + // If the player doesn't have the flag yet + if (playerFlag != 0 && !player->GetPlayerFlag(playerFlag)) { + auto* petComponent = pet->GetComponent<PetComponent>(); + if (petComponent != nullptr) { + // TODO: Pet state = 9 ?? + } + + // Shows the flag object to the player + player->SetPlayerFlag(playerFlag, true); + } + + auto* xObject = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"X")); + if (xObject != nullptr) { + xObject->Smash(xObject->GetObjectID(), eKillType::VIOLENT); + } +} + +void PetDigServer::HandleBouncerDig(const Entity* self, const Entity* owner) { + auto bounceNumber = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"BouncerNumber")); + auto bouncerSpawners = Game::zoneManager->GetSpawnersByName("PetBouncer" + bounceNumber); + auto switchSpawners = Game::zoneManager->GetSpawnersByName("PetBouncerSwitch" + bounceNumber); + + for (auto* bouncerSpawner : bouncerSpawners) { + bouncerSpawner->Activate(); + } + + for (auto* switchSpawner : switchSpawners) { + switchSpawner->Activate(); + } +} + +/** + * Progresses the Can You Dig It mission and the Pet Excavator Achievement if the player has never completed it yet + * \param owner the owner that just made a pet dig something up + */ +void PetDigServer::ProgressPetDigMissions(const Entity* owner, const Entity* chest) { + auto* missionComponent = owner->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + // Can You Dig It progress + const auto digMissionState = missionComponent->GetMissionState(843); + if (digMissionState == eMissionState::ACTIVE) { + missionComponent->ForceProgress(843, 1216, 1); + } + + // Pet Excavator progress + const auto excavatorMissionState = missionComponent->GetMissionState(505); + if (excavatorMissionState == eMissionState::ACTIVE) { + if (chest->HasVar(u"PetDig")) { + int32_t playerFlag = 1260 + chest->GetVarAs<int32_t>(u"PetDig"); + Character* player = owner->GetCharacter(); + + // check if player flag is set + if (!player->GetPlayerFlag(playerFlag)) { + missionComponent->ForceProgress(505, 767, 1); + player->SetPlayerFlag(playerFlag, 1); + } + } + } + } +} + +/** + * Some treasures spawn special pets, this handles that case + * \param owner the owner that just made a pet dig something up + * \param digInfo information regarding the treasure, will also contain info about the pet to spawn + */ +void PetDigServer::SpawnPet(Entity* self, const Entity* owner, const DigInfo digInfo) { + // Some treasures require a mission to be active + if (digInfo.requiredMission >= 0) { + auto* missionComponent = owner->GetComponent<MissionComponent>(); + if (missionComponent != nullptr && missionComponent->GetMissionState(digInfo.requiredMission) < eMissionState::ACTIVE) { + return; + } + } + + EntityInfo info{}; + info.lot = digInfo.spawnLot; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.spawnerID = self->GetSpawnerID(); + info.settings = { + new LDFData<LWOOBJID>(u"tamer", owner->GetObjectID()), + new LDFData<std::string>(u"group", "pet" + std::to_string(owner->GetObjectID())), + new LDFData<std::string>(u"spawnAnim", "spawn-pet"), + new LDFData<float>(u"spawnTimer", 1.0) + }; + + auto* spawnedPet = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(spawnedPet); +} + +Entity* PetDigServer::GetClosestTresure(NiPoint3 position) { + float closestDistance = 0; + Entity* closest = nullptr; + + for (const auto tresureId : treasures) { + auto* tresure = Game::entityManager->GetEntity(tresureId); + + if (tresure == nullptr) continue; + + float distance = Vector3::DistanceSquared(tresure->GetPosition(), position); + + if (closest == nullptr || distance < closestDistance) { + closestDistance = distance; + closest = tresure; + } + } + + return closest; +} diff --git a/dScripts/PetDigServer.h b/dScripts/02_server/Map/General/PetDigServer.h similarity index 100% rename from dScripts/PetDigServer.h rename to dScripts/02_server/Map/General/PetDigServer.h diff --git a/dScripts/02_server/Map/General/PropertyDevice.cpp b/dScripts/02_server/Map/General/PropertyDevice.cpp new file mode 100644 index 00000000..61ebe96e --- /dev/null +++ b/dScripts/02_server/Map/General/PropertyDevice.cpp @@ -0,0 +1,26 @@ +#include "PropertyDevice.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "eMissionState.h" + +void PropertyDevice::OnStartup(Entity* self) { + auto* zoneControl = Game::entityManager->GetZoneControlEntity(); + if (zoneControl != nullptr) { + zoneControl->OnFireEventServerSide(self, "CheckForPropertyOwner"); + } +} + +void PropertyDevice::OnRebuildComplete(Entity* self, Entity* target) { + auto propertyOwnerID = self->GetNetworkVar<std::string>(m_PropertyOwnerVariable); + if (propertyOwnerID == std::to_string(LWOOBJID_EMPTY)) + return; + + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + if (missionComponent->GetMissionState(m_PropertyMissionID) == eMissionState::ACTIVE) { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 641, u"create", "callhome"); + missionComponent->ForceProgress(m_PropertyMissionID, 1793, self->GetLOT()); + } + } +} diff --git a/dScripts/PropertyDevice.h b/dScripts/02_server/Map/General/PropertyDevice.h similarity index 100% rename from dScripts/PropertyDevice.h rename to dScripts/02_server/Map/General/PropertyDevice.h diff --git a/dScripts/02_server/Map/General/PropertyPlatform.cpp b/dScripts/02_server/Map/General/PropertyPlatform.cpp new file mode 100644 index 00000000..7016db94 --- /dev/null +++ b/dScripts/02_server/Map/General/PropertyPlatform.cpp @@ -0,0 +1,33 @@ +#include "PropertyPlatform.h" +#include "RebuildComponent.h" +#include "GameMessages.h" +#include "MovingPlatformComponent.h" + +void PropertyPlatform::OnRebuildComplete(Entity* self, Entity* target) { + // auto* movingPlatform = self->GetComponent<MovingPlatformComponent>(); + // if (movingPlatform != nullptr) { + // movingPlatform->StopPathing(); + // movingPlatform->SetNoAutoStart(true); + // } + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, + 0, 0, eMovementPlatformState::Stationary); +} + +void PropertyPlatform::OnUse(Entity* self, Entity* user) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + if (rebuildComponent != nullptr && rebuildComponent->GetState() == eRebuildState::COMPLETED) { + // auto* movingPlatform = self->GetComponent<MovingPlatformComponent>(); + // if (movingPlatform != nullptr) { + // movingPlatform->GotoWaypoint(1); + // } + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, + 1, 1, eMovementPlatformState::Moving); + + self->AddCallbackTimer(movementDelay + effectDelay, [self, this]() { + self->SetNetworkVar<float_t>(u"startEffect", dieDelay); + self->AddCallbackTimer(dieDelay, [self]() { + self->Smash(); + }); + }); + } +} diff --git a/dScripts/PropertyPlatform.h b/dScripts/02_server/Map/General/PropertyPlatform.h similarity index 100% rename from dScripts/PropertyPlatform.h rename to dScripts/02_server/Map/General/PropertyPlatform.h diff --git a/dScripts/02_server/Map/General/QbEnemyStunner.cpp b/dScripts/02_server/Map/General/QbEnemyStunner.cpp new file mode 100644 index 00000000..441d743c --- /dev/null +++ b/dScripts/02_server/Map/General/QbEnemyStunner.cpp @@ -0,0 +1,69 @@ +#include "QbEnemyStunner.h" +#include "SkillComponent.h" +#include "CDClientManager.h" +#include "DestroyableComponent.h" + +#include "CDObjectSkillsTable.h" +#include "CDSkillBehaviorTable.h" + +void QbEnemyStunner::OnRebuildComplete(Entity* self, Entity* target) { + auto* destroyable = self->GetComponent<DestroyableComponent>(); + + if (destroyable != nullptr) { + destroyable->SetFaction(115); + } + + auto skillComponent = self->GetComponent<SkillComponent>(); + if (!skillComponent) return; + + // Get the skill IDs of this object. + CDObjectSkillsTable* skillsTable = CDClientManager::Instance().GetTable<CDObjectSkillsTable>(); + auto skills = skillsTable->Query([=](CDObjectSkills entry) {return (entry.objectTemplate == self->GetLOT()); }); + std::map<uint32_t, uint32_t> skillBehaviorMap; + // For each skill, cast it with the associated behavior ID. + for (auto skill : skills) { + CDSkillBehaviorTable* skillBehaviorTable = CDClientManager::Instance().GetTable<CDSkillBehaviorTable>(); + CDSkillBehavior behaviorData = skillBehaviorTable->GetSkillByID(skill.skillID); + + skillBehaviorMap.insert(std::make_pair(skill.skillID, behaviorData.behaviorID)); + } + + // If there are no skills found, insert a default skill to use. + if (skillBehaviorMap.size() == 0) { + skillBehaviorMap.insert(std::make_pair(499U, 6095U)); + } + + // Start all skills associated with the object next tick + self->AddTimer("TickTime", 0); + + self->AddTimer("PlayEffect", 20); + + self->SetVar<std::map<uint32_t, uint32_t>>(u"skillBehaviorMap", skillBehaviorMap); +} + +void QbEnemyStunner::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "DieTime") { + self->Smash(); + + self->CancelAllTimers(); + } else if (timerName == "PlayEffect") { + self->SetNetworkVar(u"startEffect", 5.0f, UNASSIGNED_SYSTEM_ADDRESS); + + self->AddTimer("DieTime", 5.0f); + } else if (timerName == "TickTime") { + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + auto skillBehaviorMap = self->GetVar<std::map<uint32_t, uint32_t>>(u"skillBehaviorMap"); + if (skillBehaviorMap.size() == 0) { + // Should no skills have been found, default to the mermaid stunner + skillComponent->CalculateBehavior(499U, 6095U, LWOOBJID_EMPTY); + } else { + for (auto pair : skillBehaviorMap) { + skillComponent->CalculateBehavior(pair.first, pair.second, LWOOBJID_EMPTY); + } + } + } + self->AddTimer("TickTime", 1); + } +} diff --git a/dScripts/QbEnemyStunner.h b/dScripts/02_server/Map/General/QbEnemyStunner.h similarity index 100% rename from dScripts/QbEnemyStunner.h rename to dScripts/02_server/Map/General/QbEnemyStunner.h diff --git a/dScripts/02_server/Map/General/QbSpawner.cpp b/dScripts/02_server/Map/General/QbSpawner.cpp new file mode 100644 index 00000000..6776c8ea --- /dev/null +++ b/dScripts/02_server/Map/General/QbSpawner.cpp @@ -0,0 +1,136 @@ +#include "QbSpawner.h" +#include "BaseCombatAIComponent.h" +#include "EntityInfo.h" +#include "MovementAIComponent.h" + +void QbSpawner::OnStartup(Entity* self) { + auto mobNum = self->GetVar<int>(u"mobNum"); + auto spawnDist = self->GetVar<float>(u"spawnDist"); + auto mobTemplate = self->GetVar<LWOOBJID>(u"mobTemplate"); + auto spawnTime = self->GetVar<float>(u"spawnTime"); + + if (!mobNum) self->SetVar<int>(u"mobNum", m_DefaultMobNum); + if (!spawnDist) self->SetVar<float>(u"spawnDist", m_DefaultSpawnDist); + if (!mobTemplate) self->SetVar<LWOOBJID>(u"mobTemplate", m_DefaultMobTemplate); + if (!spawnTime) self->SetVar<float>(u"spawnTime", m_DefaultSpawnTime); + + // go ahead and setup the mob table here + std::vector<LWOOBJID> mobTable; + mobTable.assign(self->GetVar<int>(u"mobNum"), LWOOBJID_EMPTY); + + self->SetVar<std::vector<LWOOBJID>>(u"mobTable", mobTable); +} + +void QbSpawner::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + auto gateObjID = sender->GetObjectID(); + if (!gateObjID) return; + if (args == "spawnMobs") { + self->SetVar(u"gateObj", gateObjID); + auto spawnTime = self->GetVar<float>(u"spawnTime"); + self->AddTimer("SpawnMobEnemies", spawnTime); + } +} + +void QbSpawner::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "SpawnMobEnemies") { + auto mobTable = self->GetVar<std::vector<LWOOBJID>>(u"mobTable"); + + auto spawnDist = self->GetVar<float>(u"spawnDist"); + auto mobTemplate = self->GetVar<LWOOBJID>(u"mobTemplate"); + + auto gateObjID = self->GetVar<LWOOBJID>(u"gateObj"); + if (!gateObjID) return; + + auto* gate = Game::entityManager->GetEntity(gateObjID); + if (!gate) return; + + auto oPos = gate->GetPosition(); + auto oDir = gate->GetRotation().GetForwardVector(); + NiPoint3 newPos( + oPos.x + (oDir.x * spawnDist), + oPos.y, + oPos.z + (oDir.z * spawnDist) + ); + auto newRot = NiQuaternion::LookAt(newPos, oPos); + + for (int i = 0; i < mobTable.size(); i++) { + int posOffset = -10; + if (mobTable[i] == LWOOBJID_EMPTY) { + posOffset = posOffset + 5 * i; + auto newOffset = newPos; + newOffset.z = newOffset.z + posOffset; + + EntityInfo info{}; + info.lot = mobTemplate; + info.pos = newOffset; + info.rot = newRot; + info.spawnerID = self->GetObjectID(); + info.spawnerNodeID = 0; + info.settings = { + new LDFData<bool>(u"no_timed_spawn", true), + new LDFData<float>(u"aggroRadius", 70), + new LDFData<float>(u"softtetherRadius", 80), + new LDFData<float>(u"tetherRadius", 90), + new LDFData<float>(u"wanderRadius", 5), + new LDFData<int>(u"mobTableLoc", i) + }; + + auto* child = Game::entityManager->CreateEntity(info, nullptr, self); + Game::entityManager->ConstructEntity(child); + + OnChildLoaded(self, child); + } else { + auto* mob = Game::entityManager->GetEntity(mobTable[i]); + AggroTargetObject(self, mob); + } + } + + } +} + +void QbSpawner::OnChildLoaded(Entity* self, Entity* child) { + auto mobTable = self->GetVar<std::vector<LWOOBJID>>(u"mobTable"); + auto tableLoc = child->GetVar<int>(u"mobTableLoc"); + + mobTable[tableLoc] = child->GetObjectID(); + self->SetVar<std::vector<LWOOBJID>>(u"mobTable", mobTable); + + AggroTargetObject(self, child); + + const auto selfID = self->GetObjectID(); + + child->AddDieCallback([this, selfID, child]() { + auto* self = Game::entityManager->GetEntity(selfID); + OnChildRemoved(self, child); + } + ); +} + +void QbSpawner::OnChildRemoved(Entity* self, Entity* child) { + auto mobTable = self->GetVar<std::vector<LWOOBJID>>(u"mobTable"); + auto tableLoc = child->GetVar<int>(u"mobTableLoc"); + + mobTable[tableLoc] = LWOOBJID_EMPTY; + self->SetVar<std::vector<LWOOBJID>>(u"mobTable", mobTable); +} + +void QbSpawner::AggroTargetObject(Entity* self, Entity* enemy) { + auto* baseCombatAIComponent = enemy->GetComponent<BaseCombatAIComponent>(); + if (!baseCombatAIComponent) return; + + auto gateObjID = self->GetVar<LWOOBJID>(u"gateObj"); + if (gateObjID) { + auto* gate = Game::entityManager->GetEntity(gateObjID); + if (gate) { + auto* movementAIComponent = enemy->GetComponent<MovementAIComponent>(); + if (movementAIComponent) movementAIComponent->SetDestination(gate->GetPosition()); + baseCombatAIComponent->Taunt(gateObjID, 1000); + } + } + + auto playerObjID = self->GetVar<LWOOBJID>(u"player"); + if (playerObjID) { + baseCombatAIComponent->Taunt(playerObjID, 100); + } + +} diff --git a/dScripts/QbSpawner.h b/dScripts/02_server/Map/General/QbSpawner.h similarity index 100% rename from dScripts/QbSpawner.h rename to dScripts/02_server/Map/General/QbSpawner.h diff --git a/dScripts/02_server/Map/General/StoryBoxInteractServer.cpp b/dScripts/02_server/Map/General/StoryBoxInteractServer.cpp new file mode 100644 index 00000000..9a1a4908 --- /dev/null +++ b/dScripts/02_server/Map/General/StoryBoxInteractServer.cpp @@ -0,0 +1,43 @@ +#include "StoryBoxInteractServer.h" +#include "Character.h" +#include "GameMessages.h" +#include "dServer.h" +#include "Amf3.h" +#include "Entity.h" + +void StoryBoxInteractServer::OnUse(Entity* self, Entity* user) { + if (self->GetVar<bool>(u"hasCustomText")) { + const auto& customText = self->GetVar<std::string>(u"customText"); + + { + AMFArrayValue args; + + args.Insert("state", "Story"); + + GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", args); + } + + user->AddCallbackTimer(0.1f, [user, customText]() { + AMFArrayValue args; + + args.Insert("visible", true); + args.Insert("text", customText); + + GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "ToggleStoryBox", args); + }); + + return; + } + + const auto storyText = self->GetVarAsString(u"storyText"); + + int32_t boxFlag = self->GetVar<int32_t>(u"altFlagID"); + if (boxFlag <= 0) { + boxFlag = (10000 + Game::server->GetZoneID() + std::stoi(storyText.substr(storyText.length() - 2))); + } + + if (user->GetCharacter()->GetPlayerFlag(boxFlag) == false) { + user->GetCharacter()->SetPlayerFlag(boxFlag, true); + GameMessages::SendFireEventClientSide(self->GetObjectID(), user->GetSystemAddress(), u"achieve", LWOOBJID_EMPTY, 0, -1, LWOOBJID_EMPTY); + } +} diff --git a/dScripts/StoryBoxInteractServer.h b/dScripts/02_server/Map/General/StoryBoxInteractServer.h similarity index 100% rename from dScripts/StoryBoxInteractServer.h rename to dScripts/02_server/Map/General/StoryBoxInteractServer.h diff --git a/dScripts/02_server/Map/General/TokenConsoleServer.cpp b/dScripts/02_server/Map/General/TokenConsoleServer.cpp new file mode 100644 index 00000000..e13011cb --- /dev/null +++ b/dScripts/02_server/Map/General/TokenConsoleServer.cpp @@ -0,0 +1,39 @@ +#include "TokenConsoleServer.h" +#include "InventoryComponent.h" +#include "GameMessages.h" +#include "Character.h" +#include "eReplicaComponentType.h" +#include "eTerminateType.h" +#include "ePlayerFlag.h" + +//2021-05-03 - max - added script, omitted some parts related to inheritance in lua which we don't need + +void TokenConsoleServer::OnUse(Entity* self, Entity* user) { + auto* inv = static_cast<InventoryComponent*>(user->GetComponent(eReplicaComponentType::INVENTORY)); + + //make sure the user has the required amount of infected bricks + if (inv && inv->GetLotCount(6194) >= bricksToTake) { + //yeet the bricks + inv->RemoveItem(6194, bricksToTake); + + //play sound + GameMessages::SendPlayNDAudioEmitter(self, user->GetSystemAddress(), "947d0d52-c7f8-4516-8dee-e1593a7fd1d1"); + + //figure out which faction the player belongs to: + auto character = user->GetCharacter(); + if (!character) return; + // At this point the player has to be in a faction. + LOT tokenLOT = 0; + if (character->GetPlayerFlag(ePlayerFlag::VENTURE_FACTION)) //venture + tokenLOT = 8321; + else if (character->GetPlayerFlag(ePlayerFlag::ASSEMBLY_FACTION)) //assembly + tokenLOT = 8318; + else if (character->GetPlayerFlag(ePlayerFlag::PARADOX_FACTION)) //paradox + tokenLOT = 8320; + else if (character->GetPlayerFlag(ePlayerFlag::SENTINEL_FACTION)) //sentinel + tokenLOT = 8319; + inv->AddItem(tokenLOT, tokensToGive, eLootSourceType::NONE); + } + + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} diff --git a/dScripts/TokenConsoleServer.h b/dScripts/02_server/Map/General/TokenConsoleServer.h similarity index 100% rename from dScripts/TokenConsoleServer.h rename to dScripts/02_server/Map/General/TokenConsoleServer.h diff --git a/dScripts/02_server/Map/General/TouchMissionUpdateServer.cpp b/dScripts/02_server/Map/General/TouchMissionUpdateServer.cpp new file mode 100644 index 00000000..7b2495d0 --- /dev/null +++ b/dScripts/02_server/Map/General/TouchMissionUpdateServer.cpp @@ -0,0 +1,53 @@ +#include "TouchMissionUpdateServer.h" + +#include "Entity.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" + +void TouchMissionUpdateServer::OnStartup(Entity* self) { + self->SetProximityRadius(20, "touchCheck"); // Those does not have a collider for some reason? +} + +void TouchMissionUpdateServer::OnCollisionPhantom(Entity* self, Entity* target) { + int32_t missionId = self->GetVar<int32_t>(u"TouchCompleteID"); + + if (missionId == 0) { + return; + } + + auto* missionComponent = static_cast<MissionComponent*>(target->GetComponent(eReplicaComponentType::MISSION)); + + if (missionComponent == nullptr) { + return; + } + + auto* mission = missionComponent->GetMission(missionId); + + if (mission == nullptr) { + return; + } + + const auto state = mission->GetMissionState(); + + if (state >= eMissionState::COMPLETE || mission->GetCompletions() > 1) { + return; + } + + for (auto* task : mission->GetTasks()) { + if (!task->IsComplete()) { + task->Complete(); + } + } + + mission->CheckCompletion(); +} + +void TouchMissionUpdateServer::OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) { + if (name != "touchCheck" || status != "ENTER") { + return; + } + + OnCollisionPhantom(self, entering); +} diff --git a/dScripts/TouchMissionUpdateServer.h b/dScripts/02_server/Map/General/TouchMissionUpdateServer.h similarity index 100% rename from dScripts/TouchMissionUpdateServer.h rename to dScripts/02_server/Map/General/TouchMissionUpdateServer.h diff --git a/dScripts/02_server/Map/General/WishingWellServer.cpp b/dScripts/02_server/Map/General/WishingWellServer.cpp new file mode 100644 index 00000000..51f742ec --- /dev/null +++ b/dScripts/02_server/Map/General/WishingWellServer.cpp @@ -0,0 +1,47 @@ +#include "WishingWellServer.h" +#include "ScriptedActivityComponent.h" +#include "GameMessages.h" +#include "Loot.h" +#include "EntityManager.h" +#include "eTerminateType.h" + +void WishingWellServer::OnStartup(Entity* self) { +} + +void WishingWellServer::OnUse(Entity* self, Entity* user) { + auto* scriptedActivity = self->GetComponent<ScriptedActivityComponent>(); + + if (!scriptedActivity->TakeCost(user)) { + return; + } + + const auto audio = self->GetVar<std::string>(u"sound1"); + + if (!audio.empty()) { + GameMessages::SendPlayNDAudioEmitter(self, user->GetSystemAddress(), audio); + } + + LootGenerator::Instance().DropActivityLoot( + user, + self, + static_cast<uint32_t>(scriptedActivity->GetActivityID()), + GeneralUtils::GenerateRandomNumber<int32_t>(1, 1000) + ); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"StartCooldown", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); + + const auto userID = user->GetObjectID(); + + self->AddCallbackTimer(10, [self, userID]() { + auto* user = Game::entityManager->GetEntity(userID); + + if (user == nullptr) return; + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"StopCooldown", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); + }); + + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} + +void WishingWellServer::OnTimerDone(Entity* self, std::string timerName) { +} diff --git a/dScripts/WishingWellServer.h b/dScripts/02_server/Map/General/WishingWellServer.h similarity index 100% rename from dScripts/WishingWellServer.h rename to dScripts/02_server/Map/General/WishingWellServer.h diff --git a/dScripts/02_server/Map/NS/CMakeLists.txt b/dScripts/02_server/Map/NS/CMakeLists.txt new file mode 100644 index 00000000..c815a8c5 --- /dev/null +++ b/dScripts/02_server/Map/NS/CMakeLists.txt @@ -0,0 +1,13 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NS + "NsConcertChoiceBuildManager.cpp" + "NsLegoClubDoor.cpp" + "NsLupTeleport.cpp" + "NsTokenConsoleServer.cpp") + +add_subdirectory(Waves) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_NS_WAVES}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_NS ${DSCRIPTS_SOURCES_02_SERVER_MAP_NS} "Waves/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NS ${DSCRIPTS_SOURCES_02_SERVER_MAP_NS} PARENT_SCOPE) diff --git a/dScripts/02_server/Map/NS/NsConcertChoiceBuildManager.cpp b/dScripts/02_server/Map/NS/NsConcertChoiceBuildManager.cpp new file mode 100644 index 00000000..554eabac --- /dev/null +++ b/dScripts/02_server/Map/NS/NsConcertChoiceBuildManager.cpp @@ -0,0 +1,65 @@ +#include "NsConcertChoiceBuildManager.h" +#include "EntityInfo.h" +#include "EntityManager.h" + +const std::vector<Crate> NsConcertChoiceBuildManager::crates{ + { "laser", 11203, 5.0, "Concert_Laser_QB_" }, + { "rocket", 11204, 3.0, "Concert_Rocket_QB_" }, + { "speaker", 11205, 5.0, "Concert_Speaker_QB_" }, + { "spotlight", 11206, 5.0, "Concert_Spotlight_QB_" } +}; + +void NsConcertChoiceBuildManager::OnStartup(Entity* self) { + NsConcertChoiceBuildManager::SpawnCrate(self); +} + +void NsConcertChoiceBuildManager::SpawnCrate(Entity* self) { + const auto spawnNumber = self->GetVar<uint32_t>(u"spawnNumber") % crates.size(); + const auto crate = crates[spawnNumber]; + + const auto groups = self->GetGroups(); + if (groups.empty()) + return; + + // Groups are of the form CB_1, CB_2, etc. + auto group = groups.at(0); + const auto splitGroup = GeneralUtils::SplitString(group, '_'); + if (splitGroup.size() < 2) + return; + const auto groupNumber = std::stoi(splitGroup.at(1)); + + EntityInfo info{}; + info.lot = crate.lot; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.spawnerID = self->GetObjectID(); + info.settings = { + new LDFData<bool>(u"startsQBActivator", true), + new LDFData<std::string>(u"grpNameQBShowBricks", crate.group + std::to_string(groupNumber)), + new LDFData<std::u16string>(u"groupID", GeneralUtils::ASCIIToUTF16("Crate_" + group)), + new LDFData<float>(u"crateTime", crate.time), + }; + + auto* spawnedCrate = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(spawnedCrate); + + spawnedCrate->AddDieCallback([self]() { + self->CancelAllTimers(); // Don't switch if the crate was smashed + self->SetVar<LWOOBJID>(u"currentCrate", LWOOBJID_EMPTY); + }); + + self->SetVar<uint32_t>(u"spawnNumber", spawnNumber + 1); + self->SetVar<float>(u"currentTimer", crate.time); + self->SetVar<LWOOBJID>(u"currentCrate", spawnedCrate->GetObjectID()); + + // Timer that rotates the crates + self->AddCallbackTimer(crate.time, [self]() { + auto crateID = self->GetVar<LWOOBJID>(u"currentCrate"); + if (crateID != LWOOBJID_EMPTY) { + Game::entityManager->DestroyEntity(crateID); + self->SetVar<LWOOBJID>(u"currentCrate", LWOOBJID_EMPTY); + } + + SpawnCrate(self); + }); +} diff --git a/dScripts/NsConcertChoiceBuildManager.h b/dScripts/02_server/Map/NS/NsConcertChoiceBuildManager.h similarity index 100% rename from dScripts/NsConcertChoiceBuildManager.h rename to dScripts/02_server/Map/NS/NsConcertChoiceBuildManager.h diff --git a/dScripts/02_server/Map/NS/NsLegoClubDoor.cpp b/dScripts/02_server/Map/NS/NsLegoClubDoor.cpp new file mode 100644 index 00000000..35d4735c --- /dev/null +++ b/dScripts/02_server/Map/NS/NsLegoClubDoor.cpp @@ -0,0 +1,92 @@ +#include "NsLegoClubDoor.h" +#include "dZoneManager.h" +#include "GameMessages.h" +#include "Amf3.h" + +void NsLegoClubDoor::OnStartup(Entity* self) { + self->SetVar(u"currentZone", (int32_t)Game::zoneManager->GetZoneID().GetMapID()); + self->SetVar(u"choiceZone", m_ChoiceZoneID); + self->SetVar(u"teleportAnim", m_TeleportAnim); + self->SetVar(u"teleportString", m_TeleportString); + self->SetVar(u"spawnPoint", m_SpawnPoint); + + args = {}; + + args.Insert("callbackClient", std::to_string(self->GetObjectID())); + args.Insert("strIdentifier", "choiceDoor"); + args.Insert("title", "%[UI_CHOICE_DESTINATION]"); + + AMFArrayValue* choiceOptions = args.InsertArray("options"); + + { + AMFArrayValue* nsArgs = choiceOptions->PushArray(); + + nsArgs->Insert("image", "textures/ui/zone_thumnails/Nimbus_Station.dds"); + nsArgs->Insert("caption", "%[UI_CHOICE_NS]"); + nsArgs->Insert("identifier", "zoneID_1200"); + nsArgs->Insert("tooltipText", "%[UI_CHOICE_NS_HOVER]"); + } + + { + AMFArrayValue* ntArgs = choiceOptions->PushArray(); + + ntArgs->Insert("image", "textures/ui/zone_thumnails/Nexus_Tower.dds"); + ntArgs->Insert("caption", "%[UI_CHOICE_NT]"); + ntArgs->Insert("identifier", "zoneID_1900"); + ntArgs->Insert("tooltipText", "%[UI_CHOICE_NT_HOVER]"); + } + + options = choiceOptions; +} + +void NsLegoClubDoor::OnUse(Entity* self, Entity* user) { + auto* player = user; + + if (CheckChoice(self, player)) { + AMFArrayValue multiArgs; + + multiArgs.Insert("callbackClient", std::to_string(self->GetObjectID())); + multiArgs.Insert("strIdentifier", "choiceDoor"); + multiArgs.Insert("title", "%[UI_CHOICE_DESTINATION]"); + multiArgs.Insert("options", static_cast<AMFBaseValue*>(options)); + + GameMessages::SendUIMessageServerToSingleClient(player, player->GetSystemAddress(), "QueueChoiceBox", multiArgs); + + multiArgs.Remove("options", false); // We do not want the local amf to delete the options! + } else if (self->GetVar<int32_t>(u"currentZone") != m_ChoiceZoneID) { + AMFArrayValue multiArgs; + multiArgs.Insert("state", "Lobby"); + + AMFArrayValue* context = multiArgs.InsertArray("context"); + context->Insert("user", std::to_string(player->GetObjectID())); + context->Insert("callbackObj", std::to_string(self->GetObjectID())); + context->Insert("HelpVisible", "show"); + context->Insert("type", "Lego_Club_Valid"); + + GameMessages::SendUIMessageServerToSingleClient(player, player->GetSystemAddress(), "pushGameState", multiArgs); + } else { + BaseOnUse(self, player); + } +} + +void NsLegoClubDoor::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + std::u16string strIdentifier = identifier; + + if (strIdentifier == u"PlayButton" || strIdentifier == u"CloseButton") { + strIdentifier = u"TransferBox"; + } + + BaseOnMessageBoxResponse(self, sender, button, strIdentifier, userData); +} + +void NsLegoClubDoor::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { + BaseChoiceBoxRespond(self, sender, button, buttonIdentifier, identifier); +} + +void NsLegoClubDoor::OnTimerDone(Entity* self, std::string timerName) { + BaseOnTimerDone(self, timerName); +} + +void NsLegoClubDoor::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); +} diff --git a/dScripts/02_server/Map/NS/NsLegoClubDoor.h b/dScripts/02_server/Map/NS/NsLegoClubDoor.h new file mode 100644 index 00000000..5b25ba6d --- /dev/null +++ b/dScripts/02_server/Map/NS/NsLegoClubDoor.h @@ -0,0 +1,24 @@ +#pragma once +#include "CppScripts.h" +#include "ChooseYourDestinationNsToNt.h" +#include "BaseConsoleTeleportServer.h" +#include "Amf3.h" + +class NsLegoClubDoor : public CppScripts::Script, ChooseYourDestinationNsToNt, BaseConsoleTeleportServer +{ +public: + void OnStartup(Entity* self) override; + void OnUse(Entity* self, Entity* user) override; + void OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) override; + void OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) override; + void OnTimerDone(Entity* self, std::string timerName) override; + void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; + +private: + int32_t m_ChoiceZoneID = 1700; + std::string m_SpawnPoint = "NS_LEGO_Club"; + std::u16string m_TeleportAnim = u"lup-teleport"; + std::u16string m_TeleportString = u"ROCKET_TOOLTIP_USE_THE_GATEWAY_TO_TRAVEL_TO_LUP_WORLD"; + AMFArrayValue args = {}; + AMFArrayValue* options = {}; +}; diff --git a/dScripts/02_server/Map/NS/NsLupTeleport.cpp b/dScripts/02_server/Map/NS/NsLupTeleport.cpp new file mode 100644 index 00000000..c75143fc --- /dev/null +++ b/dScripts/02_server/Map/NS/NsLupTeleport.cpp @@ -0,0 +1,64 @@ +#include "NsLupTeleport.h" +#include "dZoneManager.h" +#include "GameMessages.h" +#include "Amf3.h" + +void NsLupTeleport::OnStartup(Entity* self) { + self->SetVar(u"currentZone", (int32_t)Game::zoneManager->GetZoneID().GetMapID()); + self->SetVar(u"choiceZone", m_ChoiceZoneID); + self->SetVar(u"teleportAnim", m_TeleportAnim); + self->SetVar(u"teleportString", m_TeleportString); + self->SetVar(u"spawnPoint", m_SpawnPoint); + + args = {}; + + args.Insert("callbackClient", std::to_string(self->GetObjectID())); + args.Insert("strIdentifier", "choiceDoor"); + args.Insert("title", "%[UI_CHOICE_DESTINATION]"); + + AMFArrayValue* choiceOptions = args.InsertArray("options"); + + { + AMFArrayValue* nsArgs = choiceOptions->PushArray(); + + nsArgs->Insert("image", "textures/ui/zone_thumnails/Nimbus_Station.dds"); + nsArgs->Insert("caption", "%[UI_CHOICE_NS]"); + nsArgs->Insert("identifier", "zoneID_1200"); + nsArgs->Insert("tooltipText", "%[UI_CHOICE_NS_HOVER]"); + } + + { + AMFArrayValue* ntArgs = choiceOptions->PushArray(); + + ntArgs->Insert("image", "textures/ui/zone_thumnails/Nexus_Tower.dds"); + ntArgs->Insert("caption", "%[UI_CHOICE_NT]"); + ntArgs->Insert("identifier", "zoneID_1900"); + ntArgs->Insert("tooltipText", "%[UI_CHOICE_NT_HOVER]"); + } +} + +void NsLupTeleport::OnUse(Entity* self, Entity* user) { + auto* player = user; + + if (CheckChoice(self, player)) { + GameMessages::SendUIMessageServerToSingleClient(player, player->GetSystemAddress(), "QueueChoiceBox", args); + } else { + BaseOnUse(self, player); + } +} + +void NsLupTeleport::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + BaseOnMessageBoxResponse(self, sender, button, identifier, userData); +} + +void NsLupTeleport::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { + BaseChoiceBoxRespond(self, sender, button, buttonIdentifier, identifier); +} + +void NsLupTeleport::OnTimerDone(Entity* self, std::string timerName) { + BaseOnTimerDone(self, timerName); +} + +void NsLupTeleport::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); +} diff --git a/dScripts/02_server/Map/NS/NsLupTeleport.h b/dScripts/02_server/Map/NS/NsLupTeleport.h new file mode 100644 index 00000000..385e92c7 --- /dev/null +++ b/dScripts/02_server/Map/NS/NsLupTeleport.h @@ -0,0 +1,23 @@ +#pragma once +#include "CppScripts.h" +#include "ChooseYourDestinationNsToNt.h" +#include "BaseConsoleTeleportServer.h" +#include "Amf3.h" + +class NsLupTeleport : public CppScripts::Script, ChooseYourDestinationNsToNt, BaseConsoleTeleportServer +{ +public: + void OnStartup(Entity* self) override; + void OnUse(Entity* self, Entity* user) override; + void OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) override; + void OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) override; + void OnTimerDone(Entity* self, std::string timerName) override; + void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; + +private: + int32_t m_ChoiceZoneID = 1600; + std::string m_SpawnPoint = "NS_LW"; + std::u16string m_TeleportAnim = u"lup-teleport"; + std::u16string m_TeleportString = u"UI_TRAVEL_TO_LUP_STATION"; + AMFArrayValue args = {}; +}; diff --git a/dScripts/02_server/Map/NS/NsTokenConsoleServer.cpp b/dScripts/02_server/Map/NS/NsTokenConsoleServer.cpp new file mode 100644 index 00000000..326842d2 --- /dev/null +++ b/dScripts/02_server/Map/NS/NsTokenConsoleServer.cpp @@ -0,0 +1,61 @@ +#include "NsTokenConsoleServer.h" +#include "InventoryComponent.h" +#include "GameMessages.h" +#include "Character.h" +#include "MissionComponent.h" +#include "RebuildComponent.h" +#include "eTerminateType.h" +#include "ePlayerFlag.h" + +void NsTokenConsoleServer::OnStartup(Entity* self) { + +} + +void NsTokenConsoleServer::OnUse(Entity* self, Entity* user) { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent == nullptr) { + return; + } + + if (rebuildComponent->GetState() != eRebuildState::COMPLETED) { + return; + } + + auto* inventoryComponent = user->GetComponent<InventoryComponent>(); + auto* missionComponent = user->GetComponent<MissionComponent>(); + auto* character = user->GetCharacter(); + + if (inventoryComponent == nullptr || missionComponent == nullptr || character == nullptr) { + return; + } + + if (inventoryComponent->GetLotCount(6194) < 25) { + return; + } + + inventoryComponent->RemoveItem(6194, 25); + + const auto useSound = self->GetVar<std::string>(u"sound1"); + + if (!useSound.empty()) { + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, useSound); + } + + // Player must be in faction to interact with this entity. + LOT tokenLOT = 0; + if (character->GetPlayerFlag(ePlayerFlag::VENTURE_FACTION)) //venture + tokenLOT = 8321; + else if (character->GetPlayerFlag(ePlayerFlag::ASSEMBLY_FACTION)) //assembly + tokenLOT = 8318; + else if (character->GetPlayerFlag(ePlayerFlag::PARADOX_FACTION)) //paradox + tokenLOT = 8320; + else if (character->GetPlayerFlag(ePlayerFlag::SENTINEL_FACTION)) //sentinel + tokenLOT = 8319; + + inventoryComponent->AddItem(tokenLOT, 5, eLootSourceType::NONE); + + missionComponent->ForceProgressTaskType(863, 1, 1, false); + + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} diff --git a/dScripts/NsTokenConsoleServer.h b/dScripts/02_server/Map/NS/NsTokenConsoleServer.h similarity index 100% rename from dScripts/NsTokenConsoleServer.h rename to dScripts/02_server/Map/NS/NsTokenConsoleServer.h diff --git a/dScripts/02_server/Map/NS/Waves/CMakeLists.txt b/dScripts/02_server/Map/NS/Waves/CMakeLists.txt new file mode 100644 index 00000000..bb216ee9 --- /dev/null +++ b/dScripts/02_server/Map/NS/Waves/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NS_WAVES + "ZoneNsWaves.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/NS/Waves/ZoneNsWaves.cpp b/dScripts/02_server/Map/NS/Waves/ZoneNsWaves.cpp new file mode 100644 index 00000000..184132fe --- /dev/null +++ b/dScripts/02_server/Map/NS/Waves/ZoneNsWaves.cpp @@ -0,0 +1,500 @@ +#include "ZoneNsWaves.h" + +WaveConstants ZoneNsWaves::GetConstants() { + return { + 60, + 2, + 6, + 2, + "surprise", + "intro" + }; +} + +std::vector<std::string> ZoneNsWaves::GetSpawnerNames() { + return { + "Base_MobA", + "Base_MobB", + "Base_MobC", + "MobA_01", + "MobB_01", + "MobC_01", + "MobA_02", + "MobB_02", + "MobC_02", + "MobA_03", + "MobB_03", + "MobC_03", + "Reward_01", + "Base_Reward", + "Obstacle_01", + "Boss", + "Ape_Boss", + "Geyser_01", + "Treasure_01", + "Cavalry_Boss", + "Horseman_01", + "Horseman_02", + "Horseman_03", + "Horseman_04" + }; +} + +std::vector<WaveMission> ZoneNsWaves::GetWaveMissions() { + return { + {190, 7, 1242}, + {240, 7, 1226}, + {450, 15, 1243}, + {600, 15, 1227}, + {720, 22, 1244}, + {840, 22, 1228}, + {1080, 29, 1245}, + {1200, 29, 1229}, + }; +} + +std::vector<Wave> ZoneNsWaves::GetWaves() { + return { + // Wave 1 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::stromling_minifig, 8, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::stromling_minifig, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::stromling_minifig, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::stromling_minifig, 2, GetSpawnerName(SpawnerName::gf_A) }, + } + }, + + // Wave 2 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::stromling, 8, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::gf_A) }, + } + }, + + // Wave 3 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::stromling, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::gf_A) }, + }, + }, + + // Wave 4 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 5 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::stromling, 1, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::stromling, 1, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 6 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 7 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::stromling_boss, 1, GetSpawnerName(SpawnerName::Boss) }, + }, + {1885}, + {}, + "Stromling_Boss", + 5.0f + }, + + // Wave 8 + Wave { + std::vector<MobDefinition> { + {SpawnLOTS::mushroom, 6, GetSpawnerName(SpawnerName::Reward_01) }, + {SpawnLOTS::mushroom, 3, GetSpawnerName(SpawnerName::interior_Reward) }, + }, {}, {}, "", -1.0f, + 25, + }, + + // Wave 9 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 10 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 11 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::concert_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::gf_C) }, + } + }, + + // Wave 12 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_C) }, + } + }, + + // Wave 13 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 3, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 14 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_C) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_B) }, + { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_C) }, + } + }, + + // Wave 15 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::ape_boss, 1, GetSpawnerName(SpawnerName::Ape_Boss) }, + + }, + {1886}, + {}, + "Gorilla_Boss", + 5.0f + }, + + // Wave 16 + Wave { + std::vector<MobDefinition> { + {SpawnLOTS::outhouse, 3, GetSpawnerName(SpawnerName::interior_Reward) }, + {SpawnLOTS::mushroom, 6, GetSpawnerName(SpawnerName::Reward_01) }, + }, {}, {}, "", -1.0f, + 25, + }, + + // Wave 17 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 18 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::hammerling_melee, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 19 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::hammerling, 4, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::sentry, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::gf_B) }, + } + }, + + // Wave 20 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::sentry, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::concert_C) }, + { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::gf_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::gf_C) }, + } + }, + + // Wave 21 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::ronin, 2, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::spiderling_ve, 2, GetSpawnerName(SpawnerName::interior_C) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::ronin, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::ronin, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::concert_C) }, + { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::ronin, 1, GetSpawnerName(SpawnerName::gf_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::gf_C) }, + } + }, + + // Wave 22 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::spiderling_boss, 1, GetSpawnerName(SpawnerName::Cavalry_Boss) }, + }, + {1887}, + {}, + "Spiderling_Boss", + 5.0f + }, + + // Wave 23 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::outhouse, 6, GetSpawnerName(SpawnerName::Reward_01) }, + { SpawnLOTS::outhouse, 3, GetSpawnerName(SpawnerName::interior_Reward) }, + { SpawnLOTS::maelstrom_chest, 4, GetSpawnerName(SpawnerName::Obstacle) }, + }, {}, {}, "", -1.0f, + 25, + }, + + // Wave 24 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::pirate, 3, GetSpawnerName(SpawnerName::ag_A) }, + { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::ronin, 2, GetSpawnerName(SpawnerName::interior_B) }, + } + }, + + // Wave 25 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::cavalry, 2, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::gf_B) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::gf_A) }, + { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::concert_A) }, + { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_A) }, + } + }, + + // Wave 26 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::gf_B) }, + { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::concert_B) }, + { SpawnLOTS::admiral_cp, 2, GetSpawnerName(SpawnerName::gf_C) }, + { SpawnLOTS::admiral_cp, 2, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::concert_C) }, + } + }, + + // Wave 27 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::ronin, 5, GetSpawnerName(SpawnerName::interior_A) }, + { SpawnLOTS::ronin, 4, GetSpawnerName(SpawnerName::interior_B) }, + { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::ag_C) }, + { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::gf_C) }, + { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::concert_C) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::ag_B) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::gf_B) }, + { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::concert_B) }, + } + }, + + // Wave 28 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::dragon_statue, 12, GetSpawnerName(SpawnerName::Reward_01) }, + }, {}, {}, "", -1.0f, + 30, + }, + + // Wave 29 + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::horseman_boss01, 1, GetSpawnerName(SpawnerName::Horseman_01) }, + { SpawnLOTS::horseman_boss02, 1, GetSpawnerName(SpawnerName::Horseman_02) }, + { SpawnLOTS::horseman_boss03, 1, GetSpawnerName(SpawnerName::Horseman_03) }, + { SpawnLOTS::horseman_boss04, 1, GetSpawnerName(SpawnerName::Horseman_04) }, + }, + {1888}, + {1236, 1237, 1249}, + "Horsemen_Boss", + 5.0f + }, + + // Wave 30 (treasure) + Wave { + std::vector<MobDefinition> { + { SpawnLOTS::treasure_chest, 1, GetSpawnerName(SpawnerName::Treasure_01) }, + }, {}, {}, + "Treasure_Camera", + 5.0f, + (uint32_t)-1, + true, + 60, + }, + }; +} + +std::string ZoneNsWaves::GetSpawnerName(SpawnerName spawnerName) { + switch (spawnerName) { + case interior_A: + return "Base_MobA"; + case interior_B: + return "Base_MobB"; + case interior_C: + return "Base_MobC"; + case gf_A: + return "MobA_01"; + case gf_B: + return "MobB_01"; + case gf_C: + return "MobC_01"; + case concert_A: + return "MobA_02"; + case concert_B: + return "MobB_02"; + case concert_C: + return "MobC_02"; + case ag_A: + return "MobA_03"; + case ag_B: + return "MobB_03"; + case ag_C: + return "MobC_03"; + case Reward_01: + return "Reward_01"; + case interior_Reward: + return "Base_Reward"; + case Obstacle: + return "Obstacle_01"; + case Boss: + return "Boss"; + case Ape_Boss: + return "Ape_Boss"; + case Geyser: + return "Geyser_01"; + case Treasure_01: + return "Treasure_01"; + case Cavalry_Boss: + return "Cavalry_Boss"; + case Horseman_01: + return "Horseman_01"; + case Horseman_02: + return "Horseman_02"; + case Horseman_03: + return "Horseman_03"; + case Horseman_04: + return "Horseman_04"; + default: + return ""; + } +} diff --git a/dScripts/02_server/Map/NS/Waves/ZoneNsWaves.h b/dScripts/02_server/Map/NS/Waves/ZoneNsWaves.h new file mode 100644 index 00000000..637aceb7 --- /dev/null +++ b/dScripts/02_server/Map/NS/Waves/ZoneNsWaves.h @@ -0,0 +1,71 @@ +#pragma once +#include "BaseWavesServer.h" + +#include "dCommonVars.h" + +enum SpawnerName { + interior_A, + interior_B, + interior_C, + gf_A, + gf_B, + gf_C, + concert_A, + concert_B, + concert_C, + ag_A, + ag_B, + ag_C, + Reward_01, + interior_Reward, + Obstacle, + Boss, + Ape_Boss, + Geyser, + Treasure_01, + Cavalry_Boss, + Horseman_01, + Horseman_02, + Horseman_03, + Horseman_04, +}; + +enum SpawnLOTS : LOT { + stromling = 12586, + mech = 12587, + spiderling = 12588, + pirate = 12589, + admiral = 12590, + ape_boss = 12591, + stromling_boss = 12600, + hammerling = 12602, + sentry = 12604, + spiderling_ve = 12605, + spiderling_boss = 12609, + ronin = 12610, + cavalry = 12611, + dragon_boss = 12612, + stromling_minifig = 12586, + mushroom = 12614, + maelstrom_chest = 4894, + outhouse = 12616, + dragon_statue = 12617, + treasure_chest = 12423, + hammerling_melee = 12653, + maelstrom_geyser = 10314, + ronin_statue = 12611, + horseman_boss01 = 11999, + horseman_boss02 = 12467, + horseman_boss03 = 12468, + horseman_boss04 = 12469, + admiral_cp = 13523, +}; + +class ZoneNsWaves : public BaseWavesServer { + WaveConstants GetConstants() override; + std::vector<std::string> GetSpawnerNames() override; + std::vector<WaveMission> GetWaveMissions() override; + std::vector<Wave> GetWaves() override; +private: + static std::string GetSpawnerName(SpawnerName spawnerName); +}; diff --git a/dScripts/02_server/Map/NT/CMakeLists.txt b/dScripts/02_server/Map/NT/CMakeLists.txt new file mode 100644 index 00000000..ede9b003 --- /dev/null +++ b/dScripts/02_server/Map/NT/CMakeLists.txt @@ -0,0 +1,26 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NT + "NtCombatChallengeDummy.cpp" + "NtCombatChallengeExplodingDummy.cpp" + "NtCombatChallengeServer.cpp" + "NtAssemblyTubeServer.cpp" + "NtParadoxPanelServer.cpp" + "NtImagBeamBuffer.cpp" + "NtBeamImaginationCollectors.cpp" + "NtDirtCloudServer.cpp" + "NtConsoleTeleportServer.cpp" + "SpawnStegoServer.cpp" + "SpawnSaberCatServer.cpp" + "SpawnShrakeServer.cpp" + "NtDukeServer.cpp" + "NtHaelServer.cpp" + "NtOverbuildServer.cpp" + "NtVandaServer.cpp" + "NtXRayServer.cpp" + "NtSleepingGuard.cpp" + "NtImagimeterVisibility.cpp" + "NtSentinelWalkwayServer.cpp" + "NtDarkitectRevealServer.cpp" + "NtParadoxTeleServer.cpp" + "NtVentureSpeedPadServer.cpp" + "NtVentureCannonServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/NT/NtAssemblyTubeServer.cpp b/dScripts/02_server/Map/NT/NtAssemblyTubeServer.cpp new file mode 100644 index 00000000..6ae4c767 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtAssemblyTubeServer.cpp @@ -0,0 +1,119 @@ +#include "NtAssemblyTubeServer.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" +#include "RenderComponent.h" +#include "eEndBehavior.h" +#include "eStateChangeType.h" + +void NtAssemblyTubeServer::OnStartup(Entity* self) { + self->SetProximityRadius(5, "teleport"); +} + +void NtAssemblyTubeServer::OnPlayerLoaded(Entity* self, Entity* player) { + +} + +void NtAssemblyTubeServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (status != "ENTER" || !entering->IsPlayer() || name != "teleport") return; + + auto* player = entering; + const auto playerID = player->GetObjectID(); + + RunAssemblyTube(self, player); + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + } +} + +void NtAssemblyTubeServer::RunAssemblyTube(Entity* self, Entity* player) { + const auto playerID = player->GetObjectID(); + + const auto iter = m_TeleportingPlayerTable.find(playerID); + if (iter == m_TeleportingPlayerTable.end()) m_TeleportingPlayerTable[playerID] = false; + const auto bPlayerBeingTeleported = m_TeleportingPlayerTable[playerID]; + + if (player->IsPlayer() && !bPlayerBeingTeleported) { + auto teleCinematic = self->GetVar<std::u16string>(u"Cinematic"); + + GameMessages::SendSetStunned(playerID, eStateChangeType::PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true + ); + + if (!teleCinematic.empty()) { + const auto teleCinematicUname = teleCinematic; + GameMessages::SendPlayCinematic(player->GetObjectID(), teleCinematicUname, player->GetSystemAddress(), + true, true, true, false, eEndBehavior::RETURN, false, -1, false, true + ); + } + + RenderComponent::PlayAnimation(player, u"tube-sucker", 4.0f); + + const auto animTime = 3; + + self->AddCallbackTimer(animTime, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + TeleportPlayer(self, player); + }); + } +} + +void NtAssemblyTubeServer::TeleportPlayer(Entity* self, Entity* player) { + auto destinationGroup = self->GetVar<std::u16string>(u"teleGroup"); + auto* destination = self; + + if (!destinationGroup.empty()) { + const auto& groupObjs = Game::entityManager->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(destinationGroup)); + + if (!groupObjs.empty()) { + destination = groupObjs[0]; + } + } + + const auto destPosition = destination->GetPosition(); + const auto destRotation = destination->GetRotation(); + + GameMessages::SendTeleport(player->GetObjectID(), destPosition, destRotation, player->GetSystemAddress(), true); + + RenderComponent::PlayAnimation(player, u"tube-resurrect", 4.0f); + + const auto animTime = 2; + + const auto playerID = player->GetObjectID(); + + self->AddCallbackTimer(animTime, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + UnlockPlayer(self, player); + }); + + const auto useSound = self->GetVar<std::string>(u"sound1"); + + if (!useSound.empty()) { + GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), useSound); + } +} + +void NtAssemblyTubeServer::UnlockPlayer(Entity* self, Entity* player) { + const auto playerID = player->GetObjectID(); + + m_TeleportingPlayerTable[playerID] = false; + + GameMessages::SendSetStunned(playerID, eStateChangeType::POP, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true + ); +} diff --git a/dScripts/NtAssemblyTubeServer.h b/dScripts/02_server/Map/NT/NtAssemblyTubeServer.h similarity index 100% rename from dScripts/NtAssemblyTubeServer.h rename to dScripts/02_server/Map/NT/NtAssemblyTubeServer.h diff --git a/dScripts/NtBeamImaginationCollectors.cpp b/dScripts/02_server/Map/NT/NtBeamImaginationCollectors.cpp similarity index 100% rename from dScripts/NtBeamImaginationCollectors.cpp rename to dScripts/02_server/Map/NT/NtBeamImaginationCollectors.cpp diff --git a/dScripts/NtBeamImaginationCollectors.h b/dScripts/02_server/Map/NT/NtBeamImaginationCollectors.h similarity index 100% rename from dScripts/NtBeamImaginationCollectors.h rename to dScripts/02_server/Map/NT/NtBeamImaginationCollectors.h diff --git a/dScripts/02_server/Map/NT/NtCombatChallengeDummy.cpp b/dScripts/02_server/Map/NT/NtCombatChallengeDummy.cpp new file mode 100644 index 00000000..5be5a9b3 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtCombatChallengeDummy.cpp @@ -0,0 +1,26 @@ +#include "NtCombatChallengeDummy.h" +#include "EntityManager.h" + +void NtCombatChallengeDummy::OnDie(Entity* self, Entity* killer) { + const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); + + auto* challengeObject = Game::entityManager->GetEntity(challengeObjectID); + + if (challengeObject != nullptr) { + for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { + script->OnDie(challengeObject, killer); + } + } +} + +void NtCombatChallengeDummy::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); + + auto* challengeObject = Game::entityManager->GetEntity(challengeObjectID); + + if (challengeObject != nullptr) { + for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { + script->OnHitOrHealResult(challengeObject, attacker, damage); + } + } +} diff --git a/dScripts/NtCombatChallengeDummy.h b/dScripts/02_server/Map/NT/NtCombatChallengeDummy.h similarity index 100% rename from dScripts/NtCombatChallengeDummy.h rename to dScripts/02_server/Map/NT/NtCombatChallengeDummy.h diff --git a/dScripts/02_server/Map/NT/NtCombatChallengeExplodingDummy.cpp b/dScripts/02_server/Map/NT/NtCombatChallengeExplodingDummy.cpp new file mode 100644 index 00000000..de27e106 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtCombatChallengeExplodingDummy.cpp @@ -0,0 +1,32 @@ +#include "NtCombatChallengeExplodingDummy.h" +#include "EntityManager.h" +#include "SkillComponent.h" + +void NtCombatChallengeExplodingDummy::OnDie(Entity* self, Entity* killer) { + const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); + + auto* challengeObject = Game::entityManager->GetEntity(challengeObjectID); + + if (challengeObject != nullptr) { + for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { + script->OnDie(challengeObject, killer); + } + } +} + +void NtCombatChallengeExplodingDummy::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); + + auto* challengeObject = Game::entityManager->GetEntity(challengeObjectID); + + if (challengeObject != nullptr) { + for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { + script->OnHitOrHealResult(challengeObject, attacker, damage); + } + } + auto skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(1338, 30875, attacker->GetObjectID()); + } + self->Kill(attacker); +} diff --git a/dScripts/NtCombatChallengeExplodingDummy.h b/dScripts/02_server/Map/NT/NtCombatChallengeExplodingDummy.h similarity index 100% rename from dScripts/NtCombatChallengeExplodingDummy.h rename to dScripts/02_server/Map/NT/NtCombatChallengeExplodingDummy.h diff --git a/dScripts/02_server/Map/NT/NtCombatChallengeServer.cpp b/dScripts/02_server/Map/NT/NtCombatChallengeServer.cpp new file mode 100644 index 00000000..011a67ea --- /dev/null +++ b/dScripts/02_server/Map/NT/NtCombatChallengeServer.cpp @@ -0,0 +1,209 @@ +#include "NtCombatChallengeServer.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "InventoryComponent.h" +#include "MissionComponent.h" + +void NtCombatChallengeServer::OnUse(Entity* self, Entity* user) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UI_Open", 0, 0, user->GetObjectID(), "", user->GetSystemAddress()); +} + +void NtCombatChallengeServer::OnDie(Entity* self, Entity* killer) { + if (killer != self && killer != nullptr) { + SpawnTargetDummy(self); + } +} + + +void NtCombatChallengeServer::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); + + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + auto totalDmg = self->GetVar<int32_t>(u"totalDmg"); + + totalDmg += damage; + + self->SetVar(u"totalDmg", totalDmg); + self->SetNetworkVar(u"totalDmg", totalDmg); + + GameMessages::SendPlayNDAudioEmitter(self, attacker->GetSystemAddress(), scoreSound); +} + + +void NtCombatChallengeServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UI_Close", 0, 0, sender->GetObjectID(), "", sender->GetSystemAddress()); +} + + +void NtCombatChallengeServer::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + if (identifier == u"PlayButton" && button == 1 && !self->GetNetworkVar<bool>(u"bInUse")) { + self->SetNetworkVar(u"bInUse", true); + + self->SetVar(u"playerID", sender->GetObjectID()); + + auto* inventoryComponent = sender->GetComponent<InventoryComponent>(); + + if (inventoryComponent != nullptr) { + inventoryComponent->RemoveItem(3039, 1); + } + + GameMessages::SendPlayNDAudioEmitter(self, sender->GetSystemAddress(), startSound); + + self->AddTimer("start_delay", 2.0f); + + GameMessages::SendShowActivityCountdown(self->GetObjectID(), false, false, u"", 0, sender->GetSystemAddress()); + + self->SetNetworkVar(u"toggle", true); + } else if (identifier == u"CloseButton") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UI_Close", 1, 0, sender->GetObjectID(), "", sender->GetSystemAddress()); + } +} + +void NtCombatChallengeServer::SpawnTargetDummy(Entity* self) { + const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); + + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + auto targetNumber = self->GetVar<int32_t>(u"TargetNumber"); + if (targetNumber == 0) targetNumber = 1; + + if (targetNumber > tTargets.size()) targetNumber = tTargets.size(); + + self->SetVar<int32_t>(u"TargetNumber", targetNumber + 1); + + const auto dummyLOT = tTargets[targetNumber - 1]; + + EntityInfo info{}; + info.lot = dummyLOT; + info.spawnerID = self->GetObjectID(); + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.settings = { new LDFData<std::string>(u"custom_script_server", "scripts\\02_server\\Map\\NT\\L_NT_COMBAT_CHALLENGE_DUMMY.lua") }; + + auto* dummy = Game::entityManager->CreateEntity(info); + + dummy->SetVar(u"challengeObjectID", self->GetObjectID()); + + Game::entityManager->ConstructEntity(dummy); + + self->SetVar(u"currentDummy", dummy->GetObjectID()); +} + +void NtCombatChallengeServer::SetAttackImmunity(LWOOBJID objID, bool bTurnOn) { + +} + +void NtCombatChallengeServer::OnChildLoaded(Entity* self, Entity* child) { + auto targetNumber = self->GetVar<int32_t>(u"TargetNumber"); + if (targetNumber == 0) targetNumber = 1; + self->SetVar(u"TargetNumber", targetNumber + 1); + + const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); + + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + child->SetRotation(NiQuaternion::LookAt(child->GetPosition(), player->GetPosition())); + + self->SetVar(u"currentTargetID", child->GetObjectID()); + + Game::entityManager->SerializeEntity(child); + + child->GetGroups().push_back("targets_" + std::to_string(self->GetObjectID())); +} + +void NtCombatChallengeServer::ResetGame(Entity* self) { + const auto totalDmg = self->GetVar<int32_t>(u"totalDmg"); + const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); + + auto* player = Game::entityManager->GetEntity(playerID); + + if (player != nullptr) { + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + for (const auto& mission : tMissions) { + if (totalDmg >= mission.damage) { + missionComponent->ForceProgressTaskType(mission.mission, 1, 1); + } + } + } + } + + self->SetVar(u"TargetNumber", 1); + self->SetVar(u"playerID", LWOOBJID_EMPTY); + self->SetVar(u"totalDmg", 0); + self->SetNetworkVar(u"totalDmg", false); + self->SetNetworkVar(u"update_time", 0); + + const auto& targetObjs = Game::entityManager->GetEntitiesInGroup("targets_" + std::to_string(self->GetObjectID())); + + for (auto* target : targetObjs) { + target->Smash(self->GetObjectID()); + } + + const auto currentID = self->GetVar<LWOOBJID>(u"currentDummy"); + + auto* current = Game::entityManager->GetEntity(currentID); + + if (current != nullptr) { + current->Smash(self->GetObjectID()); + } +} + +void NtCombatChallengeServer::OnActivityTimerUpdate(Entity* self, float timeRemaining) { + self->SetNetworkVar(u"update_time", std::ceil(timeRemaining)); + + if (timeRemaining <= 3) { + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, timerLowSound); + } else { + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, timerSound); + } +} + +void NtCombatChallengeServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "start_delay") { + self->SetVar(u"game_tick", gameTime); + + SpawnTargetDummy(self); + + self->AddTimer("game_tick", 1); + + self->SetNetworkVar(u"totalTime", gameTime); + } else if (timerName == "game_tick") { + auto gameTick = self->GetVar<float>(u"game_tick"); + + gameTick -= 1; + + self->SetVar(u"game_tick", gameTick); + + if (gameTick <= 0) { + ResetGame(self); + + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, stopSound); + + self->AddTimer("reset_tick", 5); + } else { + self->AddTimer("game_tick", 1); + + OnActivityTimerUpdate(self, gameTick); + } + } else if (timerName == "reset_tick") { + self->SetNetworkVar(u"toggle", false); + self->SetNetworkVar(u"bInUse", false); + } +} diff --git a/dScripts/NtCombatChallengeServer.h b/dScripts/02_server/Map/NT/NtCombatChallengeServer.h similarity index 100% rename from dScripts/NtCombatChallengeServer.h rename to dScripts/02_server/Map/NT/NtCombatChallengeServer.h diff --git a/dScripts/02_server/Map/NT/NtConsoleTeleportServer.cpp b/dScripts/02_server/Map/NT/NtConsoleTeleportServer.cpp new file mode 100644 index 00000000..8ba697ce --- /dev/null +++ b/dScripts/02_server/Map/NT/NtConsoleTeleportServer.cpp @@ -0,0 +1,28 @@ +#include "NtConsoleTeleportServer.h" +#include "Entity.h" +#include "Amf3.h" + +void NtConsoleTeleportServer::OnStartup(Entity* self) { + self->SetVar(u"teleportAnim", m_TeleportAnim); + self->SetVar(u"teleportString", m_TeleportString); +} + +void NtConsoleTeleportServer::OnUse(Entity* self, Entity* user) { + BaseOnUse(self, user); +} + +void NtConsoleTeleportServer::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + BaseOnMessageBoxResponse(self, sender, button, identifier, userData); +} + +void NtConsoleTeleportServer::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { + +} + +void NtConsoleTeleportServer::OnTimerDone(Entity* self, std::string timerName) { + BaseOnTimerDone(self, timerName); +} + +void NtConsoleTeleportServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); +} diff --git a/dScripts/NtConsoleTeleportServer.h b/dScripts/02_server/Map/NT/NtConsoleTeleportServer.h similarity index 100% rename from dScripts/NtConsoleTeleportServer.h rename to dScripts/02_server/Map/NT/NtConsoleTeleportServer.h diff --git a/dScripts/NtDarkitectRevealServer.cpp b/dScripts/02_server/Map/NT/NtDarkitectRevealServer.cpp similarity index 100% rename from dScripts/NtDarkitectRevealServer.cpp rename to dScripts/02_server/Map/NT/NtDarkitectRevealServer.cpp diff --git a/dScripts/NtDarkitectRevealServer.h b/dScripts/02_server/Map/NT/NtDarkitectRevealServer.h similarity index 100% rename from dScripts/NtDarkitectRevealServer.h rename to dScripts/02_server/Map/NT/NtDarkitectRevealServer.h diff --git a/dScripts/02_server/Map/NT/NtDirtCloudServer.cpp b/dScripts/02_server/Map/NT/NtDirtCloudServer.cpp new file mode 100644 index 00000000..92175dea --- /dev/null +++ b/dScripts/02_server/Map/NT/NtDirtCloudServer.cpp @@ -0,0 +1,46 @@ +#include "NtDirtCloudServer.h" +#include "MissionComponent.h" + +std::map<std::string, std::vector<int32_t>> NtDirtCloudServer::m_Missions = +{ + {"Dirt_Clouds_Sent", {1333,1253}}, + {"Dirt_Clouds_Assem", {1333,1276}}, + {"Dirt_Clouds_Para", {1333,1277}}, + {"Dirt_Clouds_Halls", {1333,1283}} +}; + +void NtDirtCloudServer::OnStartup(Entity* self) { + self->SetVar(u"CloudOn", true); +} + +void NtDirtCloudServer::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (message != "soapspray") { + return; + } + + if (!self->GetVar<bool>(u"CloudOn")) { + return; + } + + const auto mySpawner = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); + + if (m_Missions.count(mySpawner) == 0) { + return; + } + + const auto& myMis = m_Missions[mySpawner]; + + auto* missionComponent = caster->GetComponent<MissionComponent>(); + + if (missionComponent == nullptr) { + return; + } + + for (const auto missionID : myMis) { + missionComponent->ForceProgressTaskType(missionID, 1, 1); + } + + self->SetVar(u"CloudOn", false); + + self->Smash(self->GetObjectID(), eKillType::VIOLENT); +} diff --git a/dScripts/NtDirtCloudServer.h b/dScripts/02_server/Map/NT/NtDirtCloudServer.h similarity index 100% rename from dScripts/NtDirtCloudServer.h rename to dScripts/02_server/Map/NT/NtDirtCloudServer.h diff --git a/dScripts/02_server/Map/NT/NtDukeServer.cpp b/dScripts/02_server/Map/NT/NtDukeServer.cpp new file mode 100644 index 00000000..07d17e96 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtDukeServer.cpp @@ -0,0 +1,40 @@ +#include "NtDukeServer.h" +#include "InventoryComponent.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "ePlayerFlag.h" + +void NtDukeServer::SetVariables(Entity* self) { + self->SetVar<float_t>(m_SpyProximityVariable, 35.0f); + + self->SetVar<SpyData>(m_SpyDataVariable, { + ePlayerFlag::NT_FACTION_SPY_DUKE, 13548, 1319 + }); + + self->SetVar<std::vector<SpyDialogue>>(m_SpyDialogueTableVariable, { + { "DUKE_NT_CONVO_1", 0 }, + { "DUKE_NT_CONVO_2", 0 }, + { "DUKE_NT_CONVO_3", 0 }, + }); + + // If there's an alternating conversation, indices should be provided using the conversationID variables + self->SetVar<std::vector<LWOOBJID>>(m_SpyCinematicObjectsVariable, { self->GetObjectID() }); +} + +void NtDukeServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + + // Handles adding and removing the sword for the Crux Prime Sword mission + auto* missionComponent = target->GetComponent<MissionComponent>(); + auto* inventoryComponent = target->GetComponent<InventoryComponent>(); + + if (missionComponent != nullptr && inventoryComponent != nullptr) { + auto state = missionComponent->GetMissionState(m_SwordMissionID); + auto lotCount = inventoryComponent->GetLotCount(m_SwordLot); + + if ((state == eMissionState::AVAILABLE || state == eMissionState::ACTIVE) && lotCount < 1) { + inventoryComponent->AddItem(m_SwordLot, 1, eLootSourceType::NONE); + } else if (state == eMissionState::READY_TO_COMPLETE) { + inventoryComponent->RemoveItem(m_SwordLot, lotCount); + } + } +} diff --git a/dScripts/02_server/Map/NT/NtDukeServer.h b/dScripts/02_server/Map/NT/NtDukeServer.h new file mode 100644 index 00000000..2103ba8d --- /dev/null +++ b/dScripts/02_server/Map/NT/NtDukeServer.h @@ -0,0 +1,9 @@ +#pragma once +#include "NtFactionSpyServer.h" + +class NtDukeServer : public NtFactionSpyServer { + void SetVariables(Entity* self) override; + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + const uint32_t m_SwordMissionID = 1448; + const LOT m_SwordLot = 13777; +}; diff --git a/dScripts/02_server/Map/NT/NtHaelServer.cpp b/dScripts/02_server/Map/NT/NtHaelServer.cpp new file mode 100644 index 00000000..c726ae58 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtHaelServer.cpp @@ -0,0 +1,21 @@ +#include "NtHaelServer.h" +#include "Entity.h" +#include "ePlayerFlag.h" + +void NtHaelServer::SetVariables(Entity* self) { + self->SetVar<float_t>(m_SpyProximityVariable, 25.0f); + + self->SetVar<SpyData>(m_SpyDataVariable, { + ePlayerFlag::NT_FACTION_SPY_HAEL, 13892, 1321 + }); + + self->SetVar<std::vector<SpyDialogue>>(m_SpyDialogueTableVariable, { + { "HAEL_NT_CONVO_1", 0 }, + { "HAEL_NT_CONVO_2", 0 }, + { "HAEL_NT_CONVO_3", 0 }, + { "HAEL_NT_CONVO_4", 0 }, + }); + + // If there's an alternating conversation, indices should be provided using the conversationID variables + self->SetVar<std::vector<LWOOBJID>>(m_SpyCinematicObjectsVariable, { self->GetObjectID() }); +} diff --git a/dScripts/NtHaelServer.h b/dScripts/02_server/Map/NT/NtHaelServer.h similarity index 100% rename from dScripts/NtHaelServer.h rename to dScripts/02_server/Map/NT/NtHaelServer.h diff --git a/dScripts/02_server/Map/NT/NtImagBeamBuffer.cpp b/dScripts/02_server/Map/NT/NtImagBeamBuffer.cpp new file mode 100644 index 00000000..961e93cd --- /dev/null +++ b/dScripts/02_server/Map/NT/NtImagBeamBuffer.cpp @@ -0,0 +1,53 @@ +#include "NtImagBeamBuffer.h" +#include "EntityManager.h" +#include "SkillComponent.h" + +void NtImagBeamBuffer::OnStartup(Entity* self) { + self->SetProximityRadius(100, "ImagZone"); + + self->AddTimer("BuffImag", 2.0f); +} + +void NtImagBeamBuffer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name != "ImagZone" || !entering->IsPlayer()) { + return; + } + + if (status == "ENTER") { + const auto& iter = std::find(m_EntitiesInProximity.begin(), m_EntitiesInProximity.end(), entering->GetObjectID()); + + if (iter == m_EntitiesInProximity.end()) { + m_EntitiesInProximity.push_back(entering->GetObjectID()); + } + } else if (status == "LEAVE") { + const auto& iter = std::find(m_EntitiesInProximity.begin(), m_EntitiesInProximity.end(), entering->GetObjectID()); + + if (iter != m_EntitiesInProximity.end()) { + m_EntitiesInProximity.erase(iter); + } + } +} + +void NtImagBeamBuffer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName != "BuffImag") { + return; + } + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + return; + } + + for (const auto entityID : m_EntitiesInProximity) { + auto* entity = Game::entityManager->GetEntity(entityID); + + if (entity == nullptr) { + continue; + } + + skillComponent->CalculateBehavior(1311, 30235, entityID, true); + } + + self->AddTimer("BuffImag", 2.0f); +} diff --git a/dScripts/NtImagBeamBuffer.h b/dScripts/02_server/Map/NT/NtImagBeamBuffer.h similarity index 100% rename from dScripts/NtImagBeamBuffer.h rename to dScripts/02_server/Map/NT/NtImagBeamBuffer.h diff --git a/dScripts/02_server/Map/NT/NtImagimeterVisibility.cpp b/dScripts/02_server/Map/NT/NtImagimeterVisibility.cpp new file mode 100644 index 00000000..64493d78 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtImagimeterVisibility.cpp @@ -0,0 +1,12 @@ +#include "NtImagimeterVisibility.h" +#include "GameMessages.h" +#include "Entity.h" +#include "Character.h" +#include "ePlayerFlag.h" + +void NTImagimeterVisibility::OnRebuildComplete(Entity* self, Entity* target) { + auto* character = target->GetCharacter(); + if (character) character->SetPlayerFlag(ePlayerFlag::NT_PLINTH_REBUILD, true); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlinthBuilt", 0, 0, LWOOBJID_EMPTY, "", target->GetSystemAddress()); +} diff --git a/dScripts/NtImagimeterVisibility.h b/dScripts/02_server/Map/NT/NtImagimeterVisibility.h similarity index 100% rename from dScripts/NtImagimeterVisibility.h rename to dScripts/02_server/Map/NT/NtImagimeterVisibility.h diff --git a/dScripts/02_server/Map/NT/NtOverbuildServer.cpp b/dScripts/02_server/Map/NT/NtOverbuildServer.cpp new file mode 100644 index 00000000..e75d8add --- /dev/null +++ b/dScripts/02_server/Map/NT/NtOverbuildServer.cpp @@ -0,0 +1,31 @@ +#include "NtOverbuildServer.h" +#include "EntityManager.h" +#include "ePlayerFlag.h" + +void NtOverbuildServer::SetVariables(Entity* self) { + self->SetVar<float_t>(m_SpyProximityVariable, 30.0f); + + self->SetVar<SpyData>(m_SpyDataVariable, { + ePlayerFlag::NT_FACTION_SPY_OVERBUILD, 13891, 1320 + }); + + self->SetVar<std::vector<SpyDialogue>>(m_SpyDialogueTableVariable, { + { "OVERBUILD_NT_CONVO_1", 0 }, + { "OVERBUILD_NT_CONVO_2", 1 }, + { "OVERBUILD_NT_CONVO_3", 0 }, + { "OVERBUILD_NT_CONVO_4", 1 }, + { "OVERBUILD_NT_CONVO_5", 0 }, + { "OVERBUILD_NT_CONVO_6", 1 }, + { "OVERBUILD_NT_CONVO_7", 0 }, + }); + + // Find the second object Dr. Overbuild interacts with + LWOOBJID otherConvoObjectID = LWOOBJID_EMPTY; + for (auto* otherConvoObject : Game::entityManager->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(m_OtherEntitiesGroupVariable)))) { + otherConvoObjectID = otherConvoObject->GetObjectID(); + break; + } + + // If there's an alternating conversation, indices should be provided using the conversationID variables + self->SetVar<std::vector<LWOOBJID>>(m_SpyCinematicObjectsVariable, { self->GetObjectID(), otherConvoObjectID }); +} diff --git a/dScripts/NtOverbuildServer.h b/dScripts/02_server/Map/NT/NtOverbuildServer.h similarity index 100% rename from dScripts/NtOverbuildServer.h rename to dScripts/02_server/Map/NT/NtOverbuildServer.h diff --git a/dScripts/02_server/Map/NT/NtParadoxPanelServer.cpp b/dScripts/02_server/Map/NT/NtParadoxPanelServer.cpp new file mode 100644 index 00000000..0fe97a26 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtParadoxPanelServer.cpp @@ -0,0 +1,70 @@ +#include "NtParadoxPanelServer.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "EntityManager.h" +#include "Character.h" +#include "eMissionState.h" +#include "RenderComponent.h" +#include "eTerminateType.h" +#include "eStateChangeType.h" + +void NtParadoxPanelServer::OnUse(Entity* self, Entity* user) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"bActive", 1, 0, user->GetObjectID(), "", user->GetSystemAddress()); + + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); + + self->SetVar(u"bActive", true); + + auto* missionComponent = user->GetComponent<MissionComponent>(); + + const auto playerID = user->GetObjectID(); + + for (const auto mission : tPlayerOnMissions) { + if (missionComponent->GetMissionState(mission) != eMissionState::ACTIVE) { + continue; + } + + self->AddCallbackTimer(2, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + const auto flag = self->GetVar<int32_t>(u"flag"); + + player->GetCharacter()->SetPlayerFlag(flag, true); + + RenderComponent::PlayAnimation(player, u"rebuild-celebrate"); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SparkStop", 0, 0, player->GetObjectID(), "", player->GetSystemAddress()); + GameMessages::SendSetStunned(player->GetObjectID(), eStateChangeType::POP, player->GetSystemAddress(), LWOOBJID_EMPTY, false, false, true, false, true, true, false, false, true); + self->SetVar(u"bActive", false); + }); + RenderComponent::PlayAnimation(user, u"nexus-powerpanel", 6.0f); + GameMessages::SendSetStunned(user->GetObjectID(), eStateChangeType::PUSH, user->GetSystemAddress(), LWOOBJID_EMPTY, false, false, true, false, true, true, false, false, true); + return; + } + + RenderComponent::PlayAnimation(user, shockAnim); + + const auto dir = self->GetRotation().GetRightVector(); + + GameMessages::SendKnockback(user->GetObjectID(), self->GetObjectID(), self->GetObjectID(), 0, { dir.x * 15, 5, dir.z * 15 }); + + GameMessages::SendPlayFXEffect(self, 6432, u"create", "console_sparks", LWOOBJID_EMPTY, 1.0, 1.0, true); + + self->AddCallbackTimer(2, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"bActive", 0, 0, player->GetObjectID(), "", player->GetSystemAddress()); + + GameMessages::SendStopFXEffect(self, true, "console_sparks"); + + self->SetVar(u"bActive", false); + }); +} diff --git a/dScripts/NtParadoxPanelServer.h b/dScripts/02_server/Map/NT/NtParadoxPanelServer.h similarity index 100% rename from dScripts/NtParadoxPanelServer.h rename to dScripts/02_server/Map/NT/NtParadoxPanelServer.h diff --git a/dScripts/02_server/Map/NT/NtParadoxTeleServer.cpp b/dScripts/02_server/Map/NT/NtParadoxTeleServer.cpp new file mode 100644 index 00000000..67ff4bec --- /dev/null +++ b/dScripts/02_server/Map/NT/NtParadoxTeleServer.cpp @@ -0,0 +1,115 @@ +#include "NtParadoxTeleServer.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "RenderComponent.h" +#include "eStateChangeType.h" + +void NtParadoxTeleServer::OnStartup(Entity* self) { + self->SetProximityRadius(5, "teleport"); +} + +void NtParadoxTeleServer::OnPlayerLoaded(Entity* self, Entity* player) { + +} + +void NtParadoxTeleServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (status != "ENTER" || !entering->IsPlayer() || name != "teleport") return; + + auto* player = entering; + const auto playerID = player->GetObjectID(); + + const auto iter = m_TeleportingPlayerTable.find(playerID); + if (iter == m_TeleportingPlayerTable.end()) m_TeleportingPlayerTable[playerID] = false; + const auto bPlayerBeingTeleported = m_TeleportingPlayerTable[playerID]; + + if (player->IsPlayer() && !bPlayerBeingTeleported) { + GameMessages::SendSetStunned(playerID, eStateChangeType::PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true + ); + + auto animTime = RenderComponent::PlayAnimation(player, u"teledeath", 4.0f); + if (animTime == 0.0f) animTime = 2.0f; + + self->AddCallbackTimer(animTime, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + TeleportPlayer(self, player); + }); + } + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + } +} + +void NtParadoxTeleServer::TeleportPlayer(Entity* self, Entity* player) { + auto destinationGroup = self->GetVar<std::u16string>(u"teleGroup"); + auto* destination = self; + + if (!destinationGroup.empty()) { + const auto& groupObjs = Game::entityManager->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(destinationGroup)); + + if (!groupObjs.empty()) { + destination = groupObjs[0]; + } + } + + const auto destPosition = destination->GetPosition(); + const auto destRotation = destination->GetRotation(); + + auto teleCinematic = self->GetVar<std::u16string>(u"Cinematic"); + + if (!teleCinematic.empty()) { + const auto teleCinematicUname = teleCinematic; + GameMessages::SendPlayCinematic(player->GetObjectID(), teleCinematicUname, player->GetSystemAddress()); + } + + GameMessages::SendTeleport(player->GetObjectID(), destPosition, destRotation, player->GetSystemAddress(), true); + + RenderComponent::PlayAnimation(player, u"paradox-teleport-in", 4.0f); + + const auto animTime = 2; + + const auto playerID = player->GetObjectID(); + + self->AddCallbackTimer(animTime, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + UnlockPlayer(self, player); + }); + + const auto useSound = self->GetVar<std::string>(u"sound1"); + + if (!useSound.empty()) { + GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), useSound); + } +} + +void NtParadoxTeleServer::UnlockPlayer(Entity* self, Entity* player) { + const auto playerID = player->GetObjectID(); + + m_TeleportingPlayerTable[playerID] = false; + + GameMessages::SendSetStunned(playerID, eStateChangeType::POP, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true + ); + + auto teleCinematic = self->GetVar<std::u16string>(u"Cinematic"); + + if (!teleCinematic.empty()) { + const auto teleCinematicUname = teleCinematic; + GameMessages::SendEndCinematic(player->GetObjectID(), teleCinematicUname, player->GetSystemAddress()); + } +} diff --git a/dScripts/NtParadoxTeleServer.h b/dScripts/02_server/Map/NT/NtParadoxTeleServer.h similarity index 100% rename from dScripts/NtParadoxTeleServer.h rename to dScripts/02_server/Map/NT/NtParadoxTeleServer.h diff --git a/dScripts/02_server/Map/NT/NtSentinelWalkwayServer.cpp b/dScripts/02_server/Map/NT/NtSentinelWalkwayServer.cpp new file mode 100644 index 00000000..cac0c75f --- /dev/null +++ b/dScripts/02_server/Map/NT/NtSentinelWalkwayServer.cpp @@ -0,0 +1,45 @@ +#include "NtSentinelWalkwayServer.h" +#include "PhantomPhysicsComponent.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "ePhysicsEffectType.h" + +void NtSentinelWalkwayServer::OnStartup(Entity* self) { + auto* phantomPhysicsComponent = self->GetComponent<PhantomPhysicsComponent>(); + + if (phantomPhysicsComponent == nullptr) { + return; + } + + auto force = self->GetVar<int32_t>(u"force"); + + if (force == 0) { + force = 115; + } + + const auto forward = self->GetRotation().GetRightVector() * -1; + + phantomPhysicsComponent->SetEffectType(ePhysicsEffectType::PUSH); + phantomPhysicsComponent->SetDirectionalMultiplier(force); + phantomPhysicsComponent->SetDirection(forward); + phantomPhysicsComponent->SetPhysicsEffectActive(true); + + Game::entityManager->SerializeEntity(self); + + self->SetProximityRadius(3, "speedboost"); +} + +void NtSentinelWalkwayServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name != "speedboost" || !entering->IsPlayer() || status != "ENTER") { + return; + } + + auto* player = entering; + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + } +} diff --git a/dScripts/NtSentinelWalkwayServer.h b/dScripts/02_server/Map/NT/NtSentinelWalkwayServer.h similarity index 100% rename from dScripts/NtSentinelWalkwayServer.h rename to dScripts/02_server/Map/NT/NtSentinelWalkwayServer.h diff --git a/dScripts/02_server/Map/NT/NtSleepingGuard.cpp b/dScripts/02_server/Map/NT/NtSleepingGuard.cpp new file mode 100644 index 00000000..92a80582 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtSleepingGuard.cpp @@ -0,0 +1,36 @@ +#include "NtSleepingGuard.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "RenderComponent.h" + +void NtSleepingGuard::OnStartup(Entity* self) { + self->SetNetworkVar<bool>(u"asleep", true); +} + +void NtSleepingGuard::OnEmoteReceived(Entity* self, const int32_t emote, Entity* target) { + if (!self->GetNetworkVar<bool>(u"asleep")) + return; + + // Check if emote is in m_ValidEmotes + if (std::find(m_ValidEmotes.begin(), m_ValidEmotes.end(), emote) == m_ValidEmotes.end()) + return; + + // Set asleep to false + self->SetNetworkVar<bool>(u"asleep", false); + + RenderComponent::PlayAnimation(self, u"greet"); + + auto* missionComponent = target->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->CompleteMission(1346); + } + + self->AddTimer("AsleepAgain", 5.0f); +} + +void NtSleepingGuard::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "AsleepAgain") { + self->SetNetworkVar<bool>(u"asleep", true); + } +} diff --git a/dScripts/NtSleepingGuard.h b/dScripts/02_server/Map/NT/NtSleepingGuard.h similarity index 100% rename from dScripts/NtSleepingGuard.h rename to dScripts/02_server/Map/NT/NtSleepingGuard.h diff --git a/dScripts/02_server/Map/NT/NtVandaServer.cpp b/dScripts/02_server/Map/NT/NtVandaServer.cpp new file mode 100644 index 00000000..7750d566 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtVandaServer.cpp @@ -0,0 +1,14 @@ +#include "NtVandaServer.h" +#include "InventoryComponent.h" +#include "eMissionState.h" + +void NtVandaServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + + // Removes the alien parts after completing the mission + if (missionID == m_AlienPartMissionID && missionState == eMissionState::READY_TO_COMPLETE) { + auto* inventoryComponent = target->GetComponent<InventoryComponent>(); + for (const auto& alienPartLot : m_AlienPartLots) { + inventoryComponent->RemoveItem(alienPartLot, 1); + } + } +} diff --git a/dScripts/02_server/Map/NT/NtVandaServer.h b/dScripts/02_server/Map/NT/NtVandaServer.h new file mode 100644 index 00000000..58162cd9 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtVandaServer.h @@ -0,0 +1,8 @@ +#pragma once +#include "CppScripts.h" + +class NtVandaServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + const uint32_t m_AlienPartMissionID = 1183; + const std::vector<LOT> m_AlienPartLots = { 12479, 12480, 12481 }; +}; diff --git a/dScripts/02_server/Map/NT/NtVentureCannonServer.cpp b/dScripts/02_server/Map/NT/NtVentureCannonServer.cpp new file mode 100644 index 00000000..7e38ac12 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtVentureCannonServer.cpp @@ -0,0 +1,128 @@ +#include "NtVentureCannonServer.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "Entity.h" +#include "GeneralUtils.h" +#include "RenderComponent.h" +#include "eEndBehavior.h" +#include "eTerminateType.h" +#include "eStateChangeType.h" + +void NtVentureCannonServer::OnUse(Entity* self, Entity* user) { + auto* player = user; + const auto playerID = player->GetObjectID(); + + auto enterCinematic = self->GetVar<std::u16string>(u"EnterCinematic"); + + if (enterCinematic.empty()) { + return; + } + + self->SetNetworkVar(u"bIsInUse", true); + + GameMessages::SendSetStunned(playerID, eStateChangeType::PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true + ); + + auto destPosition = self->GetPosition(); + + destPosition.y += 5 - 1.57f; + + auto destRotation = self->GetRotation(); + + GameMessages::SendTeleport(playerID, destPosition, destRotation, player->GetSystemAddress(), true); + + RenderComponent::PlayAnimation(player, u"scale-down", 4.0f); + + const auto enterCinematicUname = enterCinematic; + GameMessages::SendPlayCinematic(player->GetObjectID(), enterCinematicUname, player->GetSystemAddress()); + + GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), "{e8bf79ce-7453-4a7d-b872-fee65e97ff15}"); + + self->AddCallbackTimer(3, [this, self]() { + self->SetNetworkVar(u"bIsInUse", false); + }); + + self->AddCallbackTimer(1.5f, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + EnterCannonEnded(self, player); + }); +} + +void NtVentureCannonServer::EnterCannonEnded(Entity* self, Entity* player) { + const auto playerID = player->GetObjectID(); + + const auto& cannonEffectGroup = Game::entityManager->GetEntitiesInGroup("cannonEffect"); + + if (!cannonEffectGroup.empty()) { + auto* cannonEffect = cannonEffectGroup[0]; + + GameMessages::SendPlayFXEffect(cannonEffect, 6036, u"create", "cannon_blast", LWOOBJID_EMPTY, 1, 1, true); + + GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(cannonEffect, u"camshake-bridge", cannonEffect->GetObjectID(), 100); + } + + FirePlayer(self, player); + + auto exitCinematic = self->GetVar<std::u16string>(u"ExitCinematic"); + + if (exitCinematic.empty()) { + UnlockCannonPlayer(self, player); + + return; + } + + const auto exitCinematicUname = exitCinematic; + GameMessages::SendPlayCinematic(player->GetObjectID(), exitCinematicUname, player->GetSystemAddress(), + true, true, true, false, eEndBehavior::RETURN, false, 0, false, false + ); + + self->AddCallbackTimer(1.5f, [this, self, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + + if (player == nullptr) { + return; + } + + ExitCannonEnded(self, player); + }); +} + +void NtVentureCannonServer::ExitCannonEnded(Entity* self, Entity* player) { + UnlockCannonPlayer(self, player); +} + +void NtVentureCannonServer::UnlockCannonPlayer(Entity* self, Entity* player) { + GameMessages::SendSetStunned(player->GetObjectID(), eStateChangeType::POP, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true + ); + + self->SetNetworkVar(u"bIsInUse", false); + + GameMessages::SendTerminateInteraction(player->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} + +void NtVentureCannonServer::FirePlayer(Entity* self, Entity* player) { + auto destinationGroup = self->GetVar<std::u16string>(u"teleGroup"); + auto* destination = self; + + if (!destinationGroup.empty()) { + const auto& groupObjs = Game::entityManager->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(destinationGroup)); + + if (!groupObjs.empty()) { + destination = groupObjs[0]; + } + } + + const auto destPosition = destination->GetPosition(); + const auto destRotation = destination->GetRotation(); + + GameMessages::SendTeleport(player->GetObjectID(), destPosition, destRotation, player->GetSystemAddress(), true); + + RenderComponent::PlayAnimation(player, u"venture-cannon-out", 4.0f); +} diff --git a/dScripts/NtVentureCannonServer.h b/dScripts/02_server/Map/NT/NtVentureCannonServer.h similarity index 100% rename from dScripts/NtVentureCannonServer.h rename to dScripts/02_server/Map/NT/NtVentureCannonServer.h diff --git a/dScripts/02_server/Map/NT/NtVentureSpeedPadServer.cpp b/dScripts/02_server/Map/NT/NtVentureSpeedPadServer.cpp new file mode 100644 index 00000000..07d33555 --- /dev/null +++ b/dScripts/02_server/Map/NT/NtVentureSpeedPadServer.cpp @@ -0,0 +1,29 @@ +#include "NtVentureSpeedPadServer.h" +#include "SkillComponent.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" + +void NtVentureSpeedPadServer::OnStartup(Entity* self) { + self->SetProximityRadius(3, "speedboost"); +} + + +void NtVentureSpeedPadServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name != "speedboost" || !entering->IsPlayer() || status != "ENTER") { + return; + } + + auto* player = entering; + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + } + + auto* skillComponent = player->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(927, 18913, player->GetObjectID(), true); + } +} diff --git a/dScripts/NtVentureSpeedPadServer.h b/dScripts/02_server/Map/NT/NtVentureSpeedPadServer.h similarity index 100% rename from dScripts/NtVentureSpeedPadServer.h rename to dScripts/02_server/Map/NT/NtVentureSpeedPadServer.h diff --git a/dScripts/NtXRayServer.cpp b/dScripts/02_server/Map/NT/NtXRayServer.cpp similarity index 100% rename from dScripts/NtXRayServer.cpp rename to dScripts/02_server/Map/NT/NtXRayServer.cpp diff --git a/dScripts/NtXRayServer.h b/dScripts/02_server/Map/NT/NtXRayServer.h similarity index 100% rename from dScripts/NtXRayServer.h rename to dScripts/02_server/Map/NT/NtXRayServer.h diff --git a/dScripts/SpawnSaberCatServer.cpp b/dScripts/02_server/Map/NT/SpawnSaberCatServer.cpp similarity index 100% rename from dScripts/SpawnSaberCatServer.cpp rename to dScripts/02_server/Map/NT/SpawnSaberCatServer.cpp diff --git a/dScripts/SpawnSaberCatServer.h b/dScripts/02_server/Map/NT/SpawnSaberCatServer.h similarity index 100% rename from dScripts/SpawnSaberCatServer.h rename to dScripts/02_server/Map/NT/SpawnSaberCatServer.h diff --git a/dScripts/SpawnShrakeServer.cpp b/dScripts/02_server/Map/NT/SpawnShrakeServer.cpp similarity index 100% rename from dScripts/SpawnShrakeServer.cpp rename to dScripts/02_server/Map/NT/SpawnShrakeServer.cpp diff --git a/dScripts/SpawnShrakeServer.h b/dScripts/02_server/Map/NT/SpawnShrakeServer.h similarity index 100% rename from dScripts/SpawnShrakeServer.h rename to dScripts/02_server/Map/NT/SpawnShrakeServer.h diff --git a/dScripts/SpawnStegoServer.cpp b/dScripts/02_server/Map/NT/SpawnStegoServer.cpp similarity index 100% rename from dScripts/SpawnStegoServer.cpp rename to dScripts/02_server/Map/NT/SpawnStegoServer.cpp diff --git a/dScripts/SpawnStegoServer.h b/dScripts/02_server/Map/NT/SpawnStegoServer.h similarity index 100% rename from dScripts/SpawnStegoServer.h rename to dScripts/02_server/Map/NT/SpawnStegoServer.h diff --git a/dScripts/02_server/Map/PR/CMakeLists.txt b/dScripts/02_server/Map/PR/CMakeLists.txt new file mode 100644 index 00000000..3a32e9f3 --- /dev/null +++ b/dScripts/02_server/Map/PR/CMakeLists.txt @@ -0,0 +1,5 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_PR + "HydrantBroken.cpp" + "PrSeagullFly.cpp" + "SpawnGryphonServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/PR/HydrantBroken.cpp b/dScripts/02_server/Map/PR/HydrantBroken.cpp new file mode 100644 index 00000000..2694b425 --- /dev/null +++ b/dScripts/02_server/Map/PR/HydrantBroken.cpp @@ -0,0 +1,37 @@ +#include "HydrantBroken.h" +#include "EntityManager.h" +#include "GameMessages.h" + +void HydrantBroken::OnStartup(Entity* self) { + self->AddTimer("playEffect", 1); + + const auto hydrant = "hydrant" + self->GetVar<std::string>(u"hydrant"); + + const auto bouncers = Game::entityManager->GetEntitiesInGroup(hydrant); + + for (auto* bouncer : bouncers) { + self->SetVar<LWOOBJID>(u"bouncer", bouncer->GetObjectID()); + + GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); + + GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"enableCollision", UNASSIGNED_SYSTEM_ADDRESS); + } + + self->AddTimer("KillBroken", 25); +} + +void HydrantBroken::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "KillBroken") { + auto* bouncer = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"bouncer")); + + if (bouncer != nullptr) { + GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); + + GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"disableCollision", UNASSIGNED_SYSTEM_ADDRESS); + } + + self->Kill(); + } else if (timerName == "playEffect") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 384, u"water", "water", LWOOBJID_EMPTY, 1, 1, true); + } +} diff --git a/dScripts/HydrantBroken.h b/dScripts/02_server/Map/PR/HydrantBroken.h similarity index 100% rename from dScripts/HydrantBroken.h rename to dScripts/02_server/Map/PR/HydrantBroken.h diff --git a/dScripts/PrSeagullFly.cpp b/dScripts/02_server/Map/PR/PrSeagullFly.cpp similarity index 100% rename from dScripts/PrSeagullFly.cpp rename to dScripts/02_server/Map/PR/PrSeagullFly.cpp diff --git a/dScripts/PrSeagullFly.h b/dScripts/02_server/Map/PR/PrSeagullFly.h similarity index 100% rename from dScripts/PrSeagullFly.h rename to dScripts/02_server/Map/PR/PrSeagullFly.h diff --git a/dScripts/02_server/Map/PR/SpawnGryphonServer.cpp b/dScripts/02_server/Map/PR/SpawnGryphonServer.cpp new file mode 100644 index 00000000..cf635fe4 --- /dev/null +++ b/dScripts/02_server/Map/PR/SpawnGryphonServer.cpp @@ -0,0 +1,29 @@ +#include "SpawnGryphonServer.h" +#include "InventoryComponent.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eTerminateType.h" + +void SpawnGryphonServer::SetVariables(Entity* self) { + self->SetVar<LOT>(u"petLOT", 12433); + self->SetVar<std::string>(u"petType", "gryphon"); + self->SetVar<uint32_t>(u"maxPets", 2); + self->SetVar<std::u16string>(u"spawnAnim", u"spawn"); + self->SetVar<std::u16string>(u"spawnCinematic", u"SentinelPet"); +} + +void SpawnGryphonServer::OnUse(Entity* self, Entity* user) { + auto* missionComponent = user->GetComponent<MissionComponent>(); + auto* inventoryComponent = user->GetComponent<InventoryComponent>(); + + // Little extra for handling the case of the egg being placed the first time + if (missionComponent != nullptr && inventoryComponent != nullptr + && missionComponent->GetMissionState(1391) == eMissionState::ACTIVE) { + inventoryComponent->RemoveItem(12483, inventoryComponent->GetLotCount(12483)); + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); + return; + } + + SpawnPetBaseServer::OnUse(self, user); +} diff --git a/dScripts/SpawnGryphonServer.h b/dScripts/02_server/Map/PR/SpawnGryphonServer.h similarity index 100% rename from dScripts/SpawnGryphonServer.h rename to dScripts/02_server/Map/PR/SpawnGryphonServer.h diff --git a/dScripts/02_server/Map/Property/AG_Med/CMakeLists.txt b/dScripts/02_server/Map/Property/AG_Med/CMakeLists.txt new file mode 100644 index 00000000..90985f9f --- /dev/null +++ b/dScripts/02_server/Map/Property/AG_Med/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY_AG_MED + "ZoneAgMedProperty.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/Property/AG_Med/ZoneAgMedProperty.cpp b/dScripts/02_server/Map/Property/AG_Med/ZoneAgMedProperty.cpp new file mode 100644 index 00000000..46ec7bbd --- /dev/null +++ b/dScripts/02_server/Map/Property/AG_Med/ZoneAgMedProperty.cpp @@ -0,0 +1,42 @@ +#include "ZoneAgMedProperty.h" +#include "Entity.h" + +void ZoneAgMedProperty::SetGameVariables(Entity* self) { + + self->SetVar<std::string>(ClaimMarkerGroup, "ClaimMarker"); + self->SetVar<std::string>(GeneratorGroup, "Generator"); + self->SetVar<std::string>(GuardGroup, "Guard"); + self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); + self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); + self->SetVar<std::string>(SpotsGroup, "Spots"); + self->SetVar<std::string>(MSCloudsGroup, "maelstrom"); + self->SetVar<std::string>(EnemiesGroup, "Enemies"); + self->SetVar<std::string>(FXManagerGroup, "FXObject"); + self->SetVar<std::string>(ImagOrbGroup, "Orb"); + self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); + + self->SetVar<std::vector<std::string>>(EnemiesSpawner, { + "StrombieWander", "Strombies", "Mechs", "OtherEnemy" + }); + self->SetVar<std::string>(ClaimMarkerSpawner, "ClaimMarker"); + self->SetVar<std::string>(GeneratorSpawner, "Generator"); + self->SetVar<std::string>(DamageFXSpawner, "MaelstromFX"); + self->SetVar<std::string>(FXSpotsSpawner, "MaelstromSpots"); + self->SetVar<std::string>(PropertyMGSpawner, "PropertyGuard"); + self->SetVar<std::string>(ImageOrbSpawner, "Orb"); + self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); + self->SetVar<std::string>(SmashablesSpawner, "Smashables"); + self->SetVar<std::string>(FXManagerSpawner, "FXObject"); + self->SetVar<std::string>(PropObjsSpawner, "BankObj"); + self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "BirdFX", "SunBeam" }); + self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, {}); + + self->SetVar<int32_t>(defeatedProperyFlag, 118); + self->SetVar<int32_t>(placedModelFlag, 119); + self->SetVar<uint32_t>(guardMissionFlag, 1293); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 1294); + self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); + self->SetVar<LOT>(generatorIdFlag, 10118); + self->SetVar<LOT>(orbIDFlag, 10226); + self->SetVar<LOT>(behaviorQBID, 10445); +} diff --git a/dScripts/ZoneAgMedProperty.h b/dScripts/02_server/Map/Property/AG_Med/ZoneAgMedProperty.h similarity index 100% rename from dScripts/ZoneAgMedProperty.h rename to dScripts/02_server/Map/Property/AG_Med/ZoneAgMedProperty.h diff --git a/dScripts/02_server/Map/Property/AG_Small/CMakeLists.txt b/dScripts/02_server/Map/Property/AG_Small/CMakeLists.txt new file mode 100644 index 00000000..3d71dc1a --- /dev/null +++ b/dScripts/02_server/Map/Property/AG_Small/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY_AG_SMALL + "ZoneAgProperty.cpp" + "EnemySpiderSpawner.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/Property/AG_Small/EnemySpiderSpawner.cpp b/dScripts/02_server/Map/Property/AG_Small/EnemySpiderSpawner.cpp new file mode 100644 index 00000000..2d526a25 --- /dev/null +++ b/dScripts/02_server/Map/Property/AG_Small/EnemySpiderSpawner.cpp @@ -0,0 +1,68 @@ +#include "EnemySpiderSpawner.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "DestroyableComponent.h" +#include "eReplicaComponentType.h" + +//---------------------------------------------- +//--Initiate egg hatching on call +//---------------------------------------------- +void EnemySpiderSpawner::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + if (args == "prepEgg") { + // Highlight eggs about to hatch with Maelstrom effect + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2856, u"maelstrom", "test", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + + // Make indestructible + auto dest = static_cast<DestroyableComponent*>(self->GetComponent(eReplicaComponentType::DESTROYABLE)); + if (dest) { + dest->SetFaction(-1); + } + Game::entityManager->SerializeEntity(self); + + // Keep track of who prepped me + self->SetI64(u"SpawnOwner", sender->GetObjectID()); + + } else if (args == "hatchEgg") { + // Final countdown to pop + self->AddTimer("StartSpawnTime", hatchTime); + } +} + +//---------------------------------------------------------------- +//--Called when timers are done +//---------------------------------------------------------------- +void EnemySpiderSpawner::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "StartSpawnTime") { + SpawnSpiderling(self); + } else if (timerName == "SpawnSpiderling") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 644, u"create", "egg_puff_b", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + + //TODO: set the aggro radius larger + + EntityInfo info{}; + info.lot = 16197; + info.pos = self->GetPosition(); + info.spawner = nullptr; + info.spawnerID = self->GetI64(u"SpawnOwner"); + info.spawnerNodeID = 0; + + Entity* newEntity = Game::entityManager->CreateEntity(info, nullptr); + if (newEntity) { + Game::entityManager->ConstructEntity(newEntity); + newEntity->GetGroups().push_back("BabySpider"); + } + + self->ScheduleKillAfterUpdate(); + } +} + +//-------------------------------------------------------------- +//Called when it is finally time to release the Spiderlings +//-------------------------------------------------------------- +void EnemySpiderSpawner::SpawnSpiderling(Entity* self) { + //Initiate the actual spawning + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2260, u"rebuild_medium", "dropdustmedium", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + self->AddTimer("SpawnSpiderling", spawnTime); +} diff --git a/dScripts/EnemySpiderSpawner.h b/dScripts/02_server/Map/Property/AG_Small/EnemySpiderSpawner.h similarity index 100% rename from dScripts/EnemySpiderSpawner.h rename to dScripts/02_server/Map/Property/AG_Small/EnemySpiderSpawner.h diff --git a/dScripts/02_server/Map/Property/AG_Small/ZoneAgProperty.cpp b/dScripts/02_server/Map/Property/AG_Small/ZoneAgProperty.cpp new file mode 100644 index 00000000..6f2f6d36 --- /dev/null +++ b/dScripts/02_server/Map/Property/AG_Small/ZoneAgProperty.cpp @@ -0,0 +1,426 @@ +#include "ZoneAgProperty.h" +#include "EntityManager.h" +#include "Character.h" +#include "Entity.h" +#include "GameMessages.h" +#include "dZoneManager.h" +#include "RenderComponent.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" + +void ZoneAgProperty::SetGameVariables(Entity* self) { + self->SetVar<std::string>(GuardGroup, "Guard"); + self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); + self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); + self->SetVar<std::string>(PropertyBorderGroup, "PropertyBorder"); + self->SetVar<std::string>(LandTargetGroup, "Land_Target"); + self->SetVar<std::string>(SpiderScreamGroup, "Spider_Scream"); + self->SetVar<std::vector<std::string>>(ROFTargetsGroup, { "ROF_Targets_00", "ROF_Targets_01", "ROF_Targets_02", "ROF_Targets_03", "ROF_Targets_04" }); + self->SetVar<std::string>(SpiderEggsGroup, "SpiderEggs"); + self->SetVar<std::string>(RocksGroup, "Rocks"); + self->SetVar<std::string>(EnemiesGroup, "SpiderBoss"); + self->SetVar<std::vector<std::string>>(ZoneVolumesGroup, { "Zone1Vol", "Zone2Vol", "Zone3Vol", "Zone4Vol", "Zone5Vol", "Zone6Vol", "Zone7Vol", "Zone8Vol", "AggroVol", "TeleVol" }); + self->SetVar<std::string>(FXManagerGroup, "FXObject"); + + self->SetVar<std::string>(EnemiesSpawner, "SpiderBoss"); + self->SetVar<std::vector<std::string>>(BossSensorSpawner, { "Zone1Vol", "Zone2Vol", "Zone3Vol", "Zone4Vol", "Zone5Vol", "Zone6Vol", "Zone7Vol", "Zone8Vol", "RFS_Targets", "AggroVol", "TeleVol" }); + self->SetVar<std::string>(LandTargetSpawner, "Land_Target"); + self->SetVar<std::string>(SpiderScreamSpawner, "Spider_Scream"); + self->SetVar<std::vector<std::string>>(ROFTargetsSpawner, { "ROF_Targets_00", "ROF_Targets_01", "ROF_Targets_02", "ROF_Targets_03", "ROF_Targets_04" }); + self->SetVar<std::string>(PropertyMGSpawner, "PropertyGuard"); + self->SetVar<std::string>(FXManagerSpawner, "FXObject"); + self->SetVar<std::string>(PropObjsSpawner, "BankObj"); + self->SetVar<std::string>(SpiderEggsSpawner, "SpiderEggs"); + self->SetVar<std::string>(RocksSpawner, "Rocks"); + self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "BirdFX", "SunBeam" }); + self->SetVar<std::vector<std::string>>(SpiderRocketSpawner, { "SpiderRocket_Bot", "SpiderRocket_Mid", "SpiderRocket_Top" }); + self->SetVar<std::string>(MailboxSpawner, "Mailbox"); + self->SetVar<std::string>(LauncherSpawner, "Launcher"); + self->SetVar<std::string>(InstancerSpawner, "Instancer"); + + self->SetVar<int32_t>(defeatedProperyFlag, 71); + self->SetVar<int32_t>(placedModelFlag, 73); + self->SetVar<uint32_t>(guardFirstMissionFlag, 891); + self->SetVar<uint32_t>(guardMissionFlag, 320); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 951); +} + +void ZoneAgProperty::OnStartup(Entity* self) { + LoadProperty(self); +} + +void ZoneAgProperty::OnPlayerLoaded(Entity* self, Entity* player) { + CheckForOwner(self); + + auto rented = self->GetVar<LWOOBJID>(u"PropertyOwner") == LWOOBJID_EMPTY; + self->SetVar<bool>(u"rented", rented); + + if (!rented) { + const auto numberOfPlayers = self->GetVar<int32_t>(u"numberOfPlayers"); + self->SetVar<int32_t>(u"numberOfPlayers", numberOfPlayers + 1); + } + + if (Game::zoneManager->GetZone()->GetZoneID().GetMapID() == 1102) { + GameMessages::SendPlay2DAmbientSound(player, GUIDMaelstrom); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"maelstromSkyOn", 0, 0, + LWOOBJID_EMPTY, "", player->GetSystemAddress()); + + self->SetNetworkVar(u"unclaimed", true); + + return; + } + + BasePlayerLoaded(self, player); +} + +void ZoneAgProperty::PropGuardCheck(Entity* self, Entity* player) { + auto* missionComponent = player->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) + return; + + const auto state = missionComponent->GetMissionState(self->GetVar<uint32_t>(guardMissionFlag)); + const auto firstState = missionComponent->GetMissionState(self->GetVar<uint32_t>(guardFirstMissionFlag)); + + if (firstState < eMissionState::COMPLETE || (state != eMissionState::COMPLETE && state != eMissionState::COMPLETE_READY_TO_COMPLETE)) + ActivateSpawner(self->GetVar<std::string>(PropertyMGSpawner)); +} + +void ZoneAgProperty::OnZoneLoadedInfo(Entity* self) { + LoadProperty(self); +} + +void ZoneAgProperty::LoadInstance(Entity* self) { + SetGameVariables(self); + + for (auto* spawner : Game::zoneManager->GetSpawnersByName(self->GetVar<std::string>(InstancerSpawner))) { + for (auto* spawnerNode : spawner->m_Info.nodes) { + spawnerNode->config.push_back( + new LDFData<std::string>(u"custom_script_server", + R"(scripts\ai\GENERAL\L_INSTANCE_EXIT_TRANSFER_PLAYER_TO_LAST_NON_INSTANCE.lua)")); + spawnerNode->config.push_back(new LDFData<std::u16string>(u"transferText", u"SPIDER_QUEEN_EXIT_QUESTION")); + } + } + + ActivateSpawner(self->GetVar<std::string>(InstancerSpawner)); +} + +void ZoneAgProperty::LoadProperty(Entity* self) { + SetGameVariables(self); + ActivateSpawner(self->GetVar<std::string>(LauncherSpawner)); + ActivateSpawner(self->GetVar<std::string>(MailboxSpawner)); +} + +void ZoneAgProperty::ProcessGroupObjects(Entity* self, std::string group) { +} + +void ZoneAgProperty::SpawnSpots(Entity* self) { + for (const auto& spot : self->GetVar<std::vector<std::string>>(ROFTargetsSpawner)) { + ActivateSpawner(spot); + } + + ActivateSpawner(self->GetVar<std::string>(LandTargetSpawner)); +} + +void ZoneAgProperty::KillSpots(Entity* self) { + for (const auto& spot : self->GetVar<std::vector<std::string>>(ROFTargetsSpawner)) { + DeactivateSpawner(spot); + } + + for (const auto& groupName : self->GetVar<std::vector<std::string>>(ROFTargetsGroup)) { + for (auto* spot : Game::entityManager->GetEntitiesInGroup(groupName)) { + spot->Kill(); + } + } + + DeactivateSpawner(self->GetVar<std::string>(LandTargetSpawner)); + for (auto* landTarget : Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(LandTargetSpawner))) { + landTarget->Kill(); + } +} + +void ZoneAgProperty::SpawnCrashedRocket(Entity* self) { + for (const auto& rocket : self->GetVar<std::vector<std::string>>(SpiderRocketSpawner)) { + ActivateSpawner(rocket); + } +} + +void ZoneAgProperty::KillCrashedRocket(Entity* self) { + for (const auto& rocket : self->GetVar<std::vector<std::string>>(SpiderRocketSpawner)) { + DeactivateSpawner(rocket); + DestroySpawner(rocket); + } +} + +void ZoneAgProperty::StartMaelstrom(Entity* self, Entity* player) { + ActivateSpawner(self->GetVar<std::string>(EnemiesSpawner)); + for (const auto& sensor : self->GetVar<std::vector<std::string>>(BossSensorSpawner)) { + ActivateSpawner(sensor); + } + + ActivateSpawner(self->GetVar<std::string>(FXManagerSpawner)); + ActivateSpawner(self->GetVar<std::string>(SpiderScreamSpawner)); + ActivateSpawner(self->GetVar<std::string>(SpiderEggsSpawner)); + ActivateSpawner(self->GetVar<std::string>(RocksSpawner)); + + SpawnCrashedRocket(self); + + for (const auto& ambient : self->GetVar<std::vector<std::string>>(AmbientFXSpawner)) { + DeactivateSpawner(ambient); + DestroySpawner(ambient); + ResetSpawner(ambient); + } + + StartTornadoFx(self); + + if (player != nullptr) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"maelstromSkyOn", 0, 0, LWOOBJID_EMPTY, + "", player->GetSystemAddress()); + } +} + +uint32_t ZoneAgProperty::RetrieveSpawnerId(Entity* self, const std::string& spawner) { + auto spawnerIDs = Game::zoneManager->GetSpawnersByName(spawner); + if (spawnerIDs.empty()) + return 0; + + return spawnerIDs[0]->m_Info.spawnerID; +} + +void ZoneAgProperty::OnTimerDone(Entity* self, std::string timerName) { + BaseTimerDone(self, timerName); +} + +void ZoneAgProperty::BaseTimerDone(Entity* self, const std::string& timerName) { + if (timerName == "GuardFlyAway") { + const auto zoneId = Game::zoneManager->GetZone()->GetWorldID(); + if (zoneId != 1150) + return; + + const auto entities = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(GuardGroup)); + if (entities.empty()) + return; + + auto* entity = entities[0]; + + GameMessages::SendNotifyClientObject(Game::entityManager->GetZoneControlEntity()->GetObjectID(), u"GuardChat", 0, 0, entity->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); + LoadProperty(self); + + self->AddTimer("KillGuard", 5); + } else if (timerName == "KillGuard") { + KillGuard(self); + } else if (timerName == "tornadoOff") { + for (auto* entity : Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup))) { + auto* renderComponent = entity->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + renderComponent->StopEffect("TornadoDebris", false); + renderComponent->StopEffect("TornadoVortex", false); + renderComponent->StopEffect("silhouette", false); + } + } + + self->AddTimer("ShowVendor", 1.2f); + self->AddTimer("ShowClearEffects", 2); + } else if (timerName == "ShowClearEffects") { + for (auto* entity : Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup))) { + auto* renderComponent = entity->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + renderComponent->PlayEffect(-1, u"beamOn", "beam"); + } + } + + self->AddTimer("killSpider", 2); + self->AddTimer("turnSkyOff", 1.5f); + self->AddTimer("killFXObject", 8); + } else if (timerName == "turnSkyOff") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SkyOff", 0, 0, LWOOBJID_EMPTY, + "", UNASSIGNED_SYSTEM_ADDRESS); + } else if (timerName == "killSpider") { + for (auto* entity : Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(EnemiesGroup))) { + entity->Kill(); + } + + for (const auto& sensor : self->GetVar<std::vector<std::string>>(BossSensorSpawner)) { + DeactivateSpawner(sensor); + DestroySpawner(sensor); + } + + DeactivateSpawner(self->GetVar<std::string>(SpiderEggsSpawner)); + DestroySpawner(self->GetVar<std::string>(SpiderEggsSpawner)); + + DeactivateSpawner(self->GetVar<std::string>(RocksSpawner)); + DestroySpawner(self->GetVar<std::string>(RocksSpawner)); + + KillSpots(self); + KillCrashedRocket(self); + + DeactivateSpawner(self->GetVar<std::string>(SpiderScreamSpawner)); + DestroySpawner(self->GetVar<std::string>(SpiderScreamSpawner)); + + for (auto* player : Game::entityManager->GetEntitiesByComponent(eReplicaComponentType::CHARACTER)) { + GameMessages::SendStop2DAmbientSound(player, true, GUIDMaelstrom); + GameMessages::SendPlay2DAmbientSound(player, GUIDPeaceful); + } + } else if (timerName == "ShowVendor") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"vendorOn", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + for (const auto& ambient : self->GetVar<std::vector<std::string>>(AmbientFXSpawner)) { + ActivateSpawner(ambient); + } + } else if (timerName == "BoundsVisOn") { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"boundsAnim", 0, 0, + LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + } else if (timerName == "runPlayerLoadedAgain") { + CheckForOwner(self); + } else if (timerName == "pollTornadoFX") { + StartTornadoFx(self); + } else if (timerName == "killFXObject") { + for (auto* entity : Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup))) { + auto* renderComponent = entity->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + renderComponent->StopEffect("beam"); + } + } + + DestroySpawner(self->GetVar<std::string>(FXManagerSpawner)); + + self->SetVar<bool>(u"FXObjectGone", true); + } else if (timerName == "ProcessGroupObj") { + // TODO + } +} + +void ZoneAgProperty::OnZonePropertyRented(Entity* self, Entity* player) { + BaseZonePropertyRented(self, player); + + auto* character = player->GetCharacter(); + if (character == nullptr) + return; + + character->SetPlayerFlag(108, true); +} + +void ZoneAgProperty::OnZonePropertyModelPlaced(Entity* self, Entity* player) { + auto* character = player->GetCharacter(); + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (!character->GetPlayerFlag(101)) { + BaseZonePropertyModelPlaced(self, player); + character->SetPlayerFlag(101, true); + if (missionComponent->GetMissionState(871) == eMissionState::ACTIVE) { + self->SetNetworkVar<std::u16string>(u"Tooltip", u"AnotherModel"); + } + + } else if (!character->GetPlayerFlag(102)) { + character->SetPlayerFlag(102, true); + if (missionComponent->GetMissionState(871) == eMissionState::ACTIVE) { + self->SetNetworkVar<std::u16string>(u"Tooltip", u"TwoMoreModels"); + } + + } else if (!character->GetPlayerFlag(103)) { + character->SetPlayerFlag(103, true); + } else if (!character->GetPlayerFlag(104)) { + character->SetPlayerFlag(104, true); + self->SetNetworkVar<std::u16string>(u"Tooltip", u"TwoMoreModelsOff"); + } else if (self->GetVar<std::string>(u"tutorial") == "place_model") { + self->SetVar<std::string>(u"tutorial", ""); + self->SetNetworkVar<std::u16string>(u"Tooltip", u"PutAway"); + } +} + +void ZoneAgProperty::OnZonePropertyModelPickedUp(Entity* self, Entity* player) { + auto* character = player->GetCharacter(); + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (!character->GetPlayerFlag(109)) { + character->SetPlayerFlag(109, true); + if (missionComponent->GetMissionState(891) == eMissionState::ACTIVE && !character->GetPlayerFlag(110)) { + self->SetNetworkVar<std::u16string>(u"Tooltip", u"Rotate"); + } + } +} + +void ZoneAgProperty::OnZonePropertyModelRemoved(Entity* self, Entity* player) { + auto* character = player->GetCharacter(); + character->SetPlayerFlag(111, true); +} + +void ZoneAgProperty::OnZonePropertyModelRemovedWhileEquipped(Entity* self, Entity* player) { + ZoneAgProperty::OnZonePropertyModelRemoved(self, player); +} + +void ZoneAgProperty::OnZonePropertyModelRotated(Entity* self, Entity* player) { + auto* character = player->GetCharacter(); + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (!character->GetPlayerFlag(110)) { + character->SetPlayerFlag(110, true); + + if (missionComponent->GetMissionState(891) == eMissionState::ACTIVE) { + self->SetNetworkVar<std::u16string>(u"Tooltip", u"PlaceModel"); + self->SetVar<std::string>(u"tutorial", "place_model"); + } + } +} + +void ZoneAgProperty::OnZonePropertyModelEquipped(Entity* self) { + self->SetNetworkVar<std::u16string>(u"PlayerAction", u"ModelEquipped"); +} + +void ZoneAgProperty::OnZonePropertyEditBegin(Entity* self) { + self->SetNetworkVar<std::u16string>(u"PlayerAction", u"Enter"); +} + +void ZoneAgProperty::OnZonePropertyEditEnd(Entity* self) { + self->SetNetworkVar<std::u16string>(u"PlayerAction", u"Exit"); +} + +void ZoneAgProperty::OnPlayerExit(Entity* self) { + // TODO: Destroy stuff +} + +void ZoneAgProperty::RemovePlayerRef(Entity* self) { + // TODO: Destroy stuff +} + +void ZoneAgProperty::BaseOnFireEventServerSide(Entity* self, Entity* sender, std::string args) { + if (args == "propertyRented") { + const auto playerId = self->GetVar<LWOOBJID>(u"playerID"); + auto* player = Game::entityManager->GetEntity(playerId); + if (player == nullptr) + return; + + OnZonePropertyRented(self, player); + } else if (args == "RetrieveZoneData") { + self->SetVar<LWOOBJID>(u"SpiderBossID", sender->GetObjectID()); + sender->SetVar<int32_t>(u"SpiderEggNetworkID", RetrieveSpawnerId(self, self->GetVar<std::string>(SpiderEggsSpawner))); + + std::vector<uint32_t> table; + + for (const auto& target : self->GetVar<std::vector<std::string>>(ROFTargetsSpawner)) { + table.push_back(RetrieveSpawnerId(self, target)); + } + + ROFTargetGroupIdTable = table; + + ProcessGroupObjects(self, self->GetVar<std::string>(LandTargetGroup)); + ProcessGroupObjects(self, self->GetVar<std::string>(SpiderScreamGroup)); + // ProcessGroupObjects(self, groups.ZoneVolumes); + } else if (args == "CheckForPropertyOwner") { + sender->SetNetworkVar<std::string>(u"PropertyOwnerID", std::to_string(self->GetVar<LWOOBJID>(u"PropertyOwner"))); + } else if (args == "ClearProperty") { + const auto playerId = self->GetVar<LWOOBJID>(u"playerID"); + auto* player = Game::entityManager->GetEntity(playerId); + if (player == nullptr) + return; + + player->GetCharacter()->SetPlayerFlag(self->GetVar<int32_t>(defeatedProperyFlag), true); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayCinematic", 0, 0, + LWOOBJID_EMPTY, destroyedCinematic, UNASSIGNED_SYSTEM_ADDRESS); + + self->AddTimer("tornadoOff", 0.5f); + } +} + +void ZoneAgProperty::NotifyDie(Entity* self) { + // TODO +} diff --git a/dScripts/ZoneAgProperty.h b/dScripts/02_server/Map/Property/AG_Small/ZoneAgProperty.h similarity index 100% rename from dScripts/ZoneAgProperty.h rename to dScripts/02_server/Map/Property/AG_Small/ZoneAgProperty.h diff --git a/dScripts/02_server/Map/Property/CMakeLists.txt b/dScripts/02_server/Map/Property/CMakeLists.txt new file mode 100644 index 00000000..74badb32 --- /dev/null +++ b/dScripts/02_server/Map/Property/CMakeLists.txt @@ -0,0 +1,22 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY + "PropertyBankInteract.cpp") + +add_subdirectory(AG_Med) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY_AG_MED}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY} "AG_Med/${file}") +endforeach() + +add_subdirectory(AG_Small) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY_AG_SMALL}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY} "AG_Small/${file}") +endforeach() + +add_subdirectory(NS_Med) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY_NS_MED}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY} "NS_Med/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY ${DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY} PARENT_SCOPE) diff --git a/dScripts/02_server/Map/Property/NS_Med/CMakeLists.txt b/dScripts/02_server/Map/Property/NS_Med/CMakeLists.txt new file mode 100644 index 00000000..230bc2cf --- /dev/null +++ b/dScripts/02_server/Map/Property/NS_Med/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_PROPERTY_NS_MED + "ZoneNsMedProperty.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/Property/NS_Med/ZoneNsMedProperty.cpp b/dScripts/02_server/Map/Property/NS_Med/ZoneNsMedProperty.cpp new file mode 100644 index 00000000..55bde71c --- /dev/null +++ b/dScripts/02_server/Map/Property/NS_Med/ZoneNsMedProperty.cpp @@ -0,0 +1,40 @@ +#include "ZoneNsMedProperty.h" +#include "Entity.h" + +void ZoneNsMedProperty::SetGameVariables(Entity* self) { + self->SetVar<std::string>(ClaimMarkerGroup, "ClaimMarker"); + self->SetVar<std::string>(GeneratorGroup, "Generator"); + self->SetVar<std::string>(GuardGroup, "Guard"); + self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); + self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); + self->SetVar<std::string>(SpotsGroup, "Spots"); + self->SetVar<std::string>(MSCloudsGroup, "maelstrom"); + self->SetVar<std::string>(EnemiesGroup, "Enemies"); + self->SetVar<std::string>(FXManagerGroup, "FXObject"); + self->SetVar<std::string>(ImagOrbGroup, "Orb"); + self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); + + self->SetVar<std::vector<std::string>>(EnemiesSpawner, + { "Admirals", "AdmiralsWander", "Mechs", "Ronin", "RoninWander" }); + self->SetVar<std::string>(ClaimMarkerSpawner, "ClaimMarker"); + self->SetVar<std::string>(GeneratorSpawner, "Generator"); + self->SetVar<std::string>(DamageFXSpawner, "MaelstromFX"); + self->SetVar<std::string>(FXSpotsSpawner, "MaelstromSpots"); + self->SetVar<std::string>(PropertyMGSpawner, "PropertyGuard"); + self->SetVar<std::string>(ImageOrbSpawner, "Orb"); + self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); + self->SetVar<std::string>(SmashablesSpawner, "Smashables"); + self->SetVar<std::string>(FXManagerSpawner, "FXObject"); + self->SetVar<std::string>(PropObjsSpawner, "BankObj"); + self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Rockets" }); + self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, { }); + + self->SetVar<int32_t>(defeatedProperyFlag, 122); + self->SetVar<int32_t>(placedModelFlag, 123); + self->SetVar<uint32_t>(guardMissionFlag, 1322); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 1294); + self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); + self->SetVar<LOT>(generatorIdFlag, 11031); + self->SetVar<LOT>(orbIDFlag, 10226); + self->SetVar<LOT>(behaviorQBID, 10445); +} diff --git a/dScripts/ZoneNsMedProperty.h b/dScripts/02_server/Map/Property/NS_Med/ZoneNsMedProperty.h similarity index 100% rename from dScripts/ZoneNsMedProperty.h rename to dScripts/02_server/Map/Property/NS_Med/ZoneNsMedProperty.h diff --git a/dScripts/02_server/Map/Property/PropertyBankInteract.cpp b/dScripts/02_server/Map/Property/PropertyBankInteract.cpp new file mode 100644 index 00000000..50fea9a0 --- /dev/null +++ b/dScripts/02_server/Map/Property/PropertyBankInteract.cpp @@ -0,0 +1,45 @@ +#include "PropertyBankInteract.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "Amf3.h" +#include "Entity.h" + +void PropertyBankInteract::OnStartup(Entity* self) { + auto* zoneControl = Game::entityManager->GetZoneControlEntity(); + if (zoneControl != nullptr) { + zoneControl->OnFireEventServerSide(self, "CheckForPropertyOwner"); + } +} + +void PropertyBankInteract::OnPlayerLoaded(Entity* self, Entity* player) { + auto* zoneControl = Game::entityManager->GetZoneControlEntity(); + if (zoneControl != nullptr) { + zoneControl->OnFireEventServerSide(self, "CheckForPropertyOwner"); + } +} + +void PropertyBankInteract::OnUse(Entity* self, Entity* user) { + + AMFArrayValue args; + + args.Insert("state", "bank"); + + GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", args); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"OpenBank", 0, 0, LWOOBJID_EMPTY, + "", user->GetSystemAddress()); +} + +void PropertyBankInteract::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + if (args == "ToggleBank") { + AMFArrayValue amfArgs; + + amfArgs.Insert("visible", false); + + GameMessages::SendUIMessageServerToSingleClient(sender, sender->GetSystemAddress(), "ToggleBank", amfArgs); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"CloseBank", 0, 0, LWOOBJID_EMPTY, + "", sender->GetSystemAddress()); + } +} diff --git a/dScripts/PropertyBankInteract.h b/dScripts/02_server/Map/Property/PropertyBankInteract.h similarity index 100% rename from dScripts/PropertyBankInteract.h rename to dScripts/02_server/Map/Property/PropertyBankInteract.h diff --git a/dScripts/02_server/Map/SS/CMakeLists.txt b/dScripts/02_server/Map/SS/CMakeLists.txt new file mode 100644 index 00000000..49db031f --- /dev/null +++ b/dScripts/02_server/Map/SS/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_SS + "SsModularBuildServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/SS/SsModularBuildServer.cpp b/dScripts/02_server/Map/SS/SsModularBuildServer.cpp new file mode 100644 index 00000000..dfb29168 --- /dev/null +++ b/dScripts/02_server/Map/SS/SsModularBuildServer.cpp @@ -0,0 +1,17 @@ +#include "SsModularBuildServer.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" + +void SsModularBuildServer::OnModularBuildExit(Entity* self, Entity* player, bool bCompleted, std::vector<LOT> modules) { + int missionNum = 1732; + + if (bCompleted) { + MissionComponent* mission = static_cast<MissionComponent*>(player->GetComponent(eReplicaComponentType::MISSION)); + Mission* rocketMission = mission->GetMission(missionNum); + + if (rocketMission->GetMissionState() == eMissionState::ACTIVE) { + mission->ForceProgress(missionNum, 2478, 1); + } + } +} diff --git a/dScripts/SsModularBuildServer.h b/dScripts/02_server/Map/SS/SsModularBuildServer.h similarity index 100% rename from dScripts/SsModularBuildServer.h rename to dScripts/02_server/Map/SS/SsModularBuildServer.h diff --git a/dScripts/02_server/Map/VE/CMakeLists.txt b/dScripts/02_server/Map/VE/CMakeLists.txt new file mode 100644 index 00000000..ac3a6461 --- /dev/null +++ b/dScripts/02_server/Map/VE/CMakeLists.txt @@ -0,0 +1,5 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_VE + "VeMissionConsole.cpp" + "VeEpsilonServer.cpp" + "VeBricksampleServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/VE/VeBricksampleServer.cpp b/dScripts/02_server/Map/VE/VeBricksampleServer.cpp new file mode 100644 index 00000000..7535d261 --- /dev/null +++ b/dScripts/02_server/Map/VE/VeBricksampleServer.cpp @@ -0,0 +1,22 @@ +#include "VeBricksampleServer.h" +#include "InventoryComponent.h" +#include "EntityManager.h" +#include "MissionComponent.h" +#include "GameMessages.h" +#include "eMissionState.h" + +void VeBricksampleServer::OnUse(Entity* self, Entity* user) { + auto* missionComponent = user->GetComponent<MissionComponent>(); + if (missionComponent != nullptr && missionComponent->GetMissionState(1183) == eMissionState::ACTIVE) { + const auto loot = self->GetVar<int32_t>(m_LootVariable); + auto* inventoryComponent = user->GetComponent<InventoryComponent>(); + + if (loot && inventoryComponent != nullptr && inventoryComponent->GetLotCount(loot) == 0) { + inventoryComponent->AddItem(loot, 1, eLootSourceType::ACTIVITY); + + for (auto* brickEntity : Game::entityManager->GetEntitiesInGroup("Bricks")) { + GameMessages::SendNotifyClientObject(brickEntity->GetObjectID(), u"Pickedup"); + } + } + } +} diff --git a/dScripts/VeBricksampleServer.h b/dScripts/02_server/Map/VE/VeBricksampleServer.h similarity index 100% rename from dScripts/VeBricksampleServer.h rename to dScripts/02_server/Map/VE/VeBricksampleServer.h diff --git a/dScripts/02_server/Map/VE/VeEpsilonServer.cpp b/dScripts/02_server/Map/VE/VeEpsilonServer.cpp new file mode 100644 index 00000000..4e8cb4e9 --- /dev/null +++ b/dScripts/02_server/Map/VE/VeEpsilonServer.cpp @@ -0,0 +1,28 @@ +#include "VeEpsilonServer.h" +#include "Character.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "eMissionState.h" +#include "Entity.h" + +void VeEpsilonServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + auto* character = target->GetCharacter(); + if (character == nullptr) + return; + + // Resets the player flags that track which consoles they've used + if ((missionID == m_ConsoleMissionID || missionID == m_ConsoleRepeatMissionID) + && (missionState == eMissionState::AVAILABLE || missionState == eMissionState::COMPLETE_AVAILABLE)) { + + for (auto i = 0; i < 10; i++) { + character->SetPlayerFlag(m_ConsoleBaseFlag + i, false); + } + } + + // Notify the client that all objects have updated + self->AddCallbackTimer(3.0f, [this]() { + for (const auto* console : Game::entityManager->GetEntitiesInGroup(m_ConsoleGroup)) { + GameMessages::SendNotifyClientObject(console->GetObjectID(), u""); + } + }); +} diff --git a/dScripts/02_server/Map/VE/VeEpsilonServer.h b/dScripts/02_server/Map/VE/VeEpsilonServer.h new file mode 100644 index 00000000..d5a53f9e --- /dev/null +++ b/dScripts/02_server/Map/VE/VeEpsilonServer.h @@ -0,0 +1,10 @@ +#pragma once +#include "CppScripts.h" + +class VeEpsilonServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + const uint32_t m_ConsoleMissionID = 1220; + const uint32_t m_ConsoleRepeatMissionID = 1225; + const int32_t m_ConsoleBaseFlag = 1010; + const std::string m_ConsoleGroup = "Consoles"; +}; diff --git a/dScripts/02_server/Map/VE/VeMissionConsole.cpp b/dScripts/02_server/Map/VE/VeMissionConsole.cpp new file mode 100644 index 00000000..35061bdf --- /dev/null +++ b/dScripts/02_server/Map/VE/VeMissionConsole.cpp @@ -0,0 +1,27 @@ +#include "VeMissionConsole.h" +#include "InventoryComponent.h" +#include "Character.h" +#include "GameMessages.h" +#include "Loot.h" +#include "eTerminateType.h" + +void VeMissionConsole::OnUse(Entity* self, Entity* user) { + LootGenerator::Instance().DropActivityLoot(user, self, 12551); + + auto* inventoryComponent = user->GetComponent<InventoryComponent>(); + if (inventoryComponent != nullptr) { + inventoryComponent->AddItem(12547, 1, eLootSourceType::ACTIVITY); // Add the panel required for pickup + } + + // The flag to set is 101<number> + const auto flagNumber = self->GetVar<std::u16string>(m_NumberVariable); + const int32_t flag = std::stoi("101" + GeneralUtils::UTF16ToWTF8(flagNumber)); + + auto* character = user->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(flag, true); + } + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u""); + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} diff --git a/dScripts/VeMissionConsole.h b/dScripts/02_server/Map/VE/VeMissionConsole.h similarity index 100% rename from dScripts/VeMissionConsole.h rename to dScripts/02_server/Map/VE/VeMissionConsole.h diff --git a/dScripts/BurningTile.cpp b/dScripts/02_server/Map/njhub/BurningTile.cpp similarity index 100% rename from dScripts/BurningTile.cpp rename to dScripts/02_server/Map/njhub/BurningTile.cpp diff --git a/dScripts/BurningTile.h b/dScripts/02_server/Map/njhub/BurningTile.h similarity index 100% rename from dScripts/BurningTile.h rename to dScripts/02_server/Map/njhub/BurningTile.h diff --git a/dScripts/02_server/Map/njhub/CMakeLists.txt b/dScripts/02_server/Map/njhub/CMakeLists.txt new file mode 100644 index 00000000..2527d1ad --- /dev/null +++ b/dScripts/02_server/Map/njhub/CMakeLists.txt @@ -0,0 +1,31 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB + "BurningTile.cpp" + "CatapultBaseServer.cpp" + "CatapultBouncerServer.cpp" + "CavePrisonCage.cpp" + "EnemySkeletonSpawner.cpp" + "FallingTile.cpp" + "FlameJetServer.cpp" + "ImaginationShrineServer.cpp" + "Lieutenant.cpp" + "MonCoreNookDoors.cpp" + "MonCoreSmashableDoors.cpp" + "NjColeNPC.cpp" + "NjDragonEmblemChestServer.cpp" + "NjEarthDragonPetServer.cpp" + "NjEarthPetServer.cpp" + "NjGarmadonCelebration.cpp" + "NjJayMissionItems.cpp" + "NjNPCMissionSpinjitzuServer.cpp" + "NjNyaMissionitems.cpp" + "NjScrollChestServer.cpp" + "NjWuNPC.cpp" + "RainOfArrows.cpp") + +add_subdirectory(boss_instance) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB_BOSS_INSTANCE}) + set(DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB ${DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB} "boss_instance/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB ${DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB} PARENT_SCOPE) diff --git a/dScripts/02_server/Map/njhub/CatapultBaseServer.cpp b/dScripts/02_server/Map/njhub/CatapultBaseServer.cpp new file mode 100644 index 00000000..8bed629c --- /dev/null +++ b/dScripts/02_server/Map/njhub/CatapultBaseServer.cpp @@ -0,0 +1,66 @@ +#include "CatapultBaseServer.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "Entity.h" +#include "RenderComponent.h" + +void CatapultBaseServer::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { + if (name == "BouncerBuilt") { + // start a timer for the arm to player the with bouncer animation + self->AddTimer("PlatAnim", .75); + + // set the bouncer so we can use it later + self->SetVar(u"Bouncer", sender->GetObjectID()); + + GameMessages::SendBouncerActiveStatus(sender->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); + } +} + +void CatapultBaseServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "PlatAnim") { + // get the arm asset + const auto arm = Game::entityManager->GetEntitiesInGroup(self->GetVarAsString(u"ArmGroup")); + + // tell the arm to the play the platform animation, which is just the arm laying there but with bouncer + for (auto* obj : arm) { + RenderComponent::PlayAnimation(obj, u"idle-platform"); + GameMessages::SendPlayNDAudioEmitter(obj, UNASSIGNED_SYSTEM_ADDRESS, "{8cccf912-69e3-4041-a20b-63e4afafc993}"); + // set the art so we can use it again + self->SetVar(u"Arm", obj->GetObjectID()); + break; + } + + // start a timer till the bouncer actually bounces + self->AddTimer("bounce", 3); + } else if (timerName == "launchAnim") { + // get the arm asset + auto* arm = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Arm")); + if (arm == nullptr) return; + + // tell the arm to player the launcher animation + auto animTime = 1; + self->AddTimer("resetArm", animTime); + RenderComponent::PlayAnimation(arm, u"launch"); + } else if (timerName == "bounce") { + auto* bouncer = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Bouncer")); + if (bouncer == nullptr) return; + + // bounce all players + bouncer->NotifyObject(bouncer, "bounceAllInProximity"); // Likely to trigger server side bounce, bodging this + // add a delay to play the animation + self->AddTimer("launchAnim", .3); + } else if (timerName == "resetArm") { + auto* arm = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Arm")); + if (arm == nullptr) return; + + // set the arm back to natural state + RenderComponent::PlayAnimation(arm, u"idle"); + + auto* bouncer = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Bouncer")); + if (bouncer == nullptr) return; + + // kill the bouncer + GameMessages::SendNotifyClientObject(bouncer->GetObjectID(), u"TimeToDie"); + bouncer->Smash(self->GetObjectID(), eKillType::VIOLENT); + } +} diff --git a/dScripts/CatapultBaseServer.h b/dScripts/02_server/Map/njhub/CatapultBaseServer.h similarity index 100% rename from dScripts/CatapultBaseServer.h rename to dScripts/02_server/Map/njhub/CatapultBaseServer.h diff --git a/dScripts/02_server/Map/njhub/CatapultBouncerServer.cpp b/dScripts/02_server/Map/njhub/CatapultBouncerServer.cpp new file mode 100644 index 00000000..98b8a29d --- /dev/null +++ b/dScripts/02_server/Map/njhub/CatapultBouncerServer.cpp @@ -0,0 +1,15 @@ +#include "CatapultBouncerServer.h" +#include "GameMessages.h" +#include "EntityManager.h" + +void CatapultBouncerServer::OnRebuildComplete(Entity* self, Entity* target) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"Built", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + + self->SetNetworkVar<bool>(u"Built", true); + + const auto base = Game::entityManager->GetEntitiesInGroup(self->GetVarAsString(u"BaseGroup")); + + for (auto* obj : base) { + obj->NotifyObject(self, "BouncerBuilt"); + } +} diff --git a/dScripts/CatapultBouncerServer.h b/dScripts/02_server/Map/njhub/CatapultBouncerServer.h similarity index 100% rename from dScripts/CatapultBouncerServer.h rename to dScripts/02_server/Map/njhub/CatapultBouncerServer.h diff --git a/dScripts/02_server/Map/njhub/CavePrisonCage.cpp b/dScripts/02_server/Map/njhub/CavePrisonCage.cpp new file mode 100644 index 00000000..9bd89067 --- /dev/null +++ b/dScripts/02_server/Map/njhub/CavePrisonCage.cpp @@ -0,0 +1,216 @@ +#include "CavePrisonCage.h" +#include "EntityManager.h" +#include "RebuildComponent.h" +#include "GameMessages.h" +#include "Character.h" +#include "dZoneManager.h" +#include "RenderComponent.h" + +void CavePrisonCage::OnStartup(Entity* self) { + const auto& myNum = self->GetVar<std::u16string>(u"myNumber"); + + if (myNum.empty()) { + return; + } + + auto* spawner = Game::zoneManager->GetSpawnersByName("PrisonCounterweight_0" + GeneralUtils::UTF16ToWTF8(myNum))[0]; + + self->SetVar<Spawner*>(u"CWSpawner", spawner); + + Setup(self, spawner); +} + +void CavePrisonCage::Setup(Entity* self, Spawner* spawner) { + SpawnCounterweight(self, spawner); + + NiPoint3 mypos = self->GetPosition(); + NiQuaternion myrot = self->GetRotation(); + + mypos.y += 1.5; + mypos.z -= 0.5; + + EntityInfo info{}; + info.lot = m_Villagers[self->GetVarAs<int32_t>(u"myNumber") - 1]; + info.pos = mypos; + info.rot = myrot; + info.spawnerID = self->GetObjectID(); + + // Spawn the villager inside the jail + auto* entity = Game::entityManager->CreateEntity(info); + + // Save the villeger ID + self->SetVar<LWOOBJID>(u"villager", entity->GetObjectID()); + + // Construct the entity + Game::entityManager->ConstructEntity(entity); +} + +void CavePrisonCage::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state != eRebuildState::RESETTING) { + return; + } + + auto* spawner = self->GetVar<Spawner*>(u"CWSpawner"); + + if (spawner == nullptr) { + return; + } + + spawner->Reset(); + + SpawnCounterweight(self, spawner); +} + +void CavePrisonCage::SpawnCounterweight(Entity* self, Spawner* spawner) { + spawner->Reset(); + + auto* counterweight = spawner->Spawn(); + + self->SetVar<LWOOBJID>(u"Counterweight", counterweight->GetObjectID()); + + auto* rebuildComponent = counterweight->GetComponent<RebuildComponent>(); + + if (rebuildComponent != nullptr) { + rebuildComponent->AddRebuildStateCallback([this, self](eRebuildState state) { + OnRebuildNotifyState(self, state); + }); + + rebuildComponent->AddRebuildCompleteCallback([this, self](Entity* user) { + // The counterweight is a simple mover, which is not implemented, so we'll just set it's position + auto* counterweight = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Counterweight")); + + if (counterweight == nullptr) { + return; + } + + // Move the counterweight down 2 units + counterweight->SetPosition(counterweight->GetPosition() + NiPoint3(0, -2, 0)); + + // Serialize the counterweight + Game::entityManager->SerializeEntity(counterweight); + + // notifyPlatformAtLastWaypoint + + // Save the userID as Builder + self->SetVar<LWOOBJID>(u"Builder", user->GetObjectID()); + + // Get the button and make sure it still exists + auto* button = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Button")); + + if (button == nullptr) { + return; + } + + // Play the 'down' animation on the button + RenderComponent::PlayAnimation(button, u"down"); + + // Setup a timer named 'buttonGoingDown' to be triggered in 5 seconds + self->AddTimer("buttonGoingDown", 5.0f); + }); + } + + if (self->GetVar<LWOOBJID>(u"Button")) { + return; + } + + GetButton(self); +} + +void CavePrisonCage::GetButton(Entity* self) { + const auto buttons = Game::entityManager->GetEntitiesInGroup("PrisonButton_0" + std::to_string(self->GetVarAs<int32_t>(u"myNumber"))); + + if (buttons.size() == 0) { + // Try again in 0.5 seconds + self->AddCallbackTimer(0.5, [this, self]() { + GetButton(self); + }); + + return; + } + + auto* button = buttons[0]; + + self->SetVar<LWOOBJID>(u"Button", button->GetObjectID()); +} + +void CavePrisonCage::OnTimerDone(Entity* self, std::string timerName) { + // the anim of the button down is over + if (timerName == "buttonGoingDown") { + // Play the 'up' animation + RenderComponent::PlayAnimation(self, u"up"); + + // Setup a timer named 'CageOpen' to be triggered in 1 second + self->AddTimer("CageOpen", 1.0f); + } else if (timerName == "CageOpen") { + // play the idle open anim + RenderComponent::PlayAnimation(self, u"idle-up"); + + // Get the villeger + auto* villager = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"villager")); + + if (villager == nullptr) { + return; + } + + GameMessages::SendNotifyClientObject(villager->GetObjectID(), u"TimeToChat", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + + // Get the builder and make sure it still exists + auto* builder = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Builder")); + + if (builder == nullptr) { + return; + } + + const auto flagNum = 2020 + self->GetVarAs<int32_t>(u"myNumber"); + + // Set the flag on the builder character + builder->GetCharacter()->SetPlayerFlag(flagNum, true); + + // Setup a timer named 'VillagerEscape' to be triggered in 5 seconds + self->AddTimer("VillagerEscape", 5.0f); + } else if (timerName == "VillagerEscape") { + // Get the villeger and make sure it still exists + auto* villager = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"villager")); + + if (villager == nullptr) { + return; + } + + // Kill the villager + villager->Kill(); + + // Setup a timer named 'SmashCounterweight' to be triggered in 2 seconds + self->AddTimer("SmashCounterweight", 2.0f); + } else if (timerName == "SmashCounterweight") { + // Get the counterweight and make sure it still exists + auto* counterweight = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Counterweight")); + + if (counterweight == nullptr) { + return; + } + + // Smash the counterweight + counterweight->Smash(); + + // Get the button and make sure it still exists + auto* button = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"Button")); + + if (button == nullptr) { + return; + } + + // Play the 'up' animation on the button + RenderComponent::PlayAnimation(button, u"up"); + + // Setup a timer named 'CageClosed' to be triggered in 1 second + self->AddTimer("CageClosed", 1.0f); + } else if (timerName == "CageClosed") { + // play the idle closed anim + RenderComponent::PlayAnimation(self, u"idle"); + + // Setup a timer named 'ResetPrison' to be triggered in 10 seconds + self->AddTimer("ResetPrison", 10.0f); + } else if (timerName == "ResetPrison") { + Setup(self, self->GetVar<Spawner*>(u"CWSpawner")); + } +} diff --git a/dScripts/CavePrisonCage.h b/dScripts/02_server/Map/njhub/CavePrisonCage.h similarity index 100% rename from dScripts/CavePrisonCage.h rename to dScripts/02_server/Map/njhub/CavePrisonCage.h diff --git a/dScripts/02_server/Map/njhub/EnemySkeletonSpawner.cpp b/dScripts/02_server/Map/njhub/EnemySkeletonSpawner.cpp new file mode 100644 index 00000000..7f4d471c --- /dev/null +++ b/dScripts/02_server/Map/njhub/EnemySkeletonSpawner.cpp @@ -0,0 +1,76 @@ +#include "EnemySkeletonSpawner.h" +#include "SkillComponent.h" +#include "RenderComponent.h" +#include "EntityManager.h" +#include "EntityInfo.h" + +void EnemySkeletonSpawner::OnStartup(Entity* self) { + self->SetProximityRadius(15, "ronin"); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(1127, 24812, LWOOBJID_EMPTY, true); + } +} + +void EnemySkeletonSpawner::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "hatchTime") { + auto* renderComponent = self->GetComponent<RenderComponent>(); + + if (renderComponent != nullptr) { + renderComponent->PlayEffect(644, u"create", "BurstFX1"); + } + + EntityInfo info{}; + info.lot = 14024; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.spawnerID = self->GetObjectID(); + + auto* spawnedEntity = Game::entityManager->CreateEntity(info); + + if (spawnedEntity == nullptr) { + return; + } + + Game::entityManager->ConstructEntity(spawnedEntity); + + spawnedEntity->AddCallbackTimer(60, [spawnedEntity]() { + spawnedEntity->Smash(spawnedEntity->GetObjectID()); + }); + + self->Smash(self->GetObjectID()); + } +} + +void EnemySkeletonSpawner::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (entering->IsPlayer() && name == "ronin" && status == "ENTER" && !self->GetVar<bool>(u"hatching")) { + StartHatching(self); + + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(305, 3568, LWOOBJID_EMPTY); + } + } +} + +void EnemySkeletonSpawner::OnHit(Entity* self, Entity* attacker) { + if (!self->GetVar<bool>(u"hatching")) { + StartHatching(self); + } +} + +void EnemySkeletonSpawner::StartHatching(Entity* self) { + self->SetVar(u"hatching", true); + + auto* renderComponent = self->GetComponent<RenderComponent>(); + + if (renderComponent != nullptr) { + renderComponent->PlayEffect(9017, u"cast", "WakeUpFX1"); + renderComponent->PlayEffect(9018, u"burst", "WakeUpFX1"); + } + + self->AddTimer("hatchTime", 2); +} diff --git a/dScripts/EnemySkeletonSpawner.h b/dScripts/02_server/Map/njhub/EnemySkeletonSpawner.h similarity index 100% rename from dScripts/EnemySkeletonSpawner.h rename to dScripts/02_server/Map/njhub/EnemySkeletonSpawner.h diff --git a/dScripts/FallingTile.cpp b/dScripts/02_server/Map/njhub/FallingTile.cpp similarity index 100% rename from dScripts/FallingTile.cpp rename to dScripts/02_server/Map/njhub/FallingTile.cpp diff --git a/dScripts/FallingTile.h b/dScripts/02_server/Map/njhub/FallingTile.h similarity index 100% rename from dScripts/FallingTile.h rename to dScripts/02_server/Map/njhub/FallingTile.h diff --git a/dScripts/FlameJetServer.cpp b/dScripts/02_server/Map/njhub/FlameJetServer.cpp similarity index 100% rename from dScripts/FlameJetServer.cpp rename to dScripts/02_server/Map/njhub/FlameJetServer.cpp diff --git a/dScripts/FlameJetServer.h b/dScripts/02_server/Map/njhub/FlameJetServer.h similarity index 100% rename from dScripts/FlameJetServer.h rename to dScripts/02_server/Map/njhub/FlameJetServer.h diff --git a/dScripts/02_server/Map/njhub/ImaginationShrineServer.cpp b/dScripts/02_server/Map/njhub/ImaginationShrineServer.cpp new file mode 100644 index 00000000..1fbfad98 --- /dev/null +++ b/dScripts/02_server/Map/njhub/ImaginationShrineServer.cpp @@ -0,0 +1,16 @@ +#include "ImaginationShrineServer.h" +#include "RebuildComponent.h" + +void ImaginationShrineServer::OnUse(Entity* self, Entity* user) { + // If the rebuild component is complete, use the shrine + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent == nullptr) { + return; + } + + if (rebuildComponent->GetState() == eRebuildState::COMPLETED) { + // Use the shrine + BaseUse(self, user); + } +} diff --git a/dScripts/ImaginationShrineServer.h b/dScripts/02_server/Map/njhub/ImaginationShrineServer.h similarity index 100% rename from dScripts/ImaginationShrineServer.h rename to dScripts/02_server/Map/njhub/ImaginationShrineServer.h diff --git a/dScripts/02_server/Map/njhub/Lieutenant.cpp b/dScripts/02_server/Map/njhub/Lieutenant.cpp new file mode 100644 index 00000000..74f0e215 --- /dev/null +++ b/dScripts/02_server/Map/njhub/Lieutenant.cpp @@ -0,0 +1,44 @@ +#include "Lieutenant.h" +#include "SkillComponent.h" +#include "dZoneManager.h" + +void Lieutenant::OnStartup(Entity* self) { + auto* skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + return; + } + + skillComponent->CalculateBehavior(1127, 24812, self->GetObjectID(), true); +} + +void Lieutenant::OnDie(Entity* self, Entity* killer) { + const auto myLOT = self->GetLOT(); + + std::string spawnerName; + + switch (myLOT) { + case 16047: + spawnerName = "EarthShrine_ERail"; + break; + case 16050: + spawnerName = "IceShrine_QBBouncer"; + break; + case 16049: + spawnerName = "LightningShrine_LRail"; + break; + default: + return; + } + + const auto spawners = Game::zoneManager->GetSpawnersByName(spawnerName); + + if (spawners.empty()) { + return; + } + + for (auto* spawner : spawners) { + spawner->Reset(); + spawner->Activate(); + } +} diff --git a/dScripts/Lieutenant.h b/dScripts/02_server/Map/njhub/Lieutenant.h similarity index 100% rename from dScripts/Lieutenant.h rename to dScripts/02_server/Map/njhub/Lieutenant.h diff --git a/dScripts/02_server/Map/njhub/MonCoreNookDoors.cpp b/dScripts/02_server/Map/njhub/MonCoreNookDoors.cpp new file mode 100644 index 00000000..a6db5064 --- /dev/null +++ b/dScripts/02_server/Map/njhub/MonCoreNookDoors.cpp @@ -0,0 +1,37 @@ +#include "MonCoreNookDoors.h" +#include "dZoneManager.h" + +void MonCoreNookDoors::OnStartup(Entity* self) { + SpawnDoor(self); +} + +void MonCoreNookDoors::SpawnDoor(Entity* self) { + const auto doorNum = self->GetVarAsString(u"number"); + + if (doorNum.empty()) { + return; + } + + const auto spawners = Game::zoneManager->GetSpawnersByName("MonCoreNookDoor0" + doorNum); + + if (spawners.empty()) { + return; + } + + auto* spawner = spawners[0]; + + spawner->Reset(); + spawner->Activate(); +} + +void MonCoreNookDoors::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + if (args == "DoorSmashed") { + self->AddTimer("RespawnDoor", 30); + } +} + +void MonCoreNookDoors::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "RespawnDoor") { + SpawnDoor(self); + } +} diff --git a/dScripts/MonCoreNookDoors.h b/dScripts/02_server/Map/njhub/MonCoreNookDoors.h similarity index 100% rename from dScripts/MonCoreNookDoors.h rename to dScripts/02_server/Map/njhub/MonCoreNookDoors.h diff --git a/dScripts/02_server/Map/njhub/MonCoreSmashableDoors.cpp b/dScripts/02_server/Map/njhub/MonCoreSmashableDoors.cpp new file mode 100644 index 00000000..f9bd6b98 --- /dev/null +++ b/dScripts/02_server/Map/njhub/MonCoreSmashableDoors.cpp @@ -0,0 +1,21 @@ +#include "MonCoreSmashableDoors.h" +#include "EntityManager.h" + +void MonCoreSmashableDoors::OnDie(Entity* self, Entity* killer) { + auto myNum = self->GetVarAsString(u"spawner_name"); + + myNum = myNum.substr(myNum.length() - 1, 1); + + auto triggerGroup = "CoreNookTrig0" + myNum; + + // Get the trigger + auto triggers = Game::entityManager->GetEntitiesInGroup(triggerGroup); + + if (triggers.empty()) { + return; + } + + for (auto trigger : triggers) { + trigger->OnFireEventServerSide(self, "DoorSmashed"); + } +} diff --git a/dScripts/MonCoreSmashableDoors.h b/dScripts/02_server/Map/njhub/MonCoreSmashableDoors.h similarity index 100% rename from dScripts/MonCoreSmashableDoors.h rename to dScripts/02_server/Map/njhub/MonCoreSmashableDoors.h diff --git a/dScripts/02_server/Map/njhub/NjColeNPC.cpp b/dScripts/02_server/Map/njhub/NjColeNPC.cpp new file mode 100644 index 00000000..b989f3ee --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjColeNPC.cpp @@ -0,0 +1,49 @@ +#include "NjColeNPC.h" +#include "MissionComponent.h" +#include "InventoryComponent.h" +#include "eMissionState.h" + +void NjColeNPC::OnEmoteReceived(Entity* self, int32_t emote, Entity* target) { + if (emote != 393) { + return; + } + + auto* inventoryComponent = target->GetComponent<InventoryComponent>(); + + if (inventoryComponent == nullptr) { + return; + } + + if (!inventoryComponent->IsEquipped(14499) && !inventoryComponent->IsEquipped(16644)) { + return; + } + + auto* missionComponent = target->GetComponent<MissionComponent>(); + + if (missionComponent == nullptr) { + return; + } + + missionComponent->ForceProgressTaskType(1818, 1, 1); +} + +void NjColeNPC::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + NjNPCMissionSpinjitzuServer::OnMissionDialogueOK(self, target, missionID, missionState); + + if (missionID == 1818 && missionState >= eMissionState::READY_TO_COMPLETE) { + auto* missionComponent = target->GetComponent<MissionComponent>(); + auto* inventoryComponent = target->GetComponent<InventoryComponent>(); + + if (missionComponent == nullptr || inventoryComponent == nullptr) { + return; + } + + if (inventoryComponent->GetLotCount(14499) > 0) { + inventoryComponent->RemoveItem(14499, 1); + } else { + return; + } + + inventoryComponent->AddItem(16644, 1, eLootSourceType::NONE); + } +} diff --git a/dScripts/02_server/Map/njhub/NjColeNPC.h b/dScripts/02_server/Map/njhub/NjColeNPC.h new file mode 100644 index 00000000..a2536e32 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjColeNPC.h @@ -0,0 +1,7 @@ +#include "NjNPCMissionSpinjitzuServer.h" + +class NjColeNPC : public NjNPCMissionSpinjitzuServer { + void OnEmoteReceived(Entity* self, int32_t emote, Entity* target) override; + + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +}; diff --git a/dScripts/02_server/Map/njhub/NjDragonEmblemChestServer.cpp b/dScripts/02_server/Map/njhub/NjDragonEmblemChestServer.cpp new file mode 100644 index 00000000..931cfe56 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjDragonEmblemChestServer.cpp @@ -0,0 +1,19 @@ +#include "NjDragonEmblemChestServer.h" +#include "Character.h" +#include "EntityInfo.h" +#include "Loot.h" +#include "Entity.h" +#include "DestroyableComponent.h" +#include "ePlayerFlag.h" + +void NjDragonEmblemChestServer::OnUse(Entity* self, Entity* user) { + auto* character = user->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(ePlayerFlag::NJ_WU_SHOW_DAILY_CHEST, false); + } + + auto* destroyable = self->GetComponent<DestroyableComponent>(); + if (destroyable != nullptr) { + LootGenerator::Instance().DropLoot(user, self, destroyable->GetLootMatrixID(), 0, 0); + } +} diff --git a/dScripts/NjDragonEmblemChestServer.h b/dScripts/02_server/Map/njhub/NjDragonEmblemChestServer.h similarity index 100% rename from dScripts/NjDragonEmblemChestServer.h rename to dScripts/02_server/Map/njhub/NjDragonEmblemChestServer.h diff --git a/dScripts/NjEarthDragonPetServer.cpp b/dScripts/02_server/Map/njhub/NjEarthDragonPetServer.cpp similarity index 100% rename from dScripts/NjEarthDragonPetServer.cpp rename to dScripts/02_server/Map/njhub/NjEarthDragonPetServer.cpp diff --git a/dScripts/NjEarthDragonPetServer.h b/dScripts/02_server/Map/njhub/NjEarthDragonPetServer.h similarity index 100% rename from dScripts/NjEarthDragonPetServer.h rename to dScripts/02_server/Map/njhub/NjEarthDragonPetServer.h diff --git a/dScripts/NjEarthPetServer.cpp b/dScripts/02_server/Map/njhub/NjEarthPetServer.cpp similarity index 100% rename from dScripts/NjEarthPetServer.cpp rename to dScripts/02_server/Map/njhub/NjEarthPetServer.cpp diff --git a/dScripts/NjEarthPetServer.h b/dScripts/02_server/Map/njhub/NjEarthPetServer.h similarity index 100% rename from dScripts/NjEarthPetServer.h rename to dScripts/02_server/Map/njhub/NjEarthPetServer.h diff --git a/dScripts/02_server/Map/njhub/NjGarmadonCelebration.cpp b/dScripts/02_server/Map/njhub/NjGarmadonCelebration.cpp new file mode 100644 index 00000000..d3e54be1 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjGarmadonCelebration.cpp @@ -0,0 +1,18 @@ +#include "NjGarmadonCelebration.h" +#include "Character.h" +#include "GameMessages.h" +#include "ePlayerFlag.h" + +void NjGarmadonCelebration::OnCollisionPhantom(Entity* self, Entity* target) { + auto* character = target->GetCharacter(); + + if (character == nullptr) { + return; + } + + if (!character->GetPlayerFlag(ePlayerFlag::NJ_GARMADON_CINEMATIC_SEEN)) { + character->SetPlayerFlag(ePlayerFlag::NJ_GARMADON_CINEMATIC_SEEN, true); + + GameMessages::SendStartCelebrationEffect(target, target->GetSystemAddress(), GarmadonCelebrationID); + } +} diff --git a/dScripts/NjGarmadonCelebration.h b/dScripts/02_server/Map/njhub/NjGarmadonCelebration.h similarity index 100% rename from dScripts/NjGarmadonCelebration.h rename to dScripts/02_server/Map/njhub/NjGarmadonCelebration.h diff --git a/dScripts/02_server/Map/njhub/NjJayMissionItems.cpp b/dScripts/02_server/Map/njhub/NjJayMissionItems.cpp new file mode 100644 index 00000000..46131446 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjJayMissionItems.cpp @@ -0,0 +1,13 @@ +#include "NjJayMissionItems.h" + +void NjJayMissionItems::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + NjNPCMissionSpinjitzuServer::OnMissionDialogueOK(self, target, missionID, missionState); + NPCAddRemoveItem::OnMissionDialogueOK(self, target, missionID, missionState); +} + +std::map<uint32_t, std::vector<ItemSetting>> NjJayMissionItems::GetSettings() { + return { + {1789, {{{14474},false, true}}}, + {1927, {{{14493},false, true}}} + }; +} diff --git a/dScripts/02_server/Map/njhub/NjJayMissionItems.h b/dScripts/02_server/Map/njhub/NjJayMissionItems.h new file mode 100644 index 00000000..bdaee3ea --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjJayMissionItems.h @@ -0,0 +1,10 @@ +#pragma once +#include "NjNPCMissionSpinjitzuServer.h" +#include "NPCAddRemoveItem.h" +#include <vector> +#include <map> + +class NjJayMissionItems : public NjNPCMissionSpinjitzuServer, NPCAddRemoveItem { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + std::map<uint32_t, std::vector<ItemSetting>> GetSettings() override; +}; diff --git a/dScripts/02_server/Map/njhub/NjNPCMissionSpinjitzuServer.cpp b/dScripts/02_server/Map/njhub/NjNPCMissionSpinjitzuServer.cpp new file mode 100644 index 00000000..37a5754c --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjNPCMissionSpinjitzuServer.cpp @@ -0,0 +1,24 @@ +#include "NjNPCMissionSpinjitzuServer.h" +#include "Character.h" +#include "EntityManager.h" +#include "eMissionState.h" + +void NjNPCMissionSpinjitzuServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + + const auto& element = self->GetVar<std::u16string>(ElementVariable); + if (missionID == ElementMissions.at(element) && missionState >= eMissionState::READY_TO_COMPLETE) { + + const auto targetID = target->GetObjectID(); + + // Wait for an animation to complete and flag that the player has learned spinjitzu + self->AddCallbackTimer(5.0f, [targetID, element]() { + auto* target = Game::entityManager->GetEntity(targetID); + if (target != nullptr) { + auto* character = target->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(ElementFlags.at(element), true); + } + } + }); + } +} diff --git a/dScripts/02_server/Map/njhub/NjNPCMissionSpinjitzuServer.h b/dScripts/02_server/Map/njhub/NjNPCMissionSpinjitzuServer.h new file mode 100644 index 00000000..116ecc93 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjNPCMissionSpinjitzuServer.h @@ -0,0 +1,25 @@ +#pragma once +#include "CppScripts.h" +#include <map> +#include "ePlayerFlag.h" + +static std::map<std::u16string, ePlayerFlag> ElementFlags = { + {u"earth", ePlayerFlag::NJ_EARTH_SPINJITZU}, + {u"lightning", ePlayerFlag::NJ_LIGHTNING_SPINJITZU}, + {u"ice", ePlayerFlag::NJ_ICE_SPINJITZU}, + {u"fire", ePlayerFlag::NJ_FIRE_SPINJITZU} +}; + +static std::map<std::u16string, uint32_t> ElementMissions = { + {u"earth", 1796}, + {u"lightning", 1952}, + {u"ice", 1959}, + {u"fire", 1962}, +}; + +class NjNPCMissionSpinjitzuServer : public CppScripts::Script { +public: + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +private: + const std::u16string ElementVariable = u"element"; +}; diff --git a/dScripts/NjNyaMissionitems.cpp b/dScripts/02_server/Map/njhub/NjNyaMissionitems.cpp similarity index 100% rename from dScripts/NjNyaMissionitems.cpp rename to dScripts/02_server/Map/njhub/NjNyaMissionitems.cpp diff --git a/dScripts/NjNyaMissionitems.h b/dScripts/02_server/Map/njhub/NjNyaMissionitems.h similarity index 100% rename from dScripts/NjNyaMissionitems.h rename to dScripts/02_server/Map/njhub/NjNyaMissionitems.h diff --git a/dScripts/02_server/Map/njhub/NjScrollChestServer.cpp b/dScripts/02_server/Map/njhub/NjScrollChestServer.cpp new file mode 100644 index 00000000..7156b368 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjScrollChestServer.cpp @@ -0,0 +1,17 @@ +#include "NjScrollChestServer.h" +#include "InventoryComponent.h" + +void NjScrollChestServer::OnUse(Entity* self, Entity* user) { + const auto keyLOT = self->GetVar<LOT>(u"KeyNum"); + const auto rewardItemLOT = self->GetVar<LOT>(u"openItemID"); + + auto* playerInventory = user->GetComponent<InventoryComponent>(); + if (playerInventory != nullptr && playerInventory->GetLotCount(keyLOT) == 1) { + + // Check for the key and remove + playerInventory->RemoveItem(keyLOT, 1); + + // Reward the player with the item set + playerInventory->AddItem(rewardItemLOT, 1, eLootSourceType::NONE); + } +} diff --git a/dScripts/NjScrollChestServer.h b/dScripts/02_server/Map/njhub/NjScrollChestServer.h similarity index 100% rename from dScripts/NjScrollChestServer.h rename to dScripts/02_server/Map/njhub/NjScrollChestServer.h diff --git a/dScripts/02_server/Map/njhub/NjWuNPC.cpp b/dScripts/02_server/Map/njhub/NjWuNPC.cpp new file mode 100644 index 00000000..1a5d30fd --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjWuNPC.cpp @@ -0,0 +1,65 @@ +#include "NjWuNPC.h" +#include "MissionComponent.h" +#include "Character.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "eMissionState.h" +#include "ePlayerFlag.h" + +void NjWuNPC::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + + // The Dragon statue daily mission + if (missionID == m_MainDragonMissionID) { + auto* character = target->GetCharacter(); + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (character == nullptr || missionComponent == nullptr) + return; + + switch (missionState) { + case eMissionState::AVAILABLE: + case eMissionState::COMPLETE_AVAILABLE: + { + // Reset the sub missions + for (const auto& subMissionID : m_SubDragonMissionIDs) { + missionComponent->RemoveMission(subMissionID); + missionComponent->AcceptMission(subMissionID); + } + + character->SetPlayerFlag(ePlayerFlag::NJ_WU_SHOW_DAILY_CHEST, false); + + // Hide the chest + for (auto* chest : Game::entityManager->GetEntitiesInGroup(m_DragonChestGroup)) { + GameMessages::SendNotifyClientObject(chest->GetObjectID(), m_ShowChestNotification, 0, -1, + target->GetObjectID(), "", target->GetSystemAddress()); + } + + return; + } + case eMissionState::READY_TO_COMPLETE: + case eMissionState::COMPLETE_READY_TO_COMPLETE: + { + character->SetPlayerFlag(ePlayerFlag::NJ_WU_SHOW_DAILY_CHEST, true); + + // Show the chest + for (auto* chest : Game::entityManager->GetEntitiesInGroup(m_DragonChestGroup)) { + GameMessages::SendNotifyClientObject(chest->GetObjectID(), m_ShowChestNotification, 1, -1, + target->GetObjectID(), "", target->GetSystemAddress()); + } + + auto playerID = target->GetObjectID(); + self->AddCallbackTimer(5.0f, [this, playerID]() { + auto* player = Game::entityManager->GetEntity(playerID); + if (player == nullptr) + return; + + // Stop the dragon effects + for (auto* dragon : Game::entityManager->GetEntitiesInGroup(m_DragonStatueGroup)) { + GameMessages::SendStopFXEffect(dragon, true, "on"); + } + }); + } + default: + return; + } + } +} diff --git a/dScripts/02_server/Map/njhub/NjWuNPC.h b/dScripts/02_server/Map/njhub/NjWuNPC.h new file mode 100644 index 00000000..f8c52303 --- /dev/null +++ b/dScripts/02_server/Map/njhub/NjWuNPC.h @@ -0,0 +1,13 @@ +#pragma once +#include "AmTemplateSkillVolume.h" + +class NjWuNPC : public AmTemplateSkillVolume { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + const uint32_t m_MainDragonMissionID = 2040; + const std::vector<uint32_t> m_SubDragonMissionIDs = { 2064, 2065, 2066, 2067 }; + + // Groups and variables + const std::string m_DragonChestGroup = "DragonEmblemChest"; + const std::string m_DragonStatueGroup = "Minidragons"; + const std::u16string m_ShowChestNotification = u"showChest"; +}; diff --git a/dScripts/02_server/Map/njhub/RainOfArrows.cpp b/dScripts/02_server/Map/njhub/RainOfArrows.cpp new file mode 100644 index 00000000..8a8f9c50 --- /dev/null +++ b/dScripts/02_server/Map/njhub/RainOfArrows.cpp @@ -0,0 +1,71 @@ +#include "RainOfArrows.h" +#include "EntityManager.h" +#include "SkillComponent.h" +#include "EntityInfo.h" +#include "GameMessages.h" + +void RainOfArrows::OnStartup(Entity* self) { + +} + +void RainOfArrows::OnRebuildComplete(Entity* self, Entity* target) { + auto myPos = self->GetPosition(); + auto myRot = self->GetRotation(); + + EntityInfo info; + info.lot = m_ArrowFXObject; + info.pos = myPos; + info.rot = myRot; + info.spawnerID = self->GetObjectID(); + + auto* entity = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(entity); + + self->SetVar<LWOOBJID>(u"ChildFX", entity->GetObjectID()); + self->SetVar<LWOOBJID>(u"playerID", target->GetObjectID()); + + self->AddTimer("ArrowsIncoming", m_ArrowDelay); + self->AddTimer("PlayArrowSound", m_ArrowDelay - 4); +} + +void RainOfArrows::OnTimerDone(Entity* self, std::string timerName) { + auto* child = Game::entityManager->GetEntity( + self->GetVar<LWOOBJID>(u"ChildFX") + ); + + auto* player = Game::entityManager->GetEntity( + self->GetVar<LWOOBJID>(u"playerID") + ); + + if (timerName == "ArrowsIncoming") { + if (child == nullptr) { + return; + } + + auto* skillComponent = child->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + return; + } + + skillComponent->CalculateBehavior( + m_ArrowSkill, + m_ArrowBehavior, + LWOOBJID_EMPTY, + true, + false, + player != nullptr ? player->GetObjectID() : LWOOBJID_EMPTY + ); + + self->AddTimer("FireSkill", 0.7f); + } else if (timerName == "PlayArrowSound") { + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, m_ArrowsGUID); + } else if (timerName == "FireSkill") { + if (child != nullptr) { + child->Smash(); + } + + self->Smash(player != nullptr ? player->GetObjectID() : LWOOBJID_EMPTY); + } +} diff --git a/dScripts/RainOfArrows.h b/dScripts/02_server/Map/njhub/RainOfArrows.h similarity index 100% rename from dScripts/RainOfArrows.h rename to dScripts/02_server/Map/njhub/RainOfArrows.h diff --git a/dScripts/02_server/Map/njhub/boss_instance/CMakeLists.txt b/dScripts/02_server/Map/njhub/boss_instance/CMakeLists.txt new file mode 100644 index 00000000..22bb541d --- /dev/null +++ b/dScripts/02_server/Map/njhub/boss_instance/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MAP_NJHUB_BOSS_INSTANCE + "NjMonastryBossInstance.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Map/njhub/boss_instance/NjMonastryBossInstance.cpp b/dScripts/02_server/Map/njhub/boss_instance/NjMonastryBossInstance.cpp new file mode 100644 index 00000000..d31f7b17 --- /dev/null +++ b/dScripts/02_server/Map/njhub/boss_instance/NjMonastryBossInstance.cpp @@ -0,0 +1,524 @@ +#include "NjMonastryBossInstance.h" +#include "RebuildComponent.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "dZoneManager.h" +#include "GameMessages.h" +#include "BaseCombatAIComponent.h" +#include "BuffComponent.h" +#include "SkillComponent.h" +#include "TeamManager.h" +#include <algorithm> +#include "RenderComponent.h" + +// // // // // // // +// Event handling // +// // // // // // // + +void NjMonastryBossInstance::OnStartup(Entity* self) { + auto spawnerNames = std::vector<std::string>{ LedgeFrakjawSpawner, LowerFrakjawSpawner, BaseEnemiesSpawner + std::to_string(1), + BaseEnemiesSpawner + std::to_string(2), BaseEnemiesSpawner + std::to_string(3), + BaseEnemiesSpawner + std::to_string(4), CounterweightSpawner }; + + // Add a notification request for all the spawned entities, corresponds to notifySpawnedObjectLoaded + for (const auto& spawnerName : spawnerNames) { + for (auto* spawner : Game::zoneManager->GetSpawnersByName(spawnerName)) { + spawner->AddEntitySpawnedCallback([self, this](Entity* entity) { + const auto lot = entity->GetLOT(); + switch (lot) { + case LedgedFrakjawLOT: + NjMonastryBossInstance::HandleLedgedFrakjawSpawned(self, entity); + return; + case CounterWeightLOT: + NjMonastryBossInstance::HandleCounterWeightSpawned(self, entity); + return; + case LowerFrakjawLOT: + NjMonastryBossInstance::HandleLowerFrakjawSpawned(self, entity); + return; + default: + NjMonastryBossInstance::HandleWaveEnemySpawned(self, entity); + return; + } + }); + } + } +} + +void NjMonastryBossInstance::OnPlayerLoaded(Entity* self, Entity* player) { + ActivityTimerStop(self, WaitingForPlayersTimer); + + // Join the player in the activity + UpdatePlayer(self, player->GetObjectID()); + + // Buff the player + auto* destroyableComponent = player->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->SetHealth((int32_t)destroyableComponent->GetMaxHealth()); + destroyableComponent->SetArmor((int32_t)destroyableComponent->GetMaxArmor()); + destroyableComponent->SetImagination((int32_t)destroyableComponent->GetMaxImagination()); + } + + // Add player ID to instance + auto totalPlayersLoaded = self->GetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable); + totalPlayersLoaded.push_back(player->GetObjectID()); + + // Properly position the player + self->SetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable, totalPlayersLoaded); + // This was always spawning all players at position one before and other values cause players to be invisible. + TeleportPlayer(player, 1); + + // Large teams face a tougher challenge + if (totalPlayersLoaded.size() >= 3) + self->SetVar<bool>(LargeTeamVariable, true); + + // Start the game if all players in the team have loaded + auto* team = TeamManager::Instance()->GetTeam(player->GetObjectID()); + if (team == nullptr || totalPlayersLoaded.size() == team->members.size()) { + StartFight(self); + return; + } + + self->AddCallbackTimer(0.0f, [self, player]() { + if (player != nullptr) { + // If we don't have enough players yet, wait for the others to load and notify the client to play a cool cinematic + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayerLoaded", 0, 0, + player->GetObjectID(), "", player->GetSystemAddress()); + } + }); + + ActivityTimerStart(self, WaitingForPlayersTimer, 45.0f, 45.0f); +} + +void NjMonastryBossInstance::OnPlayerExit(Entity* self, Entity* player) { + UpdatePlayer(self, player->GetObjectID(), true); + // Fetch the total players loaded from the vars + auto totalPlayersLoaded = self->GetVar<std::vector<LWOOBJID> >(TotalPlayersLoadedVariable); + + // Find the player to remove + auto playerToRemove = std::find(totalPlayersLoaded.begin(), totalPlayersLoaded.end(), player->GetObjectID()); + + // If we found the player remove them from out list of players + if (playerToRemove != totalPlayersLoaded.end()) { + totalPlayersLoaded.erase(playerToRemove); + } else { + Game::logger->Log("NjMonastryBossInstance", "Failed to remove player at exit."); + } + + // Set the players loaded var back + self->SetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable, totalPlayersLoaded); + + // Since this is an exit method, check if enough players have left. If enough have left + // resize the instance to account for such. + if (totalPlayersLoaded.size() <= 2) self->SetVar<bool>(LargeTeamVariable, false); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayerLeft", 0, 0, player->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); +} + +void NjMonastryBossInstance::OnActivityTimerDone(Entity* self, const std::string& name) { + auto split = GeneralUtils::SplitString(name, TimerSplitChar); + auto timerName = split[0]; + auto objectID = split.size() > 1 ? (LWOOBJID)std::stoull(split[1]) : LWOOBJID_EMPTY; + + if (timerName == WaitingForPlayersTimer) { + StartFight(self); + } else if (timerName == SpawnNextWaveTimer) { + auto* frakjaw = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(LedgeFrakjawVariable)); + if (frakjaw != nullptr) { + SummonWave(self, frakjaw); + } + } else if (timerName == SpawnWaveTimer) { + auto wave = self->GetVar<uint32_t>(WaveNumberVariable); + self->SetVar<uint32_t>(WaveNumberVariable, wave + 1); + self->SetVar<uint32_t>(TotalAliveInWaveVariable, 0); + + if (wave < m_Waves.size()) { + auto waves = m_Waves.at(wave); + auto counter = 0; + + for (const auto& waveEnemy : waves) { + const auto numberToSpawn = self->GetVar<bool>(LargeTeamVariable) + ? waveEnemy.largeNumber : waveEnemy.smallNumber; + + auto spawnIndex = counter % 4 + 1; + SpawnOnNetwork(self, waveEnemy.lot, numberToSpawn, BaseEnemiesSpawner + std::to_string(spawnIndex)); + counter++; + } + } + } else if (timerName + TimerSplitChar == UnstunTimer) { + auto* entity = Game::entityManager->GetEntity(objectID); + if (entity != nullptr) { + auto* combatAI = entity->GetComponent<BaseCombatAIComponent>(); + if (combatAI != nullptr) { + combatAI->SetDisabled(false); + } + } + } else if (timerName == SpawnCounterWeightTimer) { + auto spawners = Game::zoneManager->GetSpawnersByName(CounterweightSpawner); + if (!spawners.empty()) { + // Spawn the counter weight at a specific waypoint, there's one for each round + auto* spawner = spawners.front(); + + spawner->Spawn({ + spawner->m_Info.nodes.at((self->GetVar<uint32_t>(WaveNumberVariable) - 1) % 3) + }, true); + } + } else if (timerName == LowerFrakjawCamTimer) { + // Destroy the frakjaw on the ledge + auto* ledgeFrakjaw = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(LedgeFrakjawVariable)); + if (ledgeFrakjaw != nullptr) { + ledgeFrakjaw->Kill(); + } + + ActivityTimerStart(self, SpawnLowerFrakjawTimer, 1.0f, 1.0f); + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, + LWOOBJID_EMPTY, BottomFrakSpawn, UNASSIGNED_SYSTEM_ADDRESS); + } else if (timerName == SpawnLowerFrakjawTimer) { + auto spawners = Game::zoneManager->GetSpawnersByName(LowerFrakjawSpawner); + if (!spawners.empty()) { + auto* spawner = spawners.front(); + spawner->Activate(); + } + } else if (timerName == SpawnRailTimer) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, + LWOOBJID_EMPTY, FireRailSpawn, UNASSIGNED_SYSTEM_ADDRESS); + + auto spawners = Game::zoneManager->GetSpawnersByName(FireRailSpawner); + if (!spawners.empty()) { + auto* spawner = spawners.front(); + spawner->Activate(); + } + } else if (timerName + TimerSplitChar == FrakjawSpawnInTimer) { + auto* lowerFrakjaw = Game::entityManager->GetEntity(objectID); + if (lowerFrakjaw != nullptr) { + LowerFrakjawSummon(self, lowerFrakjaw); + } + } else if (timerName == WaveOverTimer) { + WaveOver(self); + } else if (timerName == FightOverTimer) { + FightOver(self); + } +} + +// // // // // // // // +// Custom functions // +// // // // // // // // + +void NjMonastryBossInstance::StartFight(Entity* self) { + if (self->GetVar<bool>(FightStartedVariable)) + return; + + self->SetVar<bool>(FightStartedVariable, true); + + // Activate the frakjaw spawner + for (auto* spawner : Game::zoneManager->GetSpawnersByName(LedgeFrakjawSpawner)) { + spawner->Activate(); + } +} + +void NjMonastryBossInstance::HandleLedgedFrakjawSpawned(Entity* self, Entity* ledgedFrakjaw) { + self->SetVar<LWOOBJID>(LedgeFrakjawVariable, ledgedFrakjaw->GetObjectID()); + SummonWave(self, ledgedFrakjaw); +} + +void NjMonastryBossInstance::HandleCounterWeightSpawned(Entity* self, Entity* counterWeight) { + auto* rebuildComponent = counterWeight->GetComponent<RebuildComponent>(); + if (rebuildComponent != nullptr) { + rebuildComponent->AddRebuildStateCallback([this, self, counterWeight](eRebuildState state) { + + switch (state) { + case eRebuildState::BUILDING: + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, + 0, 0, counterWeight->GetObjectID(), + BaseCounterweightQB + std::to_string(self->GetVar<uint32_t>(WaveNumberVariable)), + UNASSIGNED_SYSTEM_ADDRESS); + return; + case eRebuildState::INCOMPLETE: + GameMessages::SendNotifyClientObject(self->GetObjectID(), EndCinematicNotification, + 0, 0, LWOOBJID_EMPTY, "", + UNASSIGNED_SYSTEM_ADDRESS); + return; + case eRebuildState::RESETTING: + ActivityTimerStart(self, SpawnCounterWeightTimer, 0.0f, 0.0f); + return; + case eRebuildState::COMPLETED: { + // TODO: Move the platform? + + // The counterweight is actually a moving platform and we should listen to the last waypoint event here + // 0.5f is a rough estimate of that path, though, and results in less needed logic + self->AddCallbackTimer(0.5f, [this, self, counterWeight]() { + if (counterWeight != nullptr) { + counterWeight->Kill(); + } + + auto* frakjaw = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(LedgeFrakjawVariable)); + if (frakjaw == nullptr) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"LedgeFrakjawDead", 0, + 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + return; + } + + auto* skillComponent = frakjaw->GetComponent<SkillComponent>(); + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(1635, 39097, frakjaw->GetObjectID(), true, false); + } + + RenderComponent::PlayAnimation(frakjaw, StunnedAnimation); + GameMessages::SendPlayNDAudioEmitter(frakjaw, UNASSIGNED_SYSTEM_ADDRESS, CounterSmashAudio); + + // Before wave 4 we should lower frakjaw from the ledge + if (self->GetVar<uint32_t>(WaveNumberVariable) == 3) { + LowerFrakjaw(self, frakjaw); + return; + } + + ActivityTimerStart(self, SpawnNextWaveTimer, 2.0f, 2.0f); + }); + } + default: + return; + } + }); + } +} + +void NjMonastryBossInstance::HandleLowerFrakjawSpawned(Entity* self, Entity* lowerFrakjaw) { + RenderComponent::PlayAnimation(lowerFrakjaw, TeleportInAnimation); + self->SetVar<LWOOBJID>(LowerFrakjawVariable, lowerFrakjaw->GetObjectID()); + + auto* combatAI = lowerFrakjaw->GetComponent<BaseCombatAIComponent>(); + if (combatAI != nullptr) { + combatAI->SetDisabled(true); + } + + auto* destroyableComponent = lowerFrakjaw->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->AddOnHitCallback([this, self, lowerFrakjaw](Entity* attacker) { + NjMonastryBossInstance::HandleLowerFrakjawHit(self, lowerFrakjaw, attacker); + }); + } + + lowerFrakjaw->AddDieCallback([this, self, lowerFrakjaw]() { + NjMonastryBossInstance::HandleLowerFrakjawDied(self, lowerFrakjaw); + }); + + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"LedgeFrakjawDead", 0, 0, + LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + + if (self->GetVar<bool>(LargeTeamVariable)) { + // Double frakjaws health for large teams + if (destroyableComponent != nullptr) { + const auto doubleHealth = destroyableComponent->GetHealth() * 2; + destroyableComponent->SetHealth(doubleHealth); + destroyableComponent->SetMaxHealth((float_t)doubleHealth); + } + + ActivityTimerStart(self, FrakjawSpawnInTimer + std::to_string(lowerFrakjaw->GetObjectID()), + 2.0f, 2.0f); + ActivityTimerStart(self, UnstunTimer + std::to_string(lowerFrakjaw->GetObjectID()), + 7.0f, 7.0f); + } else { + ActivityTimerStart(self, UnstunTimer + std::to_string(lowerFrakjaw->GetObjectID()), + 5.0f, 5.0f); + } +} + +void NjMonastryBossInstance::HandleLowerFrakjawHit(Entity* self, Entity* lowerFrakjaw, Entity* attacker) { + auto* destroyableComponent = lowerFrakjaw->GetComponent<DestroyableComponent>(); + if (destroyableComponent == nullptr) + return; + + // Progress the fight to the last wave if frakjaw has less than 50% of his health left + if (destroyableComponent->GetHealth() <= (uint32_t)destroyableComponent->GetMaxHealth() / 2 && !self->GetVar<bool>(OnLastWaveVarbiale)) { + self->SetVar<bool>(OnLastWaveVarbiale, true); + + // Stun frakjaw during the cinematic + auto* combatAI = lowerFrakjaw->GetComponent<BaseCombatAIComponent>(); + if (combatAI != nullptr) { + combatAI->SetDisabled(true); + } + ActivityTimerStart(self, UnstunTimer + std::to_string(lowerFrakjaw->GetObjectID()), 5.0f, 5.0f); + + const auto trashMobsAlive = self->GetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable); + std::vector<LWOOBJID> newTrashMobs = {}; + + for (const auto& trashMobID : trashMobsAlive) { + auto* trashMob = Game::entityManager->GetEntity(trashMobID); + if (trashMob != nullptr) { + newTrashMobs.push_back(trashMobID); + + // Stun all the enemies until the cinematic is over + auto* trashMobCombatAI = trashMob->GetComponent<BaseCombatAIComponent>(); + if (trashMobCombatAI != nullptr) { + trashMobCombatAI->SetDisabled(true); + } + ActivityTimerStart(self, UnstunTimer + std::to_string(trashMobID), 5.0f, 5.0f); + } + } + + self->SetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable, newTrashMobs); + + LowerFrakjawSummon(self, lowerFrakjaw); + RemovePoison(self); + } +} + +void NjMonastryBossInstance::HandleLowerFrakjawDied(Entity* self, Entity* lowerFrakjaw) { + ActivityTimerStart(self, FightOverTimer, 2.0f, 2.0f); +} + +void NjMonastryBossInstance::HandleWaveEnemySpawned(Entity* self, Entity* waveEnemy) { + waveEnemy->AddDieCallback([this, self, waveEnemy]() { + NjMonastryBossInstance::HandleWaveEnemyDied(self, waveEnemy); + }); + + auto waveEnemies = self->GetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable); + waveEnemies.push_back(waveEnemy->GetObjectID()); + self->SetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable, waveEnemies); + + auto* combatAI = waveEnemy->GetComponent<BaseCombatAIComponent>(); + if (combatAI != nullptr) { + combatAI->SetDisabled(true); + ActivityTimerStart(self, UnstunTimer + std::to_string(waveEnemy->GetObjectID()), 3.0f, 3.0f); + } +} + +void NjMonastryBossInstance::HandleWaveEnemyDied(Entity* self, Entity* waveEnemy) { + auto waveEnemies = self->GetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable); + waveEnemies.erase(std::remove(waveEnemies.begin(), waveEnemies.end(), waveEnemy->GetObjectID()), waveEnemies.end()); + self->SetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable, waveEnemies); + + if (waveEnemies.empty()) { + ActivityTimerStart(self, WaveOverTimer, 2.0f, 2.0f); + } +} + +void NjMonastryBossInstance::TeleportPlayer(Entity* player, uint32_t position) { + for (const auto* spawnPoint : Game::entityManager->GetEntitiesInGroup("SpawnPoint" + std::to_string(position))) { + GameMessages::SendTeleport(player->GetObjectID(), spawnPoint->GetPosition(), spawnPoint->GetRotation(), + player->GetSystemAddress(), true); + } +} + +void NjMonastryBossInstance::SummonWave(Entity* self, Entity* frakjaw) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, LWOOBJID_EMPTY, + LedgeFrakSummon, UNASSIGNED_SYSTEM_ADDRESS); + RenderComponent::PlayAnimation(frakjaw, SummonAnimation); + + // Stop the music for the first, fourth and fifth wave + const auto wave = self->GetVar<uint32_t>(WaveNumberVariable); + if (wave >= 1 || wave < (m_Waves.size() - 1)) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), StopMusicNotification, 0, 0, + LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(wave - 1), + UNASSIGNED_SYSTEM_ADDRESS); + } + + // After frakjaw moves down the music stays the same + if (wave < (m_Waves.size() - 1)) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), StartMusicNotification, 0, 0, + LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(wave), + UNASSIGNED_SYSTEM_ADDRESS); + } + + ActivityTimerStart(self, SpawnWaveTimer, 4.0f, 4.0f); +} + +void NjMonastryBossInstance::LowerFrakjawSummon(Entity* self, Entity* frakjaw) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, + LWOOBJID_EMPTY, BottomFrakSummon, UNASSIGNED_SYSTEM_ADDRESS); + ActivityTimerStart(self, SpawnWaveTimer, 2.0f, 2.0f); + RenderComponent::PlayAnimation(frakjaw, SummonAnimation); +} + +void NjMonastryBossInstance::RemovePoison(Entity* self) { + const auto& totalPlayer = self->GetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable); + for (const auto& playerID : totalPlayer) { + + auto* player = Game::entityManager->GetEntity(playerID); + if (player != nullptr) { + + auto* buffComponent = player->GetComponent<BuffComponent>(); + if (buffComponent != nullptr) { + buffComponent->RemoveBuff(PoisonBuff); + } + } + } +} + +void NjMonastryBossInstance::LowerFrakjaw(Entity* self, Entity* frakjaw) { + RenderComponent::PlayAnimation(frakjaw, TeleportOutAnimation); + ActivityTimerStart(self, LowerFrakjawCamTimer, 2.0f, 2.0f); + + GameMessages::SendNotifyClientObject(frakjaw->GetObjectID(), StopMusicNotification, 0, 0, + LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(m_Waves.size() - 3), UNASSIGNED_SYSTEM_ADDRESS); + GameMessages::SendNotifyClientObject(frakjaw->GetObjectID(), StartMusicNotification, 0, 0, + LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(m_Waves.size() - 2), UNASSIGNED_SYSTEM_ADDRESS); +} + +void NjMonastryBossInstance::SpawnOnNetwork(Entity* self, const LOT& toSpawn, const uint32_t& numberToSpawn, const std::string& spawnerName) { + auto spawners = Game::zoneManager->GetSpawnersByName(spawnerName); + if (spawners.empty() || numberToSpawn <= 0) + return; + + auto* spawner = spawners.front(); + + // Spawn the lot N times + spawner->SetSpawnLot(toSpawn); + for (auto i = 0; i < numberToSpawn; i++) + spawner->Spawn({ spawner->m_Info.nodes.at(i % spawner->m_Info.nodes.size()) }, true); +} + +void NjMonastryBossInstance::WaveOver(Entity* self) { + auto wave = self->GetVar<uint32_t>(WaveNumberVariable); + if (wave >= m_Waves.size() - 1) + return; + + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, + LWOOBJID_EMPTY, BaseCounterweightSpawn + std::to_string(wave), + UNASSIGNED_SYSTEM_ADDRESS); + ActivityTimerStart(self, SpawnCounterWeightTimer, 1.5f, 1.5f); + RemovePoison(self); +} + +void NjMonastryBossInstance::FightOver(Entity* self) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"GroundFrakjawDead", 0, 0, + LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); + + // Remove all the enemies from the battlefield + for (auto i = 1; i < 5; i++) { + auto spawners = Game::zoneManager->GetSpawnersByName(BaseEnemiesSpawner + std::to_string(i)); + if (!spawners.empty()) { + auto* spawner = spawners.front(); + spawner->Deactivate(); + spawner->Reset(); + } + } + + RemovePoison(self); + ActivityTimerStart(self, SpawnRailTimer, 1.5f, 1.5f); + + // Set the music to play the victory music + GameMessages::SendNotifyClientObject(self->GetObjectID(), StopMusicNotification, 0, 0, + LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(m_Waves.size() - 2), + UNASSIGNED_SYSTEM_ADDRESS); + GameMessages::SendNotifyClientObject(self->GetObjectID(), FlashMusicNotification, 0, 0, + LWOOBJID_EMPTY, "Monastery_Frakjaw_Battle_Win", UNASSIGNED_SYSTEM_ADDRESS); + GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, + LWOOBJID_EMPTY, TreasureChestSpawning, UNASSIGNED_SYSTEM_ADDRESS); + + auto treasureChests = Game::entityManager->GetEntitiesInGroup(ChestSpawnpointGroup); + for (auto* treasureChest : treasureChests) { + auto info = EntityInfo{}; + + info.lot = ChestLOT; + info.pos = treasureChest->GetPosition(); + info.rot = treasureChest->GetRotation(); + info.spawnerID = self->GetObjectID(); + info.settings = { + new LDFData<LWOOBJID>(u"parent_tag", self->GetObjectID()) + }; + + // Finally spawn a treasure chest at the correct spawn point + auto* chestObject = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(chestObject); + } +} diff --git a/dScripts/NjMonastryBossInstance.h b/dScripts/02_server/Map/njhub/boss_instance/NjMonastryBossInstance.h similarity index 100% rename from dScripts/NjMonastryBossInstance.h rename to dScripts/02_server/Map/njhub/boss_instance/NjMonastryBossInstance.h diff --git a/dScripts/02_server/Minigame/CMakeLists.txt b/dScripts/02_server/Minigame/CMakeLists.txt new file mode 100644 index 00000000..e8cb4402 --- /dev/null +++ b/dScripts/02_server/Minigame/CMakeLists.txt @@ -0,0 +1,9 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MINIGAME) + +add_subdirectory(General) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER_MINIGAME_GENERAL}) + set(DSCRIPTS_SOURCES_02_SERVER_MINIGAME ${DSCRIPTS_SOURCES_02_SERVER_MINIGAME} "General/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_02_SERVER_MINIGAME ${DSCRIPTS_SOURCES_02_SERVER_MINIGAME} PARENT_SCOPE) diff --git a/dScripts/02_server/Minigame/General/CMakeLists.txt b/dScripts/02_server/Minigame/General/CMakeLists.txt new file mode 100644 index 00000000..55d2e595 --- /dev/null +++ b/dScripts/02_server/Minigame/General/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_02_SERVER_MINIGAME_GENERAL + "MinigameTreasureChestServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Minigame/General/MinigameTreasureChestServer.cpp b/dScripts/02_server/Minigame/General/MinigameTreasureChestServer.cpp new file mode 100644 index 00000000..e06ed1d9 --- /dev/null +++ b/dScripts/02_server/Minigame/General/MinigameTreasureChestServer.cpp @@ -0,0 +1,65 @@ +#include "MinigameTreasureChestServer.h" +#include "ScriptedActivityComponent.h" +#include "TeamManager.h" +#include "EntityManager.h" +#include "dZoneManager.h" +#include "Loot.h" + +void MinigameTreasureChestServer::OnUse(Entity* self, Entity* user) { + auto* sac = self->GetComponent<ScriptedActivityComponent>(); + if (sac == nullptr) + return; + + if (self->GetVar<bool>(u"used")) + return; + self->SetVar<bool>(u"used", true); + + if (!IsPlayerInActivity(self, user->GetObjectID())) + UpdatePlayer(self, user->GetObjectID()); + + auto* team = TeamManager::Instance()->GetTeam(user->GetObjectID()); + uint32_t activityRating = 0; + if (team != nullptr) { + for (const auto& teamMemberID : team->members) { + auto* teamMember = Game::entityManager->GetEntity(teamMemberID); + if (teamMember != nullptr) { + activityRating = CalculateActivityRating(self, teamMemberID); + + if (self->GetLOT() == frakjawChestId) activityRating = team->members.size(); + + LootGenerator::Instance().DropActivityLoot(teamMember, self, sac->GetActivityID(), activityRating); + } + } + } else { + activityRating = CalculateActivityRating(self, user->GetObjectID()); + + if (self->GetLOT() == frakjawChestId) activityRating = 1; + + LootGenerator::Instance().DropActivityLoot(user, self, sac->GetActivityID(), activityRating); + } + + sac->PlayerRemove(user->GetObjectID()); + + auto* zoneControl = Game::zoneManager->GetZoneControlObject(); + if (zoneControl != nullptr) { + zoneControl->OnFireEventServerSide(self, "Survival_Update", 0); + } + + self->Smash(self->GetObjectID()); +} + +uint32_t MinigameTreasureChestServer::CalculateActivityRating(Entity* self, LWOOBJID playerID) { + auto* team = TeamManager::Instance()->GetTeam(playerID); + return team != nullptr ? team->members.size() * 100 : ActivityManager::CalculateActivityRating(self, playerID) * 100; +} + +void MinigameTreasureChestServer::OnStartup(Entity* self) { + + // BONS treasure chest thinks it's on FV, causing it to start a lobby + if (Game::zoneManager->GetZoneID().GetMapID() == 1204) { + auto* sac = self->GetComponent<ScriptedActivityComponent>(); + if (sac != nullptr) { + sac->SetInstanceMapID(1204); + } + } +} diff --git a/dScripts/MinigameTreasureChestServer.h b/dScripts/02_server/Minigame/General/MinigameTreasureChestServer.h similarity index 100% rename from dScripts/MinigameTreasureChestServer.h rename to dScripts/02_server/Minigame/General/MinigameTreasureChestServer.h diff --git a/dScripts/02_server/Objects/AgSurvivalBuffStation.cpp b/dScripts/02_server/Objects/AgSurvivalBuffStation.cpp new file mode 100644 index 00000000..2ce44bd1 --- /dev/null +++ b/dScripts/02_server/Objects/AgSurvivalBuffStation.cpp @@ -0,0 +1,65 @@ +#include "AgSurvivalBuffStation.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "SkillComponent.h" +#include "TeamManager.h" + +void AgSurvivalBuffStation::OnRebuildComplete(Entity* self, Entity* target) { + auto destroyableComponent = self->GetComponent<DestroyableComponent>(); + // We set the faction to 1 so that the buff station sees players as friendly targets to buff + if (destroyableComponent != nullptr) destroyableComponent->SetFaction(1); + + auto skillComponent = self->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) skillComponent->CalculateBehavior(skillIdForBuffStation, behaviorIdForBuffStation, self->GetObjectID()); + + self->AddCallbackTimer(smashTimer, [self]() { + self->Smash(); + }); + self->AddTimer("DropArmor", dropArmorTimer); + self->AddTimer("DropLife", dropLifeTimer); + self->AddTimer("Dropimagination", dropImaginationTimer); + // Since all survival players should be on the same team, we get the team. + auto team = TeamManager::Instance()->GetTeam(target->GetObjectID()); + + std::vector<LWOOBJID> builderTeam; + // Not on a team + if (team == nullptr) { + builderTeam.push_back(target->GetObjectID()); + self->SetVar<std::vector<LWOOBJID>>(u"BuilderTeam", builderTeam); + return; + } + + for (auto memberID : team->members) { + builderTeam.push_back(memberID); + } + self->SetVar<std::vector<LWOOBJID>>(u"BuilderTeam", builderTeam); +} + +void AgSurvivalBuffStation::OnTimerDone(Entity* self, std::string timerName) { + uint32_t powerupToDrop = lifePowerup; + if (timerName == "DropArmor") { + powerupToDrop = armorPowerup; + self->AddTimer("DropArmor", dropArmorTimer); + } + if (timerName == "DropLife") { + powerupToDrop = lifePowerup; + self->AddTimer("DropLife", dropLifeTimer); + } + if (timerName == "Dropimagination") { + powerupToDrop = imaginationPowerup; + self->AddTimer("Dropimagination", dropImaginationTimer); + } + auto team = self->GetVar<std::vector<LWOOBJID>>(u"BuilderTeam"); + for (auto memberID : team) { + auto member = Game::entityManager->GetEntity(memberID); + if (member != nullptr && !member->GetIsDead()) { + GameMessages::SendDropClientLoot(member, self->GetObjectID(), powerupToDrop, 0, self->GetPosition()); + } else { + // If player left the team or left early erase them from the team variable. + team.erase(std::find(team.begin(), team.end(), memberID)); + self->SetVar<std::vector<LWOOBJID>>(u"BuilderTeam", team); + } + } +} diff --git a/dScripts/AgSurvivalBuffStation.h b/dScripts/02_server/Objects/AgSurvivalBuffStation.h similarity index 100% rename from dScripts/AgSurvivalBuffStation.h rename to dScripts/02_server/Objects/AgSurvivalBuffStation.h diff --git a/dScripts/02_server/Objects/CMakeLists.txt b/dScripts/02_server/Objects/CMakeLists.txt new file mode 100644 index 00000000..1b96d79f --- /dev/null +++ b/dScripts/02_server/Objects/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_02_SERVER_OBJECTS + "AgSurvivalBuffStation.cpp" + "StinkyFishTarget.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Objects/StinkyFishTarget.cpp b/dScripts/02_server/Objects/StinkyFishTarget.cpp new file mode 100644 index 00000000..9840235d --- /dev/null +++ b/dScripts/02_server/Objects/StinkyFishTarget.cpp @@ -0,0 +1,45 @@ +#include "StinkyFishTarget.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "Entity.h" + +void StinkyFishTarget::OnStartup(Entity* self) { + auto position = self->GetPosition(); + position.SetY(position.GetY() - 0.5f); + self->SetPosition(position); +} + +void StinkyFishTarget::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (message != "stinkfish" || self->GetVar<bool>(u"used")) + return; + + self->SetVar<bool>(u"used", true); + self->SetVar<LWOOBJID>(u"player", caster->GetObjectID()); + + EntityInfo entityInfo{}; + entityInfo.lot = SHARK_LOT; + entityInfo.pos = self->GetPosition(); + entityInfo.rot = self->GetRotation(); + entityInfo.spawnerID = self->GetObjectID(); + entityInfo.settings = { + new LDFData<bool>(u"no_timed_spawn", true) + }; + + auto* fish = Game::entityManager->CreateEntity(entityInfo); + Game::entityManager->ConstructEntity(fish); + + self->SetVar<LWOOBJID>(u"fish", fish->GetObjectID()); + self->AddTimer("smash", 5.0f); +} + +void StinkyFishTarget::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "smash") { + const auto playerID = self->GetVar<LWOOBJID>(u"player"); + auto* fish = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"fish")); + + if (fish) { + fish->Smash(playerID); + self->Smash(playerID); + } + } +} diff --git a/dScripts/02_server/Objects/StinkyFishTarget.h b/dScripts/02_server/Objects/StinkyFishTarget.h new file mode 100644 index 00000000..b8f9e9ae --- /dev/null +++ b/dScripts/02_server/Objects/StinkyFishTarget.h @@ -0,0 +1,11 @@ +#pragma once +#include "CppScripts.h" + +class StinkyFishTarget : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) override; + void OnTimerDone(Entity* self, std::string timerName) override; +private: + const LOT SHARK_LOT = 8570; +}; diff --git a/dScripts/02_server/Pets/CMakeLists.txt b/dScripts/02_server/Pets/CMakeLists.txt new file mode 100644 index 00000000..8820a82e --- /dev/null +++ b/dScripts/02_server/Pets/CMakeLists.txt @@ -0,0 +1,5 @@ +set(DSCRIPTS_SOURCES_02_SERVER_PETS + "PetFromDigServer.cpp" + "PetFromObjectServer.cpp" + "DamagingPets.cpp" + PARENT_SCOPE) diff --git a/dScripts/02_server/Pets/DamagingPets.cpp b/dScripts/02_server/Pets/DamagingPets.cpp new file mode 100644 index 00000000..6fd8c560 --- /dev/null +++ b/dScripts/02_server/Pets/DamagingPets.cpp @@ -0,0 +1,146 @@ +#include "DamagingPets.h" +#include "PetComponent.h" +#include "DestroyableComponent.h" +#include "BaseCombatAIComponent.h" +#include "RenderComponent.h" +#include "ePetTamingNotifyType.h" + +void DamagingPets::OnStartup(Entity* self) { + + // Make the pet hostile or non-hostile based on whether or not it is tamed + const auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent != nullptr && petComponent->GetOwner() == nullptr) { + self->AddTimer("GoEvil", 0.5f); + } +} + +void DamagingPets::OnPlayerLoaded(Entity* self, Entity* player) { + + // Makes it so that new players also see the effect + self->AddCallbackTimer(2.5f, [self]() { + if (self != nullptr) { + const auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent != nullptr && petComponent->GetOwner() == nullptr && self->GetVar<bool>(u"IsEvil")) { + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + auto counter = 1; + for (const auto petEffect : GetPetInfo(self).effect) { + renderComponent->PlayEffect(petEffect, u"create", "FXname" + std::to_string(counter)); + counter++; + } + } + } + } + }); +} + +void DamagingPets::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) { + switch (type) { + case ePetTamingNotifyType::SUCCESS: + case ePetTamingNotifyType::BEGIN: + self->CancelAllTimers(); + ClearEffects(self); + break; + case ePetTamingNotifyType::FAILED: + case ePetTamingNotifyType::QUIT: + { + self->SetNetworkVar<bool>(u"bIAmTamable", false); + self->AddTimer("GoEvil", 1.0f); + break; + } + default: + break; + } +} + +void DamagingPets::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + const auto infoForPet = GetPetInfo(self); + if (infoForPet.skill == message) { + + // Only make pets tamable that aren't tamed yet + const auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent != nullptr && petComponent->GetOwner() == nullptr && self->GetVar<bool>(u"IsEvil")) { + ClearEffects(self); + self->AddTimer("GoEvil", 30.0f); + self->SetNetworkVar<bool>(u"bIAmTamable", true); + } + } +} + +void DamagingPets::OnTimerDone(Entity* self, std::string message) { + if (message == "GoEvil") { + MakeUntamable(self); + } +} + +void DamagingPets::MakeUntamable(Entity* self) { + auto* petComponent = self->GetComponent<PetComponent>(); + + // If the pet is currently not being tamed, make it hostile + if (petComponent != nullptr && petComponent->GetStatus() != 5) { + self->SetNetworkVar<bool>(u"bIAmTamable", false); + self->SetVar<bool>(u"IsEvil", true); + petComponent->SetStatus(1); + + auto* combatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + if (combatAIComponent != nullptr) { + combatAIComponent->SetDisabled(false); + } + + // Special faction that can attack the player but the player can't attack + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->SetFaction(114); + destroyableComponent->SetHealth(5); + } + + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + auto counter = 1; + for (const auto petEffect : GetPetInfo(self).effect) { + renderComponent->PlayEffect(petEffect, u"create", "FXname" + std::to_string(counter)); + counter++; + } + } + } +} + +void DamagingPets::ClearEffects(Entity* self) { + self->SetVar<bool>(u"IsEvil", false); + + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent != nullptr) { + petComponent->SetStatus(67108866); + } + + auto* combatAIComponent = self->GetComponent<BaseCombatAIComponent>(); + if (combatAIComponent != nullptr) { + combatAIComponent->SetDisabled(true); + } + + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->SetFaction(99); + } + + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + auto counter = 1; + for (const auto petEffect : GetPetInfo(self).effect) { + renderComponent->StopEffect("FXname" + std::to_string(counter)); + counter++; + } + } +} + +PetInfo DamagingPets::GetPetInfo(Entity* self) { + const auto infoForPet = petInfo.find(self->GetLOT()); + return infoForPet != petInfo.end() ? infoForPet->second : petInfo.begin()->second; +} + +// Does not compile on Win32 with name specifiers +const std::map<LOT, PetInfo> DamagingPets::petInfo = { + { 5639, { /*.effect =*/ { 3170, 4058 }, /*.skill =*/ "waterspray"}}, // Red dragon + { 5641, { /*.effect =*/ { 3170, 4058 }, /*.skill =*/ "waterspray"}}, // Green dragon + { 3261, { /*.effect =*/ { 1490 }, /*.skill =*/ "waterspray"}}, // Skunk +}; diff --git a/dScripts/02_server/Pets/DamagingPets.h b/dScripts/02_server/Pets/DamagingPets.h new file mode 100644 index 00000000..303bff52 --- /dev/null +++ b/dScripts/02_server/Pets/DamagingPets.h @@ -0,0 +1,24 @@ +#pragma once +#include "CppScripts.h" + +/** + * Information about pets regarding which effect to play when a skill is cast + */ +struct PetInfo { + const std::vector<uint32_t> effect; + const std::string skill; +}; + +class DamagingPets : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnTimerDone(Entity* self, std::string message) override; + void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) override; + void OnSkillEventFired(Entity* self, Entity* target, const std::string& message) override; + void OnPlayerLoaded(Entity* self, Entity* player) override; +private: + static void MakeUntamable(Entity* self); + static PetInfo GetPetInfo(Entity* self); + static void ClearEffects(Entity* self); + static const std::map<LOT, PetInfo> petInfo; +}; diff --git a/dScripts/02_server/Pets/PetFromDigServer.cpp b/dScripts/02_server/Pets/PetFromDigServer.cpp new file mode 100644 index 00000000..525f3e94 --- /dev/null +++ b/dScripts/02_server/Pets/PetFromDigServer.cpp @@ -0,0 +1,45 @@ +#include "PetFromDigServer.h" +#include "PetComponent.h" +#include "ePetTamingNotifyType.h" + +void PetFromDigServer::OnStartup(Entity* self) { + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr || petComponent->GetOwner() != nullptr) + return; + + // Triggers the local dig pet script for taming etc. + auto tamer = self->GetVar<LWOOBJID>(u"tamer"); + + // Client compares this with player:GetID() which is a string, so we'll have to give it a string + self->SetNetworkVar(u"pettamer", std::to_string(tamer)); + + // Kill if the player decides that the dig pet is not worthy + self->AddTimer("killself", 45.0f); +} + +void PetFromDigServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "killself") { + + // Don't accidentally kill a pet that is already owned + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr || petComponent->GetOwner() != nullptr) + return; + + self->Smash(self->GetObjectID(), eKillType::SILENT); + } +} + +void PetFromDigServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) { + if (type == ePetTamingNotifyType::BEGIN) { + self->CancelTimer("killself"); + } else if (type == ePetTamingNotifyType::QUIT || type == ePetTamingNotifyType::FAILED) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } else if (type == ePetTamingNotifyType::SUCCESS) { + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr) + return; + // TODO: Remove custom group? + // Command the pet to the player as it may otherwise go to its spawn point which is non existant + // petComponent->Command(NiPoint3::ZERO, LWOOBJID_EMPTY, 6, 202, true); + } +} diff --git a/dScripts/02_server/Pets/PetFromDigServer.h b/dScripts/02_server/Pets/PetFromDigServer.h new file mode 100644 index 00000000..aaf31728 --- /dev/null +++ b/dScripts/02_server/Pets/PetFromDigServer.h @@ -0,0 +1,10 @@ +#pragma once +#include "CppScripts.h" + +class PetFromDigServer : public CppScripts::Script +{ +public: + void OnStartup(Entity* self) override; + void OnTimerDone(Entity* self, std::string timerName) override; + void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) override; +}; diff --git a/dScripts/02_server/Pets/PetFromObjectServer.cpp b/dScripts/02_server/Pets/PetFromObjectServer.cpp new file mode 100644 index 00000000..0a78d7ec --- /dev/null +++ b/dScripts/02_server/Pets/PetFromObjectServer.cpp @@ -0,0 +1,35 @@ +#include "PetFromObjectServer.h" +#include "PetComponent.h" +#include "ePetTamingNotifyType.h" + +void PetFromObjectServer::OnStartup(Entity* self) { + self->SetNetworkVar(u"pettamer", std::to_string(self->GetVar<LWOOBJID>(u"tamer"))); + self->AddTimer("killSelf", 45.0f); +} + +void PetFromObjectServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "killSelf") { + const auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr || petComponent->GetOwner() != nullptr) + return; + self->Smash(self->GetObjectID(), eKillType::SILENT); + } +} + +void PetFromObjectServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) { + switch (type) { + case ePetTamingNotifyType::BEGIN: + self->CancelAllTimers(); + break; + case ePetTamingNotifyType::QUIT: + case ePetTamingNotifyType::FAILED: + self->Smash(self->GetObjectID(), eKillType::SILENT); + break; + case ePetTamingNotifyType::SUCCESS: + // TODO: Remove from groups? + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UpdateSuccessPicking", 0, + 0, tamer->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); + default: + break; + } +} diff --git a/dScripts/02_server/Pets/PetFromObjectServer.h b/dScripts/02_server/Pets/PetFromObjectServer.h new file mode 100644 index 00000000..67cd5eb0 --- /dev/null +++ b/dScripts/02_server/Pets/PetFromObjectServer.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class PetFromObjectServer : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnTimerDone(Entity* self, std::string timerName) override; + void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) override; +}; diff --git a/dScripts/ActMine.cpp b/dScripts/ActMine.cpp deleted file mode 100644 index 637bd805..00000000 --- a/dScripts/ActMine.cpp +++ /dev/null @@ -1,52 +0,0 @@ -#include "ActMine.h" -#include "SkillComponent.h" -#include "DestroyableComponent.h" -#include "RebuildComponent.h" - -void ActMine::OnStartup(Entity* self) { - self->SetVar(u"RebuildComplete", false); - self->SetProximityRadius(MINE_RADIUS, "mineRadius"); -} - -void ActMine::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == eRebuildState::REBUILD_COMPLETED) { - auto* rebuild = self->GetComponent<RebuildComponent>(); - if (rebuild) { - auto* builder = rebuild->GetBuilder(); - self->SetVar(u"Builder", builder->GetObjectID()); - } - - self->SetVar(u"RebuildComplete", true); - self->SetVar(u"NumWarnings", 0); - self->AddToGroup("reset"); - } - -} - -void ActMine::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - auto* detroyable = self->GetComponent<DestroyableComponent>(); - if (!detroyable) return; - if (status == "ENTER" && self->GetVar<bool>(u"RebuildComplete") == true && detroyable->IsEnemy(entering)) { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 242, u"orange", "sirenlight_B"); - self->AddTimer("Tick", TICK_TIME); - } -} - -void ActMine::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "Tick") { - if (self->GetVar<int>(u"NumWarnings") >= MAX_WARNINGS) { - auto* skill = self->GetComponent<SkillComponent>(); - if (!skill) return; - skill->CalculateBehavior(SKILL_ID, BEHAVIOR_ID, LWOOBJID_EMPTY); - self->AddTimer("BlowedUp", BLOWED_UP_TIME); - } else { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 242, u"orange", "sirenlight_B"); - self->AddTimer("Tick", TICK_TIME); - self->SetVar(u"NumWarnings", self->GetVar<int>(u"NumWarnings") + 1); - } - } - - if (timerName == "BlowedUp") { - self->Kill(self); - } -} diff --git a/dScripts/ActNinjaTurret.cpp b/dScripts/ActNinjaTurret.cpp deleted file mode 100644 index 79e502b4..00000000 --- a/dScripts/ActNinjaTurret.cpp +++ /dev/null @@ -1,17 +0,0 @@ -#include "ActNinjaTurret.h" - -void ActNinjaTurret::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == eRebuildState::REBUILD_COMPLETED) { - self->SetVar(u"AmBuilt", true); - } else if (state == eRebuildState::REBUILD_RESETTING) { - self->SetVar(u"AmBuilt", false); - } -} - -void -ActNinjaTurret::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args == "ISpawned" && self->GetVar<bool>(u"AmBuilt")) { - sender->Smash(); - } -} diff --git a/dScripts/ActParadoxPipeFix.cpp b/dScripts/ActParadoxPipeFix.cpp deleted file mode 100644 index 1dddde64..00000000 --- a/dScripts/ActParadoxPipeFix.cpp +++ /dev/null @@ -1,61 +0,0 @@ -#include "ActParadoxPipeFix.h" -#include "EntityManager.h" -#include "RebuildComponent.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void ActParadoxPipeFix::OnRebuildComplete(Entity* self, Entity* target) { - const auto myGroup = "AllPipes"; - - const auto groupObjs = EntityManager::Instance()->GetEntitiesInGroup(myGroup); - - auto indexCount = 0; - - self->SetVar(u"PlayerID", target->GetObjectID()); - - for (auto* object : groupObjs) { - if (object == self) { - continue; - } - - auto* rebuildComponent = object->GetComponent<RebuildComponent>(); - - if (rebuildComponent->GetState() == REBUILD_COMPLETED) { - indexCount++; - } - } - - if (indexCount >= 2) { - const auto refinery = EntityManager::Instance()->GetEntitiesInGroup("Paradox"); - - if (!refinery.empty()) { - GameMessages::SendPlayFXEffect(refinery[0]->GetObjectID(), 3999, u"create", "pipeFX"); - } - - for (auto* object : groupObjs) { - auto* player = EntityManager::Instance()->GetEntity(object->GetVar<LWOOBJID>(u"PlayerID")); - - if (player != nullptr) { - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->ForceProgressTaskType(769, 1, 1, false); - } - - GameMessages::SendPlayCinematic(player->GetObjectID(), u"ParadoxPipeFinish", player->GetSystemAddress(), true, true, false, false, 0, false, 2.0f); - } - - object->SetVar(u"PlayerID", LWOOBJID_EMPTY); - } - } -} - -void ActParadoxPipeFix::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == REBUILD_RESETTING) { - const auto refinery = EntityManager::Instance()->GetEntitiesInGroup("Paradox"); - - if (!refinery.empty()) { - GameMessages::SendStopFXEffect(refinery[0], true, "pipeFX"); - } - } -} diff --git a/dScripts/ActPlayerDeathTrigger.cpp b/dScripts/ActPlayerDeathTrigger.cpp deleted file mode 100644 index 28f1ef43..00000000 --- a/dScripts/ActPlayerDeathTrigger.cpp +++ /dev/null @@ -1,7 +0,0 @@ -#include "ActPlayerDeathTrigger.h" - -void ActPlayerDeathTrigger::OnCollisionPhantom(Entity* self, Entity* target) { - if (!target->IsPlayer() || target->GetIsDead() || !target->GetPlayerReadyForUpdates()) return; //Don't kill already dead players or players not ready - - target->Smash(self->GetObjectID(), eKillType::SILENT); -} diff --git a/dScripts/ActSharkPlayerDeathTrigger.cpp b/dScripts/ActSharkPlayerDeathTrigger.cpp deleted file mode 100644 index 420fc4d7..00000000 --- a/dScripts/ActSharkPlayerDeathTrigger.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "ActSharkPlayerDeathTrigger.h" -#include "MissionComponent.h" -#include "MissionTaskType.h" -#include "Entity.h" - -void ActSharkPlayerDeathTrigger::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (args == "achieve") { - auto missionComponent = sender->GetComponent<MissionComponent>(); - if (!missionComponent) return; - - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, 8419); - - if (sender->GetIsDead() || !sender->GetPlayerReadyForUpdates()) return; //Don't kill already dead players or players not ready - - if (sender->GetCharacter()) { - sender->Smash(self->GetObjectID(), eKillType::VIOLENT, u"big-shark-death"); - } - } -} diff --git a/dScripts/ActVehicleDeathTrigger.cpp b/dScripts/ActVehicleDeathTrigger.cpp deleted file mode 100644 index 77a3b65a..00000000 --- a/dScripts/ActVehicleDeathTrigger.cpp +++ /dev/null @@ -1,52 +0,0 @@ -#include "ActVehicleDeathTrigger.h" -#include "PossessableComponent.h" -#include "GameMessages.h" -#include "RacingControlComponent.h" -#include "dZoneManager.h" -#include "EntityManager.h" -#include "PossessorComponent.h" - - -void ActVehicleDeathTrigger::OnCollisionPhantom(Entity* self, Entity* target) { - auto* possessableComponent = target->GetComponent<PossessableComponent>(); - - Entity* vehicle; - Entity* player; - - if (possessableComponent != nullptr) { - auto* player = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); - - if (player == nullptr) { - return; - } - - return; - } else if (target->IsPlayer()) { - auto* possessorComponent = target->GetComponent<PossessorComponent>(); - - if (possessorComponent == nullptr) { - return; - } - - vehicle = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); - - if (vehicle == nullptr) { - return; - } - - player = target; - } else { - return; - } - - - GameMessages::SendDie(vehicle, self->GetObjectID(), LWOOBJID_EMPTY, true, VIOLENT, u"", 0, 0, 0, true, false, 0); - - auto* zoneController = dZoneManager::Instance()->GetZoneControlObject(); - - auto* racingControlComponent = zoneController->GetComponent<RacingControlComponent>(); - - if (racingControlComponent != nullptr) { - racingControlComponent->OnRequestDie(player); - } -} diff --git a/dScripts/ActivityManager.cpp b/dScripts/ActivityManager.cpp index 18d72311..24e661e3 100644 --- a/dScripts/ActivityManager.cpp +++ b/dScripts/ActivityManager.cpp @@ -5,6 +5,7 @@ #include "GameMessages.h" #include <algorithm> #include "dLogger.h" +#include "Loot.h" bool ActivityManager::IsPlayerInActivity(Entity* self, LWOOBJID playerID) { const auto* sac = self->GetComponent<ScriptedActivityComponent>(); @@ -19,7 +20,7 @@ void ActivityManager::UpdatePlayer(Entity* self, LWOOBJID playerID, const bool r if (remove) { sac->PlayerRemove(playerID); } else { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { sac->PlayerJoin(player); SetActivityScore(self, playerID, 0); @@ -62,7 +63,7 @@ void ActivityManager::StopActivity(Entity* self, const LWOOBJID playerID, const if (quit) { UpdatePlayer(self, playerID, true); } else { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) return; @@ -72,30 +73,31 @@ void ActivityManager::StopActivity(Entity* self, const LWOOBJID playerID, const LootGenerator::Instance().GiveActivityLoot(player, self, gameID, CalculateActivityRating(self, playerID)); - // Save the new score to the leaderboard and show the leaderboard to the player - LeaderboardManager::SaveScore(playerID, gameID, score, value1); - const auto* leaderboard = LeaderboardManager::GetLeaderboard(gameID, InfoType::Standings, - false, player->GetObjectID()); - GameMessages::SendActivitySummaryLeaderboardData(self->GetObjectID(), leaderboard, player->GetSystemAddress()); - delete leaderboard; - - // Makes the leaderboard show up for the player - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"ToggleLeaderBoard", - gameID, 0, playerID, "", - player->GetSystemAddress()); - if (sac != nullptr) { sac->PlayerRemove(player->GetObjectID()); } } } +void ActivityManager::SaveScore(Entity* self, const LWOOBJID playerID, const float primaryScore, const float secondaryScore, const float tertiaryScore) const { + auto* player = Game::entityManager->GetEntity(playerID); + if (!player) return; + + auto* sac = self->GetComponent<ScriptedActivityComponent>(); + uint32_t gameID = sac != nullptr ? sac->GetActivityID() : self->GetLOT(); + // Save the new score to the leaderboard and show the leaderboard to the player + LeaderboardManager::SaveScore(playerID, gameID, primaryScore, secondaryScore, tertiaryScore); + + // Makes the leaderboard show up for the player + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"ToggleLeaderBoard", gameID, 0, playerID, "", player->GetSystemAddress()); +} + bool ActivityManager::TakeActivityCost(const Entity* self, const LWOOBJID playerID) { auto* sac = self->GetComponent<ScriptedActivityComponent>(); if (sac == nullptr) return false; - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) return false; @@ -116,7 +118,10 @@ uint32_t ActivityManager::GetActivityID(const Entity* self) { } void ActivityManager::GetLeaderboardData(Entity* self, const LWOOBJID playerID, const uint32_t activityID, uint32_t numResults) { - LeaderboardManager::SendLeaderboard(activityID, Standings, false, self->GetObjectID(), playerID); + auto* sac = self->GetComponent<ScriptedActivityComponent>(); + uint32_t gameID = sac != nullptr ? sac->GetActivityID() : self->GetLOT(); + // Save the new score to the leaderboard and show the leaderboard to the player + LeaderboardManager::SendLeaderboard(activityID, Leaderboard::InfoType::MyStanding, false, playerID, self->GetObjectID(), 0, numResults); } void ActivityManager::ActivityTimerStart(Entity* self, const std::string& timerName, const float_t updateInterval, @@ -124,7 +129,7 @@ void ActivityManager::ActivityTimerStart(Entity* self, const std::string& timerN auto* timer = new ActivityTimer{ timerName, updateInterval, stopTime }; activeTimers.push_back(timer); - Game::logger->Log("ActivityManager", "Starting timer '%s', %f, %f", timerName.c_str(), updateInterval, stopTime); + Game::logger->LogDebug("ActivityManager", "Starting timer '%s', %f, %f", timerName.c_str(), updateInterval, stopTime); self->AddTimer(GetPrefixedName(timer->name), timer->updateInterval); } @@ -205,10 +210,10 @@ void ActivityManager::OnTimerDone(Entity* self, std::string timerName) { activeTimers.erase(std::remove(activeTimers.begin(), activeTimers.end(), timer), activeTimers.end()); delete timer; - Game::logger->Log("ActivityManager", "Executing timer '%s'", activityTimerName.c_str()); + Game::logger->LogDebug("ActivityManager", "Executing timer '%s'", activityTimerName.c_str()); OnActivityTimerDone(self, activityTimerName); } else { - Game::logger->Log("ActivityManager", "Updating timer '%s'", activityTimerName.c_str()); + Game::logger->LogDebug("ActivityManager", "Updating timer '%s'", activityTimerName.c_str()); OnActivityTimerUpdate(self, timer->name, timer->stopTime - timer->runTime, timer->runTime); self->AddTimer(GetPrefixedName(timer->name), timer->updateInterval); } diff --git a/dScripts/ActivityManager.h b/dScripts/ActivityManager.h index 640cf4bf..a2202bfb 100644 --- a/dScripts/ActivityManager.h +++ b/dScripts/ActivityManager.h @@ -18,6 +18,7 @@ public: static bool TakeActivityCost(const Entity* self, LWOOBJID playerID); static uint32_t GetActivityID(const Entity* self); void StopActivity(Entity* self, LWOOBJID playerID, uint32_t score, uint32_t value1 = 0, uint32_t value2 = 0, bool quit = false); + void SaveScore(Entity* self, const LWOOBJID playerID, const float primaryScore, const float secondaryScore = 0.0f, const float tertiaryScore = 0.0f) const; virtual uint32_t CalculateActivityRating(Entity* self, LWOOBJID playerID); static void GetLeaderboardData(Entity* self, LWOOBJID playerID, uint32_t activityID, uint32_t numResults = 0); // void FreezePlayer(Entity *self, const LWOOBJID playerID, const bool state) const; diff --git a/dScripts/AgBusDoor.cpp b/dScripts/AgBusDoor.cpp deleted file mode 100644 index 4910d0c5..00000000 --- a/dScripts/AgBusDoor.cpp +++ /dev/null @@ -1,65 +0,0 @@ -#include "AgBusDoor.h" -#include "Entity.h" -#include "GameMessages.h" -#include "ProximityMonitorComponent.h" - -void AgBusDoor::OnStartup(Entity* self) { - m_Counter = 0; - m_OuterCounter = 0; - self->SetProximityRadius(75, "busDoor"); - self->SetProximityRadius(85, "busDoorOuter"); -} - -void AgBusDoor::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name != "busDoor" && name != "busDoorOuter") return; - - // Make sure only humans are taken into account - if (!entering->GetCharacter()) return; - - auto* proximityMonitorComponent = self->GetComponent<ProximityMonitorComponent>(); - - if (proximityMonitorComponent == nullptr) return; - - m_Counter = 0; - m_OuterCounter = 0; - - for (const auto& pair : proximityMonitorComponent->GetProximityObjects("busDoor")) { - auto* entity = EntityManager::Instance()->GetEntity(pair.first); - if (entity != nullptr && entity->IsPlayer()) m_Counter++; - } - - for (const auto& pair : proximityMonitorComponent->GetProximityObjects("busDoorOuter")) { - auto* entity = EntityManager::Instance()->GetEntity(pair.first); - if (entity != nullptr && entity->IsPlayer()) m_OuterCounter++; - } - - if (status == "ENTER") { - // move up when a player is inside both radii - if (m_Counter > 0) { - MoveDoor(self, true); - } - } else if (status == "LEAVE") { - // move down when no players are inside either radii - if (m_Counter <= 0) { - MoveDoor(self, false); - } - } -} - -void AgBusDoor::MoveDoor(Entity* self, bool bOpen) { - if (bOpen) { - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 1, 0); - } else { - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, 1); - self->AddTimer("dustTimer", 2.0f); - } - - //This is currently commented out because it might be the reason that people's audio is cutting out. - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, "{9a24f1fa-3177-4745-a2df-fbd996d6e1e3}"); -} - -void AgBusDoor::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "dustTimer") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 642, u"create", "busDust", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - } -} diff --git a/dScripts/AgCagedBricksServer.cpp b/dScripts/AgCagedBricksServer.cpp deleted file mode 100644 index 6d1360de..00000000 --- a/dScripts/AgCagedBricksServer.cpp +++ /dev/null @@ -1,27 +0,0 @@ -#include "AgCagedBricksServer.h" -#include "InventoryComponent.h" -#include "GameMessages.h" -#include "Character.h" -#include "EntityManager.h" - -void AgCagedBricksServer::OnUse(Entity* self, Entity* user) { - //Tell the client to spawn the baby spiderling: - auto spooders = EntityManager::Instance()->GetEntitiesInGroup("cagedSpider"); - for (auto spodder : spooders) { - GameMessages::SendFireEventClientSide(spodder->GetObjectID(), user->GetSystemAddress(), u"toggle", LWOOBJID_EMPTY, 0, 0, user->GetObjectID()); - } - - //Set the flag & mission status: - auto character = user->GetCharacter(); - - if (!character) return; - - character->SetPlayerFlag(74, true); - - //Remove the maelstrom cube: - auto inv = static_cast<InventoryComponent*>(user->GetComponent(COMPONENT_TYPE_INVENTORY)); - - if (inv) { - inv->RemoveItem(14553, 1); - } -} diff --git a/dScripts/AgFans.cpp b/dScripts/AgFans.cpp deleted file mode 100644 index 27d3b940..00000000 --- a/dScripts/AgFans.cpp +++ /dev/null @@ -1,83 +0,0 @@ -#include "AgFans.h" - -#include "EntityManager.h" -#include "GameMessages.h" -#include "PhantomPhysicsComponent.h" -#include "RenderComponent.h" - -void AgFans::OnStartup(Entity* self) { - self->SetVar<bool>(u"alive", true); - self->SetVar<bool>(u"on", false); - - ToggleFX(self, false); - - auto* renderComponent = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - - if (renderComponent == nullptr) { - return; - } - - renderComponent->PlayEffect(495, u"fanOn", "fanOn"); -} - -void AgFans::ToggleFX(Entity* self, bool hit) { - std::string fanGroup = self->GetGroups()[0]; - std::vector<Entity*> fanVolumes = EntityManager::Instance()->GetEntitiesInGroup(fanGroup); - - auto* renderComponent = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - - if (renderComponent == nullptr) { - return; - } - - if (fanVolumes.size() == 0 || !self->GetVar<bool>(u"alive")) return; - - if (self->GetVar<bool>(u"on")) { - GameMessages::SendPlayAnimation(self, u"fan-off"); - - renderComponent->StopEffect("fanOn"); - self->SetVar<bool>(u"on", false); - - for (Entity* volume : fanVolumes) { - PhantomPhysicsComponent* volumePhys = static_cast<PhantomPhysicsComponent*>(volume->GetComponent(COMPONENT_TYPE_PHANTOM_PHYSICS)); - if (!volumePhys) continue; - volumePhys->SetPhysicsEffectActive(false); - EntityManager::Instance()->SerializeEntity(volume); - if (!hit) { - Entity* fxObj = EntityManager::Instance()->GetEntitiesInGroup(fanGroup + "fx")[0]; - GameMessages::SendPlayAnimation(fxObj, u"trigger"); - } - } - } else if (!self->GetVar<bool>(u"on") && self->GetVar<bool>(u"alive")) { - GameMessages::SendPlayAnimation(self, u"fan-on"); - - renderComponent->PlayEffect(495, u"fanOn", "fanOn"); - self->SetVar<bool>(u"on", true); - - for (Entity* volume : fanVolumes) { - PhantomPhysicsComponent* volumePhys = static_cast<PhantomPhysicsComponent*>(volume->GetComponent(COMPONENT_TYPE_PHANTOM_PHYSICS)); - if (!volumePhys) continue; - volumePhys->SetPhysicsEffectActive(true); - EntityManager::Instance()->SerializeEntity(volume); - if (!hit) { - Entity* fxObj = EntityManager::Instance()->GetEntitiesInGroup(fanGroup + "fx")[0]; - GameMessages::SendPlayAnimation(fxObj, u"idle"); - } - } - } -} - -void AgFans::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args.length() == 0 || !self->GetVar<bool>(u"alive")) return; - - if ((args == "turnOn" && self->GetVar<bool>(u"on")) || (args == "turnOff" && !self->GetVar<bool>(u"on"))) return; - ToggleFX(self, false); -} - -void AgFans::OnDie(Entity* self, Entity* killer) { - if (self->GetVar<bool>(u"on")) { - ToggleFX(self, true); - } - self->SetVar<bool>(u"alive", false); -} diff --git a/dScripts/AgImagSmashable.cpp b/dScripts/AgImagSmashable.cpp deleted file mode 100644 index 593294e5..00000000 --- a/dScripts/AgImagSmashable.cpp +++ /dev/null @@ -1,42 +0,0 @@ -#include "AgImagSmashable.h" -#include "EntityManager.h" -#include "GeneralUtils.h" -#include "GameMessages.h" -#include "DestroyableComponent.h" - -void AgImagSmashable::OnDie(Entity* self, Entity* killer) { - bool maxImagGreaterThanZero = false; - - if (killer) { - DestroyableComponent* dest = static_cast<DestroyableComponent*>(killer->GetComponent(COMPONENT_TYPE_DESTROYABLE)); - if (dest) { - maxImagGreaterThanZero = dest->GetMaxImagination() > 0; - } - - if (maxImagGreaterThanZero) { - int amount = GeneralUtils::GenerateRandomNumber<int>(0, 3); - for (int i = 0; i < amount; ++i) { - GameMessages::SendDropClientLoot(killer, self->GetObjectID(), 935, 0, self->GetPosition()); - } - } - } - - CrateAnimal(self); -} - -void AgImagSmashable::CrateAnimal(Entity* self) { - int funnychance = GeneralUtils::GenerateRandomNumber<int>(0, 26); - if (funnychance == 1) { - EntityInfo info; - info.lot = 8114; - info.pos = self->GetPosition(); - info.spawner = nullptr; - info.spawnerID = self->GetSpawnerID(); - info.spawnerNodeID = 0; - - Entity* newEntity = EntityManager::Instance()->CreateEntity(info, nullptr); - if (newEntity) { - EntityManager::Instance()->ConstructEntity(newEntity); - } - } -} diff --git a/dScripts/AgJetEffectServer.cpp b/dScripts/AgJetEffectServer.cpp deleted file mode 100644 index 8c7eca3b..00000000 --- a/dScripts/AgJetEffectServer.cpp +++ /dev/null @@ -1,116 +0,0 @@ -#include "AgJetEffectServer.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "SkillComponent.h" - -void AgJetEffectServer::OnUse(Entity* self, Entity* user) { - if (inUse) { - return; - } - - GameMessages::SendNotifyClientObject( - self->GetObjectID(), - u"isInUse", - 0, - 0, - LWOOBJID_EMPTY, - "", - UNASSIGNED_SYSTEM_ADDRESS - ); - - inUse = true; - - auto entities = EntityManager::Instance()->GetEntitiesInGroup("Jet_FX"); - - if (entities.empty()) { - return; - } - - auto* effect = entities[0]; - - GameMessages::SendPlayFXEffect(effect, 641, u"create", "radarDish", LWOOBJID_EMPTY, 1, 1, true); - - self->AddTimer("radarDish", 2); - self->AddTimer("CineDone", 9); -} - -void AgJetEffectServer::OnRebuildComplete(Entity* self, Entity* target) { - auto entities = EntityManager::Instance()->GetEntitiesInGroup("Jet_FX"); - - if (entities.empty()) { - return; - } - - auto* effect = entities[0]; - - auto groups = self->GetGroups(); - - if (groups.empty()) { - return; - } - - builder = target->GetObjectID(); - - const auto group = groups[0]; - - GameMessages::SendPlayAnimation(effect, u"jetFX"); - - self->AddTimer("PlayEffect", 2.5f); - - if (group == "Base_Radar") { - self->AddTimer("CineDone", 5); - } -} - -void AgJetEffectServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "radarDish") { - GameMessages::SendStopFXEffect(self, true, "radarDish"); - - return; - } - - if (timerName == "PlayEffect") { - auto entities = EntityManager::Instance()->GetEntitiesInGroup("mortarMain"); - - if (entities.empty()) { - return; - } - - const auto size = entities.size(); - - if (size == 0) { - return; - } - - const auto selected = GeneralUtils::GenerateRandomNumber<int>(0, size - 1); - - auto* mortar = entities[selected]; - - Game::logger->Log("AgJetEffectServer", "Mortar (%i) (&d)", mortar->GetLOT(), mortar->HasComponent(COMPONENT_TYPE_SKILL)); - - mortar->SetOwnerOverride(builder); - - SkillComponent* skillComponent; - if (!mortar->TryGetComponent(COMPONENT_TYPE_SKILL, skillComponent)) { - return; - } - - skillComponent->CalculateBehavior(318, 3727, LWOOBJID_EMPTY, true); - - return; - } - - if (timerName == "CineDone") { - GameMessages::SendNotifyClientObject( - self->GetObjectID(), - u"toggleInUse", - -1, - 0, - LWOOBJID_EMPTY, - "", - UNASSIGNED_SYSTEM_ADDRESS - ); - - inUse = false; - } -} diff --git a/dScripts/AgLaserSensorServer.cpp b/dScripts/AgLaserSensorServer.cpp deleted file mode 100644 index 9efae1ca..00000000 --- a/dScripts/AgLaserSensorServer.cpp +++ /dev/null @@ -1,55 +0,0 @@ -#include "AgLaserSensorServer.h" - -#include "PhantomPhysicsComponent.h" -#include "SkillComponent.h" -#include "EntityManager.h" -#include "AgMonumentLaserServer.h" -#include "EntityManager.h" - -void AgLaserSensorServer::OnStartup(Entity* self) { - - PhantomPhysicsComponent* physComp = static_cast<PhantomPhysicsComponent*>(self->GetComponent(COMPONENT_TYPE_PHANTOM_PHYSICS)); - physComp->SetPhysicsEffectActive(true); - physComp->SetEffectType(2); // repulse (prolly should make definitions of these are in Entity.cpp) - physComp->SetDirectionalMultiplier(static_cast<float>(m_RepelForce)); - physComp->SetDirection(NiPoint3::UNIT_Y); - - m_Skill = self->GetComponent<SkillComponent>(); -} - - -void AgLaserSensorServer::OnCollisionPhantom(Entity* self, Entity* target) { - - if (!m_Skill) return; - - - Entity* laser = nullptr; - - for (auto script : EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_SCRIPT)) { - - AgMonumentLaserServer* hasLaser = (AgMonumentLaserServer*)script; - - if (hasLaser) { - const auto source = script->GetPosition(); - const auto obj = self->GetObjectID(); - - if (obj == 76690936093053 && Vector3::DistanceSquared(source, NiPoint3(149.007f, 417.083f, 218.346f)) <= 1.0f) { - laser = script; - break; - } else if (obj == 75866302318824 && Vector3::DistanceSquared(source, NiPoint3(48.6403f, 403.803f, 196.711f)) <= 1.0f) { - laser = script; - break; - } else if (obj == 75866302318822 && Vector3::DistanceSquared(source, NiPoint3(19.2155f, 420.083f, 249.226f)) <= 1.0f) { - laser = script; - break; - } else if (obj == 75866302318823 && Vector3::DistanceSquared(source, NiPoint3(-6.61596f, 404.633f, 274.323f)) <= 1.0f) { - laser = script; - break; - } - } - } - - if (laser != nullptr) { - m_Skill->CalculateBehavior(m_SkillCastID, 15714, target->GetObjectID()); - } -} diff --git a/dScripts/AgLaserSensorServer.h b/dScripts/AgLaserSensorServer.h deleted file mode 100644 index 72e09dd8..00000000 --- a/dScripts/AgLaserSensorServer.h +++ /dev/null @@ -1,15 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class SkillComponent; - -class AgLaserSensorServer : public CppScripts::Script { -public: - void OnStartup(Entity* self); - void OnCollisionPhantom(Entity* self, Entity* target); -private: - SkillComponent* m_Skill; - int m_RepelForce = -25; - int m_SkillCastID = 163; -}; - diff --git a/dScripts/AgMonumentBirds.cpp b/dScripts/AgMonumentBirds.cpp deleted file mode 100644 index ad3417a4..00000000 --- a/dScripts/AgMonumentBirds.cpp +++ /dev/null @@ -1,33 +0,0 @@ -#include "AgMonumentBirds.h" -#include "GameMessages.h" - -//-------------------------------------------------------------- -//Makes the ag birds fly away when you get close and smashes them. -//Created mrb... 6 / 3 / 11 -//Ported Max 20/07/2020 -//-------------------------------------------------------------- - -void AgMonumentBirds::OnStartup(Entity* self) { - self->SetProximityRadius(flyRadius, "MonumentBirds"); -} - -void AgMonumentBirds::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (self->GetVar<bool>(u"IsFlying")) return; - - if (name == "MonumentBirds" && status == "ENTER") { - self->AddTimer("killBird", 1.0f); - GameMessages::SendPlayAnimation(self, sOnProximityAnim); - self->SetVar<bool>(u"IsFlying", true); - self->SetVar<LWOOBJID>(u"PlayerID", entering->GetObjectID()); - } -} - -void AgMonumentBirds::OnTimerDone(Entity* self, std::string timerName) { - if (timerName != "killBird") return; - - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"PlayerID")); - - if (player == nullptr) return; - - self->ScheduleKillAfterUpdate(player); -} diff --git a/dScripts/AgMonumentLaserServer.cpp b/dScripts/AgMonumentLaserServer.cpp deleted file mode 100644 index 6efda89e..00000000 --- a/dScripts/AgMonumentLaserServer.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "AgMonumentLaserServer.h" - -void AgMonumentLaserServer::OnStartup(Entity* self) { - /* - self->SetProximityRadius(m_Radius, "MonumentLaser"); - - std::cout << "Monument Laser " << self->GetObjectID() << " is at " << self->GetPosition().GetX() - << ","<< self->GetPosition().GetY() << "," << self->GetPosition().GetZ() << std::endl; - */ -} - -void AgMonumentLaserServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - /* - if (status == "ENTER") { - - std::cout << "Monument laser ID: " << self->GetObjectID() << std::endl; - } - */ -} - diff --git a/dScripts/AgMonumentLaserServer.h b/dScripts/AgMonumentLaserServer.h deleted file mode 100644 index 56979c55..00000000 --- a/dScripts/AgMonumentLaserServer.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class AgMonumentLaserServer : public CppScripts::Script { -public: - void OnStartup(Entity* self); - void OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status); -private: - float m_Radius = 25.0f; -}; diff --git a/dScripts/AgMonumentRaceCancel.cpp b/dScripts/AgMonumentRaceCancel.cpp deleted file mode 100644 index 2e744434..00000000 --- a/dScripts/AgMonumentRaceCancel.cpp +++ /dev/null @@ -1,9 +0,0 @@ -#include "AgMonumentRaceCancel.h" -#include "EntityManager.h" - -void AgMonumentRaceCancel::OnCollisionPhantom(Entity* self, Entity* target) { - auto managers = EntityManager::Instance()->GetEntitiesInGroup("race_manager"); - if (!managers.empty()) { - managers[0]->OnFireEventServerSide(target, "course_cancel"); - } -} diff --git a/dScripts/AgMonumentRaceGoal.cpp b/dScripts/AgMonumentRaceGoal.cpp deleted file mode 100644 index 78bbaee5..00000000 --- a/dScripts/AgMonumentRaceGoal.cpp +++ /dev/null @@ -1,15 +0,0 @@ -#include "AgMonumentRaceGoal.h" -#include "EntityManager.h" - - -void AgMonumentRaceGoal::OnStartup(Entity* self) { - self->SetProximityRadius(15, "RaceGoal"); -} - -void AgMonumentRaceGoal::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name == "RaceGoal" && entering->IsPlayer() && status == "ENTER") { - auto* manager = EntityManager::Instance()->GetEntitiesInGroup("race_manager")[0]; - - manager->OnFireEventServerSide(entering, "course_finish"); - } -} diff --git a/dScripts/AgPicnicBlanket.cpp b/dScripts/AgPicnicBlanket.cpp deleted file mode 100644 index 30fb2950..00000000 --- a/dScripts/AgPicnicBlanket.cpp +++ /dev/null @@ -1,16 +0,0 @@ -#include "AgPicnicBlanket.h" -#include "GameMessages.h" - -void AgPicnicBlanket::OnUse(Entity* self, Entity* user) { - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); - if (self->GetVar<bool>(u"active")) - return; - self->SetVar<bool>(u"active", true); - - auto lootTable = std::unordered_map<LOT, int32_t>{ {935, 3} }; - LootGenerator::Instance().DropLoot(user, self, lootTable, 0, 0); - - self->AddCallbackTimer(5.0f, [self]() { - self->SetVar<bool>(u"active", false); - }); -} diff --git a/dScripts/AgPropGuard.cpp b/dScripts/AgPropGuard.cpp deleted file mode 100644 index 51f76a27..00000000 --- a/dScripts/AgPropGuard.cpp +++ /dev/null @@ -1,39 +0,0 @@ -#include "AgPropGuard.h" -#include "Entity.h" -#include "Character.h" -#include "EntityManager.h" -#include "InventoryComponent.h" -#include "MissionComponent.h" -#include "Item.h" - -void AgPropGuard::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - auto* character = target->GetCharacter(); - auto* missionComponent = target->GetComponent<MissionComponent>(); - auto* inventoryComponent = target->GetComponent<InventoryComponent>(); - - const auto state = missionComponent->GetMissionState(320); - if (missionID == 768 && missionState == MissionState::MISSION_STATE_AVAILABLE) { - if (!character->GetPlayerFlag(71)) { - // TODO: Cinematic "MissionCam" - } - } else if (missionID == 768 && missionState >= MissionState::MISSION_STATE_READY_TO_COMPLETE) { - //remove the inventory items - for (int item : gearSets) { - auto* id = inventoryComponent->FindItemByLot(item); - - if (id) { - inventoryComponent->UnEquipItem(id); - inventoryComponent->RemoveItem(id->GetLot(), id->GetCount()); - } - } - } else if ( - (missionID == 320 && state == MissionState::MISSION_STATE_AVAILABLE) /*|| - (state == MissionState::MISSION_STATE_COMPLETE && missionID == 891 && missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE)*/ - ) { - //GameMessages::SendNotifyClientObject(EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(), u"GuardChat", target->GetObjectID(), 0, target->GetObjectID(), "", target->GetSystemAddress()); - - target->GetCharacter()->SetPlayerFlag(113, true); - - EntityManager::Instance()->GetZoneControlEntity()->AddTimer("GuardFlyAway", 1.0f); - } -} diff --git a/dScripts/AgPropGuard.h b/dScripts/AgPropGuard.h deleted file mode 100644 index f68573dd..00000000 --- a/dScripts/AgPropGuard.h +++ /dev/null @@ -1,11 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class AgPropGuard final : public CppScripts::Script -{ -public: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - -private: - std::vector<int> gearSets = { 14359,14321,14353,14315 }; -}; diff --git a/dScripts/AgPropguards.cpp b/dScripts/AgPropguards.cpp deleted file mode 100644 index 674c4bdd..00000000 --- a/dScripts/AgPropguards.cpp +++ /dev/null @@ -1,56 +0,0 @@ -#include "AgPropguards.h" -#include "Character.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "dZoneManager.h" - -void AgPropguards::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - auto* character = target->GetCharacter(); - if (character == nullptr) - return; - - const auto flag = GetFlagForMission(missionID); - if (flag == 0) - return; - - if ((missionState == MissionState::MISSION_STATE_AVAILABLE || missionState == MissionState::MISSION_STATE_ACTIVE) - && !character->GetPlayerFlag(flag)) { - // If the player just started the mission, play a cinematic highlighting the target - GameMessages::SendPlayCinematic(target->GetObjectID(), u"MissionCam", target->GetSystemAddress()); - } else if (missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { - // Makes the guard disappear once the mission has been completed - const auto zoneControlID = EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(); - GameMessages::SendNotifyClientObject(zoneControlID, u"GuardChat", 0, 0, self->GetObjectID(), - "", UNASSIGNED_SYSTEM_ADDRESS); - - self->AddCallbackTimer(5.0f, [self]() { - auto spawnerName = self->GetVar<std::string>(u"spawner_name"); - if (spawnerName.empty()) - spawnerName = "Guard"; - - auto spawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName); - for (auto* spawner : spawners) { - spawner->Deactivate(); - } - - self->Smash(); - }); - } -} - -uint32_t AgPropguards::GetFlagForMission(uint32_t missionID) { - switch (missionID) { - case 872: - return 97; - case 873: - return 98; - case 874: - return 99; - case 1293: - return 118; - case 1322: - return 122; - default: - return 0; - } -} diff --git a/dScripts/AgPropguards.h b/dScripts/AgPropguards.h deleted file mode 100644 index 511b7b6a..00000000 --- a/dScripts/AgPropguards.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class AgPropguards : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -private: - static uint32_t GetFlagForMission(uint32_t missionID); -}; diff --git a/dScripts/AgQbElevator.cpp b/dScripts/AgQbElevator.cpp deleted file mode 100644 index ccb4d3b1..00000000 --- a/dScripts/AgQbElevator.cpp +++ /dev/null @@ -1,60 +0,0 @@ -#include "AgQbElevator.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void AgQbElevator::OnStartup(Entity* self) { - -} - -//when the QB is finished being built by a player -void AgQbElevator::OnRebuildComplete(Entity* self, Entity* target) { - self->SetProximityRadius(proxRadius, "elevatorProx"); - self->SetI64(u"qbPlayer", target->GetObjectID()); - - float delayTime = killTime - endTime; - if (delayTime < 1) delayTime = 1; - - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, - 0, 0, MovementPlatformState::Stationary); - - //add a timer that will kill the QB if no players get on in the killTime - self->AddTimer("startKillTimer", killTime); -} - -void AgQbElevator::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - //make sure we haven't already started pathing. - if (self->GetBoolean(u"qbPlayerRdy")) return; - - if (status == "ENTER") { - Entity* builder = EntityManager::Instance()->GetEntity(self->GetI64(u"qbPlayer")); - if (builder && builder == entering) { - //the builder has entered so cancel the start timer and just start moving - self->SetBoolean(u"qbPlayerRdy", true); - self->CancelTimer("StartElevator"); - - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, - 1, 1, MovementPlatformState::Moving); - } else if (!self->GetBoolean(u"StartTimer")) { - self->SetBoolean(u"StartTimer", true); - self->AddTimer("StartElevator", startTime); - } - } -} - -void AgQbElevator::OnTimerDone(Entity* self, std::string timerName) { - - if (timerName == "StartElevator") { - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, - 1, 1, MovementPlatformState::Moving); - } else if (timerName == "startKillTimer") { - killTimerStartup(self); - } else if (timerName == "KillTimer") { - self->Smash(self->GetObjectID(), VIOLENT); - } -} - -void AgQbElevator::killTimerStartup(Entity* self) const { - self->CancelAllTimers(); - self->AddTimer("KillTimer", endTime); - self->SetNetworkVar<float>(u"startEffect", endTime); // Blinking effect -} diff --git a/dScripts/AgQbWall.cpp b/dScripts/AgQbWall.cpp deleted file mode 100644 index d6222419..00000000 --- a/dScripts/AgQbWall.cpp +++ /dev/null @@ -1,15 +0,0 @@ -#include "AgQbWall.h" - -void AgQbWall::OnRebuildComplete(Entity* self, Entity* player) { - self->SetVar(u"player", player->GetObjectID()); - auto targetWallSpawners = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner")); - if (targetWallSpawners != "") { - auto groupObjs = EntityManager::Instance()->GetEntitiesInGroup(targetWallSpawners); - for (auto* obj : groupObjs) { - if (obj) { - obj->SetVar(u"player", player->GetObjectID()); - obj->OnFireEventServerSide(self, "spawnMobs"); - } - } - } -} diff --git a/dScripts/AgSalutingNpcs.cpp b/dScripts/AgSalutingNpcs.cpp deleted file mode 100644 index 4e4d8b2c..00000000 --- a/dScripts/AgSalutingNpcs.cpp +++ /dev/null @@ -1,11 +0,0 @@ -#include "AgSalutingNpcs.h" -#include "GameMessages.h" - - -void AgSalutingNpcs::OnEmoteReceived(Entity* self, const int32_t emote, Entity* target) { - if (emote != 356) { - return; - } - - GameMessages::SendPlayAnimation(self, u"salutePlayer"); -} diff --git a/dScripts/AgShipPlayerShockServer.cpp b/dScripts/AgShipPlayerShockServer.cpp deleted file mode 100644 index 2bed8152..00000000 --- a/dScripts/AgShipPlayerShockServer.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "AgShipPlayerShockServer.h" -#include "GameMessages.h" - -void AgShipPlayerShockServer::OnUse(Entity* self, Entity* user) { - GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); - if (active) { - return; - } - active = true; - GameMessages::SendPlayAnimation(user, shockAnim); - GameMessages::SendKnockback(user->GetObjectID(), self->GetObjectID(), self->GetObjectID(), 0, NiPoint3(-20, 10, -20)); - - GameMessages::SendPlayFXEffect(self, 1430, u"create", "console_sparks", LWOOBJID_EMPTY, 1.0, 1.0, true); - self->AddTimer("FXTime", fxTime); -} - -void AgShipPlayerShockServer::OnTimerDone(Entity* self, std::string timerName) { - GameMessages::SendStopFXEffect(self, true, "console_sparks"); - active = false; -} diff --git a/dScripts/AgSpaceStuff.cpp b/dScripts/AgSpaceStuff.cpp deleted file mode 100644 index 80a87e70..00000000 --- a/dScripts/AgSpaceStuff.cpp +++ /dev/null @@ -1,103 +0,0 @@ -#include "AgSpaceStuff.h" -#include "GeneralUtils.h" -#include "GameMessages.h" -#include "EntityManager.h" - -void AgSpaceStuff::OnStartup(Entity* self) { - self->AddTimer("FloaterScale", 5.0f); - - EntityInfo info{}; - - info.pos = { -418, 585, -30 }; - info.lot = 33; - info.spawnerID = self->GetObjectID(); - - auto* ref = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(ref); - - self->SetVar(u"ShakeObject", ref->GetObjectID()); - - self->AddTimer("ShipShakeIdle", 2.0f); - self->SetVar(u"RandomTime", 10); -} - -void AgSpaceStuff::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "FloaterScale") { - int scaleType = GeneralUtils::GenerateRandomNumber<int>(1, 5); - - GameMessages::SendPlayAnimation(self, u"scale_0" + GeneralUtils::to_u16string(scaleType)); - self->AddTimer("FloaterPath", 0.4); - } else if (timerName == "FloaterPath") { - int pathType = GeneralUtils::GenerateRandomNumber<int>(1, 4); - int randTime = GeneralUtils::GenerateRandomNumber<int>(20, 25); - - GameMessages::SendPlayAnimation(self, u"path_0" + (GeneralUtils::to_u16string(pathType))); - self->AddTimer("FloaterScale", randTime); - } else if (timerName == "ShipShakeExplode") { - DoShake(self, true); - } else if (timerName == "ShipShakeIdle") { - DoShake(self, false); - } -} - -void AgSpaceStuff::DoShake(Entity* self, bool explodeIdle) { - - if (!explodeIdle) { - auto* ref = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"ShakeObject")); - - const auto randomTime = self->GetVar<int>(u"RandomTime"); - auto time = GeneralUtils::GenerateRandomNumber<int>(0, randomTime + 1); - - if (time < randomTime / 2) { - time += randomTime / 2; - } - - self->AddTimer("ShipShakeIdle", static_cast<float>(time)); - - if (ref) - GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(ref, FXName, ref->GetObjectID(), 500.0f); - - auto* debrisObject = GetEntityInGroup(DebrisFX); - - if (debrisObject) - GameMessages::SendPlayFXEffect(debrisObject, -1, u"DebrisFall", "Debris", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - - const auto randomFx = GeneralUtils::GenerateRandomNumber<int>(0, 3); - - auto* shipFxObject = GetEntityInGroup(ShipFX); - if (shipFxObject) { - std::string effectType = "shipboom" + std::to_string(randomFx); - GameMessages::SendPlayFXEffect(shipFxObject, 559, GeneralUtils::ASCIIToUTF16(effectType), "FX", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - } - - self->AddTimer("ShipShakeExplode", 5.0f); - - auto* shipFxObject2 = GetEntityInGroup(ShipFX2); - if (shipFxObject2) - GameMessages::SendPlayAnimation(shipFxObject2, u"explosion"); - } else { - auto* shipFxObject = GetEntityInGroup(ShipFX); - auto* shipFxObject2 = GetEntityInGroup(ShipFX2); - - if (shipFxObject) - GameMessages::SendPlayAnimation(shipFxObject, u"idle"); - - if (shipFxObject2) - GameMessages::SendPlayAnimation(shipFxObject2, u"idle"); - } -} - -Entity* AgSpaceStuff::GetEntityInGroup(const std::string& group) { - auto entities = EntityManager::Instance()->GetEntitiesInGroup(group); - Entity* en = nullptr; - - for (auto entity : entities) { - if (entity) { - en = entity; - break; - } - } - - return en; -} diff --git a/dScripts/AgStromlingProperty.cpp b/dScripts/AgStromlingProperty.cpp deleted file mode 100644 index 36d8d378..00000000 --- a/dScripts/AgStromlingProperty.cpp +++ /dev/null @@ -1,16 +0,0 @@ -#include "AgStromlingProperty.h" -#include "MovementAIComponent.h" - -void AgStromlingProperty::OnStartup(Entity* self) { - auto movementInfo = MovementAIInfo{ - "Wander", - 71, - 3, - 100, - 1, - 4 - }; - - auto* movementAIComponent = new MovementAIComponent(self, movementInfo); - self->AddComponent(COMPONENT_TYPE_MOVEMENT_AI, movementAIComponent); -} diff --git a/dScripts/AgSurvivalBuffStation.cpp b/dScripts/AgSurvivalBuffStation.cpp deleted file mode 100644 index 4d3482c4..00000000 --- a/dScripts/AgSurvivalBuffStation.cpp +++ /dev/null @@ -1,65 +0,0 @@ -#include "AgSurvivalBuffStation.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "SkillComponent.h" -#include "TeamManager.h" - -void AgSurvivalBuffStation::OnRebuildComplete(Entity* self, Entity* target) { - auto destroyableComponent = self->GetComponent<DestroyableComponent>(); - // We set the faction to 1 so that the buff station sees players as friendly targets to buff - if (destroyableComponent != nullptr) destroyableComponent->SetFaction(1); - - auto skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) skillComponent->CalculateBehavior(skillIdForBuffStation, behaviorIdForBuffStation, self->GetObjectID()); - - self->AddCallbackTimer(smashTimer, [self]() { - self->Smash(); - }); - self->AddTimer("DropArmor", dropArmorTimer); - self->AddTimer("DropLife", dropLifeTimer); - self->AddTimer("Dropimagination", dropImaginationTimer); - // Since all survival players should be on the same team, we get the team. - auto team = TeamManager::Instance()->GetTeam(target->GetObjectID()); - - std::vector<LWOOBJID> builderTeam; - // Not on a team - if (team == nullptr) { - builderTeam.push_back(target->GetObjectID()); - self->SetVar<std::vector<LWOOBJID>>(u"BuilderTeam", builderTeam); - return; - } - - for (auto memberID : team->members) { - builderTeam.push_back(memberID); - } - self->SetVar<std::vector<LWOOBJID>>(u"BuilderTeam", builderTeam); -} - -void AgSurvivalBuffStation::OnTimerDone(Entity* self, std::string timerName) { - uint32_t powerupToDrop = lifePowerup; - if (timerName == "DropArmor") { - powerupToDrop = armorPowerup; - self->AddTimer("DropArmor", dropArmorTimer); - } - if (timerName == "DropLife") { - powerupToDrop = lifePowerup; - self->AddTimer("DropLife", dropLifeTimer); - } - if (timerName == "Dropimagination") { - powerupToDrop = imaginationPowerup; - self->AddTimer("Dropimagination", dropImaginationTimer); - } - auto team = self->GetVar<std::vector<LWOOBJID>>(u"BuilderTeam"); - for (auto memberID : team) { - auto member = EntityManager::Instance()->GetEntity(memberID); - if (member != nullptr && !member->GetIsDead()) { - GameMessages::SendDropClientLoot(member, self->GetObjectID(), powerupToDrop, 0, self->GetPosition()); - } else { - // If player left the team or left early erase them from the team variable. - team.erase(std::find(team.begin(), team.end(), memberID)); - self->SetVar<std::vector<LWOOBJID>>(u"BuilderTeam", team); - } - } -} diff --git a/dScripts/AmBlueX.cpp b/dScripts/AmBlueX.cpp deleted file mode 100644 index 97f80e77..00000000 --- a/dScripts/AmBlueX.cpp +++ /dev/null @@ -1,55 +0,0 @@ -#include "AmBlueX.h" -#include "SkillComponent.h" -#include "EntityManager.h" -#include "Character.h" - -void AmBlueX::OnUse(Entity* self, Entity* user) { - auto* skillComponent = user->GetComponent<SkillComponent>(); - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(m_SwordSkill, m_SwordBehavior, self->GetObjectID()); - } -} - -void AmBlueX::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (message == "FireDukesStrike") { - self->SetNetworkVar<bool>(m_XUsedVariable, true); - self->SetNetworkVar<bool>(m_StartEffectVariable, true); - - auto* character = caster->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(self->GetVar<int32_t>(m_FlagVariable), true); - } - - EntityInfo info{}; - info.lot = m_FXObject; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.spawnerID = self->GetObjectID(); - - auto* fxObject = EntityManager::Instance()->CreateEntity(info, nullptr, self); - EntityManager::Instance()->ConstructEntity(fxObject); - - auto fxObjectID = fxObject->GetObjectID(); - auto playerID = caster->GetObjectID(); - - // Add a callback for the bomb to explode - self->AddCallbackTimer(m_BombTime, [this, self, fxObjectID, playerID]() { - auto* fxObject = EntityManager::Instance()->GetEntity(fxObjectID); - auto* player = EntityManager::Instance()->GetEntity(playerID); - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) - return; - - // Cast the skill that destroys the object - if (player != nullptr) { - skillComponent->CalculateBehavior(m_AOESkill, m_AOEBehavior, LWOOBJID_EMPTY, false, false, playerID); - } else { - skillComponent->CalculateBehavior(m_AOESkill, m_AOEBehavior, LWOOBJID_EMPTY); - } - - fxObject->Smash(); - self->Smash(); - }); - } -} diff --git a/dScripts/AmBridge.cpp b/dScripts/AmBridge.cpp deleted file mode 100644 index 88f30642..00000000 --- a/dScripts/AmBridge.cpp +++ /dev/null @@ -1,28 +0,0 @@ -#include "AmBridge.h" -#include "EntityManager.h" - -void AmBridge::OnStartup(Entity* self) { - -} - -void AmBridge::OnRebuildComplete(Entity* self, Entity* target) { - const auto consoles = EntityManager::Instance()->GetEntitiesInGroup("Console" + GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"bridge"))); - - if (consoles.empty()) { - return; - } - - auto* console = consoles[0]; - - console->NotifyObject(self, "BridgeBuilt"); - - self->AddTimer("SmashBridge", 50); -} - -void AmBridge::OnTimerDone(Entity* self, std::string timerName) { - if (timerName != "SmashBridge") { - return; - } - - self->Smash(self->GetObjectID(), VIOLENT); -} diff --git a/dScripts/AmConsoleTeleportServer.cpp b/dScripts/AmConsoleTeleportServer.cpp deleted file mode 100644 index f3931e0d..00000000 --- a/dScripts/AmConsoleTeleportServer.cpp +++ /dev/null @@ -1,30 +0,0 @@ -#include "AmConsoleTeleportServer.h" -#include "ChooseYourDestinationNsToNt.h" -#include "AMFFormat.h" - -void AmConsoleTeleportServer::OnStartup(Entity* self) { - self->SetVar(u"teleportAnim", m_TeleportAnim); - self->SetVar(u"teleportString", m_TeleportString); - self->SetVar(u"teleportEffectID", m_TeleportEffectID); - self->SetVar(u"teleportEffectTypes", m_TeleportEffectTypes); -} - -void AmConsoleTeleportServer::OnUse(Entity* self, Entity* user) { - BaseOnUse(self, user); -} - -void AmConsoleTeleportServer::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - BaseOnMessageBoxResponse(self, sender, button, identifier, userData); -} - -void AmConsoleTeleportServer::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { - -} - -void AmConsoleTeleportServer::OnTimerDone(Entity* self, std::string timerName) { - BaseOnTimerDone(self, timerName); -} - -void AmConsoleTeleportServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); -} diff --git a/dScripts/AmDarklingDragon.cpp b/dScripts/AmDarklingDragon.cpp deleted file mode 100644 index 8ec49d3e..00000000 --- a/dScripts/AmDarklingDragon.cpp +++ /dev/null @@ -1,152 +0,0 @@ -#include "AmDarklingDragon.h" -#include "BaseCombatAIComponent.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "SkillComponent.h" -#include "BaseCombatAIComponent.h" - -void AmDarklingDragon::OnStartup(Entity* self) { - self->SetVar<int32_t>(u"weakspot", 0); - - auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - - if (baseCombatAIComponent != nullptr) { - baseCombatAIComponent->SetStunImmune(true); - } -} - -void AmDarklingDragon::OnDie(Entity* self, Entity* killer) { - if (self->GetVar<bool>(u"bDied")) { - return; - } - - self->SetVar<bool>(u"bDied", true); - - auto golemId = self->GetVar<LWOOBJID>(u"Golem"); - - auto* golem = EntityManager::Instance()->GetEntity(golemId); - - if (golem != nullptr) { - golem->Smash(self->GetObjectID()); - } -} - -void AmDarklingDragon::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - GameMessages::SendPlayFXEffect(self, -1, u"gothit", "", LWOOBJID_EMPTY, 1, 1, true); - - if (true) { - auto weakpoint = self->GetVar<int32_t>(u"weakspot"); - - if (weakpoint == 1) { - self->Smash(attacker->GetObjectID()); - } - } - - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - - if (destroyableComponent != nullptr) { - if (destroyableComponent->GetArmor() > 0) return; - - auto weakpoint = self->GetVar<int32_t>(u"weakpoint"); - - if (weakpoint == 0) { - self->AddTimer("ReviveTimer", 12); - - auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (baseCombatAIComponent != nullptr) { - baseCombatAIComponent->SetDisabled(true); - baseCombatAIComponent->SetStunned(true); - } - - if (skillComponent != nullptr) { - skillComponent->Interrupt(); - } - - self->SetVar<int32_t>(u"weakpoint", 2); - - GameMessages::SendPlayAnimation(self, u"stunstart", 1.7f); - - self->AddTimer("timeToStunLoop", 1); - - auto position = self->GetPosition(); - auto forward = self->GetRotation().GetForwardVector(); - auto backwards = forward * -1; - - forward.x *= 10; - forward.z *= 10; - - auto rotation = self->GetRotation(); - - auto objectPosition = NiPoint3(); - - objectPosition.y = position.y; - objectPosition.x = position.x - (backwards.x * 8); - objectPosition.z = position.z - (backwards.z * 8); - - auto golem = self->GetVar<int32_t>(u"DragonSmashingGolem"); - - EntityInfo info{}; - info.lot = golem != 0 ? golem : 8340; - info.pos = objectPosition; - info.rot = rotation; - info.spawnerID = self->GetObjectID(); - info.settings = { - new LDFData<std::string>(u"rebuild_activators", - std::to_string(objectPosition.x + forward.x) + "\x1f" + - std::to_string(objectPosition.y) + "\x1f" + - std::to_string(objectPosition.z + forward.z) - ), - new LDFData<int32_t>(u"respawn", 100000), - new LDFData<float>(u"rebuild_reset_time", 15), - new LDFData<bool>(u"no_timed_spawn", true), - new LDFData<LWOOBJID>(u"Dragon", self->GetObjectID()) - }; - - auto* golemObject = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(golemObject); - } - } -} - -void AmDarklingDragon::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "ReviveHeldTimer") { - self->AddTimer("backToAttack", 2.5); - } else if (timerName == "ExposeWeakSpotTimer") { - self->SetVar<int32_t>(u"weakspot", 1); - } else if (timerName == "timeToStunLoop") { - GameMessages::SendPlayAnimation(self, u"stunloop", 1.8f); - } else if (timerName == "ReviveTimer") { - GameMessages::SendPlayAnimation(self, u"stunend", 2.0f); - self->AddTimer("backToAttack", 1); - } else if (timerName == "backToAttack") { - auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - auto* skillComponent = self->GetComponent<SkillComponent>(); - if (baseCombatAIComponent != nullptr) { - baseCombatAIComponent->SetDisabled(false); - baseCombatAIComponent->SetStunned(false); - } - if (skillComponent != nullptr) { - skillComponent->Interrupt(); - } - self->SetVar<int32_t>(u"weakspot", -1); - GameMessages::SendNotifyObject(self->GetObjectID(), self->GetObjectID(), u"DragonRevive", UNASSIGNED_SYSTEM_ADDRESS); - } -} - -void AmDarklingDragon::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - if (args != "rebuildDone") return; - - self->AddTimer("ExposeWeakSpotTimer", 3.8f); - - self->CancelTimer("ReviveTimer"); - - self->AddTimer("ReviveHeldTimer", 10.5f); - - self->SetVar<LWOOBJID>(u"Golem", sender->GetObjectID()); - - GameMessages::SendPlayAnimation(self, u"quickbuildhold", 1.9f); -} diff --git a/dScripts/AmDrawBridge.cpp b/dScripts/AmDrawBridge.cpp deleted file mode 100644 index 20636b13..00000000 --- a/dScripts/AmDrawBridge.cpp +++ /dev/null @@ -1,121 +0,0 @@ -#include "AmDrawBridge.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "SimplePhysicsComponent.h" - -void AmDrawBridge::OnStartup(Entity* self) { - self->SetNetworkVar(u"InUse", false); - self->SetVar(u"BridgeDown", false); -} - -void AmDrawBridge::OnUse(Entity* self, Entity* user) { - auto* bridge = GetBridge(self); - - if (bridge == nullptr) { - return; - } - - if (!self->GetNetworkVar<bool>(u"InUse")) { - self->SetNetworkVar(u"startEffect", 5); - - self->AddTimer("ChangeBridge", 5); - - self->SetNetworkVar(u"InUse", true); - } - - auto* player = user; - - GameMessages::SendTerminateInteraction(player->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} - -void AmDrawBridge::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "ChangeBridge") { - auto* bridge = GetBridge(self); - - if (bridge == nullptr) { - return; - } - - if (!self->GetVar<bool>(u"BridgeDown")) { - self->SetVar(u"BridgeDown", true); - - MoveBridgeDown(self, bridge, true); - } else { - self->SetVar(u"BridgeDown", false); - - MoveBridgeDown(self, bridge, false); - } - - self->SetNetworkVar(u"BridgeLeaving", true); - self->SetVar(u"BridgeDown", false); - } else if (timerName == "SmashEffectBridge") { - self->SetNetworkVar(u"SmashBridge", 5); - } else if (timerName == "rotateBridgeDown") { - auto* bridge = GetBridge(self); - - if (bridge == nullptr) { - return; - } - - self->SetNetworkVar(u"BridgeLeaving", false); - - auto* simplePhysicsComponent = bridge->GetComponent<SimplePhysicsComponent>(); - - if (simplePhysicsComponent == nullptr) { - return; - } - - simplePhysicsComponent->SetAngularVelocity(NiPoint3::ZERO); - - EntityManager::Instance()->SerializeEntity(bridge); - } -} - -void AmDrawBridge::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { - if (name == "BridgeBuilt") { - self->SetVar(u"BridgeID", sender->GetObjectID()); - - self->AddTimer("SmashEffectBridge", 45); - - self->SetNetworkVar(u"BridgeDead", true); - - sender->AddDieCallback([this, self, sender]() { - NotifyDie(self, sender); - }); - } -} - -void AmDrawBridge::MoveBridgeDown(Entity* self, Entity* bridge, bool down) { - auto* simplePhysicsComponent = bridge->GetComponent<SimplePhysicsComponent>(); - - if (simplePhysicsComponent == nullptr) { - return; - } - - auto forwardVect = simplePhysicsComponent->GetRotation().GetForwardVector(); - - auto degrees = down ? 90.0f : -90.0f; - - const auto travelTime = 2.0f; - - forwardVect = forwardVect * (float)((degrees / travelTime) * (3.14f / 180.0f)); - - simplePhysicsComponent->SetAngularVelocity(forwardVect); - - EntityManager::Instance()->SerializeEntity(bridge); - - self->AddTimer("rotateBridgeDown", travelTime); -} - -void AmDrawBridge::NotifyDie(Entity* self, Entity* other) { - self->SetNetworkVar(u"InUse", false); - self->SetVar(u"BridgeDown", false); - - self->CancelAllTimers(); -} - -Entity* AmDrawBridge::GetBridge(Entity* self) { - const auto bridgeID = self->GetVar<LWOOBJID>(u"BridgeID"); - - return EntityManager::Instance()->GetEntity(bridgeID); -} diff --git a/dScripts/AmDropshipComputer.cpp b/dScripts/AmDropshipComputer.cpp deleted file mode 100644 index 909beaaf..00000000 --- a/dScripts/AmDropshipComputer.cpp +++ /dev/null @@ -1,81 +0,0 @@ -#include "AmDropshipComputer.h" -#include "MissionComponent.h" -#include "RebuildComponent.h" -#include "InventoryComponent.h" -#include "dZoneManager.h" - -void AmDropshipComputer::OnStartup(Entity* self) { - self->AddTimer("reset", 45.0f); -} - -void AmDropshipComputer::OnUse(Entity* self, Entity* user) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent == nullptr || rebuildComponent->GetState() != REBUILD_COMPLETED) { - return; - } - - auto* missionComponent = user->GetComponent<MissionComponent>(); - auto* inventoryComponent = user->GetComponent<InventoryComponent>(); - - if (missionComponent == nullptr || inventoryComponent == nullptr) { - return; - } - - if (inventoryComponent->GetLotCount(m_NexusTalonDataCard) != 0 || missionComponent->GetMission(979)->GetMissionState() == MissionState::MISSION_STATE_COMPLETE) { - return; - } - - inventoryComponent->AddItem(m_NexusTalonDataCard, 1, eLootSourceType::LOOT_SOURCE_NONE); -} - -void AmDropshipComputer::OnDie(Entity* self, Entity* killer) { - const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); - - int32_t pipeNum = 0; - if (!GeneralUtils::TryParse<int32_t>(myGroup.substr(10, 1), pipeNum)) { - return; - } - - const auto pipeGroup = myGroup.substr(0, 10); - - const auto nextPipeNum = pipeNum + 1; - - const auto samePipeSpawners = dZoneManager::Instance()->GetSpawnersByName(myGroup); - - if (!samePipeSpawners.empty()) { - samePipeSpawners[0]->SoftReset(); - - samePipeSpawners[0]->Deactivate(); - } - - if (killer != nullptr && killer->IsPlayer()) { - const auto nextPipe = pipeGroup + std::to_string(nextPipeNum); - - const auto nextPipeSpawners = dZoneManager::Instance()->GetSpawnersByName(nextPipe); - - if (!nextPipeSpawners.empty()) { - nextPipeSpawners[0]->Activate(); - } - } else { - const auto nextPipe = pipeGroup + "1"; - - const auto firstPipeSpawners = dZoneManager::Instance()->GetSpawnersByName(nextPipe); - - if (!firstPipeSpawners.empty()) { - firstPipeSpawners[0]->Activate(); - } - } -} - -void AmDropshipComputer::OnTimerDone(Entity* self, std::string timerName) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent == nullptr) { - return; - } - - if (timerName == "reset" && rebuildComponent->GetState() == REBUILD_OPEN) { - self->Smash(self->GetObjectID(), SILENT); - } -} diff --git a/dScripts/AmShieldGenerator.cpp b/dScripts/AmShieldGenerator.cpp deleted file mode 100644 index d16acb87..00000000 --- a/dScripts/AmShieldGenerator.cpp +++ /dev/null @@ -1,144 +0,0 @@ -#include "AmShieldGenerator.h" -#include "EntityManager.h" -#include "DestroyableComponent.h" -#include "GameMessages.h" -#include "MovementAIComponent.h" -#include "BaseCombatAIComponent.h" -#include "SkillComponent.h" - -void AmShieldGenerator::OnStartup(Entity* self) { - self->SetProximityRadius(20, "shield"); - self->SetProximityRadius(21, "buffer"); - - StartShield(self); -} - -void AmShieldGenerator::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - auto* destroyableComponent = entering->GetComponent<DestroyableComponent>(); - - if (status == "ENTER" && name == "shield") { - if (destroyableComponent->HasFaction(4)) { - EnemyEnteredShield(self, entering); - } - } - - if (name != "buffer" || !entering->IsPlayer()) { - return; - } - - auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - if (status == "ENTER") { - const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); - - if (iter == entitiesInProximity.end()) { - entitiesInProximity.push_back(entering->GetObjectID()); - } - } else if (status == "LEAVE") { - const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); - - if (iter != entitiesInProximity.end()) { - entitiesInProximity.erase(iter); - } - } - - self->SetVar<std::vector<LWOOBJID>>(u"Players", entitiesInProximity); -} - -void AmShieldGenerator::OnDie(Entity* self, Entity* killer) { - self->CancelAllTimers(); - - auto* child = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Child")); - - if (child != nullptr) { - child->Kill(); - } -} - -void AmShieldGenerator::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "BuffPlayers") { - BuffPlayers(self); - - self->AddTimer("BuffPlayers", 3.0f); - } else if (timerName == "PlayFX") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 5351, u"generatorOn", "generatorOn"); - - self->AddTimer("PlayFX", 1.5f); - } else if (timerName == "RefreshEnemies") { - auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); - - for (const auto enemyID : enemiesInProximity) { - auto* enemy = EntityManager::Instance()->GetEntity(enemyID); - - if (enemy != nullptr) { - EnemyEnteredShield(self, enemy); - } - } - - self->AddTimer("RefreshEnemies", 1.5f); - } -} - -void AmShieldGenerator::StartShield(Entity* self) { - self->AddTimer("PlayFX", 1.5f); - self->AddTimer("BuffPlayers", 3.0f); - self->AddTimer("RefreshEnemies", 1.5f); - - const auto myPos = self->GetPosition(); - const auto myRot = self->GetRotation(); - - EntityInfo info{}; - info.lot = 13111; - info.pos = myPos; - info.rot = myRot; - info.spawnerID = self->GetObjectID(); - - auto* child = EntityManager::Instance()->CreateEntity(info); - - self->SetVar(u"Child", child->GetObjectID()); - - BuffPlayers(self); -} - -void AmShieldGenerator::BuffPlayers(Entity* self) { - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - for (const auto playerID : entitiesInProximity) { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - skillComponent->CalculateBehavior(1200, 27024, playerID, true); - } -} - -void AmShieldGenerator::EnemyEnteredShield(Entity* self, Entity* intruder) { - auto* baseCombatAIComponent = intruder->GetComponent<BaseCombatAIComponent>(); - auto* movementAIComponent = intruder->GetComponent<MovementAIComponent>(); - - if (baseCombatAIComponent == nullptr || movementAIComponent == nullptr) { - return; - } - - auto dir = intruder->GetRotation().GetForwardVector() * -1; - dir.y += 15; - dir.x *= 50; - dir.z *= 50; - - // TODO: Figure out how todo knockback, I'll stun them for now - - if (NiPoint3::DistanceSquared(self->GetPosition(), movementAIComponent->GetCurrentPosition()) < 20 * 20) { - baseCombatAIComponent->Stun(2.0f); - movementAIComponent->SetDestination(baseCombatAIComponent->GetStartPosition()); - } - - baseCombatAIComponent->ClearThreat(); -} diff --git a/dScripts/AmShieldGeneratorQuickbuild.cpp b/dScripts/AmShieldGeneratorQuickbuild.cpp deleted file mode 100644 index 9f27b904..00000000 --- a/dScripts/AmShieldGeneratorQuickbuild.cpp +++ /dev/null @@ -1,202 +0,0 @@ -#include "AmShieldGeneratorQuickbuild.h" -#include "EntityManager.h" -#include "DestroyableComponent.h" -#include "GameMessages.h" -#include "MovementAIComponent.h" -#include "BaseCombatAIComponent.h" -#include "SkillComponent.h" -#include "RebuildComponent.h" -#include "MissionComponent.h" - -void AmShieldGeneratorQuickbuild::OnStartup(Entity* self) { - self->SetProximityRadius(20, "shield"); - self->SetProximityRadius(21, "buffer"); -} - -void AmShieldGeneratorQuickbuild::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - auto* destroyableComponent = entering->GetComponent<DestroyableComponent>(); - - if (name == "shield") { - if (!destroyableComponent->HasFaction(4) || entering->IsPlayer()) { - return; - } - - auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); - - if (status == "ENTER") { - EnemyEnteredShield(self, entering); - - const auto& iter = std::find(enemiesInProximity.begin(), enemiesInProximity.end(), entering->GetObjectID()); - - if (iter == enemiesInProximity.end()) { - enemiesInProximity.push_back(entering->GetObjectID()); - } - } else if (status == "LEAVE") { - const auto& iter = std::find(enemiesInProximity.begin(), enemiesInProximity.end(), entering->GetObjectID()); - - if (iter != enemiesInProximity.end()) { - enemiesInProximity.erase(iter); - } - } - - self->SetVar<std::vector<LWOOBJID>>(u"Enemies", enemiesInProximity); - } - - if (name != "buffer" || !entering->IsPlayer()) { - return; - } - - auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - if (status == "ENTER") { - const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); - - if (iter == entitiesInProximity.end()) { - entitiesInProximity.push_back(entering->GetObjectID()); - } - } else if (status == "LEAVE") { - const auto& iter = std::find(entitiesInProximity.begin(), entitiesInProximity.end(), entering->GetObjectID()); - - if (iter != entitiesInProximity.end()) { - entitiesInProximity.erase(iter); - } - } - - self->SetVar<std::vector<LWOOBJID>>(u"Players", entitiesInProximity); -} - -void AmShieldGeneratorQuickbuild::OnDie(Entity* self, Entity* killer) { - self->CancelAllTimers(); - - auto* child = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Child")); - - if (child != nullptr) { - child->Kill(); - } -} - -void AmShieldGeneratorQuickbuild::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "BuffPlayers") { - BuffPlayers(self); - - self->AddTimer("BuffPlayers", 3.0f); - } else if (timerName == "PlayFX") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 5351, u"generatorOn", "generatorOn"); - - self->AddTimer("PlayFX", 1.5f); - } else if (timerName == "RefreshEnemies") { - auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); - - for (const auto enemyID : enemiesInProximity) { - auto* enemy = EntityManager::Instance()->GetEntity(enemyID); - - if (enemy != nullptr) { - EnemyEnteredShield(self, enemy); - } - } - - self->AddTimer("RefreshEnemies", 1.5f); - } -} - -void AmShieldGeneratorQuickbuild::OnRebuildComplete(Entity* self, Entity* target) { - StartShield(self); - - auto enemiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Enemies"); - - for (const auto enemyID : enemiesInProximity) { - auto* enemy = EntityManager::Instance()->GetEntity(enemyID); - - if (enemy != nullptr) { - enemy->Smash(); - } - } - - auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - for (const auto playerID : entitiesInProximity) { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - continue; - } - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent == nullptr) { - return; - } - - missionComponent->ForceProgressTaskType(987, 1, 1, false); - } -} - -void AmShieldGeneratorQuickbuild::StartShield(Entity* self) { - self->AddTimer("PlayFX", 1.5f); - self->AddTimer("BuffPlayers", 3.0f); - self->AddTimer("RefreshEnemies", 1.5f); - - const auto myPos = self->GetPosition(); - const auto myRot = self->GetRotation(); - - EntityInfo info{}; - info.lot = 13111; - info.pos = myPos; - info.rot = myRot; - info.spawnerID = self->GetObjectID(); - - auto* child = EntityManager::Instance()->CreateEntity(info); - - self->SetVar(u"Child", child->GetObjectID()); - - BuffPlayers(self); -} - -void AmShieldGeneratorQuickbuild::BuffPlayers(Entity* self) { - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - auto entitiesInProximity = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - for (const auto playerID : entitiesInProximity) { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - skillComponent->CalculateBehavior(1200, 27024, playerID, true); - } -} - -void AmShieldGeneratorQuickbuild::EnemyEnteredShield(Entity* self, Entity* intruder) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent == nullptr || rebuildComponent->GetState() != REBUILD_COMPLETED) { - return; - } - - auto* baseCombatAIComponent = intruder->GetComponent<BaseCombatAIComponent>(); - auto* movementAIComponent = intruder->GetComponent<MovementAIComponent>(); - - if (baseCombatAIComponent == nullptr || movementAIComponent == nullptr) { - return; - } - - auto dir = intruder->GetRotation().GetForwardVector() * -1; - dir.y += 15; - dir.x *= 50; - dir.z *= 50; - - // TODO: Figure out how todo knockback, I'll stun them for now - - if (NiPoint3::DistanceSquared(self->GetPosition(), movementAIComponent->GetCurrentPosition()) < 20 * 20) { - baseCombatAIComponent->Stun(2.0f); - movementAIComponent->SetDestination(baseCombatAIComponent->GetStartPosition()); - } - - baseCombatAIComponent->ClearThreat(); -} diff --git a/dScripts/AmSkullkinDrill.cpp b/dScripts/AmSkullkinDrill.cpp deleted file mode 100644 index 321660e2..00000000 --- a/dScripts/AmSkullkinDrill.cpp +++ /dev/null @@ -1,323 +0,0 @@ -#include "AmSkullkinDrill.h" -#include "GameMessages.h" -#include "MovingPlatformComponent.h" -#include "DestroyableComponent.h" -#include "ProximityMonitorComponent.h" -#include "MissionComponent.h" - -void AmSkullkinDrill::OnStartup(Entity* self) { - self->SetNetworkVar(u"bIsInUse", false); - self->SetVar(u"bActive", true); - - GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"spin", "active"); - - auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); - - if (movingPlatformComponent == nullptr) { - return; - } - - movingPlatformComponent->SetSerialized(true); - - movingPlatformComponent->GotoWaypoint(0); - - auto* standObj = GetStandObj(self); - - if (standObj != nullptr) { - standObj->SetVar(u"bActive", true); - } - - self->SetProximityRadius(5, "spin_distance"); -} - -Entity* AmSkullkinDrill::GetStandObj(Entity* self) { - const auto& myGroup = self->GetGroups(); - - if (myGroup.empty()) { - return nullptr; - } - - std::string groupName = "Drill_Stand_"; - - groupName.push_back(myGroup[0][myGroup[0].size() - 1]); - - const auto standObjs = EntityManager::Instance()->GetEntitiesInGroup(groupName); - - if (standObjs.empty()) { - return nullptr; - } - - return standObjs[0]; -} - -void AmSkullkinDrill::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (message != "NinjagoSpinEvent" || self->GetNetworkVar<bool>(u"bIsInUse")) { - return; - } - - auto* proximityMonitorComponent = self->GetComponent<ProximityMonitorComponent>(); - - if (proximityMonitorComponent == nullptr || !proximityMonitorComponent->IsInProximity("spin_distance", caster->GetObjectID())) { - return; - } - - self->SetVar(u"activaterID", caster->GetObjectID()); - - self->SetNetworkVar(u"bIsInUse", true); - - TriggerDrill(self); -} - -void AmSkullkinDrill::TriggerDrill(Entity* self) { - GameMessages::SendPlayAnimation(self, u"slowdown"); - - self->AddTimer("killDrill", 10.0f); - - auto* standObj = GetStandObj(self); - - if (standObj != nullptr) { - standObj->SetVar(u"bActive", false); - } - - auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); - - if (movingPlatformComponent == nullptr) { - return; - } - - movingPlatformComponent->GotoWaypoint(1); -} - -void AmSkullkinDrill::OnWaypointReached(Entity* self, uint32_t waypointIndex) { - if (waypointIndex == 1) { - auto myPos = self->GetPosition(); - auto myRot = self->GetRotation(); - - myPos.y -= 21; - - EntityInfo info = {}; - info.lot = 12346; - info.pos = myPos; - info.rot = myRot; - info.scale = 3; // Needs the scale, otherwise attacks fail - info.spawnerID = self->GetObjectID(); - - auto* child = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(child); - - self->SetVar(u"ChildSmash", child->GetObjectID()); - - child->AddDieCallback([this, self]() { - const auto& userID = self->GetVar<LWOOBJID>(u"activaterID"); - - auto* player = EntityManager::Instance()->GetEntity(userID); - - if (player == nullptr) { - return; - } - - OnHitOrHealResult(self, player, 1); - }); - } - - OnArrived(self, waypointIndex); -} - -void AmSkullkinDrill::OnUse(Entity* self, Entity* user) { - if (self->GetNetworkVar<bool>(u"bIsInUse")) { - return; - } - - self->SetNetworkVar(u"bIsInUse", true); - - GameMessages::SendPlayFXEffect(user->GetObjectID(), 5499, u"on-anim", "tornado"); - GameMessages::SendPlayFXEffect(user->GetObjectID(), 5502, u"on-anim", "staff"); - - const auto userID = user->GetObjectID(); - - self->SetVar(u"userID", userID); - self->SetVar(u"activaterID", userID); - - PlayAnim(self, user, "spinjitzu-staff-windup"); - PlayCinematic(self); - - FreezePlayer(self, user, true); -} - -void AmSkullkinDrill::FreezePlayer(Entity* self, Entity* player, bool bFreeze) { - eStunState eChangeType = POP; - - if (bFreeze) { - if (player->GetIsDead()) { - return; - } - - eChangeType = PUSH; - } else { - if (player->GetIsDead()) { - // - } - } - - GameMessages::SendSetStunned(player->GetObjectID(), eChangeType, player->GetSystemAddress(), self->GetObjectID(), - true, false, true, false, true, false, true - ); -} - -void AmSkullkinDrill::OnArrived(Entity* self, uint32_t waypointIndex) { - auto* standObj = GetStandObj(self); - - if (waypointIndex == 1) { - GameMessages::SendPlayAnimation(self, u"no-spin"); - GameMessages::SendStopFXEffect(self, true, "active"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"indicator", "indicator"); - - self->SetVar(u"bActive", false); - - const auto playerID = self->GetVar<LWOOBJID>(u"userID"); - - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player != nullptr) { - PlayAnim(self, player, "spinjitzu-staff-end"); - } - - if (standObj != nullptr) { - standObj->SetVar(u"bActive", false); - } - - return; - } else { - GameMessages::SendPlayAnimation(self, u"idle"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"spin", "active"); - GameMessages::SendStopFXEffect(self, true, "indicator"); - } -} - -void AmSkullkinDrill::PlayCinematic(Entity* self) { - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"userID")); - - if (player == nullptr) { - return; - } - - const auto& cine = self->GetVar<std::u16string>(u"cinematic"); - - if (cine.empty()) { - return; - } - - GameMessages::SendPlayCinematic(player->GetObjectID(), cine, player->GetSystemAddress()); -} - -void AmSkullkinDrill::PlayAnim(Entity* self, Entity* player, const std::string& animName) { - const auto animTime = animName == "spinjitzu-staff-end" ? 0.5f : 1.0f; - - GameMessages::SendPlayAnimation(player, GeneralUtils::ASCIIToUTF16(animName)); - - self->AddTimer("AnimDone_" + animName, animTime); -} - -void AmSkullkinDrill::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - - if (destroyableComponent == nullptr || !attacker->IsPlayer()) { - return; - } - - if (self->GetVar<bool>(u"bActive")) { - return; - } - - const auto activaterID = self->GetVar<LWOOBJID>(u"activaterID"); - - auto* activator = EntityManager::Instance()->GetEntity(activaterID); - - // TODO: Missions - if (activator != nullptr) { - auto* missionComponent = activator->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - for (const auto missionID : m_MissionsToUpdate) { - missionComponent->ForceProgressValue(missionID, 1, self->GetLOT()); - } - } - } - - self->Smash(attacker->GetObjectID(), SILENT); - - self->CancelAllTimers(); - - auto* standObj = GetStandObj(self); - - if (standObj != nullptr) { - GameMessages::SendPlayFXEffect(standObj->GetObjectID(), 4946, u"explode", "explode"); - } -} - -void AmSkullkinDrill::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "killDrill") { - const auto childID = self->GetVar<LWOOBJID>(u"ChildSmash"); - - auto* child = EntityManager::Instance()->GetEntity(childID); - - if (child != nullptr) { - child->Smash(self->GetObjectID(), SILENT); - } - - self->SetNetworkVar(u"bIsInUse", false); - self->SetVar(u"bActive", true); - self->SetVar(u"activaterID", LWOOBJID_EMPTY); - - auto* standObj = GetStandObj(self); - - if (standObj != nullptr) { - standObj->SetVar(u"bActive", true); - } - - auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); - - if (movingPlatformComponent == nullptr) { - return; - } - - movingPlatformComponent->GotoWaypoint(0); - - return; - } - - const auto& data = GeneralUtils::SplitString(timerName, '_'); - - if (data.empty()) { - return; - } - - if (data[0] == "AnimDone") { - const auto& animName = data[1]; - - const auto playerID = self->GetVar<LWOOBJID>(u"userID"); - - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - if (animName == "spinjitzu-staff-windup") { - TriggerDrill(self); - - GameMessages::SendPlayAnimation(player, u"spinjitzu-staff-loop"); - } else if (animName == "spinjitzu-staff-end") { - FreezePlayer(self, player, false); - - self->SetVar(u"userID", LWOOBJID_EMPTY); - - GameMessages::SendStopFXEffect(player, true, "tornado"); - GameMessages::SendStopFXEffect(player, true, "staff"); - } - - } else if (data[0] == "TryUnFreezeAgain") { - - } -} diff --git a/dScripts/AmSkullkinDrillStand.cpp b/dScripts/AmSkullkinDrillStand.cpp deleted file mode 100644 index 628d616a..00000000 --- a/dScripts/AmSkullkinDrillStand.cpp +++ /dev/null @@ -1,35 +0,0 @@ -#include "AmSkullkinDrillStand.h" -#include "GameMessages.h" -#include "dpEntity.h" - -void AmSkullkinDrillStand::OnStartup(Entity* self) { - self->SetVar(u"bActive", true); - - self->SetProximityRadius(new dpEntity(self->GetObjectID(), { 6, 14, 6 }), "knockback"); -} - -void AmSkullkinDrillStand::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { - -} - -void AmSkullkinDrillStand::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (!self->GetVar<bool>(u"bActive")) { - return; - } - - if (!entering->IsPlayer() || status != "ENTER" || name != "knockback") { - return; - } - - auto myPos = self->GetPosition(); - - auto objPos = entering->GetPosition(); - - NiPoint3 newVec = { (objPos.x - myPos.x) * 4.5f, 15, (objPos.z - myPos.z) * 4.5f }; - - GameMessages::SendKnockback(entering->GetObjectID(), self->GetObjectID(), self->GetObjectID(), 0, newVec); - - GameMessages::SendPlayFXEffect(entering->GetObjectID(), 1378, u"create", "pushBack"); - - GameMessages::SendPlayAnimation(entering, u"knockback-recovery"); -} diff --git a/dScripts/AmSkullkinTower.cpp b/dScripts/AmSkullkinTower.cpp deleted file mode 100644 index 180bbdac..00000000 --- a/dScripts/AmSkullkinTower.cpp +++ /dev/null @@ -1,243 +0,0 @@ -#include "AmSkullkinTower.h" -#include "EntityManager.h" -#include "DestroyableComponent.h" -#include "MovingPlatformComponent.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void AmSkullkinTower::OnStartup(Entity* self) { - self->SetProximityRadius(20, "Tower"); - - // onPhysicsComponentReady - - auto* movingPlatformComponent = self->GetComponent<MovingPlatformComponent>(); - - if (movingPlatformComponent != nullptr) { - movingPlatformComponent->StopPathing(); - } - - SpawnLegs(self, "Left"); - SpawnLegs(self, "Right"); - SpawnLegs(self, "Rear"); -} - -void AmSkullkinTower::SpawnLegs(Entity* self, const std::string& loc) { - auto pos = self->GetPosition(); - auto rot = self->GetRotation(); - pos.y += self->GetVarAs<float>(u"vert_offset"); - - auto newRot = rot; - auto offset = self->GetVarAs<float>(u"hort_offset"); - - auto legLOT = self->GetVar<LOT>(u"legLOT"); - - if (legLOT == 0) { - return; - } - - std::vector<LDFBaseData*> config = { new LDFData<std::string>(u"Leg", loc) }; - - EntityInfo info{}; - info.lot = legLOT; - info.spawnerID = self->GetObjectID(); - info.settings = config; - info.rot = newRot; - - if (loc == "Right") { - const auto dir = rot.GetForwardVector(); - pos.x += dir.x * offset; - pos.z += dir.z * offset; - info.pos = pos; - } else if (loc == "Rear") { - const auto dir = rot.GetRightVector(); - pos.x += dir.x * offset; - pos.z += dir.z * offset; - info.pos = pos; - } else if (loc == "Left") { - const auto dir = rot.GetForwardVector() * -1; - pos.x += dir.x * offset; - pos.z += dir.z * offset; - info.pos = pos; - } - - info.rot = NiQuaternion::LookAt(info.pos, self->GetPosition()); - - auto* entity = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(entity); - - OnChildLoaded(self, entity); -} - -void AmSkullkinTower::OnChildLoaded(Entity* self, Entity* child) { - auto legTable = self->GetVar<std::vector<LWOOBJID>>(u"legTable"); - - legTable.push_back(child->GetObjectID()); - - self->SetVar(u"legTable", legTable); - - const auto selfID = self->GetObjectID(); - - child->AddDieCallback([this, selfID, child]() { - auto* self = EntityManager::Instance()->GetEntity(selfID); - auto* destroyableComponent = child->GetComponent<DestroyableComponent>(); - - if (destroyableComponent == nullptr || self == nullptr) { - return; - } - - NotifyDie(self, child, destroyableComponent->GetKiller()); - }); -} - -void AmSkullkinTower::NotifyDie(Entity* self, Entity* other, Entity* killer) { - auto players = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - const auto& iter = std::find(players.begin(), players.end(), killer->GetObjectID()); - - if (iter == players.end()) { - players.push_back(killer->GetObjectID()); - } - - self->SetVar(u"Players", players); - - OnChildRemoved(self, other); -} - -void AmSkullkinTower::OnChildRemoved(Entity* self, Entity* child) { - auto legTable = self->GetVar<std::vector<LWOOBJID>>(u"legTable"); - - const auto& iter = std::find(legTable.begin(), legTable.end(), child->GetObjectID()); - - if (iter != legTable.end()) { - legTable.erase(iter); - } - - self->SetVar(u"legTable", legTable); - - if (legTable.size() == 2) { - GameMessages::SendPlayAnimation(self, u"wobble-1"); - } else if (legTable.size() == 1) { - GameMessages::SendPlayAnimation(self, u"wobble-2"); - } else if (legTable.empty()) { - const auto animTime = 2.5f; - - GameMessages::SendPlayAnimation(self, u"fall"); - - self->AddTimer("spawnGuys", animTime - 0.2f); - - self->CancelTimer("RespawnLeg"); - self->CancelTimer("RespawnLeg"); - self->CancelTimer("RespawnLeg"); - - std::vector<int32_t> missionIDs; - - auto missionsString = self->GetVar<std::u16string>(u"missions"); - - if (!missionsString.empty()) { - // Split the missions string by '_' - const auto missions = GeneralUtils::SplitString( - GeneralUtils::UTF16ToWTF8(missionsString), - '_' - ); - - for (const auto& mission : missions) { - int32_t missionID = 0; - - if (!GeneralUtils::TryParse(mission, missionID)) { - continue; - } - - missionIDs.push_back(missionID); - } - } - - const auto& players = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - for (const auto& playerID : players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - continue; - } - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent == nullptr) { - continue; - } - - for (const auto missionID : missionIDs) { - missionComponent->ForceProgressValue(missionID, 1, self->GetLOT()); - } - - //missionComponent->ForceProgressValue(1305, 1, self->GetLOT()); - } - } - - auto deadLegs = self->GetVar<std::vector<std::string>>(u"DeadLegs"); - - const auto& leg = child->GetVar<std::string>(u"Leg"); - - const auto& legIter = std::find(deadLegs.begin(), deadLegs.end(), leg); - - if (legIter == deadLegs.end()) { - deadLegs.push_back(leg); - } - - self->SetVar(u"DeadLegs", deadLegs); - - self->AddTimer("RespawnLeg", 20); -} - -void AmSkullkinTower::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (status != "LEAVE") { - return; - } - - auto players = self->GetVar<std::vector<LWOOBJID>>(u"Players"); - - const auto& iter = std::find(players.begin(), players.end(), entering->GetObjectID()); - - if (iter != players.end()) { - players.erase(iter); - } - - self->SetVar(u"Players", players); -} - -void AmSkullkinTower::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "RespawnLeg") { - auto deadLegs = self->GetVar<std::vector<std::string>>(u"DeadLegs"); - - if (deadLegs.empty()) { - return; - } - - SpawnLegs(self, deadLegs[0]); - - deadLegs.erase(deadLegs.begin()); - - self->SetVar<std::vector<std::string>>(u"DeadLegs", deadLegs); - } else if (timerName == "spawnGuys") { - EntityInfo info{}; - info.lot = self->GetVar<LOT>(u"enemyToSpawn"); - auto pos = self->GetPosition(); - pos.y += 7; - info.pos = pos; - info.rot = self->GetRotation(); - info.spawnerID = self->GetObjectID(); - - for (size_t i = 0; i < 2; i++) { - info.pos.x += i * 2; // Just to set the apart a bit - - auto* entity = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(entity); - } - - self->AddTimer("killTower", 0.7f); - } else if (timerName == "killTower") { - self->Smash(self->GetObjectID()); - } -} diff --git a/dScripts/AmTeapotServer.cpp b/dScripts/AmTeapotServer.cpp deleted file mode 100644 index 32abafd6..00000000 --- a/dScripts/AmTeapotServer.cpp +++ /dev/null @@ -1,15 +0,0 @@ -#include "AmTeapotServer.h" -#include "InventoryComponent.h" -#include "GameMessages.h" - - -void AmTeapotServer::OnUse(Entity* self, Entity* user) { - auto* inventoryComponent = user->GetComponent<InventoryComponent>(); - if (!inventoryComponent) return; - - if (inventoryComponent->GetLotCount(BLUE_FLOWER_LEAVES) >= 10) { - inventoryComponent->RemoveItem(BLUE_FLOWER_LEAVES, 10); - inventoryComponent->AddItem(WU_S_IMAGINATION_TEA, 1); - } - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} diff --git a/dScripts/BankInteractServer.cpp b/dScripts/BankInteractServer.cpp deleted file mode 100644 index db5ebb98..00000000 --- a/dScripts/BankInteractServer.cpp +++ /dev/null @@ -1,23 +0,0 @@ -#include "BankInteractServer.h" -#include "GameMessages.h" - -void BankInteractServer::OnUse(Entity* self, Entity* user) { - AMFArrayValue args; - AMFStringValue* bank = new AMFStringValue(); - bank->SetStringValue("bank"); - args.InsertValue("state", bank); - - GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", &args); -} - -void BankInteractServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (args == "ToggleBank") { - AMFArrayValue args; - args.InsertValue("visible", new AMFFalseValue()); - - GameMessages::SendUIMessageServerToSingleClient(sender, sender->GetSystemAddress(), "ToggleBank", &args); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"CloseBank", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - } -} diff --git a/dScripts/BaseConsoleTeleportServer.cpp b/dScripts/BaseConsoleTeleportServer.cpp index a8538de5..27d2bcfc 100644 --- a/dScripts/BaseConsoleTeleportServer.cpp +++ b/dScripts/BaseConsoleTeleportServer.cpp @@ -1,6 +1,10 @@ #include "BaseConsoleTeleportServer.h" #include "GameMessages.h" #include "Player.h" +#include "RenderComponent.h" +#include "EntityManager.h" +#include "eTerminateType.h" +#include "eStateChangeType.h" void BaseConsoleTeleportServer::BaseOnUse(Entity* self, Entity* user) { auto* player = user; @@ -15,7 +19,8 @@ void BaseConsoleTeleportServer::BaseOnMessageBoxResponse(Entity* self, Entity* s if (button == 1) { - GameMessages::SendSetStunned(player->GetObjectID(), PUSH, player->GetSystemAddress(), player->GetObjectID(), + GameMessages::SendSetStunned( + player->GetObjectID(), eStateChangeType::PUSH, player->GetSystemAddress(), player->GetObjectID(), true, true, true, true, true, true, true ); @@ -30,19 +35,17 @@ void BaseConsoleTeleportServer::BaseOnMessageBoxResponse(Entity* self, Entity* s } const auto& teleIntroAnim = self->GetVar<std::u16string>(u"teleportAnim"); - + auto animTime = 3.32999992370605f; if (!teleIntroAnim.empty()) { - GameMessages::SendPlayAnimation(player, teleIntroAnim); + animTime = RenderComponent::PlayAnimation(player, teleIntroAnim); } - const auto animTime = 3.32999992370605f; - UpdatePlayerTable(self, player, true); const auto playerID = player->GetObjectID(); self->AddCallbackTimer(animTime, [playerID, self]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) { return; @@ -51,7 +54,7 @@ void BaseConsoleTeleportServer::BaseOnMessageBoxResponse(Entity* self, Entity* s GameMessages::SendDisplayZoneSummary(playerID, player->GetSystemAddress(), false, false, self->GetObjectID()); }); } else if (button == -1 || button == 0) { - GameMessages::SendTerminateInteraction(player->GetObjectID(), FROM_INTERACTION, player->GetObjectID()); + GameMessages::SendTerminateInteraction(player->GetObjectID(), eTerminateType::FROM_INTERACTION, player->GetObjectID()); } } @@ -82,13 +85,12 @@ void BaseConsoleTeleportServer::TransferPlayer(Entity* self, Entity* player, int return; } - // Ignoring extra effects for now - - /*GameMessages::SendSetStunned(player->GetObjectID(), POP, player->GetSystemAddress(), player->GetObjectID(), + GameMessages::SendSetStunned( + player->GetObjectID(), eStateChangeType::POP, player->GetSystemAddress(), player->GetObjectID(), true, true, true, true, true, true, true - );*/ + ); - GameMessages::SendTerminateInteraction(player->GetObjectID(), FROM_INTERACTION, player->GetObjectID()); + GameMessages::SendTerminateInteraction(player->GetObjectID(), eTerminateType::FROM_INTERACTION, player->GetObjectID()); const auto& teleportZone = self->GetVar<std::u16string>(u"transferZoneID"); diff --git a/dScripts/BaseEnemyApe.cpp b/dScripts/BaseEnemyApe.cpp deleted file mode 100644 index 3d0b8e25..00000000 --- a/dScripts/BaseEnemyApe.cpp +++ /dev/null @@ -1,134 +0,0 @@ -#include "BaseEnemyApe.h" -#include "BaseCombatAIComponent.h" -#include "DestroyableComponent.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "SkillComponent.h" - -void BaseEnemyApe::OnStartup(Entity* self) { - self->SetVar<uint32_t>(u"timesStunned", 2); - self->SetVar<bool>(u"knockedOut", false); -} - -void BaseEnemyApe::OnDie(Entity* self, Entity* killer) { - auto* anchor = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"QB")); - if (anchor != nullptr && !anchor->GetIsDead()) { - anchor->Smash(self->GetObjectID(), SILENT); - } -} - -void BaseEnemyApe::OnSkillCast(Entity* self, uint32_t skillID) { - const auto groundPoundSkill = self->GetVar<uint32_t>(u"GroundPoundSkill") != 0 ? self->GetVar<uint32_t>(u"GroundPoundSkill") : 725; - const auto spawnQuickBuildTime = self->GetVar<float_t>(u"spawnQBTime") != 0.0f ? self->GetVar<float_t>(u"spawnQBTime") : 5.0f; - - if (skillID == groundPoundSkill && self->GetVar<LWOOBJID>(u"QB") == LWOOBJID_EMPTY) { - self->AddTimer("spawnQBTime", spawnQuickBuildTime); - } -} - -void BaseEnemyApe::OnHit(Entity* self, Entity* attacker) { - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr && destroyableComponent->GetArmor() < 1 && !self->GetBoolean(u"knockedOut")) { - StunApe(self, true); - self->CancelTimer("spawnQBTime"); - - GameMessages::SendPlayAnimation(self, u"disable", 1.7f); - - const auto reviveTime = self->GetVar<float_t>(u"reviveTime") != 0.0f - ? self->GetVar<float_t>(u"reviveTime") : 12.0f; - self->AddTimer("reviveTime", reviveTime); - } -} - -void BaseEnemyApe::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "reviveTime") { - - // Revives the ape, giving it back some armor - const auto timesStunned = self->GetVar<uint32_t>(u"timesStunned"); - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->SetArmor(destroyableComponent->GetMaxArmor() / timesStunned); - } - EntityManager::Instance()->SerializeEntity(self); - self->SetVar<uint32_t>(u"timesStunned", timesStunned + 1); - StunApe(self, false); - - } else if (timerName == "spawnQBTime" && self->GetVar<LWOOBJID>(u"QB") == LWOOBJID_EMPTY) { - // Spawn QB in front of ape. - const auto position = self->GetPosition(); - const auto rotation = self->GetRotation(); - - const auto backwardVector = rotation.GetForwardVector() * -1; - const auto objectPosition = NiPoint3( - position.GetX() - (backwardVector.GetX() * 8), - position.GetY(), - position.GetZ() - (backwardVector.GetZ() * 8) - ); - - EntityInfo entityInfo{}; - - entityInfo.pos = position; - entityInfo.rot = rotation; - entityInfo.pos.SetY(entityInfo.pos.GetY() + 13.0f); - - entityInfo.spawnerID = self->GetObjectID(); - entityInfo.lot = self->GetVar<LOT>(u"QuickbuildAnchorLOT") != 0 - ? self->GetVar<LOT>(u"QuickbuildAnchorLOT") : 7549; - entityInfo.settings = { - new LDFData<std::string>(u"rebuild_activators", - std::to_string(objectPosition.GetX()) + "\x1f" + - std::to_string(objectPosition.GetY()) + "\x1f" + - std::to_string(objectPosition.GetZ()) - ), - new LDFData<bool>(u"no_timed_spawn", true), - new LDFData<LWOOBJID>(u"ape", self->GetObjectID()) - }; - - auto* anchor = EntityManager::Instance()->CreateEntity(entityInfo); - EntityManager::Instance()->ConstructEntity(anchor); - self->SetVar<LWOOBJID>(u"QB", anchor->GetObjectID()); - - } else if (timerName == "anchorDamageTimer") { - - // Attacks the ape with some god skill - const auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"smasher")); - if (player == nullptr) { - return; - } - - auto* skillComponent = self->GetComponent<SkillComponent>(); - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(1273, 29446, self->GetObjectID(), true, false, player->GetObjectID()); - } - - self->SetVar<LWOOBJID>(u"QB", LWOOBJID_EMPTY); - } -} - -void BaseEnemyApe::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args == "rebuildDone" && sender != nullptr) { - self->SetVar<LWOOBJID>(u"smasher", sender->GetObjectID()); - const auto anchorDamageDelayTime = self->GetVar<float_t>(u"AnchorDamageDelayTime") != 0.0f ? self->GetVar<float_t>(u"AnchorDamageDelayTime") : 0.5f; - self->AddTimer("anchorDamageTimer", anchorDamageDelayTime); - } -} - -void BaseEnemyApe::StunApe(Entity* self, bool stunState) { - auto* combatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - if (combatAIComponent != nullptr) { - combatAIComponent->SetDisabled(stunState); - combatAIComponent->SetStunned(stunState); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - if (skillComponent != nullptr) { - skillComponent->Interrupt(); - } - - GameMessages::SendSetStunned(self->GetObjectID(), stunState ? PUSH : POP, UNASSIGNED_SYSTEM_ADDRESS, self->GetObjectID(), - true, true, true, true, true, - true, true, true, true); - - self->SetBoolean(u"knockedOut", stunState); - } -} diff --git a/dScripts/BaseEnemyMech.cpp b/dScripts/BaseEnemyMech.cpp deleted file mode 100644 index 8017be2c..00000000 --- a/dScripts/BaseEnemyMech.cpp +++ /dev/null @@ -1,43 +0,0 @@ -#include "BaseEnemyMech.h" -#include "Entity.h" -#include "ControllablePhysicsComponent.h" -#include "EntityManager.h" -#include "dpWorld.h" -#include "GeneralUtils.h" -#include "DestroyableComponent.h" - -void BaseEnemyMech::OnStartup(Entity* self) { - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->SetFaction(4); - } -} - -void BaseEnemyMech::OnDie(Entity* self, Entity* killer) { - ControllablePhysicsComponent* controlPhys = static_cast<ControllablePhysicsComponent*>(self->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); - if (!controlPhys) return; - - NiPoint3 newLoc = { controlPhys->GetPosition().x, dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(controlPhys->GetPosition()), controlPhys->GetPosition().z }; - - EntityInfo info = EntityInfo(); - std::vector<LDFBaseData*> cfg; - std::u16string activatorPosStr; - activatorPosStr += (GeneralUtils::to_u16string(controlPhys->GetPosition().x)); - activatorPosStr.push_back(0x1f); - activatorPosStr += (GeneralUtils::to_u16string(controlPhys->GetPosition().y)); - activatorPosStr.push_back(0x1f); - activatorPosStr += (GeneralUtils::to_u16string(controlPhys->GetPosition().z)); - - LDFBaseData* activatorPos = new LDFData<std::u16string>(u"rebuild_activators", activatorPosStr); - cfg.push_back(activatorPos); - info.lot = qbTurretLOT; - info.pos = newLoc; - info.rot = controlPhys->GetRotation(); - info.spawnerID = self->GetObjectID(); - info.settings = cfg; - - Entity* turret = EntityManager::Instance()->CreateEntity(info, nullptr); - if (turret) { - EntityManager::Instance()->ConstructEntity(turret); - } -} diff --git a/dScripts/BaseFootRaceManager.cpp b/dScripts/BaseFootRaceManager.cpp deleted file mode 100644 index 699ee096..00000000 --- a/dScripts/BaseFootRaceManager.cpp +++ /dev/null @@ -1,48 +0,0 @@ -#include "BaseFootRaceManager.h" -#include "EntityManager.h" -#include "Character.h" - -void BaseFootRaceManager::OnStartup(Entity* self) { - // TODO: Add to FootRaceStarter group -} - -void BaseFootRaceManager::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - const auto splitArguments = GeneralUtils::SplitString(args, '_'); - if (splitArguments.size() > 1) { - - const auto eventName = splitArguments[0]; - const auto player = EntityManager::Instance()->GetEntity(std::stoull(splitArguments[1])); - - if (player != nullptr) { - if (eventName == "updatePlayer") { - UpdatePlayer(self, player->GetObjectID()); - } else if (IsPlayerInActivity(self, player->GetObjectID())) { - if (eventName == "initialActivityScore") { - auto* character = player->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(115, true); - } - - SetActivityScore(self, player->GetObjectID(), 1); - } else if (eventName == "updatePlayerTrue") { - auto* character = player->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(115, false); - } - - UpdatePlayer(self, player->GetObjectID(), true); - } else if (eventName == "PlayerWon") { - auto* character = player->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(115, false); - if (param2 != -1) // Certain footraces set a flag - character->SetPlayerFlag(param2, true); - } - - StopActivity(self, player->GetObjectID(), 0, param1); - } - } - } - } -} diff --git a/dScripts/BasePropertyServer.cpp b/dScripts/BasePropertyServer.cpp index a182c9c1..fe20b09d 100644 --- a/dScripts/BasePropertyServer.cpp +++ b/dScripts/BasePropertyServer.cpp @@ -8,6 +8,8 @@ #include "RenderComponent.h" #include "PropertyManagementComponent.h" #include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" void BasePropertyServer::SetGameVariables(Entity* self) { self->SetVar<std::string>(ClaimMarkerGroup, ""); @@ -36,8 +38,8 @@ void BasePropertyServer::SetGameVariables(Entity* self) { self->SetVar<std::vector<std::string>>(AmbientFXSpawner, {}); self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, {}); - self->SetVar<uint32_t>(defeatedProperyFlag, 0); - self->SetVar<uint32_t>(placedModelFlag, 0); + self->SetVar<int32_t>(defeatedProperyFlag, 0); + self->SetVar<int32_t>(placedModelFlag, 0); self->SetVar<uint32_t>(guardMissionFlag, 0); self->SetVar<uint32_t>(brickLinkMissionIDFlag, 0); self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); @@ -47,7 +49,7 @@ void BasePropertyServer::SetGameVariables(Entity* self) { } void BasePropertyServer::CheckForOwner(Entity* self) { - if (EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(PropertyPlaqueGroup)).empty()) { + if (Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(PropertyPlaqueGroup)).empty()) { self->AddTimer(RunPlayerLoadedAgainTimer, 0.5f); return; } @@ -85,19 +87,19 @@ void BasePropertyServer::BasePlayerLoaded(Entity* self, Entity* player) { self->SetNetworkVar(PropertyOwnerIDVariable, propertyOwner); if (rented) { - auto plaques = EntityManager::Instance()->GetEntitiesInGroup("PropertyVendor"); + auto plaques = Game::entityManager->GetEntitiesInGroup("PropertyVendor"); for (auto* plaque : plaques) { - EntityManager::Instance()->DestructEntity(plaque); + Game::entityManager->DestructEntity(plaque); } - const auto& mapID = dZoneManager::Instance()->GetZone()->GetZoneID(); + const auto& mapID = Game::zoneManager->GetZone()->GetZoneID(); if (propertyOwner > 0) { auto* missionComponent = player->GetComponent<MissionComponent>(); if (missionComponent != nullptr) { missionComponent->Progress( - MissionTaskType::MISSION_TASK_TYPE_VISIT_PROPERTY, + eMissionTaskType::VISIT_PROPERTY, mapID.GetMapID(), mapID.GetCloneID() ); @@ -125,7 +127,7 @@ void BasePropertyServer::BasePlayerLoaded(Entity* self, Entity* player) { if (player->GetObjectID() != propertyOwner) return; } else { - const auto defeatedFlag = player->GetCharacter()->GetPlayerFlag(self->GetVar<uint32_t>(defeatedProperyFlag)); + const auto defeatedFlag = player->GetCharacter()->GetPlayerFlag(self->GetVar<int32_t>(defeatedProperyFlag)); self->SetNetworkVar(UnclaimedVariable, true); self->SetVar<LWOOBJID>(PlayerIDVariable, player->GetObjectID()); @@ -150,7 +152,7 @@ void BasePropertyServer::PropGuardCheck(Entity* self, Entity* player) { auto* missionComponent = player->GetComponent<MissionComponent>(); if (missionComponent != nullptr - && missionComponent->GetMissionState(self->GetVar<uint32_t>(guardMissionFlag)) != MissionState::MISSION_STATE_COMPLETE) { + && missionComponent->GetMissionState(self->GetVar<uint32_t>(guardMissionFlag)) != eMissionState::COMPLETE) { ActivateSpawner(self->GetVar<std::string>(PropertyMGSpawner)); } } @@ -162,9 +164,9 @@ void BasePropertyServer::BaseZonePropertyRented(Entity* self, Entity* player) co self->AddTimer(BoundsVisOnTimer, 2); self->SetVar<LWOOBJID>(PropertyOwnerVariable, player->GetObjectID()); - auto plaques = EntityManager::Instance()->GetEntitiesInGroup("PropertyVendor"); + auto plaques = Game::entityManager->GetEntitiesInGroup("PropertyVendor"); for (auto* plaque : plaques) { - EntityManager::Instance()->DestructEntity(plaque); + Game::entityManager->DestructEntity(plaque); } auto brickLinkMissionID = self->GetVar<uint32_t>(brickLinkMissionIDFlag); @@ -232,7 +234,7 @@ void BasePropertyServer::StartMaelstrom(Entity* self, Entity* player) { } void BasePropertyServer::StartTornadoFx(Entity* self) const { - const auto entities = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); + const auto entities = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); if (entities.empty()) { self->AddTimer("pollTornadoFX", 0.1f); return; @@ -257,7 +259,7 @@ void BasePropertyServer::BasePlayerExit(Entity* self, Entity* player) { } void BasePropertyServer::KillGuard(Entity* self) { - const auto entities = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(GuardGroup)); + const auto entities = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(GuardGroup)); if (entities.empty()) return; @@ -272,47 +274,37 @@ void BasePropertyServer::RequestDie(Entity* self, Entity* other) { if (destroyable == nullptr) return; - destroyable->Smash(other->GetObjectID(), SILENT); + destroyable->Smash(other->GetObjectID(), eKillType::SILENT); } void BasePropertyServer::ActivateSpawner(const std::string& spawnerName) { - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(spawnerName)) { + for (auto* spawner : Game::zoneManager->GetSpawnersByName(spawnerName)) { spawner->Activate(); } } void BasePropertyServer::DeactivateSpawner(const std::string& spawnerName) { - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(spawnerName)) { + for (auto* spawner : Game::zoneManager->GetSpawnersByName(spawnerName)) { spawner->Deactivate(); } } void BasePropertyServer::TriggerSpawner(const std::string& spawnerName) { - for (auto* spawner : dZoneManager::Instance()->GetSpawnersInGroup(spawnerName)) { + for (auto* spawner : Game::zoneManager->GetSpawnersInGroup(spawnerName)) { spawner->Spawn(); } } void BasePropertyServer::ResetSpawner(const std::string& spawnerName) { - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(spawnerName)) { + for (auto* spawner : Game::zoneManager->GetSpawnersByName(spawnerName)) { spawner->Reset(); } } void BasePropertyServer::DestroySpawner(const std::string& spawnerName) { - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(spawnerName)) { - for (auto* node : spawner->m_Info.nodes) { - for (const auto& element : node->entities) { - auto* entity = EntityManager::Instance()->GetEntity(element); - if (entity == nullptr) - continue; - - entity->Kill(); - } - - node->entities.clear(); - } - + for (auto* spawner : Game::zoneManager->GetSpawnersByName(spawnerName)) { + if (!spawner) return; + spawner->DestroyAllEntities(); spawner->Deactivate(); } } @@ -330,18 +322,18 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam } else if (timerName == StartQuickbuildTimer) { HandleQuickBuildTimer(self); } else if (timerName == "GuardFlyAway") { - const auto zoneId = dZoneManager::Instance()->GetZone()->GetWorldID(); + const auto zoneId = Game::zoneManager->GetZone()->GetWorldID(); // No guard for the spider instance fight - if (dZoneManager::Instance()->GetZoneID().GetMapID() == 1150) + if (Game::zoneManager->GetZoneID().GetMapID() == 1150) return; - const auto entities = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(GuardGroup)); + const auto entities = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(GuardGroup)); if (entities.empty()) return; auto* guard = entities[0]; - GameMessages::SendNotifyClientObject(EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(), + GameMessages::SendNotifyClientObject(Game::entityManager->GetZoneControlEntity()->GetObjectID(), u"GuardChat", 0, 0, guard->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); @@ -349,7 +341,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam } else if (timerName == KillGuardTimer) { KillGuard(self); } else if (timerName == TornadoOffTimer) { - auto fxManagers = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); + auto fxManagers = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); for (auto* fxManager : fxManagers) { auto* renderComponent = fxManager->GetComponent<RenderComponent>(); @@ -362,7 +354,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam self->AddTimer(ShowClearEffectsTimer, 2); } else if (timerName == ShowClearEffectsTimer) { - auto fxManagers = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); + auto fxManagers = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); for (auto* fxManager : fxManagers) { auto* renderComponent = fxManager->GetComponent<RenderComponent>(); @@ -374,11 +366,11 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam self->AddTimer(TurnSkyOffTimer, 1.5f); self->AddTimer(KillFXObjectTimer, 8.0f); } else if (timerName == TurnSkyOffTimer) { - auto* controller = dZoneManager::Instance()->GetZoneControlObject(); + auto* controller = Game::zoneManager->GetZoneControlObject(); GameMessages::SendNotifyClientObject(controller->GetObjectID(), u"SkyOff", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); } else if (timerName == KillStrombiesTimer) { - const auto enemies = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(EnemiesGroup)); + const auto enemies = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(EnemiesGroup)); for (auto* enemy : enemies) { RequestDie(self, enemy); } @@ -386,7 +378,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam DestroySpawner(self->GetVar<std::string>(SmashablesSpawner)); KillSpots(self); - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); if (player == nullptr) return; @@ -401,7 +393,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam DestroySpawner(behaviorObjectSpawner); } - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(ImagOrbGroup))) { + for (auto* entity : Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(ImagOrbGroup))) { entity->Smash(); } @@ -409,7 +401,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam self->AddTimer(ShowVendorTimer, 1.0f); } else if (timerName == ShowVendorTimer) { - GameMessages::SendNotifyClientObject(EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(), + GameMessages::SendNotifyClientObject(Game::entityManager->GetZoneControlEntity()->GetObjectID(), u"vendorOn", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); @@ -424,7 +416,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam } else if (timerName == PollTornadoFXTimer) { StartTornadoFx(self); } else if (timerName == KillFXObjectTimer) { - const auto fxManagers = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); + const auto fxManagers = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup)); if (fxManagers.empty()) { self->AddTimer(KillFXObjectTimer, 1.0f); return; @@ -444,7 +436,7 @@ void BasePropertyServer::BaseTimerDone(Entity* self, const std::string& timerNam void BasePropertyServer::HandleOrbsTimer(Entity* self) { self->SetVar<bool>(CollidedVariable, false); - auto orbs = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(ImagOrbGroup)); + auto orbs = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(ImagOrbGroup)); if (orbs.empty()) { self->AddTimer(StartOrbTimer, 0.5f); return; @@ -463,16 +455,16 @@ void BasePropertyServer::HandleOrbsTimer(Entity* self) { } DestroySpawner(self->GetVar<std::string>(GeneratorFXSpawner)); - GameMessages::SendNotifyClientObject(EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(), + GameMessages::SendNotifyClientObject(Game::entityManager->GetZoneControlEntity()->GetObjectID(), u"PlayCinematic", 0, 0, LWOOBJID_EMPTY, "DestroyMaelstrom", UNASSIGNED_SYSTEM_ADDRESS); // Notifies the client that the property has been claimed with a flag, completes missions too - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); if (player != nullptr) { auto* character = player->GetCharacter(); if (character != nullptr) { - character->SetPlayerFlag(self->GetVar<uint32_t>(defeatedProperyFlag), true); + character->SetPlayerFlag(self->GetVar<int32_t>(defeatedProperyFlag), true); } } @@ -484,7 +476,7 @@ void BasePropertyServer::HandleOrbsTimer(Entity* self) { } void BasePropertyServer::HandleGeneratorTimer(Entity* self) { - auto generators = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(GeneratorGroup)); + auto generators = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(GeneratorGroup)); if (generators.empty()) { self->AddTimer(StartGeneratorTimer, 0.5f); return; @@ -504,7 +496,7 @@ void BasePropertyServer::HandleGeneratorTimer(Entity* self) { } void BasePropertyServer::HandleQuickBuildTimer(Entity* self) { - auto claimMarkers = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(ClaimMarkerGroup)); + auto claimMarkers = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(ClaimMarkerGroup)); if (claimMarkers.empty()) { self->AddTimer(StartQuickbuildTimer, 0.5f); return; diff --git a/dScripts/BaseRandomServer.cpp b/dScripts/BaseRandomServer.cpp index 8cc4f993..293fb79a 100644 --- a/dScripts/BaseRandomServer.cpp +++ b/dScripts/BaseRandomServer.cpp @@ -52,7 +52,7 @@ void BaseRandomServer::SpawnSection(Entity* self, const std::string& sectionName } void BaseRandomServer::SetSpawnerNetwork(Entity* self, const std::string& spawnerName, int32_t spawnNum, LOT spawnLOT) { - const auto& spawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName); + const auto& spawners = Game::zoneManager->GetSpawnersByName(spawnerName); if (spawnLOT == 11217 && spawnNum > 1) { spawnNum = 1; diff --git a/dScripts/BaseSurvivalServer.cpp b/dScripts/BaseSurvivalServer.cpp index 270b6741..ffa86968 100644 --- a/dScripts/BaseSurvivalServer.cpp +++ b/dScripts/BaseSurvivalServer.cpp @@ -4,9 +4,12 @@ #include "EntityManager.h" #include "dZoneManager.h" #include "Player.h" -#include "MissionTaskType.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" #include "MissionComponent.h" #include "Character.h" +#include "Game.h" +#include "dConfig.h" void BaseSurvivalServer::SetGameVariables(Entity* self) { this->constants = std::move(GetConstants()); @@ -211,7 +214,7 @@ void BaseSurvivalServer::OnActivityTimerDone(Entity* self, const std::string& na ActivityTimerStart(self, PlaySpawnSoundTimer, 3, 3); } else if (name == PlaySpawnSoundTimer) { for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), spawnSoundGUID); } @@ -220,7 +223,7 @@ void BaseSurvivalServer::OnActivityTimerDone(Entity* self, const std::string& na } void BaseSurvivalServer::ResetStats(LWOOBJID playerID) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { // Boost all the player stats when loading in @@ -283,7 +286,7 @@ void BaseSurvivalServer::StartWaves(Entity* self) { state.waitingPlayers.clear(); for (const auto& playerID : state.players) { - const auto player = EntityManager::Instance()->GetEntity(playerID); + const auto player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { state.waitingPlayers.push_back(playerID); UpdatePlayer(self, playerID); @@ -310,7 +313,7 @@ bool BaseSurvivalServer::CheckAllPlayersDead() { auto deadPlayers = 0; for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr || player->GetIsDead()) { deadPlayers++; } @@ -322,9 +325,9 @@ bool BaseSurvivalServer::CheckAllPlayersDead() { void BaseSurvivalServer::SetPlayerSpawnPoints() { auto spawnerIndex = 1; for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { - auto possibleSpawners = EntityManager::Instance()->GetEntitiesInGroup("P" + std::to_string(spawnerIndex) + "_Spawn"); + auto possibleSpawners = Game::entityManager->GetEntitiesInGroup("P" + std::to_string(spawnerIndex) + "_Spawn"); if (!possibleSpawners.empty()) { auto* spawner = possibleSpawners.at(0); GameMessages::SendTeleport(playerID, spawner->GetPosition(), spawner->GetRotation(), player->GetSystemAddress(), true); @@ -347,12 +350,13 @@ void BaseSurvivalServer::GameOver(Entity* self) { SpawnerReset(spawnerNetworks.rewardNetworks); for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) continue; const auto score = GetActivityValue(self, playerID, 0); const auto time = GetActivityValue(self, playerID, 1); + SaveScore(self, playerID, score, time); GameMessages::SendNotifyClientZoneObject(self->GetObjectID(), u"Update_ScoreBoard", time, 0, playerID, std::to_string(score), UNASSIGNED_SYSTEM_ADDRESS); @@ -361,16 +365,16 @@ void BaseSurvivalServer::GameOver(Entity* self) { // Update all mission progression auto* missionComponent = player->GetComponent<MissionComponent>(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MINIGAME, time, self->GetObjectID(), + missionComponent->Progress(eMissionTaskType::PERFORM_ACTIVITY, time, self->GetObjectID(), self->GetVar<std::string>(MissionTypeVariable)); for (const auto& survivalMission : missionsToUpdate) { auto* mission = missionComponent->GetMission(survivalMission.first); if (mission != nullptr && (uint32_t)time >= survivalMission.second - && (mission->GetMissionState() == MissionState::MISSION_STATE_ACTIVE - || mission->GetMissionState() == MissionState::MISSION_STATE_COMPLETE_ACTIVE)) { + && (mission->GetMissionState() == eMissionState::ACTIVE + || mission->GetMissionState() == eMissionState::COMPLETE_ACTIVE)) { - mission->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); + mission->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); } } } @@ -398,7 +402,7 @@ void BaseSurvivalServer::SpawnerReset(SpawnerNetworkCollection& spawnerNetworkCo for (auto& spawner : spawnerNetworkCollection.networks) { for (auto& spawnerName : spawner.names) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName + spawner.number); + auto spawners = Game::zoneManager->GetSpawnersByName(spawnerName + spawner.number); if (!spawners.empty()) { auto* spawnerObject = spawners.at(0); @@ -427,7 +431,7 @@ void BaseSurvivalServer::SpawnerUpdate(Entity* self, SpawnerNetworkCollection& s // If we want to spawn something specific now if (amount != 0) { auto spawnerNetwork = spawnerNetworkCollection.networks.at(0); - auto possibleSpawners = dZoneManager::Instance()->GetSpawnersByName(spawnerNetwork.names.at(0) + spawnerNetwork.number); + auto possibleSpawners = Game::zoneManager->GetSpawnersByName(spawnerNetwork.names.at(0) + spawnerNetwork.number); if (!possibleSpawners.empty()) { SpawnNow(possibleSpawners.at(0), amount); return; @@ -443,7 +447,7 @@ void BaseSurvivalServer::SpawnerUpdate(Entity* self, SpawnerNetworkCollection& s const auto& name = spawnerNetwork.names.at(i); const auto& toSpawn = newSet.at(i); - auto possibleSpawners = dZoneManager::Instance()->GetSpawnersByName(name + spawnerNetwork.number); + auto possibleSpawners = Game::zoneManager->GetSpawnersByName(name + spawnerNetwork.number); if (!possibleSpawners.empty()) { SpawnNow(possibleSpawners.front(), toSpawn); } @@ -468,7 +472,7 @@ std::vector<uint32_t> BaseSurvivalServer::GetRandomMobSet(SpawnerNetworkCollecti if (mobSets.sets.find(spawnerNetworkCollection.mobSetName) != mobSets.sets.end()) { auto mobSet = mobSets.sets.at(spawnerNetworkCollection.mobSetName); if (setNumber < mobSet.size()) { - return mobSet.at(setNumber).at(rand() % mobSet.at(setNumber).size()); + return mobSet.at(setNumber).at(GeneralUtils::GenerateRandomNumber<int32_t>(0, mobSet.at(setNumber).size() - 1)); } } @@ -483,7 +487,7 @@ SpawnerNetwork BaseSurvivalServer::GetRandomSpawner(SpawnerNetworkCollection& sp } if (!validSpawners.empty()) { - auto spawner = validSpawners.at(rand() % validSpawners.size()); + auto spawner = validSpawners.at(GeneralUtils::GenerateRandomNumber<int32_t>(0, validSpawners.size() - 1)); spawner.isActive = true; return spawner; } @@ -494,7 +498,7 @@ SpawnerNetwork BaseSurvivalServer::GetRandomSpawner(SpawnerNetworkCollection& sp void BaseSurvivalServer::ActivateSpawnerNetwork(SpawnerNetworkCollection& spawnerNetworkCollection) { for (auto& spawner : spawnerNetworkCollection.networks) { for (const auto& spawnerName : spawner.names) { - auto possibleSpawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName + spawner.number); + auto possibleSpawners = Game::zoneManager->GetSpawnersByName(spawnerName + spawner.number); if (!possibleSpawners.empty()) { auto* spawnerObject = possibleSpawners.at(0); spawnerObject->Activate(); @@ -508,7 +512,7 @@ void BaseSurvivalServer::UpdateMobLots(SpawnerNetworkCollection& spawnerNetworkC for (auto& spawner : spawnerNetworkCollection.networks) { for (auto& spawnerName : spawner.names) { if (!spawnerName.empty()) { - auto spawnerObjects = dZoneManager::Instance()->GetSpawnersByName(spawnerName + spawner.number); + auto spawnerObjects = Game::zoneManager->GetSpawnersByName(spawnerName + spawner.number); if (!spawnerObjects.empty()) { auto splitName = GeneralUtils::SplitString(spawnerName, '_'); auto cleanName = splitName.size() > 1 ? splitName.at(1) : splitName.at(0); diff --git a/dScripts/BaseWavesGenericEnemy.cpp b/dScripts/BaseWavesGenericEnemy.cpp index 3270e2d8..1e66bfeb 100644 --- a/dScripts/BaseWavesGenericEnemy.cpp +++ b/dScripts/BaseWavesGenericEnemy.cpp @@ -6,7 +6,7 @@ void BaseWavesGenericEnemy::OnStartup(Entity* self) { } void BaseWavesGenericEnemy::OnDie(Entity* self, Entity* killer) { - auto* zoneControlObject = dZoneManager::Instance()->GetZoneControlObject(); + auto* zoneControlObject = Game::zoneManager->GetZoneControlObject(); if (zoneControlObject != nullptr) { zoneControlObject->OnFireEventServerSide(killer, "Survival_Update", GetPoints()); } diff --git a/dScripts/BaseWavesServer.cpp b/dScripts/BaseWavesServer.cpp index 8c32f984..1090d8c5 100644 --- a/dScripts/BaseWavesServer.cpp +++ b/dScripts/BaseWavesServer.cpp @@ -4,7 +4,8 @@ #include "EntityManager.h" #include "dZoneManager.h" #include "Player.h" -#include "MissionTaskType.h" +#include "eMissionTaskType.h" +#include "eMissionState.h" #include "MissionComponent.h" #include "Character.h" @@ -194,7 +195,7 @@ void BaseWavesServer::OnActivityTimerDone(Entity* self, const std::string& name) ActivityTimerStart(self, PlaySpawnSoundTimer, 3, 3); } else if (name == PlaySpawnSoundTimer) { for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), spawnSoundGUID); } @@ -215,7 +216,7 @@ void BaseWavesServer::OnActivityTimerDone(Entity* self, const std::string& name) } else if (name == GameOverWinTimer) { GameOver(self, true); } else if (name == CinematicDoneTimer) { - for (auto* boss : EntityManager::Instance()->GetEntitiesInGroup("boss")) { + for (auto* boss : Game::entityManager->GetEntitiesInGroup("boss")) { boss->OnFireEventServerSide(self, "startAI"); } } @@ -223,7 +224,7 @@ void BaseWavesServer::OnActivityTimerDone(Entity* self, const std::string& name) // Done void BaseWavesServer::ResetStats(LWOOBJID playerID) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { // Boost all the player stats when loading in @@ -283,7 +284,7 @@ void BaseWavesServer::StartWaves(Entity* self) { state.waitingPlayers.clear(); for (const auto& playerID : state.players) { - const auto player = EntityManager::Instance()->GetEntity(playerID); + const auto player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { state.waitingPlayers.push_back(playerID); @@ -308,7 +309,7 @@ bool BaseWavesServer::CheckAllPlayersDead() { auto deadPlayers = 0; for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr || player->GetIsDead()) { deadPlayers++; } @@ -321,9 +322,9 @@ bool BaseWavesServer::CheckAllPlayersDead() { void BaseWavesServer::SetPlayerSpawnPoints(const LWOOBJID& specificPlayerID) { auto spawnerIndex = 1; for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr && (specificPlayerID == LWOOBJID_EMPTY || playerID == specificPlayerID)) { - auto possibleSpawners = EntityManager::Instance()->GetEntitiesInGroup("P" + std::to_string(spawnerIndex) + "_Spawn"); + auto possibleSpawners = Game::entityManager->GetEntitiesInGroup("P" + std::to_string(spawnerIndex) + "_Spawn"); if (!possibleSpawners.empty()) { auto* spawner = possibleSpawners.at(0); GameMessages::SendTeleport(playerID, spawner->GetPosition(), spawner->GetRotation(), player->GetSystemAddress(), true); @@ -352,7 +353,7 @@ void BaseWavesServer::GameOver(Entity* self, bool won) { ClearSpawners(); for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player == nullptr) continue; @@ -373,10 +374,11 @@ void BaseWavesServer::GameOver(Entity* self, bool won) { // Update all mission progression auto* missionComponent = player->GetComponent<MissionComponent>(); if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MINIGAME, time, self->GetObjectID(), self->GetVar<std::string>(MissionTypeVariable)); + missionComponent->Progress(eMissionTaskType::PERFORM_ACTIVITY, time, self->GetObjectID(), self->GetVar<std::string>(MissionTypeVariable)); } StopActivity(self, playerID, wave, time, score); + SaveScore(self, playerID, wave, time); } } @@ -391,7 +393,7 @@ void BaseWavesServer::GameWon(Entity* self) { // Done void BaseWavesServer::SpawnNow(const std::string& spawnerName, uint32_t amount, LOT spawnLot) { - const auto spawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName); + const auto spawners = Game::zoneManager->GetSpawnersByName(spawnerName); for (auto* spawner : spawners) { if (spawnLot != LOT_NULL) { spawner->SetSpawnLot(spawnLot); @@ -428,8 +430,8 @@ void BaseWavesServer::SpawnWave(Entity* self) { } for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); - if (player != nullptr) { + auto* player = Game::entityManager->GetEntity(playerID); + if (player && player->GetIsDead()) { player->Resurrect(); } } @@ -470,7 +472,7 @@ bool BaseWavesServer::UpdateSpawnedEnemies(Entity* self, LWOOBJID enemyID, uint3 state.currentSpawned--; - auto* enemy = EntityManager::Instance()->GetEntity(enemyID); + auto* enemy = Game::entityManager->GetEntity(enemyID); if (enemy != nullptr && enemy->IsPlayer() && IsPlayerInActivity(self, enemyID)) { SetActivityValue(self, enemyID, 0, GetActivityValue(self, enemyID, 0) + score); } @@ -498,7 +500,7 @@ bool BaseWavesServer::UpdateSpawnedEnemies(Entity* self, LWOOBJID enemyID, uint3 const auto soloWaveMissions = waves.at(completedWave).soloMissions; for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr && !player->GetIsDead()) { SetActivityValue(self, playerID, 1, currentTime); SetActivityValue(self, playerID, 2, state.waveNumber); @@ -510,15 +512,15 @@ bool BaseWavesServer::UpdateSpawnedEnemies(Entity* self, LWOOBJID enemyID, uint3 // Get the mission state auto missionState = missionComponent->GetMissionState(missionID); // For some reason these achievements are not accepted by default, so we accept them here if they arent already. - if (missionState != MissionState::MISSION_STATE_COMPLETE && missionState != MissionState::MISSION_STATE_UNKNOWN) { + if (missionState != eMissionState::COMPLETE && missionState != eMissionState::UNKNOWN) { missionComponent->AcceptMission(missionID); missionState = missionComponent->GetMissionState(missionID); } - if (missionState != MissionState::MISSION_STATE_COMPLETE) { + if (missionState != eMissionState::COMPLETE) { auto mission = missionComponent->GetMission(missionID); if (mission != nullptr) { - mission->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); + mission->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); } } } @@ -528,15 +530,15 @@ bool BaseWavesServer::UpdateSpawnedEnemies(Entity* self, LWOOBJID enemyID, uint3 // Get the mission state auto missionState = missionComponent->GetMissionState(missionID); // For some reason these achievements are not accepted by default, so we accept them here if they arent already. - if (missionState != MissionState::MISSION_STATE_COMPLETE && missionState != MissionState::MISSION_STATE_UNKNOWN) { + if (missionState != eMissionState::COMPLETE && missionState != eMissionState::UNKNOWN) { missionComponent->AcceptMission(missionID); missionState = missionComponent->GetMissionState(missionID); } - if (missionState != MissionState::MISSION_STATE_COMPLETE) { + if (missionState != eMissionState::COMPLETE) { auto mission = missionComponent->GetMission(missionID); if (mission != nullptr) { - mission->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); + mission->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); } } } @@ -557,21 +559,21 @@ bool BaseWavesServer::UpdateSpawnedEnemies(Entity* self, LWOOBJID enemyID, uint3 // Done void BaseWavesServer::UpdateMissionForAllPlayers(Entity* self, uint32_t missionID) { for (const auto& playerID : state.players) { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (player != nullptr) { auto* missionComponent = player->GetComponent<MissionComponent>(); if (missionComponent == nullptr) return; // Get the mission state auto missionState = missionComponent->GetMissionState(missionID); // For some reason these achievements are not accepted by default, so we accept them here if they arent already. - if (missionState != MissionState::MISSION_STATE_COMPLETE && missionState != MissionState::MISSION_STATE_UNKNOWN) { + if (missionState != eMissionState::COMPLETE && missionState != eMissionState::UNKNOWN) { missionComponent->AcceptMission(missionID); missionState = missionComponent->GetMissionState(missionID); } - if (missionState != MissionState::MISSION_STATE_COMPLETE) { + if (missionState != eMissionState::COMPLETE) { auto mission = missionComponent->GetMission(missionID); if (mission != nullptr) { - mission->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); + mission->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); } } } @@ -580,7 +582,7 @@ void BaseWavesServer::UpdateMissionForAllPlayers(Entity* self, uint32_t missionI void BaseWavesServer::ClearSpawners() { for (const auto& spawnerName : spawners) { - const auto spawnerObjects = dZoneManager::Instance()->GetSpawnersByName(spawnerName); + const auto spawnerObjects = Game::zoneManager->GetSpawnersByName(spawnerName); for (auto* spawnerObject : spawnerObjects) { spawnerObject->Reset(); diff --git a/dScripts/BootyDigServer.cpp b/dScripts/BootyDigServer.cpp deleted file mode 100644 index d38791d8..00000000 --- a/dScripts/BootyDigServer.cpp +++ /dev/null @@ -1,52 +0,0 @@ -#include "BootyDigServer.h" -#include "EntityManager.h" -#include "RenderComponent.h" -#include "MissionComponent.h" -#include "MissionTaskType.h" - -void BootyDigServer::OnStartup(Entity* self) { - auto* zoneControlObject = EntityManager::Instance()->GetZoneControlEntity(); - if (zoneControlObject != nullptr) { - zoneControlObject->OnFireEventServerSide(self, "CheckForPropertyOwner"); - } -} - -void BootyDigServer::OnPlayerLoaded(Entity* self, Entity* player) { - auto* zoneControlObject = EntityManager::Instance()->GetZoneControlEntity(); - if (zoneControlObject != nullptr) { - zoneControlObject->OnFireEventServerSide(self, "CheckForPropertyOwner"); - } -} - -void -BootyDigServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - - auto propertyOwner = self->GetNetworkVar<std::string>(u"PropertyOwnerID"); - auto* player = self->GetParentEntity(); - if (player == nullptr) - return; - - if (args == "ChestReady" && (propertyOwner == std::to_string(LWOOBJID_EMPTY) || player->GetVar<bool>(u"bootyDug"))) { - self->Smash(self->GetObjectID(), SILENT); - } else if (args == "ChestOpened") { - // Make sure players only dig up one booty per instance - player->SetVar<bool>(u"bootyDug", true); - - auto* missionComponent = player->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - auto* mission = missionComponent->GetMission(1881); - if (mission != nullptr && (mission->GetMissionState() == MissionState::MISSION_STATE_ACTIVE || mission->GetMissionState() == MissionState::MISSION_STATE_COMPLETE_ACTIVE)) { - mission->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) - renderComponent->PlayEffect(7730, u"cast", "bootyshine"); - - LootGenerator::Instance().DropLoot(player, self, 231, 75, 75); - } - } - } else if (args == "ChestDead") { - self->Smash(player->GetObjectID(), SILENT); - } -} diff --git a/dScripts/BossSpiderQueenEnemyServer.cpp b/dScripts/BossSpiderQueenEnemyServer.cpp deleted file mode 100644 index ef81c888..00000000 --- a/dScripts/BossSpiderQueenEnemyServer.cpp +++ /dev/null @@ -1,710 +0,0 @@ -#include "BossSpiderQueenEnemyServer.h" - -#include "GeneralUtils.h" - -#include "MissionComponent.h" -#include "EntityManager.h" -#include "Entity.h" -#include "dZoneManager.h" - -#include "DestroyableComponent.h" -#include "ControllablePhysicsComponent.h" -#include "BaseCombatAIComponent.h" - -#include "GameMessages.h" -#include "SkillComponent.h" - -#include <vector> - -//---------------------------------------------------------------- -//--On Startup, process necessary AI events -//---------------------------------------------------------------- -void BossSpiderQueenEnemyServer::OnStartup(Entity* self) { - // Make immune to stuns - //self:SetStunImmunity{ StateChangeType = "PUSH", bImmuneToStunAttack = true, bImmuneToStunMove = true, bImmuneToStunTurn = true, bImmuneToStunUseItem = true, bImmuneToStunEquip = true, bImmuneToStunInteract = true, bImmuneToStunJump = true } - - // Make immune to knockbacks and pulls - //self:SetStatusImmunity{ StateChangeType = "PUSH", bImmuneToPullToPoint = true, bImmuneToKnockback = true, bImmuneToInterrupt = true } - - //Get our components: - destroyable = static_cast<DestroyableComponent*>(self->GetComponent(COMPONENT_TYPE_DESTROYABLE)); - controllable = static_cast<ControllablePhysicsComponent*>(self->GetComponent(COMPONENT_TYPE_CONTROLLABLE_PHYSICS)); - combat = static_cast<BaseCombatAIComponent*>(self->GetComponent(COMPONENT_TYPE_BASE_COMBAT_AI)); - - if (!destroyable || !controllable) return; - - // Determine Spider Boss health transition thresholds - int spiderBossHealth = destroyable->GetMaxHealth(); - int transitionTickHealth = spiderBossHealth / 3; - - int Stage2HealthThreshold = spiderBossHealth - transitionTickHealth; - int Stage3HealthThreshold = spiderBossHealth - (2 * transitionTickHealth); - ThresholdTable = { Stage2HealthThreshold, Stage3HealthThreshold }; - - originRotation = controllable->GetRotation(); - combat->SetStunImmune(true); - - m_CurrentBossStage = 1; - - // Obtain faction and collision group to save for subsequent resets - //self : SetVar("SBFactionList", self:GetFaction().factionList) - //self : SetVar("SBCollisionGroup", self:GetCollisionGroup().colGroup) -} - -void BossSpiderQueenEnemyServer::OnDie(Entity* self, Entity* killer) { - if (dZoneManager::Instance()->GetZoneID().GetMapID() == instanceZoneID) { - auto* missionComponent = killer->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) - return; - - missionComponent->CompleteMission(instanceMissionID); - } - - Game::logger->Log("BossSpiderQueenEnemyServer", "Starting timer..."); - - // There is suppose to be a 0.1 second delay here but that may be admitted? - auto* controller = EntityManager::Instance()->GetZoneControlEntity(); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", 10, 0, 0, "", UNASSIGNED_SYSTEM_ADDRESS); - - self->SetPosition({ 10000, 0, 10000 }); - - EntityManager::Instance()->SerializeEntity(self); - - controller->OnFireEventServerSide(self, "ClearProperty"); -} - -void BossSpiderQueenEnemyServer::WithdrawSpider(Entity* self, const bool withdraw) { - const auto withdrawn = self->GetBoolean(u"isWithdrawn"); - - if (withdrawn == withdraw) { - return; - } - - if (withdraw) { - //Move spider away from battle zone - // Disabled because we cant option the reset collition group right now - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", 10, 0, 0, "", UNASSIGNED_SYSTEM_ADDRESS); - - //First rotate for anim - NiQuaternion rot = NiQuaternion::IDENTITY; - - controllable->SetStatic(false); - - controllable->SetRotation(rot); - - controllable->SetStatic(true); - - controllable->SetDirtyPosition(true); - - rot = controllable->GetRotation(); - - EntityManager::Instance()->SerializeEntity(self); - - auto* baseCombatAi = self->GetComponent<BaseCombatAIComponent>(); - - baseCombatAi->SetDisabled(true); - - float animTime = PlayAnimAndReturnTime(self, spiderWithdrawAnim); - float withdrawTime = animTime - 0.25f; - - combat->SetStunImmune(false); - combat->Stun(withdrawTime + 6.0f); - combat->SetStunImmune(true); - - //TODO: Set faction to -1 and set immunity - destroyable->SetFaction(-1); - destroyable->SetIsImmune(true); - EntityManager::Instance()->SerializeEntity(self); - - self->AddTimer("WithdrawComplete", withdrawTime + 1.0f); - waitForIdle = true; - } else { - controllable->SetStatic(false); - - //Cancel all remaining timers for say idle anims: - self->CancelAllTimers(); - - auto* baseCombatAi = self->GetComponent<BaseCombatAIComponent>(); - - baseCombatAi->SetDisabled(false); - - // Move the Spider to its ground location - // preparing its stage attacks, and removing invulnerability - //destroyable->SetIsImmune(false); - - // Run the advance animation and prepare a timer for resuming AI - float animTime = PlayAnimAndReturnTime(self, spiderAdvanceAnim); - animTime += 1.f; - - float attackPause = animTime - 0.4f; - - destroyable->SetFaction(4); - destroyable->SetIsImmune(false); - - //Advance stage - m_CurrentBossStage++; - - //Reset the current wave death counter - m_DeathCounter = 0; - - EntityManager::Instance()->SerializeEntity(self); - - // Prepare a timer for post leap attack - self->AddTimer("AdvanceAttack", attackPause); - - // Prepare a timer for post leap - self->AddTimer("AdvanceComplete", animTime); - } - - self->SetBoolean(u"isWithdrawn", withdraw); -} - -void BossSpiderQueenEnemyServer::SpawnSpiderWave(Entity* self, int spiderCount) { - // The Spider Queen Boss is withdrawing and requesting the spawn - // of a hatchling wave - - /*auto SpiderEggNetworkID = self->GetI64(u"SpiderEggNetworkID"); - if (SpiderEggNetworkID == 0) return;*/ - - // Clamp invalid Spiderling number requests to the maximum amount of eggs available - if ((spiderCount > maxSpiderEggCnt) || (spiderCount < 0)) - spiderCount = maxSpiderEggCnt; - - // Reset our wave manager reference variables - hatchCounter = spiderCount; - hatchList = {}; - - Game::logger->Log("SpiderQueen", "Trying to spawn %i spiders", hatchCounter); - - - // Run the wave manager - SpiderWaveManager(self); - -} - -void BossSpiderQueenEnemyServer::SpiderWaveManager(Entity* self) { - auto SpiderEggNetworkID = self->GetI64(u"SpiderEggNetworkID"); - - // Reset the spider egg spawner network to ensure a maximum number of eggs - //SpiderEggNetworkID:SpawnerReset() - - // Obtain a list of all the eggs on the egg spawner network - - //auto spiderEggList = SpiderEggNetworkID:SpawnerGetAllObjectIDsSpawned().objects; - - //if (table.maxn(spiderEggList) <= 0) { - // self->AddTimer("PollSpiderWaveManager", 1.0f); - // return; - //} - // - //// A check for (wave mangement across multiple spawn iterations - //if(hatchCounter < spiderWaveCnt) { - // // We have already prepped some objects for (hatching, - // // remove them from our list for (random egg pulls - // for (i, sVal in ipairs(spiderEggList) { - // if(hatchList[sVal:GetID()]) { - // // We have found a prepped egg, remove it from the spiderEggList - // spiderEggList[i] = nil - // } - // } - - //} - - - - std::vector<LWOOBJID> spiderEggs{}; - - auto spooders = EntityManager::Instance()->GetEntitiesInGroup("EGG"); - for (auto spodder : spooders) { - spiderEggs.push_back(spodder->GetObjectID()); - } - - // Select a number of random spider eggs from the list equal to the - // current number needed to complete the current wave - for (int i = 0; i < hatchCounter; i++) { - // Select a random spider egg - auto randomEggLoc = GeneralUtils::GenerateRandomNumber<int>(0, spiderEggs.size() - 1); - auto randomEgg = spiderEggs[randomEggLoc]; - - //Just a quick check to try and prevent dupes: - for (auto en : hatchList) { - if (en == randomEgg) { - randomEggLoc++; - randomEgg = spiderEggs[randomEggLoc]; - } - } - - if (randomEgg) { - auto* eggEntity = EntityManager::Instance()->GetEntity(randomEgg); - - if (eggEntity == nullptr) { - continue; - } - - // Prep the selected spider egg - //randomEgg:FireEvent{s}erID=self, args="prepEgg"} - eggEntity->OnFireEventServerSide(self, "prepEgg"); - Game::logger->Log("SpiderQueen", "Prepping egg %llu", eggEntity->GetObjectID()); - - // Add the prepped egg to our hatchList - hatchList.push_back(eggEntity->GetObjectID()); - - // Decrement the hatchCounter - hatchCounter = hatchCounter - 1; - } - - // Remove it from our spider egg list - //table.remove(spiderEggList, randomEggLoc); - spiderEggs[randomEggLoc] = LWOOBJID_EMPTY; - - if (spiderEggs.size() <= 0 || (hatchCounter <= 0)) { - break; - } - } - - if (hatchCounter > 0) { - // We still have more eggs to hatch, poll the SpiderWaveManager again - self->AddTimer("PollSpiderWaveManager", 1.0f); - - } else { - // We have successfully readied a full wave - // initiate hatching! - for (auto egg : hatchList) { - auto* eggEntity = EntityManager::Instance()->GetEntity(egg); - - if (eggEntity == nullptr) { - continue; - } - - eggEntity->OnFireEventServerSide(self, "hatchEgg"); - Game::logger->Log("SpiderQueen", "hatching egg %llu", eggEntity->GetObjectID()); - - auto time = PlayAnimAndReturnTime(self, spiderWithdrawIdle); - combat->SetStunImmune(false); - combat->Stun(time += 6.0f); - combat->SetStunImmune(true); - - //self->AddTimer("disableWaitForIdle", defaultAnimPause); - self->AddTimer("checkForSpiders", 6.0f); - - } - - hatchList.clear(); - - } - -} - -void BossSpiderQueenEnemyServer::ToggleForSpecial(Entity* self, const bool state) { - self->SetBoolean(u"stoppedFlag", state); - - combat->SetDisabled(state); -} - -void BossSpiderQueenEnemyServer::RunRainOfFire(Entity* self) { - if (self->GetBoolean(u"stoppedFlag")) { - self->AddTimer("ROF", GeneralUtils::GenerateRandomNumber<float>(10, 20)); - - return; - } - - ToggleForSpecial(self, true); - - impactList.clear(); - - auto index = 0u; - for (const auto& rofGroup : ROFTargetGroupIDTable) { - const auto spawners = dZoneManager::Instance()->GetSpawnersInGroup(rofGroup); - - std::vector<LWOOBJID> spawned; - - for (auto* spawner : spawners) { - for (const auto* node : spawner->m_Info.nodes) { - spawned.insert(spawned.end(), node->entities.begin(), node->entities.end()); - } - } - - if (index == 0) { - impactList.insert(impactList.end(), spawned.begin(), spawned.end()); - } else { - const auto randomIndex = GeneralUtils::GenerateRandomNumber<int32_t>(0, spawned.size() - 1); - - impactList.push_back(spawned[randomIndex]); - } - - index++; - } - - const auto animTime = PlayAnimAndReturnTime(self, spiderROFAnim); - - self->AddTimer("StartROF", animTime); -} - -void BossSpiderQueenEnemyServer::RainOfFireManager(Entity* self) { - if (!impactList.empty()) { - auto* entity = EntityManager::Instance()->GetEntity(impactList[0]); - - impactList.erase(impactList.begin()); - - if (entity == nullptr) { - Game::logger->Log("BossSpiderQueenEnemyServer", "Failed to find impact!"); - - return; - } - - auto* skillComponent = entity->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - Game::logger->Log("BossSpiderQueenEnemyServer", "Failed to find impact skill component!"); - - return; - } - - skillComponent->CalculateBehavior(1376, 32168, LWOOBJID_EMPTY, true); - - self->AddTimer("PollROFManager", 0.5f); - - return; - } - - ToggleForSpecial(self, false); - - self->AddTimer("ROF", GeneralUtils::GenerateRandomNumber<float>(20, 40)); -} - -void BossSpiderQueenEnemyServer::RapidFireShooterManager(Entity* self) { - if (attackTargetTable.empty()) { - const auto animationTime = PlayAnimAndReturnTime(self, spiderJeerAnim); - - self->AddTimer("RFSTauntComplete", animationTime); - - ToggleForSpecial(self, false); - - return; - } - - const auto target = attackTargetTable[0]; - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - skillComponent->CalculateBehavior(1394, 32612, target, true); - - attackTargetTable.erase(attackTargetTable.begin()); - - self->AddTimer("PollRFSManager", 0.3f); -} - -void BossSpiderQueenEnemyServer::RunRapidFireShooter(Entity* self) { - /* - const auto targets = EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_CHARACTER); - */ - - const auto targets = self->GetTargetsInPhantom(); - - if (self->GetBoolean(u"stoppedFlag")) { - self->AddTimer("RFS", GeneralUtils::GenerateRandomNumber<float>(5, 10)); - - return; - } - - if (targets.empty()) { - Game::logger->Log("BossSpiderQueenEnemyServer", "Failed to find RFS targets"); - - self->AddTimer("RFS", GeneralUtils::GenerateRandomNumber<float>(5, 10)); - - return; - } - - ToggleForSpecial(self, true); - - const auto randomTarget = GeneralUtils::GenerateRandomNumber<int32_t>(0, targets.size() - 1); - - auto attackFocus = targets[randomTarget]; - - attackTargetTable.push_back(attackFocus); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - skillComponent->CalculateBehavior(1480, 36652, attackFocus, true); - - RapidFireShooterManager(self); - - PlayAnimAndReturnTime(self, spiderSingleShot); - - Game::logger->Log("BossSpiderQueenEnemyServer", "Ran RFS"); - - self->AddTimer("RFS", GeneralUtils::GenerateRandomNumber<float>(10, 15)); -} - -void BossSpiderQueenEnemyServer::OnTimerDone(Entity* self, const std::string timerName) { - if (timerName == "PollSpiderWaveManager") { - //Call the manager again to attempt to finish prepping a Spiderling wave - //Run the wave manager - SpiderWaveManager(self); - } else if (timerName == "disableWaitForIdle") { waitForIdle = false; } else if (timerName == "checkForSpiders") { - //Don't do anything if we ain't withdrawn: - const auto withdrawn = self->GetBoolean(u"isWithdrawn"); - if (!withdrawn) return; - - NiQuaternion rot = NiQuaternion::IDENTITY; - - //First rotate for anim - controllable->SetStatic(false); - controllable->SetRotation(rot); - controllable->SetStatic(true); - EntityManager::Instance()->SerializeEntity(self); - - //Play the Spider Boss' mountain idle anim - auto time = PlayAnimAndReturnTime(self, spiderWithdrawIdle); - combat->SetStunImmune(false); - combat->Stun(time); - combat->SetStunImmune(true); - - rot = controllable->GetRotation(); - - //If there are still baby spiders, don't do anyhting either - const auto spiders = EntityManager::Instance()->GetEntitiesInGroup("BabySpider"); - if (spiders.size() > 0) - self->AddTimer("checkForSpiders", time); - else - WithdrawSpider(self, false); - } else if (timerName == "PollROFManager") { - //Call the manager again to attempt to initiate an impact on another random location - //Run the ROF Manager - RainOfFireManager(self); - - } else if (timerName == "PollRFSManager") { - //Call the manager again to attempt to initiate a rapid fire shot at the next sequential target - //Run the ROF Manager - RapidFireShooterManager(self); - - } else if (timerName == "StartROF") { - //Re-enable Spider Boss - //ToggleForSpecial(self, false); - - RainOfFireManager(self); - - } else if (timerName == "PollSpiderSkillManager") { - //Call the skill manager again to attempt to run the current Spider Boss - //stage's special attack again - //SpiderSkillManager(self, true); - PlayAnimAndReturnTime(self, spiderJeerAnim); - - } else if (timerName == "RFS") { - RunRapidFireShooter(self); - } else if (timerName == "ROF") { - RunRainOfFire(self); - } else if (timerName == "RFSTauntComplete") { - //Determine an appropriate random time to check our manager again - // local spiderCooldownDelay = math.random(s1DelayMin, s1DelayMax) - - //Set a timer based on our random cooldown determination - //to pulse the SpiderSkillManager again - - //GAMEOBJ:GetTimer():AddTimerWithCancel(spiderCooldownDelay, "PollSpiderSkillManager", self) - - //Re-enable Spider Boss - //ToggleForSpecial(self, false); - - } else if (timerName == "WithdrawComplete") { - //Play the Spider Boss' mountain idle anim - PlayAnimAndReturnTime(self, spiderWithdrawIdle); - - //The Spider Boss has retreated, hatch a wave! - int currentStage = m_CurrentBossStage; - - //Prepare a Spiderling wave and initiate egg hatch events - //self->SetVar(u"SpiderWaveCount", ) - - //TODO: Actually spawn the spiders here - hatchCounter = 2; - if (currentStage > 1) hatchCounter++; - - SpawnSpiderWave(self, spiderWaveCntTable[currentStage - 1]); - - } else if (timerName == "AdvanceAttack") { - //TODO: Can we even do knockbacks yet? @Wincent01 - // Yes ^ - - //Fire the melee smash skill to throw players back - /*local landingTarget = self:GetVar("LandingTarget") or false - - if((landingTarget) and (landingTarget:Exists())) { - local advSmashFlag = landingTarget:CastSkill{skillID = bossLandingSkill} - landingTarget:PlayEmbeddedEffectOnAllClientsNearObject{radius = 100, fromObjectID = landingTarget, effectName = "camshake-bridge"} - }*/ - - auto landingTarget = self->GetI64(u"LandingTarget"); - auto landingEntity = EntityManager::Instance()->GetEntity(landingTarget); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(bossLandingSkill, 37739, LWOOBJID_EMPTY); - } - - if (landingEntity) { - auto* landingSkill = landingEntity->GetComponent<SkillComponent>(); - - if (landingSkill != nullptr) { - landingSkill->CalculateBehavior(bossLandingSkill, 37739, LWOOBJID_EMPTY, true); - } - } - - GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(self, u"camshake-bridge", self->GetObjectID(), 100.0f); - - } else if (timerName == "AdvanceComplete") { - //Reset faction and collision - /*local SBFactionList = self:GetVar("SBFactionList") - local SBCollisionGroup = self:GetVar("SBCollisionGroup") - - for i, fVal in ipairs(SBFactionList) { - if(i == 1) { - //Our first faction - flush and add - self:SetFaction{faction = fVal} - else - //Add - self:ModifyFaction{factionID = fVal, bAddFaction = true} - } - }*/ - - /* - auto SBCollisionGroup = self->GetI32(u"SBCollisionGroup"); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", SBCollisionGroup, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - */ - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SetColGroup", 11, 0, 0, "", UNASSIGNED_SYSTEM_ADDRESS); - - //Wind up, telegraphing next round - float animTime = PlayAnimAndReturnTime(self, spiderJeerAnim); - self->AddTimer("AdvanceTauntComplete", animTime); - - } else if (timerName == "AdvanceTauntComplete") { - - //Declare a default special Spider Boss skill cooldown - int spiderCooldownDelay = 10; - - if (m_CurrentBossStage == 2) { - spiderCooldownDelay = GeneralUtils::GenerateRandomNumber<int>(s1DelayMin, s1DelayMax); - } else if (m_CurrentBossStage == 3) { - spiderCooldownDelay = GeneralUtils::GenerateRandomNumber<int>(s2DelayMin, s2DelayMax); - } - - //Set a timer based on our random cooldown determination - //to pulse the SpiderSkillManager - self->AddTimer("PollSpiderSkillManager", spiderCooldownDelay); - - //Remove current status immunity - /*self:SetStatusImmunity{ StateChangeType = "POP", bImmuneToSpeed = true, bImmuneToBasicAttack = true, bImmuneToDOT = true} - - self:SetStunned{StateChangeType = "POP", - bCantMove = true, - bCantJump = true, - bCantTurn = true, - bCantAttack = true, - bCantUseItem = true, - bCantEquip = true, - bCantInteract = true, - bIgnoreImmunity = true}*/ - - destroyable->SetIsImmune(false); - destroyable->SetFaction(4); - - EntityManager::Instance()->SerializeEntity(self); - - } else if (timerName == "Clear") { - EntityManager::Instance()->FireEventServerSide(self, "ClearProperty"); - self->CancelAllTimers(); - } else if (timerName == "UnlockSpecials") { - //We no longer need to lock specials - self->SetBoolean(u"bSpecialLock", false); - - //Did we queue a spcial attack? - if (self->GetBoolean(u"bSpecialQueued")) { - self->SetBoolean(u"bSpecialQueued", false); - //SpiderSkillManager(self, true); - } - } -} - -void BossSpiderQueenEnemyServer::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - if (m_CurrentBossStage > 0 && !self->HasTimer("RFS")) { - self->AddTimer("RFS", 5.0f); - } - - if (m_CurrentBossStage > 0 && !self->HasTimer("ROF")) { - self->AddTimer("ROF", 10.0f); - } - - if (m_CurrentBossStage > ThresholdTable.size()) { - return; - } - - int currentThreshold = ThresholdTable[m_CurrentBossStage - 1]; - - if (destroyable->GetHealth() <= currentThreshold) { - auto isWithdrawn = self->GetBoolean(u"isWithdrawn"); - - if (!isWithdrawn) { - self->CancelAllTimers(); - - self->SetBoolean(u"isSpecialAttacking", false); - self->SetBoolean(u"bSpecialLock", false); - - WithdrawSpider(self, true); - } - } -} - -void BossSpiderQueenEnemyServer::OnUpdate(Entity* self) { - auto isWithdrawn = self->GetBoolean(u"isWithdrawn"); - - if (!isWithdrawn) return; - - if (controllable->GetRotation() == NiQuaternion::IDENTITY) { - return; - } - - controllable->SetStatic(false); - controllable->SetRotation(NiQuaternion::IDENTITY); - controllable->SetStatic(true); - - EntityManager::Instance()->SerializeEntity(self); - - //if (waitForIdle) return; - - ////Play the Spider Boss' mountain idle anim - //PlayAnimAndReturnTime(self, spiderWithdrawIdle); - - ////If there are still baby spiders, don't do anyhting either - //auto spooders = EntityManager::Instance()->GetEntitiesInGroup("BabySpider"); - //if (spooders.size() > 0) return; - //else - // WithdrawSpider(self, false); -} - -//---------------------------------------------- -//--Utility function capable of playing a priority -//-- animation on a targetand returning either the -//-- anim time, or a desired default -//---------------------------------------------- -float BossSpiderQueenEnemyServer::PlayAnimAndReturnTime(Entity* self, const std::u16string& animID) { - //TODO: Get the actual animation time - - // Get the anim time - float animTimer = defaultAnimPause; //self:GetAnimationTime{animationID = animID}.time - - // If we have an animation play it - if (animTimer > 0) { - GameMessages::SendPlayAnimation(self, animID); - } - - // If the anim time is less than the the default time use default - if (animTimer < defaultAnimPause) { - animTimer = defaultAnimPause; - } - - return animTimer; -} diff --git a/dScripts/BuccaneerValiantShip.cpp b/dScripts/BuccaneerValiantShip.cpp deleted file mode 100644 index 3db214b5..00000000 --- a/dScripts/BuccaneerValiantShip.cpp +++ /dev/null @@ -1,18 +0,0 @@ -#include "BuccaneerValiantShip.h" -#include "SkillComponent.h" - -void BuccaneerValiantShip::OnStartup(Entity* self) { - self->AddCallbackTimer(1.0F, [self]() { - auto* skillComponent = self->GetComponent<SkillComponent>(); - auto* owner = self->GetOwner(); - - if (skillComponent != nullptr && owner != nullptr) { - skillComponent->CalculateBehavior(982, 20577, LWOOBJID_EMPTY, true, false, owner->GetObjectID()); - - // Kill self if missed - self->AddCallbackTimer(1.1F, [self]() { - self->Kill(); - }); - } - }); -} diff --git a/dScripts/CMakeLists.txt b/dScripts/CMakeLists.txt index cb70cd44..ac600dbc 100644 --- a/dScripts/CMakeLists.txt +++ b/dScripts/CMakeLists.txt @@ -1,265 +1,54 @@ -set(DSCRIPT_SOURCES "ActivityManager.cpp" - "ActMine.cpp" - "ActNinjaTurret.cpp" - "ActParadoxPipeFix.cpp" - "ActPlayerDeathTrigger.cpp" - "ActSharkPlayerDeathTrigger.cpp" - "ActVehicleDeathTrigger.cpp" - "AgBugsprayer.cpp" - "AgBusDoor.cpp" - "AgCagedBricksServer.cpp" - "AgDarkSpiderling.cpp" - "AgFans.cpp" - "AgImagSmashable.cpp" - "AgJetEffectServer.cpp" - "AgLaserSensorServer.cpp" - "AgMonumentBirds.cpp" - "AgMonumentLaserServer.cpp" - "AgMonumentRaceCancel.cpp" - "AgMonumentRaceGoal.cpp" - "AgPicnicBlanket.cpp" - "AgPropGuard.cpp" - "AgPropguards.cpp" - "AgQbElevator.cpp" - "AgQbWall.cpp" - "AgSalutingNpcs.cpp" - "AgShipPlayerDeathTrigger.cpp" - "AgShipPlayerShockServer.cpp" - "AgSpaceStuff.cpp" - "AgStagePlatforms.cpp" - "AgStromlingProperty.cpp" - "AgSurvivalBuffStation.cpp" - "AgSurvivalMech.cpp" - "AgSurvivalSpiderling.cpp" - "AgSurvivalStromling.cpp" - "AgTurret.cpp" - "AllCrateChicken.cpp" - "AmBlueX.cpp" - "AmBridge.cpp" - "AmConsoleTeleportServer.cpp" - "AmDarklingDragon.cpp" - "AmDarklingMech.cpp" - "AmDrawBridge.cpp" - "AmDropshipComputer.cpp" - "AmScrollReaderServer.cpp" - "AmShieldGenerator.cpp" - "AmShieldGeneratorQuickbuild.cpp" - "AmSkeletonEngineer.cpp" - "AmSkullkinDrill.cpp" - "AmSkullkinDrillStand.cpp" - "AmSkullkinTower.cpp" - "AmTeapotServer.cpp" - "AmTemplateSkillVolume.cpp" - "AnvilOfArmor.cpp" - "BankInteractServer.cpp" +set(DSCRIPTS_SOURCES + "ActivityManager.cpp" "BaseConsoleTeleportServer.cpp" - "BaseEnemyApe.cpp" - "BaseEnemyMech.cpp" - "BaseFootRaceManager.cpp" - "BaseInteractDropLootServer.cpp" "BasePropertyServer.cpp" "BaseRandomServer.cpp" "BaseSurvivalServer.cpp" "BaseWavesGenericEnemy.cpp" "BaseWavesServer.cpp" - "Binoculars.cpp" - "BootyDigServer.cpp" - "BossSpiderQueenEnemyServer.cpp" - "BuccaneerValiantShip.cpp" - "BurningTile.cpp" - "CatapultBaseServer.cpp" - "CatapultBouncerServer.cpp" - "CauldronOfLife.cpp" - "CavePrisonCage.cpp" "ChooseYourDestinationNsToNt.cpp" - "ClRing.cpp" "CppScripts.cpp" - "CrabServer.cpp" - "DamagingPets.cpp" "Darkitect.cpp" - "DLUVanityNPC.cpp" - "EnemyNjBuff.cpp" - "EnemyRoninSpawner.cpp" - "EnemySkeletonSpawner.cpp" - "EnemySpiderSpawner.cpp" - "ExplodingAsset.cpp" - "FallingTile.cpp" - "FireFirstSkillonStartup.cpp" - "FlameJetServer.cpp" - "ForceVolumeServer.cpp" - "FountainOfImagination.cpp" - "FvBounceOverWall.cpp" - "FvBrickPuzzleServer.cpp" - "FvCandle.cpp" - "FvConsoleLeftQuickbuild.cpp" - "FvConsoleRightQuickbuild.cpp" - "FvDragonSmashingGolemQb.cpp" - "FvFacilityBrick.cpp" - "FvFacilityPipes.cpp" - "FvFlyingCreviceDragon.cpp" - "FvFong.cpp" - "FvFreeGfNinjas.cpp" - "FvHorsemenTrigger.cpp" - "FvMaelstromCavalry.cpp" - "FvMaelstromGeyser.cpp" - "FvMaelstromDragon.cpp" - "FvNinjaGuard.cpp" - "FvPandaServer.cpp" - "FvPandaSpawnerServer.cpp" - "FvPassThroughWall.cpp" - "FvRaceSmashEggImagineServer.cpp" - "GfApeSmashingQB.cpp" - "GfArchway.cpp" - "GfBanana.cpp" - "GfBananaCluster.cpp" - "GfCampfire.cpp" - "GfCaptainsCannon.cpp" - "GfJailkeepMission.cpp" - "GfJailWalls.cpp" - "GfMaelstromGeyser.cpp" - "GfOrgan.cpp" - "GfParrotCrash.cpp" - "GfTikiTorch.cpp" - "GrowingFlower.cpp" - "HydrantBroken.cpp" - "HydrantSmashable.cpp" - "ImaginationBackpackHealServer.cpp" - "ImaginationShrineServer.cpp" - "ImgBrickConsoleQB.cpp" - "InstanceExitTransferPlayerToLastNonInstance.cpp" - "InvalidScript.cpp" - "LegoDieRoll.cpp" - "Lieutenant.cpp" - "LupGenericInteract.cpp" - "MaestromExtracticatorServer.cpp" - "MailBoxServer.cpp" - "MastTeleport.cpp" - "MinigameTreasureChestServer.cpp" - "MonCoreNookDoors.cpp" - "MonCoreSmashableDoors.cpp" - "NjColeNPC.cpp" - "NjDragonEmblemChestServer.cpp" - "NjEarthDragonPetServer.cpp" - "NjEarthPetServer.cpp" - "NjGarmadonCelebration.cpp" - "NjhubLavaPlayerDeathTrigger.cpp" - "NjIceRailActivator.cpp" - "NjJayMissionItems.cpp" - "NjMonastryBossInstance.cpp" - "NjNPCMissionSpinjitzuServer.cpp" - "NjNyaMissionitems.cpp" - "NjRailActivatorsServer.cpp" - "NjRailPostServer.cpp" - "NjRailSwitch.cpp" - "NjScrollChestServer.cpp" - "NjWuNPC.cpp" "NPCAddRemoveItem.cpp" - "NpcAgCourseStarter.cpp" - "NpcCowboyServer.cpp" - "NpcEpsilonServer.cpp" - "NpcNjAssistantServer.cpp" - "NpcNpSpacemanBob.cpp" - "NpcPirateServer.cpp" - "NpcWispServer.cpp" - "NsConcertChoiceBuild.cpp" - "NsConcertChoiceBuildManager.cpp" - "NsConcertInstrument.cpp" - "NsConcertQuickBuild.cpp" - "NsGetFactionMissionServer.cpp" - "NsJohnnyMissionServer.cpp" - "NsLegoClubDoor.cpp" - "NsLupTeleport.cpp" - "NsModularBuild.cpp" - "NsQbImaginationStatue.cpp" - "NsTokenConsoleServer.cpp" - "NtAssemblyTubeServer.cpp" - "NtBeamImaginationCollectors.cpp" - "NtCombatChallengeDummy.cpp" - "NtCombatChallengeExplodingDummy.cpp" - "NtCombatChallengeServer.cpp" - "NtConsoleTeleportServer.cpp" - "NtDarkitectRevealServer.cpp" - "NtDirtCloudServer.cpp" - "NtDukeServer.cpp" "NtFactionSpyServer.cpp" - "NtHaelServer.cpp" - "NtImagBeamBuffer.cpp" - "NtImagimeterVisibility.cpp" - "NtOverbuildServer.cpp" - "NtParadoxPanelServer.cpp" - "NtParadoxTeleServer.cpp" - "NtSentinelWalkwayServer.cpp" - "NtSleepingGuard.cpp" - "NtVandaServer.cpp" - "NtVentureCannonServer.cpp" - "NtVentureSpeedPadServer.cpp" - "NtXRayServer.cpp" - "PersonalFortress.cpp" - "PetDigBuild.cpp" - "PetDigServer.cpp" - "PetFromDigServer.cpp" - "PetFromObjectServer.cpp" - "PirateRep.cpp" - "PropertyBankInteract.cpp" - "PropertyDeathPlane.cpp" - "PropertyDevice.cpp" - "PropertyFXDamage.cpp" - "PropertyPlatform.cpp" - "PrSeagullFly.cpp" - "PrWhistle.cpp" - "QbEnemyStunner.cpp" - "QbSpawner.cpp" - "RaceImagineCrateServer.cpp" - "RaceImaginePowerup.cpp" - "RaceMaelstromGeiser.cpp" - "RaceSmashServer.cpp" - "RainOfArrows.cpp" - "RandomSpawnerFin.cpp" - "RandomSpawnerPit.cpp" - "RandomSpawnerStr.cpp" - "RandomSpawnerZip.cpp" - "RemoveRentalGear.cpp" - "RockHydrantBroken.cpp" - "RockHydrantSmashable.cpp" "ScriptComponent.cpp" "ScriptedPowerupSpawner.cpp" - "SGCannon.cpp" - "SpawnGryphonServer.cpp" - "SpawnLionServer.cpp" - "SpawnPetBaseServer.cpp" - "SpawnSaberCatServer.cpp" - "SpawnShrakeServer.cpp" - "SpawnStegoServer.cpp" - "SpecialImaginePowerupSpawner.cpp" - "SpiderBossTreasureChestServer.cpp" - "SsModularBuildServer.cpp" - "StinkyFishTarget.cpp" - "StoryBoxInteractServer.cpp" - "Sunflower.cpp" - "TokenConsoleServer.cpp" - "TouchMissionUpdateServer.cpp" - "TreasureChestDragonServer.cpp" - "TriggerAmbush.cpp" - "VeBricksampleServer.cpp" - "VeEpsilonServer.cpp" - "VeMech.cpp" - "VeMissionConsole.cpp" - "WaveBossApe.cpp" - "WaveBossHammerling.cpp" - "WaveBossHorsemen.cpp" - "WaveBossSpiderling.cpp" - "WblGenericZone.cpp" - "WblRobotCitizen.cpp" - "WhFans.cpp" - "WildAmbients.cpp" - "WishingWellServer.cpp" - "ZoneAgMedProperty.cpp" - "ZoneAgProperty.cpp" - "ZoneAgSpiderQueen.cpp" - "ZoneAgSurvival.cpp" - "ZoneFvProperty.cpp" - "ZoneGfProperty.cpp" - "ZoneNsMedProperty.cpp" - "ZoneNsProperty.cpp" - "ZoneNsWaves.cpp" - "ZoneSGServer.cpp" PARENT_SCOPE) + "SpawnPetBaseServer.cpp") + +add_subdirectory(02_server) + +foreach(file ${DSCRIPTS_SOURCES_02_SERVER}) + set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} "02_server/${file}") +endforeach() + +add_subdirectory(ai) + +foreach(file ${DSCRIPTS_SOURCES_AI}) + set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} "ai/${file}") +endforeach() + +add_subdirectory(client) + +foreach(file ${DSCRIPTS_SOURCES_CLIENT}) + set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} "client/${file}") +endforeach() + +add_subdirectory(EquipmentScripts) + +foreach(file ${DSCRIPTS_SOURCES_EQUIPMENTSCRIPTS}) + set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} "EquipmentScripts/${file}") +endforeach() + +add_subdirectory(EquipmentTriggers) + +foreach(file ${DSCRIPTS_SOURCES_EQUIPMENTTRIGGERSSCRIPTS}) + set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} "EquipmentTriggers/${file}") +endforeach() + +add_subdirectory(zone) + +foreach(file ${DSCRIPTS_SOURCES_ZONE}) + set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} "zone/${file}") +endforeach() + +set(DSCRIPTS_SOURCES ${DSCRIPTS_SOURCES} PARENT_SCOPE) diff --git a/dScripts/CatapultBaseServer.cpp b/dScripts/CatapultBaseServer.cpp deleted file mode 100644 index fda103b0..00000000 --- a/dScripts/CatapultBaseServer.cpp +++ /dev/null @@ -1,64 +0,0 @@ -#include "CatapultBaseServer.h" -#include "GameMessages.h" -#include "EntityManager.h" - -void CatapultBaseServer::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { - if (name == "BouncerBuilt") { - // start a timer for the arm to player the with bouncer animation - self->AddTimer("PlatAnim", .75); - - // set the bouncer so we can use it later - self->SetVar(u"Bouncer", sender->GetObjectID()); - - GameMessages::SendBouncerActiveStatus(sender->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); - } -} - -void CatapultBaseServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "PlatAnim") { - // get the arm asset - const auto arm = EntityManager::Instance()->GetEntitiesInGroup(self->GetVarAsString(u"ArmGroup")); - - // tell the arm to the play the platform animation, which is just the arm laying there but with bouncer - for (auto* obj : arm) { - GameMessages::SendPlayAnimation(obj, u"idle-platform"); - GameMessages::SendPlayNDAudioEmitter(obj, UNASSIGNED_SYSTEM_ADDRESS, "{8cccf912-69e3-4041-a20b-63e4afafc993}"); - // set the art so we can use it again - self->SetVar(u"Arm", obj->GetObjectID()); - break; - } - - // start a timer till the bouncer actually bounces - self->AddTimer("bounce", 3); - } else if (timerName == "launchAnim") { - // get the arm asset - auto* arm = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Arm")); - if (arm == nullptr) return; - - // tell the arm to player the launcher animation - auto animTime = 1; - self->AddTimer("resetArm", animTime); - GameMessages::SendPlayAnimation(arm, u"launch"); - } else if (timerName == "bounce") { - auto* bouncer = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Bouncer")); - if (bouncer == nullptr) return; - - // bounce all players - bouncer->NotifyObject(bouncer, "bounceAllInProximity"); // Likely to trigger server side bounce, bodging this - // add a delay to play the animation - self->AddTimer("launchAnim", .3); - } else if (timerName == "resetArm") { - auto* arm = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Arm")); - if (arm == nullptr) return; - - // set the arm back to natural state - GameMessages::SendPlayAnimation(arm, u"idle"); - - auto* bouncer = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Bouncer")); - if (bouncer == nullptr) return; - - // kill the bouncer - GameMessages::SendNotifyClientObject(bouncer->GetObjectID(), u"TimeToDie"); - bouncer->Smash(self->GetObjectID(), VIOLENT); - } -} diff --git a/dScripts/CatapultBouncerServer.cpp b/dScripts/CatapultBouncerServer.cpp deleted file mode 100644 index 89faf001..00000000 --- a/dScripts/CatapultBouncerServer.cpp +++ /dev/null @@ -1,15 +0,0 @@ -#include "CatapultBouncerServer.h" -#include "GameMessages.h" -#include "EntityManager.h" - -void CatapultBouncerServer::OnRebuildComplete(Entity* self, Entity* target) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"Built", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - - self->SetNetworkVar<bool>(u"Built", true); - - const auto base = EntityManager::Instance()->GetEntitiesInGroup(self->GetVarAsString(u"BaseGroup")); - - for (auto* obj : base) { - obj->NotifyObject(self, "BouncerBuilt"); - } -} diff --git a/dScripts/CavePrisonCage.cpp b/dScripts/CavePrisonCage.cpp deleted file mode 100644 index 0df91884..00000000 --- a/dScripts/CavePrisonCage.cpp +++ /dev/null @@ -1,215 +0,0 @@ -#include "CavePrisonCage.h" -#include "EntityManager.h" -#include "RebuildComponent.h" -#include "GameMessages.h" -#include "Character.h" -#include "dZoneManager.h" - -void CavePrisonCage::OnStartup(Entity* self) { - const auto& myNum = self->GetVar<std::u16string>(u"myNumber"); - - if (myNum.empty()) { - return; - } - - auto* spawner = dZoneManager::Instance()->GetSpawnersByName("PrisonCounterweight_0" + GeneralUtils::UTF16ToWTF8(myNum))[0]; - - self->SetVar<Spawner*>(u"CWSpawner", spawner); - - Setup(self, spawner); -} - -void CavePrisonCage::Setup(Entity* self, Spawner* spawner) { - SpawnCounterweight(self, spawner); - - NiPoint3 mypos = self->GetPosition(); - NiQuaternion myrot = self->GetRotation(); - - mypos.y += 1.5; - mypos.z -= 0.5; - - EntityInfo info{}; - info.lot = m_Villagers[self->GetVarAs<int32_t>(u"myNumber") - 1]; - info.pos = mypos; - info.rot = myrot; - info.spawnerID = self->GetObjectID(); - - // Spawn the villager inside the jail - auto* entity = EntityManager::Instance()->CreateEntity(info); - - // Save the villeger ID - self->SetVar<LWOOBJID>(u"villager", entity->GetObjectID()); - - // Construct the entity - EntityManager::Instance()->ConstructEntity(entity); -} - -void CavePrisonCage::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state != eRebuildState::REBUILD_RESETTING) { - return; - } - - auto* spawner = self->GetVar<Spawner*>(u"CWSpawner"); - - if (spawner == nullptr) { - return; - } - - spawner->Reset(); - - SpawnCounterweight(self, spawner); -} - -void CavePrisonCage::SpawnCounterweight(Entity* self, Spawner* spawner) { - spawner->Reset(); - - auto* counterweight = spawner->Spawn(); - - self->SetVar<LWOOBJID>(u"Counterweight", counterweight->GetObjectID()); - - auto* rebuildComponent = counterweight->GetComponent<RebuildComponent>(); - - if (rebuildComponent != nullptr) { - rebuildComponent->AddRebuildStateCallback([this, self](eRebuildState state) { - OnRebuildNotifyState(self, state); - }); - - rebuildComponent->AddRebuildCompleteCallback([this, self](Entity* user) { - // The counterweight is a simple mover, which is not implemented, so we'll just set it's position - auto* counterweight = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Counterweight")); - - if (counterweight == nullptr) { - return; - } - - // Move the counterweight down 2 units - counterweight->SetPosition(counterweight->GetPosition() + NiPoint3(0, -2, 0)); - - // Serialize the counterweight - EntityManager::Instance()->SerializeEntity(counterweight); - - // notifyPlatformAtLastWaypoint - - // Save the userID as Builder - self->SetVar<LWOOBJID>(u"Builder", user->GetObjectID()); - - // Get the button and make sure it still exists - auto* button = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Button")); - - if (button == nullptr) { - return; - } - - // Play the 'down' animation on the button - GameMessages::SendPlayAnimation(button, u"down"); - - // Setup a timer named 'buttonGoingDown' to be triggered in 5 seconds - self->AddTimer("buttonGoingDown", 5.0f); - }); - } - - if (self->GetVar<LWOOBJID>(u"Button")) { - return; - } - - GetButton(self); -} - -void CavePrisonCage::GetButton(Entity* self) { - const auto buttons = EntityManager::Instance()->GetEntitiesInGroup("PrisonButton_0" + std::to_string(self->GetVarAs<int32_t>(u"myNumber"))); - - if (buttons.size() == 0) { - // Try again in 0.5 seconds - self->AddCallbackTimer(0.5, [this, self]() { - GetButton(self); - }); - - return; - } - - auto* button = buttons[0]; - - self->SetVar<LWOOBJID>(u"Button", button->GetObjectID()); -} - -void CavePrisonCage::OnTimerDone(Entity* self, std::string timerName) { - // the anim of the button down is over - if (timerName == "buttonGoingDown") { - // Play the 'up' animation - GameMessages::SendPlayAnimation(self, u"up"); - - // Setup a timer named 'CageOpen' to be triggered in 1 second - self->AddTimer("CageOpen", 1.0f); - } else if (timerName == "CageOpen") { - // play the idle open anim - GameMessages::SendPlayAnimation(self, u"idle-up"); - - // Get the villeger - auto* villager = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"villager")); - - if (villager == nullptr) { - return; - } - - GameMessages::SendNotifyClientObject(villager->GetObjectID(), u"TimeToChat", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - - // Get the builder and make sure it still exists - auto* builder = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Builder")); - - if (builder == nullptr) { - return; - } - - const auto flagNum = 2020 + self->GetVarAs<int32_t>(u"myNumber"); - - // Set the flag on the builder character - builder->GetCharacter()->SetPlayerFlag(flagNum, true); - - // Setup a timer named 'VillagerEscape' to be triggered in 5 seconds - self->AddTimer("VillagerEscape", 5.0f); - } else if (timerName == "VillagerEscape") { - // Get the villeger and make sure it still exists - auto* villager = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"villager")); - - if (villager == nullptr) { - return; - } - - // Kill the villager - villager->Kill(); - - // Setup a timer named 'SmashCounterweight' to be triggered in 2 seconds - self->AddTimer("SmashCounterweight", 2.0f); - } else if (timerName == "SmashCounterweight") { - // Get the counterweight and make sure it still exists - auto* counterweight = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Counterweight")); - - if (counterweight == nullptr) { - return; - } - - // Smash the counterweight - counterweight->Smash(); - - // Get the button and make sure it still exists - auto* button = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"Button")); - - if (button == nullptr) { - return; - } - - // Play the 'up' animation on the button - GameMessages::SendPlayAnimation(button, u"up"); - - // Setup a timer named 'CageClosed' to be triggered in 1 second - self->AddTimer("CageClosed", 1.0f); - } else if (timerName == "CageClosed") { - // play the idle closed anim - GameMessages::SendPlayAnimation(self, u"idle"); - - // Setup a timer named 'ResetPrison' to be triggered in 10 seconds - self->AddTimer("ResetPrison", 10.0f); - } else if (timerName == "ResetPrison") { - Setup(self, self->GetVar<Spawner*>(u"CWSpawner")); - } -} diff --git a/dScripts/ChooseYourDestinationNsToNt.cpp b/dScripts/ChooseYourDestinationNsToNt.cpp index e50d70c5..f9ca0a79 100644 --- a/dScripts/ChooseYourDestinationNsToNt.cpp +++ b/dScripts/ChooseYourDestinationNsToNt.cpp @@ -1,6 +1,7 @@ #include "ChooseYourDestinationNsToNt.h" #include "Character.h" #include "GameMessages.h" +#include "eTerminateType.h" bool ChooseYourDestinationNsToNt::CheckChoice(Entity* self, Entity* player) { const auto choiceZoneID = self->GetVar<int32_t>(u"choiceZone"); @@ -59,6 +60,6 @@ void ChooseYourDestinationNsToNt::BaseChoiceBoxRespond(Entity* self, Entity* sen GameMessages::SendDisplayMessageBox(sender->GetObjectID(), true, self->GetObjectID(), u"TransferBox", 0, strText, u"", sender->GetSystemAddress()); } else { - GameMessages::SendTerminateInteraction(sender->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); + GameMessages::SendTerminateInteraction(sender->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); } } diff --git a/dScripts/CppScripts.cpp b/dScripts/CppScripts.cpp index 04d24bd5..20619548 100644 --- a/dScripts/CppScripts.cpp +++ b/dScripts/CppScripts.cpp @@ -160,7 +160,6 @@ #include "AgSalutingNpcs.h" #include "BossSpiderQueenEnemyServer.h" #include "RockHydrantSmashable.h" -#include "SpecialImaginePowerupSpawner.h" // Misc Scripts #include "ExplodingAsset.h" @@ -278,6 +277,10 @@ #include "ImaginationBackpackHealServer.h" #include "LegoDieRoll.h" #include "BuccaneerValiantShip.h" +#include "GemPack.h" +#include "ShardArmor.h" +#include "TeslaPack.h" +#include "StunImmunity.h" // Survival scripts #include "AgSurvivalStromling.h" @@ -293,6 +296,24 @@ #include "LupGenericInteract.h" #include "WblRobotCitizen.h" +// Alpha Scripts +#include "TriggerGas.h" +#include "ActNinjaSensei.h" + +// pickups +#include "SpecialCoinSpawner.h" +#include "SpecialPowerupSpawner.h" +#include "SpecialSpeedBuffSpawner.h" + +// Wild Scripts +#include "WildAndScared.h" +#include "WildGfGlowbug.h" +#include "WildAmbientCrab.h" +#include "WildPants.h" +#include "WildNinjaStudent.h" +#include "WildNinjaSensei.h" +#include "WildNinjaBricks.h" + //Big bad global bc this is a namespace and not a class: InvalidScript* invalidToReturn = new InvalidScript(); std::map<std::string, CppScripts::Script*> m_Scripts; @@ -369,8 +390,6 @@ CppScripts::Script* CppScripts::GetScript(Entity* parent, const std::string& scr script = new RemoveRentalGear(); else if (scriptName == "scripts\\02_server\\Map\\AG\\L_NPC_NJ_ASSISTANT_SERVER.lua") script = new NpcNjAssistantServer(); - else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_IMAGINE-POWERUP-SPAWNER.lua") - script = new SpecialImaginePowerupSpawner(); else if (scriptName == "scripts\\ai\\AG\\L_AG_SALUTING_NPCS.lua") script = new AgSalutingNpcs(); else if (scriptName == "scripts\\ai\\AG\\L_AG_JET_EFFECT_SERVER.lua") @@ -839,6 +858,14 @@ CppScripts::Script* CppScripts::GetScript(Entity* parent, const std::string& scr script = new BuccaneerValiantShip(); else if (scriptName == "scripts\\EquipmentScripts\\FireFirstSkillonStartup.lua") script = new FireFirstSkillonStartup(); + else if (scriptName == "scripts\\equipmenttriggers\\gempack.lua") + script = new GemPack(); + else if (scriptName == "scripts\\equipmenttriggers\\shardarmor.lua") + script = new ShardArmor(); + else if (scriptName == "scripts\\equipmenttriggers\\coilbackpack.lua") + script = new TeslaPack(); + else if (scriptName == "scripts\\EquipmentScripts\\stunImmunity.lua") + script = new StunImmunity(); // FB else if (scriptName == "scripts\\ai\\NS\\WH\\L_ROCKHYDRANT_BROKEN.lua") @@ -854,17 +881,67 @@ CppScripts::Script* CppScripts::GetScript(Entity* parent, const std::string& scr else if (scriptName.rfind("scripts\\zone\\LUPs\\RobotCity Intro\\WBL_RCIntro_RobotCitizen", 0) == 0) script = new WblRobotCitizen(); - //Ignore these scripts: - else if (scriptName == "scripts\\02_server\\Enemy\\General\\L_SUSPEND_LUA_AI.lua") - script = invalidToReturn; - else if (scriptName == "scripts\\02_server\\Enemy\\General\\L_BASE_ENEMY_SPIDERLING.lua") - script = invalidToReturn; - else if (script == invalidToReturn) { - if (scriptName.length() > 0) - Game::logger->Log("CppScripts", "Lot %i attempted to load CppScript for '%s', but returned InvalidScript.", parent->GetLOT(), scriptName.c_str()); - // information not really needed for sys admins but is for developers + // Alpha + if (scriptName == "scripts\\ai\\FV\\L_TRIGGER_GAS.lua") + script = new TriggerGas(); + else if (scriptName == "scripts\\ai\\FV\\L_ACT_NINJA_SENSEI.lua") + script = new ActNinjaSensei(); - script = invalidToReturn; + // pickups + if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_1_BRONZE-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(1); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_1_GOLD-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(10000); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_1_SILVER-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(100); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_10_BRONZE-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(10); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_10_GOLD-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(100000); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_10_SILVER-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(1000); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_25_BRONZE-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(25); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_25_GOLD-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(250000); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_25_SILVER-COIN-SPAWNER.lua") + script = new SpecialCoinSpawner(2500); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_IMAGINE-POWERUP-SPAWNER.lua") + script = new SpecialPowerupSpawner(13); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_IMAGINE-POWERUP-SPAWNER-2PT.lua") + script = new SpecialPowerupSpawner(129); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_LIFE-POWERUP-SPAWNER.lua") + script = new SpecialPowerupSpawner(5); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_ARMOR-POWERUP-SPAWNER.lua") + script = new SpecialPowerupSpawner(747); + else if (scriptName == "scripts\\ai\\SPEC\\L_SPECIAL_SPEED_BUFF_SPAWNER.lua") + script = new SpecialSpeedBuffSpawner(); + + // Wild + if (scriptName == "scripts\\ai\\WILD\\L_WILD_GF_RAT.lua" || scriptName == "scripts\\ai\\WILD\\L_WILD_GF_SNAIL.lua") + script = new WildAndScared(); + else if (scriptName == "scripts\\ai\\WILD\\L_WILD_GF_GLOWBUG.lua") + script = new WildGfGlowbug(); + else if (scriptName == "scripts\\ai\\WILD\\L_WILD_AMBIENT_CRAB.lua") + script = new WildAmbientCrab(); + else if (scriptName == "scripts\\ai\\WILD\\L_WILD_PANTS.lua") + script = new WildPants(); + else if (scriptName == "scripts\\ai\\WILD\\L_WILD_NINJA_BRICKS.lua") + script = new WildNinjaBricks(); + else if (scriptName == "scripts\\ai\\WILD\\L_WILD_NINJA_STUDENT.lua") + script = new WildNinjaStudent(); + else if (scriptName == "scripts\\ai\\WILD\\L_WILD_NINJA_SENSEI.lua") + script = new WildNinjaSensei(); + + // handle invalid script reporting if the path is greater than zero and it's not an ignored script + // information not really needed for sys admins but is for developers + else if (script == invalidToReturn) { + if ((scriptName.length() > 0) && !((scriptName == "scripts\\02_server\\Enemy\\General\\L_SUSPEND_LUA_AI.lua") || + (scriptName == "scripts\\02_server\\Enemy\\General\\L_BASE_ENEMY_SPIDERLING.lua") || + (scriptName =="scripts\\ai\\FV\\L_ACT_NINJA_STUDENT.lua") || + (scriptName == "scripts\\ai\\WILD\\L_WILD_GF_FROG.lua") || + (scriptName == "scripts\\empty.lua") + )) Game::logger->LogDebug("CppScripts", "LOT %i attempted to load CppScript for '%s', but returned InvalidScript.", parent->GetLOT(), scriptName.c_str()); } m_Scripts[scriptName] = script; diff --git a/dScripts/CppScripts.h b/dScripts/CppScripts.h index 916d2638..1b9d0591 100644 --- a/dScripts/CppScripts.h +++ b/dScripts/CppScripts.h @@ -1,12 +1,15 @@ #pragma once -#include "dCommonVars.h" -#include "MissionState.h" + +#include <cstdint> #include <string> #include <vector> class User; class Entity; class NiPoint3; +enum class eMissionState : int32_t; +enum class ePetTamingNotifyType : uint32_t; +enum class eRebuildState : uint32_t; namespace CppScripts { /** @@ -43,12 +46,19 @@ namespace CppScripts { */ virtual void OnCollisionPhantom(Entity* self, Entity* target) {}; + /** + * Invoked upon an entity leaving the phantom collider on self. + * + * Equivalent to 'function onOffCollisionPhantom(self, msg)' + */ + virtual void OnOffCollisionPhantom(Entity* self, Entity* target) {}; + /** * Invoked when a player accepted a mission. * * Equivalent to 'function onMissionDialogueOK(self, msg)' */ - virtual void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) {}; + virtual void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) {}; /** * Invoked when the client or the server invoked an event server-side. @@ -172,6 +182,13 @@ namespace CppScripts { */ virtual void OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) {}; + /** + * Invoked when self has received either a hit or heal. Only used for scripts subscribed to an entity. + * + * Equivalent to 'function notifyHitOrHealResult(self, msg)' + */ + virtual void NotifyHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) {}; + /** * Invoked when a player has responsed to a mission. * @@ -254,7 +271,7 @@ namespace CppScripts { * * Equivalent to 'function onNotifyPetTamingMinigame(self, msg)' */ - virtual void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) {}; + virtual void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) {}; /** * Invoked when a player responded to a message box. @@ -316,6 +333,31 @@ namespace CppScripts { virtual void OnCinematicUpdate(Entity* self, Entity* sender, eCinematicEvent event, const std::u16string& pathName, float_t pathTime, float_t totalTime, int32_t waypoint) { }; + + /** + * Used by items to tell their owner that they were equipped. + * + * @param itemOwner The owner of the item + * @param itemObjId The items Object ID + */ + virtual void OnFactionTriggerItemEquipped(Entity* itemOwner, LWOOBJID itemObjId) {}; + + /** + * Used by items to tell their owner that they were unequipped. + * + * @param itemOwner The owner of the item + * @param itemObjId The items Object ID + */ + virtual void OnFactionTriggerItemUnequipped(Entity* itemOwner, LWOOBJID itemObjId) {}; + + /** + * Handles exiting a scripted activity + * + * @param sender + * @param player the player to remove + * @param canceled if it was done via the cancel button + */ + virtual void OnRequestActivityExit(Entity* sender, LWOOBJID player, bool canceled){}; }; Script* GetScript(Entity* parent, const std::string& scriptName); diff --git a/dScripts/CrabServer.cpp b/dScripts/CrabServer.cpp deleted file mode 100644 index 890b8ed9..00000000 --- a/dScripts/CrabServer.cpp +++ /dev/null @@ -1,43 +0,0 @@ -#include "CrabServer.h" -#include "PetComponent.h" - -void CrabServer::OnStartup(Entity* self) { - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr || petComponent->GetOwner() != nullptr) - return; - - // Triggers the local crab script for taming etc. - auto tamer = self->GetVar<LWOOBJID>(u"tamer"); - // Client compares this with player:GetID() which is a string, so we'll have to give it a string - self->SetNetworkVar(u"crabtamer", std::to_string(tamer)); - - // Kill if the player decides that the crab is not worthy - self->AddTimer("killself", 45.0f); -} - -void CrabServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "killself") { - - // Don't accidentally kill a pet that is already owned - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr || petComponent->GetOwner() != nullptr) - return; - - self->Smash(self->GetObjectID(), SILENT); - } -} - -void CrabServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) { - if (type == NOTIFY_TYPE_BEGIN) { - self->CancelTimer("killself"); - } else if (type == NOTIFY_TYPE_QUIT || type == NOTIFY_TYPE_FAILED) { - self->Smash(self->GetObjectID(), SILENT); - } else if (type == NOTIFY_TYPE_SUCCESS) { - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr) - return; - // TODO: Remove custom group? - // Command the pet to the player as it may otherwise go to its spawn point which is non existant - // petComponent->Command(NiPoint3::ZERO, LWOOBJID_EMPTY, 6, 202, true); - } -} diff --git a/dScripts/CrabServer.h b/dScripts/CrabServer.h deleted file mode 100644 index 28533cb5..00000000 --- a/dScripts/CrabServer.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class CrabServer : public CppScripts::Script -{ -public: - void OnStartup(Entity* self) override; - void OnTimerDone(Entity* self, std::string timerName) override; - void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) override; -}; diff --git a/dScripts/DLUVanityNPC.cpp b/dScripts/DLUVanityNPC.cpp deleted file mode 100644 index e3db2353..00000000 --- a/dScripts/DLUVanityNPC.cpp +++ /dev/null @@ -1,46 +0,0 @@ -#include "DLUVanityNPC.h" -#include "GameMessages.h" -#include "dServer.h" -#include "VanityUtilities.h" - -void DLUVanityNPC::OnStartup(Entity* self) { - m_NPC = VanityUtilities::GetNPC("averysumner - Destroyer of Worlds"); - - if (m_NPC == nullptr) { - return; - } - - if (self->GetVar<bool>(u"teleport")) { - self->AddTimer("setupTeleport", 15.0f); - } -} - -void DLUVanityNPC::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "setupTeleport") { - GameMessages::SendPlayAnimation(self, u"interact"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportBeam", "teleportBeam"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportRings", "teleportRings"); - - self->AddTimer("teleport", 2.0f); - self->AddTimer("stopFX", 2.0f); - } else if (timerName == "stopFX") { - GameMessages::SendStopFXEffect(self, true, "teleportBeam"); - GameMessages::SendStopFXEffect(self, true, "teleportRings"); - } else if (timerName == "teleport") { - std::vector<VanityNPCLocation>& locations = m_NPC->m_Locations[Game::server->GetZoneID()]; - - selectLocation: - VanityNPCLocation& newLocation = locations[GeneralUtils::GenerateRandomNumber<size_t>(0, locations.size() - 1)]; - - if (self->GetPosition() == newLocation.m_Position) { - goto selectLocation; // cry about it - } - - self->SetPosition(newLocation.m_Position); - self->SetRotation(newLocation.m_Rotation); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportBeam", "teleportBeam"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 6478, u"teleportRings", "teleportRings"); - self->AddTimer("stopFX", 2.0f); - self->AddTimer("setupTeleport", 15.0f); - } -} diff --git a/dScripts/DamagingPets.cpp b/dScripts/DamagingPets.cpp deleted file mode 100644 index f7eada00..00000000 --- a/dScripts/DamagingPets.cpp +++ /dev/null @@ -1,145 +0,0 @@ -#include "DamagingPets.h" -#include "PetComponent.h" -#include "DestroyableComponent.h" -#include "BaseCombatAIComponent.h" -#include "RenderComponent.h" - -void DamagingPets::OnStartup(Entity* self) { - - // Make the pet hostile or non-hostile based on whether or not it is tamed - const auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent != nullptr && petComponent->GetOwner() == nullptr) { - self->AddTimer("GoEvil", 0.5f); - } -} - -void DamagingPets::OnPlayerLoaded(Entity* self, Entity* player) { - - // Makes it so that new players also see the effect - self->AddCallbackTimer(2.5f, [self]() { - if (self != nullptr) { - const auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent != nullptr && petComponent->GetOwner() == nullptr && self->GetVar<bool>(u"IsEvil")) { - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - auto counter = 1; - for (const auto petEffect : GetPetInfo(self).effect) { - renderComponent->PlayEffect(petEffect, u"create", "FXname" + std::to_string(counter)); - counter++; - } - } - } - } - }); -} - -void DamagingPets::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) { - switch (type) { - case NOTIFY_TYPE_SUCCESS: - case NOTIFY_TYPE_BEGIN: - self->CancelAllTimers(); - ClearEffects(self); - break; - case NOTIFY_TYPE_FAILED: - case NOTIFY_TYPE_QUIT: - { - self->SetNetworkVar<bool>(u"bIAmTamable", false); - self->AddTimer("GoEvil", 1.0f); - break; - } - default: - break; - } -} - -void DamagingPets::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - const auto infoForPet = GetPetInfo(self); - if (infoForPet.skill == message) { - - // Only make pets tamable that aren't tamed yet - const auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent != nullptr && petComponent->GetOwner() == nullptr && self->GetVar<bool>(u"IsEvil")) { - ClearEffects(self); - self->AddTimer("GoEvil", 30.0f); - self->SetNetworkVar<bool>(u"bIAmTamable", true); - } - } -} - -void DamagingPets::OnTimerDone(Entity* self, std::string message) { - if (message == "GoEvil") { - MakeUntamable(self); - } -} - -void DamagingPets::MakeUntamable(Entity* self) { - auto* petComponent = self->GetComponent<PetComponent>(); - - // If the pet is currently not being tamed, make it hostile - if (petComponent != nullptr && petComponent->GetStatus() != 5) { - self->SetNetworkVar<bool>(u"bIAmTamable", false); - self->SetVar<bool>(u"IsEvil", true); - petComponent->SetStatus(1); - - auto* combatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - if (combatAIComponent != nullptr) { - combatAIComponent->SetDisabled(false); - } - - // Special faction that can attack the player but the player can't attack - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->SetFaction(114); - destroyableComponent->SetHealth(5); - } - - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - auto counter = 1; - for (const auto petEffect : GetPetInfo(self).effect) { - renderComponent->PlayEffect(petEffect, u"create", "FXname" + std::to_string(counter)); - counter++; - } - } - } -} - -void DamagingPets::ClearEffects(Entity* self) { - self->SetVar<bool>(u"IsEvil", false); - - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent != nullptr) { - petComponent->SetStatus(67108866); - } - - auto* combatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - if (combatAIComponent != nullptr) { - combatAIComponent->SetDisabled(true); - } - - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->SetFaction(99); - } - - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - auto counter = 1; - for (const auto petEffect : GetPetInfo(self).effect) { - renderComponent->StopEffect("FXname" + std::to_string(counter)); - counter++; - } - } -} - -PetInfo DamagingPets::GetPetInfo(Entity* self) { - const auto infoForPet = petInfo.find(self->GetLOT()); - return infoForPet != petInfo.end() ? infoForPet->second : petInfo.begin()->second; -} - -// Does not compile on Win32 with name specifiers -const std::map<LOT, PetInfo> DamagingPets::petInfo = { - { 5639, { /*.effect =*/ { 3170, 4058 }, /*.skill =*/ "waterspray"}}, // Red dragon - { 5641, { /*.effect =*/ { 3170, 4058 }, /*.skill =*/ "waterspray"}}, // Green dragon - { 3261, { /*.effect =*/ { 1490 }, /*.skill =*/ "waterspray"}}, // Skunk -}; diff --git a/dScripts/DamagingPets.h b/dScripts/DamagingPets.h deleted file mode 100644 index 2ce0ddc1..00000000 --- a/dScripts/DamagingPets.h +++ /dev/null @@ -1,24 +0,0 @@ -#pragma once -#include "CppScripts.h" - -/** - * Information about pets regarding which effect to play when a skill is cast - */ -struct PetInfo { - const std::vector<uint32_t> effect; - const std::string skill; -}; - -class DamagingPets : public CppScripts::Script { -public: - void OnStartup(Entity* self) override; - void OnTimerDone(Entity* self, std::string message) override; - void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) override; - void OnSkillEventFired(Entity* self, Entity* target, const std::string& message) override; - void OnPlayerLoaded(Entity* self, Entity* player) override; -private: - static void MakeUntamable(Entity* self); - static PetInfo GetPetInfo(Entity* self); - static void ClearEffects(Entity* self); - static const std::map<LOT, PetInfo> petInfo; -}; diff --git a/dScripts/Darkitect.cpp b/dScripts/Darkitect.cpp index 323d0247..2881bde8 100644 --- a/dScripts/Darkitect.cpp +++ b/dScripts/Darkitect.cpp @@ -4,6 +4,7 @@ #include "EntityManager.h" #include "GameMessages.h" #include "Character.h" +#include "eMissionState.h" void Darkitect::Reveal(Entity* self, Entity* player) { const auto playerID = player->GetObjectID(); @@ -11,7 +12,7 @@ void Darkitect::Reveal(Entity* self, Entity* player) { GameMessages::SendNotifyClientObject(self->GetObjectID(), u"reveal", 0, 0, playerID, "", player->GetSystemAddress()); self->AddCallbackTimer(20, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); + auto* player = Game::entityManager->GetEntity(playerID); if (!player) return; @@ -24,11 +25,11 @@ void Darkitect::Reveal(Entity* self, Entity* player) { destroyableComponent->SetHealth(1); destroyableComponent->SetImagination(0); - if (missionComponent->GetMissionState(1295) == MissionState::MISSION_STATE_ACTIVE) { + if (missionComponent->GetMissionState(1295) == eMissionState::ACTIVE) { character->SetPlayerFlag(1911, true); } - EntityManager::Instance()->SerializeEntity(player); + Game::entityManager->SerializeEntity(player); } }); } diff --git a/dScripts/EnemyRoninSpawner.cpp b/dScripts/EnemyRoninSpawner.cpp deleted file mode 100644 index b6cae3bb..00000000 --- a/dScripts/EnemyRoninSpawner.cpp +++ /dev/null @@ -1,68 +0,0 @@ -#include "EnemyRoninSpawner.h" -#include "SkillComponent.h" -#include "RenderComponent.h" -#include "EntityManager.h" - -void EnemyRoninSpawner::OnStartup(Entity* self) { - self->SetProximityRadius(15, "ronin"); -} - -void EnemyRoninSpawner::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "hatchTime") { - auto* renderComponent = self->GetComponent<RenderComponent>(); - - if (renderComponent != nullptr) { - renderComponent->PlayEffect(644, u"create", "BurstFX1"); - } - - EntityInfo info{}; - info.lot = 7815; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.spawnerID = self->GetObjectID(); - - auto* spawnedEntity = EntityManager::Instance()->CreateEntity(info); - - if (spawnedEntity == nullptr) { - return; - } - - EntityManager::Instance()->ConstructEntity(spawnedEntity); - - spawnedEntity->AddCallbackTimer(60, [spawnedEntity]() { - spawnedEntity->Smash(spawnedEntity->GetObjectID()); - }); - - self->Smash(self->GetObjectID()); - } -} - -void EnemyRoninSpawner::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (entering->IsPlayer() && name == "ronin" && status == "ENTER" && !self->GetVar<bool>(u"hatching")) { - StartHatching(self); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(305, 3568, LWOOBJID_EMPTY); - } - } -} - -void EnemyRoninSpawner::OnHit(Entity* self, Entity* attacker) { - if (!self->GetVar<bool>(u"hatching")) { - StartHatching(self); - } -} - -void EnemyRoninSpawner::StartHatching(Entity* self) { - self->SetVar(u"hatching", true); - - auto* renderComponent = self->GetComponent<RenderComponent>(); - - if (renderComponent != nullptr) { - renderComponent->PlayEffect(2260, u"rebuild_medium", "WakeUpFX1"); - } - - self->AddTimer("hatchTime", 2); -} diff --git a/dScripts/EnemySkeletonSpawner.cpp b/dScripts/EnemySkeletonSpawner.cpp deleted file mode 100644 index 451fa1d5..00000000 --- a/dScripts/EnemySkeletonSpawner.cpp +++ /dev/null @@ -1,75 +0,0 @@ -#include "EnemySkeletonSpawner.h" -#include "SkillComponent.h" -#include "RenderComponent.h" -#include "EntityManager.h" - -void EnemySkeletonSpawner::OnStartup(Entity* self) { - self->SetProximityRadius(15, "ronin"); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(1127, 24812, LWOOBJID_EMPTY, true); - } -} - -void EnemySkeletonSpawner::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "hatchTime") { - auto* renderComponent = self->GetComponent<RenderComponent>(); - - if (renderComponent != nullptr) { - renderComponent->PlayEffect(644, u"create", "BurstFX1"); - } - - EntityInfo info{}; - info.lot = 14024; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.spawnerID = self->GetObjectID(); - - auto* spawnedEntity = EntityManager::Instance()->CreateEntity(info); - - if (spawnedEntity == nullptr) { - return; - } - - EntityManager::Instance()->ConstructEntity(spawnedEntity); - - spawnedEntity->AddCallbackTimer(60, [spawnedEntity]() { - spawnedEntity->Smash(spawnedEntity->GetObjectID()); - }); - - self->Smash(self->GetObjectID()); - } -} - -void EnemySkeletonSpawner::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (entering->IsPlayer() && name == "ronin" && status == "ENTER" && !self->GetVar<bool>(u"hatching")) { - StartHatching(self); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(305, 3568, LWOOBJID_EMPTY); - } - } -} - -void EnemySkeletonSpawner::OnHit(Entity* self, Entity* attacker) { - if (!self->GetVar<bool>(u"hatching")) { - StartHatching(self); - } -} - -void EnemySkeletonSpawner::StartHatching(Entity* self) { - self->SetVar(u"hatching", true); - - auto* renderComponent = self->GetComponent<RenderComponent>(); - - if (renderComponent != nullptr) { - renderComponent->PlayEffect(9017, u"cast", "WakeUpFX1"); - renderComponent->PlayEffect(9018, u"burst", "WakeUpFX1"); - } - - self->AddTimer("hatchTime", 2); -} diff --git a/dScripts/EnemySpiderSpawner.cpp b/dScripts/EnemySpiderSpawner.cpp deleted file mode 100644 index 96302a33..00000000 --- a/dScripts/EnemySpiderSpawner.cpp +++ /dev/null @@ -1,72 +0,0 @@ -#include "EnemySpiderSpawner.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "DestroyableComponent.h" - -//---------------------------------------------- -//--Initiate egg hatching on call -//---------------------------------------------- -void EnemySpiderSpawner::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (args == "prepEgg") { - // Highlight eggs about to hatch with Maelstrom effect - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2856, u"maelstrom", "test", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - - // Make indestructible - auto dest = static_cast<DestroyableComponent*>(self->GetComponent(COMPONENT_TYPE_DESTROYABLE)); - if (dest) { - dest->SetFaction(-1); - } - EntityManager::Instance()->SerializeEntity(self); - - // Keep track of who prepped me - self->SetI64(u"SpawnOwner", sender->GetObjectID()); - - } else if (args == "hatchEgg") { - // Final countdown to pop - self->AddTimer("StartSpawnTime", hatchTime); - } -} - -//---------------------------------------------------------------- -//--Called when timers are done -//---------------------------------------------------------------- -void EnemySpiderSpawner::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "StartSpawnTime") { - SpawnSpiderling(self); - } else if (timerName == "SpawnSpiderling") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 644, u"create", "egg_puff_b", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - - //TODO: set the aggro radius larger - - EntityInfo info{}; - info.lot = 16197; - info.pos = self->GetPosition(); - info.spawner = nullptr; - info.spawnerID = self->GetI64(u"SpawnOwner"); - info.spawnerNodeID = 0; - - Entity* newEntity = EntityManager::Instance()->CreateEntity(info, nullptr); - if (newEntity) { - EntityManager::Instance()->ConstructEntity(newEntity); - newEntity->GetGroups().push_back("BabySpider"); - - /* - auto* movementAi = newEntity->GetComponent<MovementAIComponent>(); - - movementAi->SetDestination(newEntity->GetPosition()); - */ - } - - self->ScheduleKillAfterUpdate(); - } -} - -//-------------------------------------------------------------- -//Called when it is finally time to release the Spiderlings -//-------------------------------------------------------------- -void EnemySpiderSpawner::SpawnSpiderling(Entity* self) { - //Initiate the actual spawning - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2260, u"rebuild_medium", "dropdustmedium", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - self->AddTimer("SpawnSpiderling", spawnTime); -} diff --git a/dScripts/AnvilOfArmor.cpp b/dScripts/EquipmentScripts/AnvilOfArmor.cpp similarity index 100% rename from dScripts/AnvilOfArmor.cpp rename to dScripts/EquipmentScripts/AnvilOfArmor.cpp diff --git a/dScripts/AnvilOfArmor.h b/dScripts/EquipmentScripts/AnvilOfArmor.h similarity index 100% rename from dScripts/AnvilOfArmor.h rename to dScripts/EquipmentScripts/AnvilOfArmor.h diff --git a/dScripts/EquipmentScripts/BuccaneerValiantShip.cpp b/dScripts/EquipmentScripts/BuccaneerValiantShip.cpp new file mode 100644 index 00000000..15954622 --- /dev/null +++ b/dScripts/EquipmentScripts/BuccaneerValiantShip.cpp @@ -0,0 +1,18 @@ +#include "BuccaneerValiantShip.h" +#include "SkillComponent.h" + +void BuccaneerValiantShip::OnStartup(Entity* self) { + self->AddCallbackTimer(1.0F, [self]() { + auto* skillComponent = self->GetComponent<SkillComponent>(); + auto* owner = self->GetOwner(); + + if (skillComponent != nullptr && owner != nullptr) { + skillComponent->CalculateBehavior(982, 20577, LWOOBJID_EMPTY, true, false, owner->GetObjectID()); + + // Kill self if missed + self->AddCallbackTimer(1.1F, [self]() { + self->Smash(); + }); + } + }); +} diff --git a/dScripts/BuccaneerValiantShip.h b/dScripts/EquipmentScripts/BuccaneerValiantShip.h similarity index 100% rename from dScripts/BuccaneerValiantShip.h rename to dScripts/EquipmentScripts/BuccaneerValiantShip.h diff --git a/dScripts/EquipmentScripts/CMakeLists.txt b/dScripts/EquipmentScripts/CMakeLists.txt new file mode 100644 index 00000000..696be03d --- /dev/null +++ b/dScripts/EquipmentScripts/CMakeLists.txt @@ -0,0 +1,10 @@ +set(DSCRIPTS_SOURCES_EQUIPMENTSCRIPTS + "Sunflower.cpp" + "AnvilOfArmor.cpp" + "FountainOfImagination.cpp" + "CauldronOfLife.cpp" + "PersonalFortress.cpp" + "BuccaneerValiantShip.cpp" + "FireFirstSkillonStartup.cpp" + "StunImmunity.cpp" + PARENT_SCOPE) diff --git a/dScripts/CauldronOfLife.cpp b/dScripts/EquipmentScripts/CauldronOfLife.cpp similarity index 100% rename from dScripts/CauldronOfLife.cpp rename to dScripts/EquipmentScripts/CauldronOfLife.cpp diff --git a/dScripts/CauldronOfLife.h b/dScripts/EquipmentScripts/CauldronOfLife.h similarity index 100% rename from dScripts/CauldronOfLife.h rename to dScripts/EquipmentScripts/CauldronOfLife.h diff --git a/dScripts/EquipmentScripts/FireFirstSkillonStartup.cpp b/dScripts/EquipmentScripts/FireFirstSkillonStartup.cpp new file mode 100644 index 00000000..389f3621 --- /dev/null +++ b/dScripts/EquipmentScripts/FireFirstSkillonStartup.cpp @@ -0,0 +1,26 @@ +#include "FireFirstSkillonStartup.h" +#include "Entity.h" +#include "SkillComponent.h" +#include "CDClientDatabase.h" +#include "CDObjectSkillsTable.h" +#include "CDSkillBehaviorTable.h" +#include "CDClientManager.h" + +void FireFirstSkillonStartup::OnStartup(Entity* self) { + auto skillComponent = self->GetComponent<SkillComponent>(); + if (!skillComponent) return; + + // Get the skill IDs of this object. + CDObjectSkillsTable* skillsTable = CDClientManager::Instance().GetTable<CDObjectSkillsTable>(); + std::vector<CDObjectSkills> skills = skillsTable->Query([=](CDObjectSkills entry) {return (entry.objectTemplate == self->GetLOT()); }); + + // For each skill, cast it with the associated behavior ID. + for (auto skill : skills) { + CDSkillBehaviorTable* skillBehaviorTable = CDClientManager::Instance().GetTable<CDSkillBehaviorTable>(); + CDSkillBehavior behaviorData = skillBehaviorTable->GetSkillByID(skill.skillID); + + // Should parent entity be null, make the originator self. + const auto target = self->GetParentEntity() ? self->GetParentEntity()->GetObjectID() : self->GetObjectID(); + skillComponent->CalculateBehavior(skill.skillID, behaviorData.behaviorID, LWOOBJID_EMPTY, false, false, target); + } +} diff --git a/dScripts/FireFirstSkillonStartup.h b/dScripts/EquipmentScripts/FireFirstSkillonStartup.h similarity index 100% rename from dScripts/FireFirstSkillonStartup.h rename to dScripts/EquipmentScripts/FireFirstSkillonStartup.h diff --git a/dScripts/FountainOfImagination.cpp b/dScripts/EquipmentScripts/FountainOfImagination.cpp similarity index 100% rename from dScripts/FountainOfImagination.cpp rename to dScripts/EquipmentScripts/FountainOfImagination.cpp diff --git a/dScripts/FountainOfImagination.h b/dScripts/EquipmentScripts/FountainOfImagination.h similarity index 100% rename from dScripts/FountainOfImagination.h rename to dScripts/EquipmentScripts/FountainOfImagination.h diff --git a/dScripts/EquipmentScripts/PersonalFortress.cpp b/dScripts/EquipmentScripts/PersonalFortress.cpp new file mode 100644 index 00000000..7e6d3566 --- /dev/null +++ b/dScripts/EquipmentScripts/PersonalFortress.cpp @@ -0,0 +1,58 @@ +#include "PersonalFortress.h" +#include "GameMessages.h" +#include "SkillComponent.h" +#include "DestroyableComponent.h" +#include "ControllablePhysicsComponent.h" +#include "EntityManager.h" +#include "eStateChangeType.h" + +void PersonalFortress::OnStartup(Entity* self) { + auto* owner = self->GetOwner(); + self->AddTimer("FireSkill", 1.5); + + auto* destroyableComponent = owner->GetComponent<DestroyableComponent>(); + if (destroyableComponent) destroyableComponent->SetStatusImmunity( + eStateChangeType::PUSH, + true, true, true, true, true, false, true, false, false + ); + + auto* controllablePhysicsComponent = owner->GetComponent<ControllablePhysicsComponent>(); + if (controllablePhysicsComponent) controllablePhysicsComponent->SetStunImmunity( + eStateChangeType::PUSH, LWOOBJID_EMPTY, + true, true, true, true, true, true + ); + + GameMessages::SendSetStunned(owner->GetObjectID(), eStateChangeType::PUSH, owner->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true, true, true + ); + + Game::entityManager->SerializeEntity(owner); +} + +void PersonalFortress::OnDie(Entity* self, Entity* killer) { + auto* owner = self->GetOwner(); + auto* destroyableComponent = owner->GetComponent<DestroyableComponent>(); + if (destroyableComponent) destroyableComponent->SetStatusImmunity( + eStateChangeType::POP, + true, true, true, true, true, false, true, false, false + ); + + auto* controllablePhysicsComponent = owner->GetComponent<ControllablePhysicsComponent>(); + if (controllablePhysicsComponent) controllablePhysicsComponent->SetStunImmunity( + eStateChangeType::POP, LWOOBJID_EMPTY, + true, true, true, true, true, true + ); + + GameMessages::SendSetStunned(owner->GetObjectID(), eStateChangeType::POP, owner->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true, true, true + ); + + Game::entityManager->SerializeEntity(owner); +} + +void PersonalFortress::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "FireSkill") { + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent) skillComponent->CalculateBehavior(650, 13364, LWOOBJID_EMPTY, true, false); + } +} diff --git a/dScripts/PersonalFortress.h b/dScripts/EquipmentScripts/PersonalFortress.h similarity index 100% rename from dScripts/PersonalFortress.h rename to dScripts/EquipmentScripts/PersonalFortress.h diff --git a/dScripts/EquipmentScripts/StunImmunity.cpp b/dScripts/EquipmentScripts/StunImmunity.cpp new file mode 100644 index 00000000..0ec956f0 --- /dev/null +++ b/dScripts/EquipmentScripts/StunImmunity.cpp @@ -0,0 +1,20 @@ +#include "StunImmunity.h" +#include "DestroyableComponent.h" +#include "ControllablePhysicsComponent.h" +#include "eStateChangeType.h" + +void StunImmunity::OnStartup(Entity* self) { + auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); + if (destroyableComponent) { + destroyableComponent->SetStatusImmunity( + eStateChangeType::PUSH, false, false, true, true, false, false, false, false, true + ); + } + + auto* controllablePhysicsComponent = self->GetComponent<ControllablePhysicsComponent>(); + if (controllablePhysicsComponent) { + controllablePhysicsComponent->SetStunImmunity( + eStateChangeType::PUSH, self->GetObjectID(), true + ); + } +} diff --git a/dScripts/EquipmentScripts/StunImmunity.h b/dScripts/EquipmentScripts/StunImmunity.h new file mode 100644 index 00000000..0faaf061 --- /dev/null +++ b/dScripts/EquipmentScripts/StunImmunity.h @@ -0,0 +1,6 @@ +#pragma once +#include "CppScripts.h" + +class StunImmunity : public CppScripts::Script { + void OnStartup(Entity* self) override; +}; diff --git a/dScripts/Sunflower.cpp b/dScripts/EquipmentScripts/Sunflower.cpp similarity index 100% rename from dScripts/Sunflower.cpp rename to dScripts/EquipmentScripts/Sunflower.cpp diff --git a/dScripts/Sunflower.h b/dScripts/EquipmentScripts/Sunflower.h similarity index 100% rename from dScripts/Sunflower.h rename to dScripts/EquipmentScripts/Sunflower.h diff --git a/dScripts/EquipmentTriggers/CMakeLists.txt b/dScripts/EquipmentTriggers/CMakeLists.txt new file mode 100644 index 00000000..416ef553 --- /dev/null +++ b/dScripts/EquipmentTriggers/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_EQUIPMENTTRIGGERSSCRIPTS + "CoilBackpackBase.cpp" + PARENT_SCOPE) diff --git a/dScripts/EquipmentTriggers/CoilBackpackBase.cpp b/dScripts/EquipmentTriggers/CoilBackpackBase.cpp new file mode 100644 index 00000000..4e323a08 --- /dev/null +++ b/dScripts/EquipmentTriggers/CoilBackpackBase.cpp @@ -0,0 +1,25 @@ +#include "CoilBackpackBase.h" + +#include "Entity.h" +#include "SkillComponent.h" + +void CoilBackpackBase::OnFactionTriggerItemEquipped(Entity* itemOwner, LWOOBJID itemObjId) { + itemOwner->Subscribe(itemObjId, this, "HitOrHealResult"); + itemOwner->SetVar<uint8_t>(u"coilCount", 0); +} + +void CoilBackpackBase::NotifyHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { + if (damage > 0) { + self->SetVar<uint8_t>(u"coilCount", self->GetVar<uint8_t>(u"coilCount") + 1); + if (self->GetVar<uint8_t>(u"coilCount") > 4) { + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (!skillComponent) return; + skillComponent->CastSkill(m_SkillId); + self->SetVar<uint8_t>(u"coilCount", 0); + } + } +} + +void CoilBackpackBase::OnFactionTriggerItemUnequipped(Entity* itemOwner, LWOOBJID itemObjId) { + itemOwner->Unsubscribe(itemObjId, "HitOrHealResult"); +} diff --git a/dScripts/EquipmentTriggers/CoilBackpackBase.h b/dScripts/EquipmentTriggers/CoilBackpackBase.h new file mode 100644 index 00000000..2d641346 --- /dev/null +++ b/dScripts/EquipmentTriggers/CoilBackpackBase.h @@ -0,0 +1,19 @@ +#ifndef __GemPackBase__H__ +#define __GemPackBase__H__ + +#include "CppScripts.h" + +class CoilBackpackBase: public CppScripts::Script { +public: + CoilBackpackBase(uint32_t skillId) { + m_SkillId = skillId; + }; + + void OnFactionTriggerItemEquipped(Entity* itemOwner, LWOOBJID itemObjId) override; + void NotifyHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) override; + void OnFactionTriggerItemUnequipped(Entity* itemOwner, LWOOBJID itemObjId) override; +private: + uint32_t m_SkillId = 0; +}; + +#endif //!__GemPackBase__H__ diff --git a/dScripts/EquipmentTriggers/GemPack.h b/dScripts/EquipmentTriggers/GemPack.h new file mode 100644 index 00000000..d71181ab --- /dev/null +++ b/dScripts/EquipmentTriggers/GemPack.h @@ -0,0 +1,13 @@ +#ifndef __GEMPACK__H__ +#define __GEMPACK__H__ + +#include "CoilBackpackBase.h" + +class GemPack : public CoilBackpackBase { +public: + GemPack() : CoilBackpackBase(skillId) {}; +private: + static const uint32_t skillId = 1488; +}; + +#endif //!__GEMPACK__H__ diff --git a/dScripts/EquipmentTriggers/ShardArmor.h b/dScripts/EquipmentTriggers/ShardArmor.h new file mode 100644 index 00000000..5486db54 --- /dev/null +++ b/dScripts/EquipmentTriggers/ShardArmor.h @@ -0,0 +1,13 @@ +#ifndef __SHARDARMOR__H__ +#define __SHARDARMOR__H__ + +#include "CoilBackpackBase.h" + +class ShardArmor : public CoilBackpackBase { +public: + ShardArmor() : CoilBackpackBase(skillId) {}; +private: + static const uint32_t skillId = 1249; +}; + +#endif //!__SHARDARMOR__H__ diff --git a/dScripts/EquipmentTriggers/TeslaPack.h b/dScripts/EquipmentTriggers/TeslaPack.h new file mode 100644 index 00000000..3ba09f5c --- /dev/null +++ b/dScripts/EquipmentTriggers/TeslaPack.h @@ -0,0 +1,13 @@ +#ifndef __TESLAPACK__H__ +#define __TESLAPACK__H__ + +#include "CoilBackpackBase.h" + +class TeslaPack : public CoilBackpackBase { +public: + TeslaPack() : CoilBackpackBase(skillId) {}; +private: + static const uint32_t skillId = 1001; +}; + +#endif //!__TESLAPACK__H__ diff --git a/dScripts/ExplodingAsset.cpp b/dScripts/ExplodingAsset.cpp deleted file mode 100644 index 46ff1522..00000000 --- a/dScripts/ExplodingAsset.cpp +++ /dev/null @@ -1,109 +0,0 @@ -#include "ExplodingAsset.h" -#include "DestroyableComponent.h" -#include "GameMessages.h" -#include "MissionComponent.h" -#include "SkillComponent.h" - -//TODO: this has to be updated so that you only get killed if you're in a certain radius. -//And so that all entities in a certain radius are killed, not just the attacker. - -void ExplodingAsset::OnStartup(Entity* self) { - self->SetProximityRadius(20.0f, "outRadius"); - self->SetVar<int32_t>(u"playersNearChest", 0); - self->SetProximityRadius(10.0f, "crateHitters"); -} - -void ExplodingAsset::OnHit(Entity* self, Entity* attacker) { - std::vector<Entity*> entities; - entities.push_back(attacker); - - if (!self->GetBoolean(u"bIsHit")) { - for (Entity* en : entities) { - if (en->GetObjectID() == attacker->GetObjectID()) { - if (Vector3::DistanceSquared(en->GetPosition(), self->GetPosition()) > 10 * 10) continue; - - auto* destroyable = en->GetComponent<DestroyableComponent>(); - if (destroyable == nullptr) { - continue; - } - - destroyable->Smash(attacker->GetObjectID()); - } - } - } - - attacker = attacker->GetOwner(); - self->SetBoolean(u"bIsHit", true); - self->SetOwnerOverride(attacker->GetObjectID()); - - GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(self, u"camshake", self->GetObjectID(), 16); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(147, 4721, LWOOBJID_EMPTY, true); - } - - const auto missionID = self->GetVar<int32_t>(u"missionID"); - auto achievementIDs = self->GetVar<std::u16string>(u"achieveID"); - - // Progress all scripted missions related to this asset - auto* missionComponent = attacker->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - if (missionID != 0) { - missionComponent->ForceProgressValue(missionID, - static_cast<uint32_t>(MissionTaskType::MISSION_TASK_TYPE_SCRIPT), - self->GetLOT(), false); - } - - if (!achievementIDs.empty()) { - for (const auto& achievementID : GeneralUtils::SplitString(achievementIDs, u'_')) { - missionComponent->ForceProgressValue(std::stoi(GeneralUtils::UTF16ToWTF8(achievementID)), - static_cast<uint32_t>(MissionTaskType::MISSION_TASK_TYPE_SCRIPT), - self->GetLOT()); - } - } - } - - self->ScheduleKillAfterUpdate(); -} - -void ExplodingAsset::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - /* - if msg.objId:BelongsToFaction{factionID = 1}.bIsInFaction then - if (msg.status == "ENTER") then - self:PlayAnimation{ animationID = "bounce" } - self:PlayFXEffect{ name = "bouncin", effectType = "anim" } - self:SetVar("playersNearChest", (self:GetVar("playersNearChest") + 1 )) - elseif (msg.status == "LEAVE") then - self:SetVar("playersNearChest", (self:GetVar("playersNearChest") - 1 )) - if self:GetVar("playersNearChest") < 1 then - self:PlayAnimation{ animationID = "idle" } - self:StopFXEffect{ name = "bouncin" } - self:SetVar("playersNearChest", 0) - end - end - end - */ - - auto* destuctableComponent = entering->GetComponent<DestroyableComponent>(); - - if (destuctableComponent == nullptr) return; - - const auto& factions = destuctableComponent->GetFactionIDs(); - - if (!std::count(factions.begin(), factions.end(), 1)) return; - - if (status == "ENTER") { - GameMessages::SendPlayAnimation(self, u"bounce"); - GameMessages::SendPlayFXEffect(self, -1, u"anim", "bouncin", LWOOBJID_EMPTY, 1, 1, true); - self->SetVar(u"playersNearChest", self->GetVar<int32_t>(u"playersNearChest") + 1); - } else if (status == "LEAVE") { - self->SetVar(u"playersNearChest", self->GetVar<int32_t>(u"playersNearChest") - 1); - - if (self->GetVar<int32_t>(u"playersNearChest") < 1) { - GameMessages::SendPlayAnimation(self, u"idle"); - GameMessages::SendStopFXEffect(self, true, "bouncin"); - self->SetVar<int32_t>(u"playersNearChest", 0); - } - } -} diff --git a/dScripts/FireFirstSkillonStartup.cpp b/dScripts/FireFirstSkillonStartup.cpp deleted file mode 100644 index cd0d94f2..00000000 --- a/dScripts/FireFirstSkillonStartup.cpp +++ /dev/null @@ -1,24 +0,0 @@ -#include "FireFirstSkillonStartup.h" -#include "Entity.h" -#include "SkillComponent.h" -#include "CDClientDatabase.h" -#include "CDObjectSkillsTable.h" - -void FireFirstSkillonStartup::OnStartup(Entity* self) { - auto skillComponent = self->GetComponent<SkillComponent>(); - if (!skillComponent) return; - - // Get the skill IDs of this object. - CDObjectSkillsTable* skillsTable = CDClientManager::Instance()->GetTable<CDObjectSkillsTable>("ObjectSkills"); - std::vector<CDObjectSkills> skills = skillsTable->Query([=](CDObjectSkills entry) {return (entry.objectTemplate == self->GetLOT()); }); - - // For each skill, cast it with the associated behavior ID. - for (auto skill : skills) { - CDSkillBehaviorTable* skillBehaviorTable = CDClientManager::Instance()->GetTable<CDSkillBehaviorTable>("SkillBehavior"); - CDSkillBehavior behaviorData = skillBehaviorTable->GetSkillByID(skill.skillID); - - // Should parent entity be null, make the originator self. - const auto target = self->GetParentEntity() ? self->GetParentEntity()->GetObjectID() : self->GetObjectID(); - skillComponent->CalculateBehavior(skill.skillID, behaviorData.behaviorID, LWOOBJID_EMPTY, false, false, target); - } -} diff --git a/dScripts/ForceVolumeServer.cpp b/dScripts/ForceVolumeServer.cpp deleted file mode 100644 index fbaad6ee..00000000 --- a/dScripts/ForceVolumeServer.cpp +++ /dev/null @@ -1,21 +0,0 @@ -#include "ForceVolumeServer.h" -#include "PhantomPhysicsComponent.h" -#include "EntityManager.h" - -void ForceVolumeServer::OnStartup(Entity* self) { - auto* phantomPhysicsComponent = self->GetComponent<PhantomPhysicsComponent>(); - - if (phantomPhysicsComponent == nullptr) return; - - const auto forceAmount = self->GetVar<float>(u"ForceAmt"); - const auto forceX = self->GetVar<float>(u"ForceX"); - const auto forceY = self->GetVar<float>(u"ForceY"); - const auto forceZ = self->GetVar<float>(u"ForceZ"); - - phantomPhysicsComponent->SetEffectType(0); // PUSH - phantomPhysicsComponent->SetDirectionalMultiplier(forceAmount); - phantomPhysicsComponent->SetDirection({ forceX, forceY, forceZ }); - phantomPhysicsComponent->SetPhysicsEffectActive(true); - - EntityManager::Instance()->SerializeEntity(self); -} diff --git a/dScripts/FvBrickPuzzleServer.cpp b/dScripts/FvBrickPuzzleServer.cpp deleted file mode 100644 index cea6146b..00000000 --- a/dScripts/FvBrickPuzzleServer.cpp +++ /dev/null @@ -1,68 +0,0 @@ -#include "FvBrickPuzzleServer.h" -#include "GeneralUtils.h" -#include "dZoneManager.h" -#include "Spawner.h" -#include "RebuildComponent.h" - -void FvBrickPuzzleServer::OnStartup(Entity* self) { - const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); - - int32_t pipeNum = 0; - if (!GeneralUtils::TryParse<int32_t>(myGroup.substr(10, 1), pipeNum)) { - return; - } - - if (pipeNum != 1) { - self->AddTimer("reset", 30); - } -} - -void FvBrickPuzzleServer::OnDie(Entity* self, Entity* killer) { - const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); - - int32_t pipeNum = 0; - if (!GeneralUtils::TryParse<int32_t>(myGroup.substr(10, 1), pipeNum)) { - return; - } - - const auto pipeGroup = myGroup.substr(0, 10); - - const auto nextPipeNum = pipeNum + 1; - - const auto samePipeSpawners = dZoneManager::Instance()->GetSpawnersByName(myGroup); - - if (!samePipeSpawners.empty()) { - samePipeSpawners[0]->SoftReset(); - - samePipeSpawners[0]->Deactivate(); - } - - if (killer != nullptr && killer->IsPlayer()) { - const auto nextPipe = pipeGroup + std::to_string(nextPipeNum); - - const auto nextPipeSpawners = dZoneManager::Instance()->GetSpawnersByName(nextPipe); - - if (!nextPipeSpawners.empty()) { - nextPipeSpawners[0]->Activate(); - } - } else { - const auto nextPipe = pipeGroup + "1"; - - const auto firstPipeSpawners = dZoneManager::Instance()->GetSpawnersByName(nextPipe); - - if (!firstPipeSpawners.empty()) { - firstPipeSpawners[0]->Activate(); - } - } - -} - -void FvBrickPuzzleServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "reset") { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent != nullptr && rebuildComponent->GetState() == REBUILD_OPEN) { - self->Smash(self->GetObjectID(), SILENT); - } - } -} diff --git a/dScripts/FvCandle.cpp b/dScripts/FvCandle.cpp deleted file mode 100644 index efd717ee..00000000 --- a/dScripts/FvCandle.cpp +++ /dev/null @@ -1,55 +0,0 @@ -#include "FvCandle.h" -#include "MissionComponent.h" -#include "RenderComponent.h" - -std::vector<int32_t> FvCandle::m_Missions = { 850, 1431, 1529, 1566, 1603 }; - -void FvCandle::OnStartup(Entity* self) { - auto* render = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - if (render == nullptr) - return; - - render->PlayEffect(2108, u"create", "candle_light", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - self->SetI32(u"Smoke", static_cast<int32_t>(5)); - self->SetBoolean(u"AmHit", false); -} - -void FvCandle::OnHit(Entity* self, Entity* attacker) { - BlowOutCandle(self, attacker); -} - -void FvCandle::BlowOutCandle(Entity* self, Entity* blower) { - if (self->GetBoolean(u"AmHit")) - return; - - auto* render = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - if (render == nullptr) - return; - - auto* missionComponent = blower->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - for (const auto mission : m_Missions) { - missionComponent->ForceProgressTaskType(mission, 1, 1); - } - } - - //Update mission tasks here - self->SetBoolean(u"AmHit", true); - - render->StopEffect("candle_light", false); - render->PlayEffect(2109, u"create", "candle_smoke", LWOOBJID_EMPTY, 1.0f, 1.0f, true); - - self->AddTimer("SmokeTime", self->GetI32(u"Smoke")); -} - -void FvCandle::OnTimerDone(Entity* self, std::string timerName) { - self->SetBoolean(u"AmHit", false); - - auto* render = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - if (render == nullptr) - return; - - render->StopEffect("candle_smoke", false); - render->PlayEffect(2108, u"create", "candle_light", LWOOBJID_EMPTY, 1.0f, 1.0f, true); -} diff --git a/dScripts/FvConsoleLeftQuickbuild.cpp b/dScripts/FvConsoleLeftQuickbuild.cpp deleted file mode 100644 index b998b9ec..00000000 --- a/dScripts/FvConsoleLeftQuickbuild.cpp +++ /dev/null @@ -1,47 +0,0 @@ -#include "FvConsoleLeftQuickbuild.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void FvConsoleLeftQuickbuild::OnStartup(Entity* self) { - self->SetVar(u"IAmBuilt", false); - self->SetVar(u"AmActive", false); -} - -void FvConsoleLeftQuickbuild::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == REBUILD_COMPLETED) { - self->SetVar(u"IAmBuilt", true); - - const auto objects = EntityManager::Instance()->GetEntitiesInGroup("Facility"); - - if (!objects.empty()) { - objects[0]->NotifyObject(self, "ConsoleLeftUp"); - } - } else if (state == REBUILD_RESETTING) { - self->SetVar(u"IAmBuilt", false); - self->SetVar(u"AmActive", false); - - const auto objects = EntityManager::Instance()->GetEntitiesInGroup("Facility"); - - if (!objects.empty()) { - objects[0]->NotifyObject(self, "ConsoleLeftDown"); - } - } -} - -void FvConsoleLeftQuickbuild::OnUse(Entity* self, Entity* user) { - if (self->GetVar<bool>(u"AmActive")) { - return; - } - - if (self->GetVar<bool>(u"IAmBuilt")) { - self->SetVar(u"AmActive", true); - - const auto objects = EntityManager::Instance()->GetEntitiesInGroup("Facility"); - - if (!objects.empty()) { - objects[0]->NotifyObject(self, "ConsoleLeftActive"); - } - } - - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} diff --git a/dScripts/FvConsoleRightQuickbuild.cpp b/dScripts/FvConsoleRightQuickbuild.cpp deleted file mode 100644 index ea047cd8..00000000 --- a/dScripts/FvConsoleRightQuickbuild.cpp +++ /dev/null @@ -1,47 +0,0 @@ -#include "FvConsoleRightQuickbuild.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void FvConsoleRightQuickbuild::OnStartup(Entity* self) { - self->SetVar(u"IAmBuilt", false); - self->SetVar(u"AmActive", false); -} - -void FvConsoleRightQuickbuild::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == REBUILD_COMPLETED) { - self->SetVar(u"IAmBuilt", true); - - const auto objects = EntityManager::Instance()->GetEntitiesInGroup("Facility"); - - if (!objects.empty()) { - objects[0]->NotifyObject(self, "ConsoleRightUp"); - } - } else if (state == REBUILD_RESETTING) { - self->SetVar(u"IAmBuilt", false); - self->SetVar(u"AmActive", false); - - const auto objects = EntityManager::Instance()->GetEntitiesInGroup("Facility"); - - if (!objects.empty()) { - objects[0]->NotifyObject(self, "ConsoleRightDown"); - } - } -} - -void FvConsoleRightQuickbuild::OnUse(Entity* self, Entity* user) { - if (self->GetVar<bool>(u"AmActive")) { - return; - } - - if (self->GetVar<bool>(u"IAmBuilt")) { - self->SetVar(u"AmActive", true); - - const auto objects = EntityManager::Instance()->GetEntitiesInGroup("Facility"); - - if (!objects.empty()) { - objects[0]->NotifyObject(self, "ConsoleRightActive"); - } - } - - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} diff --git a/dScripts/FvDragonSmashingGolemQb.cpp b/dScripts/FvDragonSmashingGolemQb.cpp deleted file mode 100644 index a9d38aa5..00000000 --- a/dScripts/FvDragonSmashingGolemQb.cpp +++ /dev/null @@ -1,30 +0,0 @@ -#include "FvDragonSmashingGolemQb.h" -#include "GameMessages.h" -#include "EntityManager.h" - -void FvDragonSmashingGolemQb::OnStartup(Entity* self) { - self->AddTimer("GolemBreakTimer", 10.5f); -} - -void FvDragonSmashingGolemQb::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "GolemBreakTimer") { - self->Smash(); - } -} - -void FvDragonSmashingGolemQb::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == eRebuildState::REBUILD_COMPLETED) { - GameMessages::SendPlayAnimation(self, u"dragonsmash"); - - const auto dragonId = self->GetVar<LWOOBJID>(u"Dragon"); - - auto* dragon = EntityManager::Instance()->GetEntity(dragonId); - - if (dragon != nullptr) { - dragon->OnFireEventServerSide(self, "rebuildDone"); - } - - self->CancelTimer("GolemBreakTimer"); - self->AddTimer("GolemBreakTimer", 10.5f); - } -} diff --git a/dScripts/FvFacilityBrick.cpp b/dScripts/FvFacilityBrick.cpp deleted file mode 100644 index 1ae910e4..00000000 --- a/dScripts/FvFacilityBrick.cpp +++ /dev/null @@ -1,97 +0,0 @@ -#include "FvFacilityBrick.h" -#include "GameMessages.h" -#include "dZoneManager.h" -#include "EntityManager.h" - -void FvFacilityBrick::OnStartup(Entity* self) { - self->SetVar(u"ConsoleLEFTActive", false); - self->SetVar(u"ConsoleRIGHTtActive", false); -} - -void FvFacilityBrick::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { - auto* brickSpawner = dZoneManager::Instance()->GetSpawnersByName("ImaginationBrick")[0]; - auto* bugSpawner = dZoneManager::Instance()->GetSpawnersByName("MaelstromBug")[0]; - auto* canisterSpawner = dZoneManager::Instance()->GetSpawnersByName("BrickCanister")[0]; - - if (name == "ConsoleLeftUp") { - GameMessages::SendStopFXEffect(self, true, "LeftPipeOff"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2775, u"create", "LeftPipeEnergy"); - } else if (name == "ConsoleLeftDown") { - self->SetVar(u"ConsoleLEFTActive", false); - - GameMessages::SendStopFXEffect(self, true, "LeftPipeEnergy"); - GameMessages::SendStopFXEffect(self, true, "LeftPipeOn"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2774, u"create", "LeftPipeOff"); - } else if (name == "ConsoleLeftActive") { - self->SetVar(u"ConsoleLEFTActive", true); - - GameMessages::SendStopFXEffect(self, true, "LeftPipeEnergy"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2776, u"create", "LeftPipeOn"); - } - - else if (name == "ConsoleRightUp") { - GameMessages::SendStopFXEffect(self, true, "RightPipeOff"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2778, u"create", "RightPipeEnergy"); - } else if (name == "ConsoleRightDown") { - self->SetVar(u"ConsoleRIGHTActive", false); - - GameMessages::SendStopFXEffect(self, true, "RightPipeEnergy"); - GameMessages::SendStopFXEffect(self, true, "RightPipeOn"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2777, u"create", "RightPipeOff"); - } else if (name == "ConsoleRightActive") { - self->SetVar(u"ConsoleRIGHTActive", true); - - GameMessages::SendStopFXEffect(self, true, "RightPipeOff"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2779, u"create", "RightPipeEnergy"); - } - - if (self->GetVar<bool>(u"ConsoleLEFTActive") && self->GetVar<bool>(u"ConsoleRIGHTActive")) { - auto* object = EntityManager::Instance()->GetEntitiesInGroup("Brick")[0]; - - if (object != nullptr) { - GameMessages::SendPlayFXEffect(object->GetObjectID(), 122, u"create", "bluebrick"); - GameMessages::SendPlayFXEffect(object->GetObjectID(), 1034, u"cast", "imaginationexplosion"); - } - - object = EntityManager::Instance()->GetEntitiesInGroup("Canister")[0]; - - if (object != nullptr) { - object->Smash(self->GetObjectID(), SILENT); - } - - canisterSpawner->Reset(); - canisterSpawner->Deactivate(); - } else if (self->GetVar<bool>(u"ConsoleLEFTActive") || self->GetVar<bool>(u"ConsoleRIGHTActive")) { - brickSpawner->Activate(); - - auto* object = EntityManager::Instance()->GetEntitiesInGroup("Brick")[0]; - - if (object != nullptr) { - GameMessages::SendStopFXEffect(object, true, "bluebrick"); - } - - bugSpawner->Reset(); - bugSpawner->Deactivate(); - - canisterSpawner->Reset(); - canisterSpawner->Activate(); - } else { - brickSpawner->Reset(); - brickSpawner->Deactivate(); - - bugSpawner->Reset(); - bugSpawner->Activate(); - } -} - -void FvFacilityBrick::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args != "PlayFX") { - return; - } - - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2774, u"create", "LeftPipeOff"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2777, u"create", "RightPipeOff"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2750, u"create", "imagination_canister"); - GameMessages::SendPlayFXEffect(self->GetObjectID(), 2751, u"create", "canister_light_filler"); -} diff --git a/dScripts/FvFlyingCreviceDragon.cpp b/dScripts/FvFlyingCreviceDragon.cpp deleted file mode 100644 index 16eda512..00000000 --- a/dScripts/FvFlyingCreviceDragon.cpp +++ /dev/null @@ -1,107 +0,0 @@ -#include "FvFlyingCreviceDragon.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "SkillComponent.h" -#include "GeneralUtils.h" - -void FvFlyingCreviceDragon::OnStartup(Entity* self) { - self->AddTimer("waypoint", 5); -} - -void FvFlyingCreviceDragon::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "waypoint") { - auto point = self->GetVar<int32_t>(u"waypoint"); - - if (point >= 20) { - point = 0; - } - - self->SetVar<int32_t>(u"waypoint", point + 1); - - self->AddTimer("waypoint", 5); - - OnArrived(self); - - return; - } - - std::string groupName = ""; - - if (timerName == "platform1attack") { - groupName = "dragonFireballs1"; - } else if (timerName == "platform3attack") { - groupName = "dragonFireballs3"; - } - - const auto& group = EntityManager::Instance()->GetEntitiesInGroup(groupName); - - if (group.empty()) { - return; - } - - auto* skillComponent = group[0]->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY, true); - } - - auto minionCount = 1; - for (size_t i = 1; i < group.size(); i++) { - if (minionCount == 4) { - return; - } - - if (/*GeneralUtils::GenerateRandomNumber<int32_t>(1, 5) > 3*/ true) { - skillComponent = group[i]->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY); - - ++minionCount; - } - } - } -} - -void FvFlyingCreviceDragon::OnArrived(Entity* self) { - auto point = self->GetVar<int32_t>(u"waypoint"); - - if (point == 4) { - GameMessages::SendPlayAnimation(self, u"attack1", 2); - self->AddTimer("platform1attack", 1.75f); - } else if (point == 12) { - GameMessages::SendPlayAnimation(self, u"attack2", 2); - - const auto& group2 = EntityManager::Instance()->GetEntitiesInGroup("dragonFireballs2"); - - if (group2.empty()) { - return; - } - - auto* skillComponent = group2[0]->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY); - } - - auto minionCount = 1; - for (size_t i = 1; i < group2.size(); i++) { - if (minionCount == 4) { - return; - } - - if (GeneralUtils::GenerateRandomNumber<int32_t>(1, 5) > 3) { - skillComponent = group2[i]->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY, true); - - ++minionCount; - } - } - } - } else if (point == 16) { - GameMessages::SendPlayAnimation(self, u"attack3", 2); - self->AddTimer("platform3attack", 0.5f); - } -} diff --git a/dScripts/FvFong.cpp b/dScripts/FvFong.cpp deleted file mode 100644 index 7f63e5e5..00000000 --- a/dScripts/FvFong.cpp +++ /dev/null @@ -1,10 +0,0 @@ -#include "FvFong.h" -#include "Darkitect.h" -#include "MissionState.h" - -void FvFong::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID == 734 && missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE) { - Darkitect Baron; - Baron.Reveal(self, target); - } -} diff --git a/dScripts/FvFong.h b/dScripts/FvFong.h deleted file mode 100644 index 074e6d8c..00000000 --- a/dScripts/FvFong.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class FvFong : public CppScripts::Script -{ -public: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -}; diff --git a/dScripts/FvFreeGfNinjas.cpp b/dScripts/FvFreeGfNinjas.cpp deleted file mode 100644 index 1751b6a7..00000000 --- a/dScripts/FvFreeGfNinjas.cpp +++ /dev/null @@ -1,42 +0,0 @@ -#include "FvFreeGfNinjas.h" -#include "Character.h" -#include "MissionComponent.h" - -void FvFreeGfNinjas::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID == 705 && missionState == MissionState::MISSION_STATE_AVAILABLE) { - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) - return; - - missionComponent->AcceptMission(701); - missionComponent->AcceptMission(702); - missionComponent->AcceptMission(703); - missionComponent->AcceptMission(704); - - auto* character = target->GetCharacter(); - if (character != nullptr) - character->SetPlayerFlag(68, true); - } else if (missionID == 786) { - auto* character = target->GetCharacter(); - if (character != nullptr) - character->SetPlayerFlag(81, true); - } -} - -void FvFreeGfNinjas::OnUse(Entity* self, Entity* user) { - // To allow player who already have the mission to progress. - auto* missionComponent = user->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) - return; - - if (missionComponent->GetMissionState(705) == MissionState::MISSION_STATE_ACTIVE) { - auto* character = user->GetCharacter(); - if (character != nullptr) - character->SetPlayerFlag(68, true); - - missionComponent->AcceptMission(701, true); - missionComponent->AcceptMission(702, true); - missionComponent->AcceptMission(703, true); - missionComponent->AcceptMission(704, true); - } -} diff --git a/dScripts/FvFreeGfNinjas.h b/dScripts/FvFreeGfNinjas.h deleted file mode 100644 index c7e4876e..00000000 --- a/dScripts/FvFreeGfNinjas.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class FvFreeGfNinjas : public CppScripts::Script { -public: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - void OnUse(Entity* self, Entity* user) override; -}; diff --git a/dScripts/FvHorsemenTrigger.cpp b/dScripts/FvHorsemenTrigger.cpp deleted file mode 100644 index e87d9629..00000000 --- a/dScripts/FvHorsemenTrigger.cpp +++ /dev/null @@ -1,53 +0,0 @@ -#include "FvHorsemenTrigger.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void FvHorsemenTrigger::OnStartup(Entity* self) { - self->SetProximityRadius(40, "horsemenTrigger"); - - self->SetVar<std::vector<LWOOBJID>>(u"players", {}); -} - -void FvHorsemenTrigger::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name != "horsemenTrigger" || !entering->IsPlayer()) { - return; - } - - auto players = self->GetVar<std::vector<LWOOBJID>>(u"players"); - - const auto& iter = std::find(players.begin(), players.end(), entering->GetObjectID()); - - if (status == "ENTER" && iter == players.end()) { - players.push_back(entering->GetObjectID()); - } else if (status == "LEAVE" && iter != players.end()) { - players.erase(iter); - } - - self->SetVar<std::vector<LWOOBJID>>(u"players", players); -} - -void -FvHorsemenTrigger::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - auto players = self->GetVar<std::vector<LWOOBJID>>(u"players"); - - if (args == "HorsemenDeath") { - for (const auto& playerId : self->GetVar<std::vector<LWOOBJID>>(u"players")) { - auto* player = EntityManager::Instance()->GetEntity(playerId); - - if (player == nullptr) { - continue; - } - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent == nullptr) { - continue; - } - - for (const auto missionId : m_Missions) { - missionComponent->ForceProgressTaskType(missionId, 1, 1); - } - } - } -} diff --git a/dScripts/FvMaelstromCavalry.cpp b/dScripts/FvMaelstromCavalry.cpp deleted file mode 100644 index 9214667e..00000000 --- a/dScripts/FvMaelstromCavalry.cpp +++ /dev/null @@ -1,30 +0,0 @@ -#include "FvMaelstromCavalry.h" -#include "EntityManager.h" - -void FvMaelstromCavalry::OnStartup(Entity* self) { - for (const auto& group : self->GetGroups()) { - const auto& objects = EntityManager::Instance()->GetEntitiesInGroup(group); - - for (auto* obj : objects) { - if (obj->GetLOT() != 8551) continue; - - obj->OnFireEventServerSide(self, "ISpawned"); - } - } -} - -void FvMaelstromCavalry::OnDie(Entity* self, Entity* killer) { - if (killer == nullptr) { - return; - } - - if (killer->GetLOT() != 8665) { - return; - } - - const auto& triggers = EntityManager::Instance()->GetEntitiesInGroup("HorsemenTrigger"); - - for (auto* trigger : triggers) { - trigger->OnFireEventServerSide(self, "HorsemenDeath"); - } -} diff --git a/dScripts/FvMaelstromDragon.cpp b/dScripts/FvMaelstromDragon.cpp deleted file mode 100644 index 47ddb4bf..00000000 --- a/dScripts/FvMaelstromDragon.cpp +++ /dev/null @@ -1,175 +0,0 @@ -#include "FvMaelstromDragon.h" -#include "EntityManager.h" -#include "SkillComponent.h" -#include "BaseCombatAIComponent.h" -#include "DestroyableComponent.h" - -void FvMaelstromDragon::OnStartup(Entity* self) { - self->SetVar<int32_t>(u"weakspot", 0); - - auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - - if (baseCombatAIComponent != nullptr) { - baseCombatAIComponent->SetStunImmune(true); - } -} - -void FvMaelstromDragon::OnDie(Entity* self, Entity* killer) { - if (self->GetVar<bool>(u"bDied")) { - return; - } - - self->SetVar<bool>(u"bDied", true); - - auto position = self->GetPosition(); - auto rotation = self->GetRotation(); - - auto chestObject = 11229; - - EntityInfo info{}; - info.lot = chestObject; - info.pos = position; - info.rot = rotation; - info.spawnerID = self->GetObjectID(); - - auto* chest = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(chest); - - auto golemId = self->GetVar<LWOOBJID>(u"Golem"); - - auto* golem = EntityManager::Instance()->GetEntity(golemId); - - if (golem != nullptr) { - golem->Smash(self->GetObjectID()); - } -} - -void FvMaelstromDragon::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - GameMessages::SendPlayFXEffect(self, -1, u"gothit", "", LWOOBJID_EMPTY, 1, 1, true); - - if (true) { - auto weakpoint = self->GetVar<int32_t>(u"weakspot"); - - if (weakpoint == 1) { - self->Smash(attacker->GetObjectID()); - } - } - - auto* destroyableComponent = self->GetComponent<DestroyableComponent>(); - - if (destroyableComponent != nullptr) { - Game::logger->Log("FvMaelstromDragon", "Hit %i", destroyableComponent->GetArmor()); - - if (destroyableComponent->GetArmor() > 0) return; - - auto weakpoint = self->GetVar<int32_t>(u"weakpoint"); - - if (weakpoint == 0) { - Game::logger->Log("FvMaelstromDragon", "Activating weakpoint"); - - self->AddTimer("ReviveTimer", 12); - - auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (baseCombatAIComponent != nullptr) { - baseCombatAIComponent->SetDisabled(true); - baseCombatAIComponent->SetStunned(true); - } - - if (skillComponent != nullptr) { - skillComponent->Interrupt(); - } - - self->SetVar<int32_t>(u"weakpoint", 2); - - GameMessages::SendPlayAnimation(self, u"stunstart", 1.7f); - - self->AddTimer("timeToStunLoop", 1); - - auto position = self->GetPosition(); - auto forward = self->GetRotation().GetForwardVector(); - auto backwards = forward * -1; - - forward.x *= 10; - forward.z *= 10; - - auto rotation = self->GetRotation(); - - auto objectPosition = NiPoint3(); - - objectPosition.y = position.y; - objectPosition.x = position.x - (backwards.x * 8); - objectPosition.z = position.z - (backwards.z * 8); - - auto golem = self->GetVar<int32_t>(u"DragonSmashingGolem"); - - EntityInfo info{}; - info.lot = golem != 0 ? golem : 8340; - info.pos = objectPosition; - info.rot = rotation; - info.spawnerID = self->GetObjectID(); - info.settings = { - new LDFData<std::string>(u"rebuild_activators", - std::to_string(objectPosition.x + forward.x) + "\x1f" + - std::to_string(objectPosition.y) + "\x1f" + - std::to_string(objectPosition.z + forward.z) - ), - new LDFData<int32_t>(u"respawn", 100000), - new LDFData<float>(u"rebuild_reset_time", 15), - new LDFData<bool>(u"no_timed_spawn", true), - new LDFData<LWOOBJID>(u"Dragon", self->GetObjectID()) - }; - - auto* golemObject = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(golemObject); - } - } -} - -void FvMaelstromDragon::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "ReviveHeldTimer") { - self->AddTimer("backToAttack", 2.5); - } else if (timerName == "ExposeWeakSpotTimer") { - self->SetVar<int32_t>(u"weakspot", 1); - } else if (timerName == "timeToStunLoop") { - GameMessages::SendPlayAnimation(self, u"stunloop", 1.8f); - } else if (timerName == "ReviveTimer") { - GameMessages::SendPlayAnimation(self, u"stunend", 2.0f); - self->AddTimer("backToAttack", 1); - } else if (timerName == "backToAttack") { - auto* baseCombatAIComponent = self->GetComponent<BaseCombatAIComponent>(); - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (baseCombatAIComponent != nullptr) { - baseCombatAIComponent->SetDisabled(false); - baseCombatAIComponent->SetStunned(false); - } - - if (skillComponent != nullptr) { - skillComponent->Interrupt(); - } - - self->SetVar<int32_t>(u"weakspot", -1); - - GameMessages::SendNotifyObject(self->GetObjectID(), self->GetObjectID(), u"DragonRevive", UNASSIGNED_SYSTEM_ADDRESS); - } -} - -void -FvMaelstromDragon::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args != "rebuildDone") return; - - self->AddTimer("ExposeWeakSpotTimer", 3.8f); - - self->CancelTimer("ReviveTimer"); - - self->AddTimer("ReviveHeldTimer", 10.5f); - - self->SetVar<LWOOBJID>(u"Golem", sender->GetObjectID()); - - GameMessages::SendPlayAnimation(self, u"quickbuildhold", 1.9f); -} diff --git a/dScripts/FvNinjaGuard.cpp b/dScripts/FvNinjaGuard.cpp deleted file mode 100644 index 6a841ccf..00000000 --- a/dScripts/FvNinjaGuard.cpp +++ /dev/null @@ -1,42 +0,0 @@ -#include "FvNinjaGuard.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void FvNinjaGuard::OnStartup(Entity* self) { - if (self->GetLOT() == 7412) { - m_LeftGuard = self->GetObjectID(); - } else if (self->GetLOT() == 11128) { - m_RightGuard = self->GetObjectID(); - } -} - -void FvNinjaGuard::OnEmoteReceived(Entity* self, const int32_t emote, Entity* target) { - if (emote != 392) { - GameMessages::SendPlayAnimation(self, u"no"); - - return; - } - - GameMessages::SendPlayAnimation(self, u"scared"); - - auto* missionComponent = target->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr && missionComponent->HasMission(737)) { - missionComponent->ForceProgressTaskType(737, 5, 1, false); - } - - if (self->GetLOT() == 7412) { - auto* rightGuard = EntityManager::Instance()->GetEntity(m_RightGuard); - - if (rightGuard != nullptr) { - GameMessages::SendPlayAnimation(rightGuard, u"laugh_rt"); - } - } else if (self->GetLOT() == 11128) { - auto* leftGuard = EntityManager::Instance()->GetEntity(m_LeftGuard); - - if (leftGuard != nullptr) { - GameMessages::SendPlayAnimation(leftGuard, u"laugh_lt"); - } - } -} - diff --git a/dScripts/FvPandaServer.cpp b/dScripts/FvPandaServer.cpp deleted file mode 100644 index bea93a6c..00000000 --- a/dScripts/FvPandaServer.cpp +++ /dev/null @@ -1,35 +0,0 @@ -#include "FvPandaServer.h" -#include "PetComponent.h" -#include "Character.h" - -void FvPandaServer::OnStartup(Entity* self) { - const auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent != nullptr && petComponent->GetOwner() == nullptr) { - self->SetNetworkVar<std::string>(u"pandatamer", std::to_string(self->GetVar<LWOOBJID>(u"tamer"))); - self->AddTimer("killSelf", 45); - } -} - -void FvPandaServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) { - if (type == NOTIFY_TYPE_BEGIN) { - self->CancelAllTimers(); - } else if (type == NOTIFY_TYPE_QUIT || type == NOTIFY_TYPE_FAILED) { - self->Smash(); - } else if (type == NOTIFY_TYPE_SUCCESS) { - // TODO: Remove from groups - - auto* character = tamer->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(82, true); - } - } -} - -void FvPandaServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "killSelf") { - const auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent != nullptr && petComponent->GetOwner() == nullptr) { - self->Smash(self->GetObjectID(), SILENT); - } - } -} diff --git a/dScripts/FvPandaServer.h b/dScripts/FvPandaServer.h deleted file mode 100644 index 4948fdf4..00000000 --- a/dScripts/FvPandaServer.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class FvPandaServer : public CppScripts::Script { - void OnStartup(Entity* self) override; - void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) override; - void OnTimerDone(Entity* self, std::string timerName) override; -}; diff --git a/dScripts/FvPandaSpawnerServer.cpp b/dScripts/FvPandaSpawnerServer.cpp deleted file mode 100644 index e1f3fb96..00000000 --- a/dScripts/FvPandaSpawnerServer.cpp +++ /dev/null @@ -1,48 +0,0 @@ -#include "FvPandaSpawnerServer.h" -#include "Character.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "ScriptedActivityComponent.h" - -void FvPandaSpawnerServer::OnCollisionPhantom(Entity* self, Entity* target) { - auto* character = target->GetCharacter(); - if (character != nullptr && character->GetPlayerFlag(81)) { - - auto raceObjects = EntityManager::Instance()->GetEntitiesInGroup("PandaRaceObject"); - if (raceObjects.empty()) - return; - - // Check if the player is currently in a footrace - auto* scriptedActivityComponent = raceObjects.at(0)->GetComponent<ScriptedActivityComponent>(); - if (scriptedActivityComponent == nullptr || !scriptedActivityComponent->IsPlayedBy(target)) - return; - - // If the player already spawned a panda - auto playerPandas = EntityManager::Instance()->GetEntitiesInGroup("panda" + std::to_string(target->GetObjectID())); - if (!playerPandas.empty()) { - GameMessages::SendFireEventClientSide(self->GetObjectID(), target->GetSystemAddress(), u"playerPanda", - target->GetObjectID(), 0, 0, target->GetObjectID()); - return; - } - - // If there's already too many spawned pandas - auto pandas = EntityManager::Instance()->GetEntitiesInGroup("pandas"); - if (pandas.size() > 4) { - GameMessages::SendFireEventClientSide(self->GetObjectID(), target->GetSystemAddress(), u"tooManyPandas", - target->GetObjectID(), 0, 0, target->GetObjectID()); - return; - } - - EntityInfo info{}; - info.spawnerID = target->GetObjectID(); - info.pos = self->GetPosition(); - info.lot = 5643; - info.settings = { - new LDFData<LWOOBJID>(u"tamer", target->GetObjectID()), - new LDFData<std::u16string>(u"groupID", u"panda" + (GeneralUtils::to_u16string(target->GetObjectID())) + u";pandas") - }; - - auto* panda = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(panda); - } -} diff --git a/dScripts/FvRaceSmashEggImagineServer.cpp b/dScripts/FvRaceSmashEggImagineServer.cpp deleted file mode 100644 index 32dbf619..00000000 --- a/dScripts/FvRaceSmashEggImagineServer.cpp +++ /dev/null @@ -1,38 +0,0 @@ -#include "FvRaceSmashEggImagineServer.h" -#include "CharacterComponent.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "PossessableComponent.h" -#include "RacingTaskParam.h" -#include "MissionComponent.h" - -void FvRaceSmashEggImagineServer::OnDie(Entity* self, Entity* killer) { - if (killer != nullptr) { - auto* destroyableComponent = killer->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->SetImagination(destroyableComponent->GetImagination() + 10); - EntityManager::Instance()->SerializeEntity(killer); - } - - // get possessor to progress statistics and tasks. - auto* possessableComponent = killer->GetComponent<PossessableComponent>(); - if (possessableComponent != nullptr) { - - auto* possessor = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); - if (possessor != nullptr) { - - auto* missionComponent = possessor->GetComponent<MissionComponent>(); - auto* characterComponent = possessor->GetComponent<CharacterComponent>(); - if (characterComponent != nullptr) { - characterComponent->UpdatePlayerStatistic(ImaginationPowerUpsCollected); - characterComponent->UpdatePlayerStatistic(RacingSmashablesSmashed); - } - if (missionComponent == nullptr) return; - // Dragon eggs have their own smash server so we handle mission progression for them here. - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, 0, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_SMASHABLES); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, self->GetLOT(), (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_SMASH_SPECIFIC_SMASHABLE); - } - } - - } -} diff --git a/dScripts/GfApeSmashingQB.cpp b/dScripts/GfApeSmashingQB.cpp deleted file mode 100644 index 5bba6450..00000000 --- a/dScripts/GfApeSmashingQB.cpp +++ /dev/null @@ -1,22 +0,0 @@ -#include "GfApeSmashingQB.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void GfApeSmashingQB::OnStartup(Entity* self) { - self->SetNetworkVar<LWOOBJID>(u"lootTagOwner", self->GetVar<LWOOBJID>(u"lootTagOwner")); -} - -void GfApeSmashingQB::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "anchorBreakTime") { - self->Smash(); - } -} - -void GfApeSmashingQB::OnRebuildComplete(Entity* self, Entity* target) { - auto* ape = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"ape")); - if (ape != nullptr) { - ape->OnFireEventServerSide(target, "rebuildDone"); - GameMessages::SendPlayAnimation(self, u"smash", 1.7f); - self->AddTimer("anchorBreakTime", 1.0f); - } -} diff --git a/dScripts/GfBanana.cpp b/dScripts/GfBanana.cpp deleted file mode 100644 index 3a71eded..00000000 --- a/dScripts/GfBanana.cpp +++ /dev/null @@ -1,93 +0,0 @@ -#include "GfBanana.h" - -#include "Entity.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" - -void GfBanana::SpawnBanana(Entity* self) { - auto position = self->GetPosition(); - const auto rotation = self->GetRotation(); - - position.y += 12; - position.x -= rotation.GetRightVector().x * 5; - position.z -= rotation.GetRightVector().z * 5; - - EntityInfo info{}; - - info.pos = position; - info.rot = rotation; - info.lot = 6909; - info.spawnerID = self->GetObjectID(); - - auto* entity = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(entity); - - self->SetVar(u"banana", entity->GetObjectID()); - - entity->AddDieCallback([self]() { - self->SetVar(u"banana", LWOOBJID_EMPTY); - - self->AddTimer("bananaTimer", 30); - }); -} - -void GfBanana::OnStartup(Entity* self) { - SpawnBanana(self); -} - -void GfBanana::OnHit(Entity* self, Entity* attacker) { - auto* destroyable = self->GetComponent<DestroyableComponent>(); - - destroyable->SetHealth(9999); - - const auto bananaId = self->GetVar<LWOOBJID>(u"banana"); - - if (bananaId == LWOOBJID_EMPTY) return; - - auto* bananaEntity = EntityManager::Instance()->GetEntity(bananaId); - - if (bananaEntity == nullptr) { - self->SetVar(u"banana", LWOOBJID_EMPTY); - - self->AddTimer("bananaTimer", 30); - - return; - } - - bananaEntity->SetPosition(bananaEntity->GetPosition() - NiPoint3::UNIT_Y * 8); - - auto* bananaDestroyable = bananaEntity->GetComponent<DestroyableComponent>(); - - bananaDestroyable->SetHealth(0); - - bananaDestroyable->Smash(attacker->GetObjectID()); - - /* - auto position = self->GetPosition(); - const auto rotation = self->GetRotation(); - - position.y += 12; - position.x -= rotation.GetRightVector().x * 5; - position.z -= rotation.GetRightVector().z * 5; - - EntityInfo info {}; - - info.pos = position; - info.rot = rotation; - info.lot = 6718; - info.spawnerID = self->GetObjectID(); - - auto* entity = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(entity, UNASSIGNED_SYSTEM_ADDRESS); - */ - - EntityManager::Instance()->SerializeEntity(self); -} - -void GfBanana::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "bananaTimer") { - SpawnBanana(self); - } -} diff --git a/dScripts/GfCampfire.cpp b/dScripts/GfCampfire.cpp deleted file mode 100644 index 95e29a9a..00000000 --- a/dScripts/GfCampfire.cpp +++ /dev/null @@ -1,100 +0,0 @@ -#include "GfCampfire.h" -#include "RenderComponent.h" -#include "SkillComponent.h" -#include "MissionComponent.h" -#include "RenderComponent.h" -#include "EntityManager.h" - -void GfCampfire::OnStartup(Entity* self) { - self->SetI32(u"counter", static_cast<int32_t>(0)); - self->SetProximityRadius(2.0f, "placeholder"); - self->SetBoolean(u"isBurning", true); - - auto* render = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - if (render == nullptr) - return; - - render->PlayEffect(295, u"running", "Burn"); -} - -void GfCampfire::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args == "physicsReady") { - auto* render = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - - render->PlayEffect(295, u"running", "Burn"); - } -} - -void GfCampfire::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - auto* skill = self->GetComponent<SkillComponent>(); - - if (self->GetBoolean(u"isBurning")) { - if (status == "ENTER") { - if (entering->GetCharacter()) { - int32_t counter = self->GetI32(u"counter"); - counter = counter + 1; - self->SetI32(u"counter", counter); - - if (counter == 1) { - skill->CalculateBehavior(m_skillCastId, 115, entering->GetObjectID()); - self->AddTimer("TimeBetweenCast", FIRE_COOLDOWN); - - //self->SetVar<LWOOBJID>("target", entering->GetObjectID()); - - auto* missionComponet = entering->GetComponent<MissionComponent>(); - - if (missionComponet != nullptr) { - missionComponet->ForceProgress(440, 658, 1); - } - } - } - } else { - int32_t counter = self->GetI32(u"counter"); - if (counter > 0) { - counter = counter - 1; - self->SetI32(u"counter", counter); - if (counter == 0) { - self->CancelAllTimers(); - } - } - } - } -} - -void GfCampfire::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (message == "waterspray" && self->GetVar<bool>(u"isBurning")) { - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - renderComponent->StopEffect("Burn"); - renderComponent->PlayEffect(295, u"idle", "Off"); - - self->SetVar<bool>(u"isBurning", false); - self->AddTimer("FireRestart", 37); - } - } -} - -void GfCampfire::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "TimeBetweenCast") { - /* - self->AddTimer("TimeBetweenCast", FIRE_COOLDOWN); - - const auto targetId = self->GetVar<LWOOBJID>("target"); - - auto* entering = EntityManager::Instance()->GetEntity(targetId); - - if (entering == nullptr) - { - - } - */ - } else if (timerName == "FireRestart" && !self->GetVar<bool>(u"isBurning")) { - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - renderComponent->StopEffect("Off"); - renderComponent->PlayEffect(295, u"running", "Burn"); - self->SetVar<bool>(u"isBurning", true); - } - } -} diff --git a/dScripts/GfCaptainsCannon.cpp b/dScripts/GfCaptainsCannon.cpp deleted file mode 100644 index 7d67c781..00000000 --- a/dScripts/GfCaptainsCannon.cpp +++ /dev/null @@ -1,83 +0,0 @@ -#include "GfCaptainsCannon.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void GfCaptainsCannon::OnUse(Entity* self, Entity* user) { - if (self->GetVar<bool>(u"bIsInUse")) { - return; - } - - self->SetVar<LWOOBJID>(u"userID", user->GetObjectID()); - - self->SetVar<bool>(u"bIsInUse", true); - self->SetNetworkVar<bool>(u"bIsInUse", true); - - GameMessages::SendSetStunned(user->GetObjectID(), PUSH, user->GetSystemAddress(), - LWOOBJID_EMPTY, true, true, true, true, true, true, true, true - ); - - auto position = self->GetPosition(); - auto forward = self->GetRotation().GetForwardVector(); - - position.x += forward.x * -3; - position.z += forward.z * -3; - - auto rotation = self->GetRotation(); - - GameMessages::SendTeleport(user->GetObjectID(), position, rotation, user->GetSystemAddress()); - - GameMessages::SendPlayAnimation(user, u"cannon-strike-no-equip"); - - GameMessages::SendPlayFXEffect(user->GetObjectID(), 6039, u"hook", "hook", LWOOBJID_EMPTY, 1, 1, true); - - self->AddTimer("FireCannon", 1.667f); -} - -void GfCaptainsCannon::OnTimerDone(Entity* self, std::string timerName) { - const auto playerId = self->GetVar<LWOOBJID>(u"userID"); - - auto* player = EntityManager::Instance()->GetEntity(playerId); - - if (player == nullptr) { - self->SetVar<bool>(u"bIsInUse", false); - self->SetNetworkVar<bool>(u"bIsInUse", false); - - return; - } - - if (timerName == "FireCannon") { - float cinematicTime = 6.3f; - - GameMessages::SendPlayCinematic(playerId, u"Cannon_Cam", player->GetSystemAddress()); - - self->AddTimer("cinematicTimer", cinematicTime); - - const auto sharkObjects = EntityManager::Instance()->GetEntitiesInGroup("SharkCannon"); - - for (auto* shark : sharkObjects) { - if (shark->GetLOT() != m_SharkItemID) continue; - - GameMessages::SendPlayAnimation(shark, u"cannon"); - } - - GameMessages::SendPlay2DAmbientSound(player, "{7457d85c-4537-4317-ac9d-2f549219ea87}"); - } else if (timerName == "cinematicTimer") { - GameMessages::SendSetStunned(playerId, POP, player->GetSystemAddress(), - LWOOBJID_EMPTY, true, true, true, true, true, true, true, true - ); - - self->SetVar<bool>(u"bIsInUse", false); - self->SetNetworkVar<bool>(u"bIsInUse", false); - - GameMessages::SendStopFXEffect(player, true, "hook"); - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->ForceProgress(601, 910, 1); - } - - GameMessages::SendTerminateInteraction(playerId, FROM_INTERACTION, self->GetObjectID()); - } -} diff --git a/dScripts/GfJailWalls.cpp b/dScripts/GfJailWalls.cpp deleted file mode 100644 index 56710832..00000000 --- a/dScripts/GfJailWalls.cpp +++ /dev/null @@ -1,29 +0,0 @@ -#include "GfJailWalls.h" -#include "dZoneManager.h" -#include "GeneralUtils.h" - -void GfJailWalls::OnRebuildComplete(Entity* self, Entity* target) { - const auto wall = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"Wall")); - - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName("Jail0" + wall)) { - spawner->Deactivate(); - } - - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName("JailCaptain0" + wall)) { - spawner->Deactivate(); - } -} - -void GfJailWalls::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state != eRebuildState::REBUILD_RESETTING) return; - - const auto wall = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"Wall")); - - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName("Jail0" + wall)) { - spawner->Activate(); - } - - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName("JailCaptain0" + wall)) { - spawner->Activate(); - } -} diff --git a/dScripts/GfJailkeepMission.cpp b/dScripts/GfJailkeepMission.cpp deleted file mode 100644 index eaa8c73d..00000000 --- a/dScripts/GfJailkeepMission.cpp +++ /dev/null @@ -1,38 +0,0 @@ -#include "GfJailkeepMission.h" -#include "MissionComponent.h" -#include "Character.h" - -void GfJailkeepMission::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) - return; - - if (missionID == 385 && missionState == MissionState::MISSION_STATE_AVAILABLE) { - missionComponent->AcceptMission(386, true); - missionComponent->AcceptMission(387, true); - missionComponent->AcceptMission(388, true); - missionComponent->AcceptMission(390, true); - } else if (missionID == 385 && missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { - auto* character = target->GetCharacter(); - if (character != nullptr && character->GetPlayerFlag(68)) { - missionComponent->AcceptMission(701); - missionComponent->AcceptMission(702); - missionComponent->AcceptMission(703); - missionComponent->AcceptMission(704); - } - } -} - -void GfJailkeepMission::OnUse(Entity* self, Entity* user) { - auto* missionComponent = user->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) - return; - - if (missionComponent->GetMissionState(385) == MissionState::MISSION_STATE_ACTIVE) { - missionComponent->AcceptMission(386, true); - missionComponent->AcceptMission(387, true); - missionComponent->AcceptMission(388, true); - missionComponent->AcceptMission(390, true); - } -} - diff --git a/dScripts/GfJailkeepMission.h b/dScripts/GfJailkeepMission.h deleted file mode 100644 index a120d29b..00000000 --- a/dScripts/GfJailkeepMission.h +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class GfJailkeepMission final : public CppScripts::Script -{ -public: - void OnUse(Entity* self, Entity* user) override; - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -}; diff --git a/dScripts/GfOrgan.cpp b/dScripts/GfOrgan.cpp deleted file mode 100644 index 372ed3a2..00000000 --- a/dScripts/GfOrgan.cpp +++ /dev/null @@ -1,21 +0,0 @@ -#include "GfOrgan.h" -#include "GameMessages.h" - -void GfOrgan::OnUse(Entity* self, Entity* user) { - if (self->GetBoolean(u"bIsInUse")) { - m_canUse = false; - return; - } - - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, "{15d5f8bd-139a-4c31-8904-970c480cd70f}"); - self->SetBoolean(u"bIsInUse", true); - self->AddTimer("reset", 5.0f); - - GameMessages::SendPlayAnimation(user, u"jig"); -} - -void GfOrgan::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "reset") { - self->SetBoolean(u"bIsInUse", false); - } -} diff --git a/dScripts/GfTikiTorch.cpp b/dScripts/GfTikiTorch.cpp deleted file mode 100644 index 3a06b054..00000000 --- a/dScripts/GfTikiTorch.cpp +++ /dev/null @@ -1,75 +0,0 @@ -#include "GfTikiTorch.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "MissionComponent.h" -#include "RenderComponent.h" - -void GfTikiTorch::OnStartup(Entity* self) { - LightTorch(self); -} - -void GfTikiTorch::OnUse(Entity* self, Entity* killer) { - if (self->GetBoolean(u"isInUse")) { - self->SetBoolean(u"isInUse", false); - return; - } - - GameMessages::SendPlayAnimation(self, u"interact"); - self->SetI64(u"userID", killer->GetObjectID()); - - for (int i = 0; i < m_numspawn; i++) { - GameMessages::SendDropClientLoot(killer, self->GetObjectID(), 935, 0, self->GetPosition()); - } - - self->AddTimer("InteractionCooldown", 4); -} - -void GfTikiTorch::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "Relight") { - LightTorch(self); - } else if (timerName == "InteractionCooldown") { - Entity* player = EntityManager::Instance()->GetEntity(self->GetI64(u"userID")); - - if (player != nullptr && player->GetCharacter()) { - GameMessages::SendTerminateInteraction(player->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); - } - - self->SetBoolean(u"isInUse", false); - - self->SetI64(u"userID", 0); - } -} - -void GfTikiTorch::LightTorch(Entity* self) { - auto* renderComponent = static_cast<RenderComponent*>(self->GetComponent(COMPONENT_TYPE_RENDER)); - if (renderComponent == nullptr) - return; - - self->SetBoolean(u"isInUse", false); - - renderComponent->PlayEffect(611, u"fire", "tikitorch"); - self->SetBoolean(u"isBurning", true); -} - -void GfTikiTorch::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (self->GetBoolean(u"isBurning") && message == "waterspray") { - GameMessages::SendPlayAnimation(self, u"water"); - - auto* renderComponent = self->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - renderComponent->StopEffect("tikitorch"); - renderComponent->PlayEffect(611, u"water", "water"); - renderComponent->PlayEffect(611, u"steam", "steam"); - } - - auto* casterMissionComponent = caster->GetComponent<MissionComponent>(); - if (casterMissionComponent != nullptr) { - for (const auto missionID : m_missions) { - casterMissionComponent->ForceProgressTaskType(missionID, static_cast<uint32_t>(MissionTaskType::MISSION_TASK_TYPE_SCRIPT), 1); - } - } - - self->AddTimer("Relight", 7.0f); - self->SetBoolean(u"isBurning", false); - } -} diff --git a/dScripts/GrowingFlower.cpp b/dScripts/GrowingFlower.cpp deleted file mode 100644 index 7b495841..00000000 --- a/dScripts/GrowingFlower.cpp +++ /dev/null @@ -1,35 +0,0 @@ -#include "GrowingFlower.h" -#include "MissionComponent.h" - -void GrowingFlower::OnSkillEventFired(Entity* self, Entity* target, const std::string& message) { - if (!self->GetVar<bool>(u"blooming") && (message == "waterspray" || message == "shovelgrow")) { - self->SetVar<bool>(u"blooming", true); - self->SetNetworkVar(u"blooming", true); - self->AddTimer("FlowerDie", GrowingFlower::aliveTime); - - const auto mission1 = self->GetVar<int32_t>(u"missionID"); - const auto mission2 = self->GetVar<int32_t>(u"missionID2"); - - LootGenerator::Instance().DropActivityLoot(target, self, self->GetLOT(), 0); - - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - for (const auto mission : achievementIDs) - missionComponent->ForceProgressTaskType(mission, static_cast<uint32_t>(MissionTaskType::MISSION_TASK_TYPE_SCRIPT), 1); - - if (mission1 && missionComponent->GetMissionState(mission1) == MissionState::MISSION_STATE_ACTIVE) - missionComponent->ForceProgressTaskType(mission1, static_cast<uint32_t>(MissionTaskType::MISSION_TASK_TYPE_SCRIPT), 1); - - if (mission2 && missionComponent->GetMissionState(mission2) == MissionState::MISSION_STATE_ACTIVE) - missionComponent->ForceProgressTaskType(mission2, static_cast<uint32_t>(MissionTaskType::MISSION_TASK_TYPE_SCRIPT), 1); - } - } -} - -void GrowingFlower::OnTimerDone(Entity* self, std::string message) { - if (message == "FlowerDie") { - self->Smash(); - } -} - -const std::vector<uint32_t> GrowingFlower::achievementIDs = { 143, 152, 153, 1409, 1507, 1544, 1581, 1845 }; diff --git a/dScripts/HydrantBroken.cpp b/dScripts/HydrantBroken.cpp deleted file mode 100644 index 1409c9fb..00000000 --- a/dScripts/HydrantBroken.cpp +++ /dev/null @@ -1,37 +0,0 @@ -#include "HydrantBroken.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void HydrantBroken::OnStartup(Entity* self) { - self->AddTimer("playEffect", 1); - - const auto hydrant = "hydrant" + self->GetVar<std::string>(u"hydrant"); - - const auto bouncers = EntityManager::Instance()->GetEntitiesInGroup(hydrant); - - for (auto* bouncer : bouncers) { - self->SetVar<LWOOBJID>(u"bouncer", bouncer->GetObjectID()); - - GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); - - GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"enableCollision", UNASSIGNED_SYSTEM_ADDRESS); - } - - self->AddTimer("KillBroken", 25); -} - -void HydrantBroken::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "KillBroken") { - auto* bouncer = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"bouncer")); - - if (bouncer != nullptr) { - GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); - - GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"disableCollision", UNASSIGNED_SYSTEM_ADDRESS); - } - - self->Kill(); - } else if (timerName == "playEffect") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 384, u"water", "water", LWOOBJID_EMPTY, 1, 1, true); - } -} diff --git a/dScripts/HydrantSmashable.cpp b/dScripts/HydrantSmashable.cpp deleted file mode 100644 index 20baa58b..00000000 --- a/dScripts/HydrantSmashable.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "HydrantSmashable.h" -#include "EntityManager.h" -#include "GeneralUtils.h" - -void HydrantSmashable::OnDie(Entity* self, Entity* killer) { - const auto hydrantName = self->GetVar<std::u16string>(u"hydrant"); - - LDFBaseData* data = new LDFData<std::string>(u"hydrant", GeneralUtils::UTF16ToWTF8(hydrantName)); - - EntityInfo info{}; - info.lot = HYDRANT_BROKEN; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.settings = { data }; - info.spawnerID = self->GetSpawnerID(); - - auto* hydrant = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(hydrant); -} diff --git a/dScripts/ImaginationBackpackHealServer.cpp b/dScripts/ImaginationBackpackHealServer.cpp deleted file mode 100644 index 1e0d35cf..00000000 --- a/dScripts/ImaginationBackpackHealServer.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "ImaginationBackpackHealServer.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void ImaginationBackpackHealServer::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (message == "CastImaginationBackpack") { - auto healMission = self->GetVar<int32_t>(u"FXOffMis"); - if (healMission == 0) - healMission = self->GetVar<int32_t>(u"FXOnMis"); - if (healMission == 0) - return; - - auto* missionComponent = caster->GetComponent<MissionComponent>(); - if (missionComponent != nullptr && missionComponent->GetMissionState(healMission) == MissionState::MISSION_STATE_ACTIVE) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"ClearMaelstrom", 0, 0, - caster->GetObjectID(), "", caster->GetSystemAddress()); - } - } -} diff --git a/dScripts/ImaginationShrineServer.cpp b/dScripts/ImaginationShrineServer.cpp deleted file mode 100644 index 267d0480..00000000 --- a/dScripts/ImaginationShrineServer.cpp +++ /dev/null @@ -1,16 +0,0 @@ -#include "ImaginationShrineServer.h" -#include "RebuildComponent.h" - -void ImaginationShrineServer::OnUse(Entity* self, Entity* user) { - // If the rebuild component is complete, use the shrine - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent == nullptr) { - return; - } - - if (rebuildComponent->GetState() == REBUILD_COMPLETED) { - // Use the shrine - BaseUse(self, user); - } -} diff --git a/dScripts/ImgBrickConsoleQB.cpp b/dScripts/ImgBrickConsoleQB.cpp deleted file mode 100644 index 4a12324f..00000000 --- a/dScripts/ImgBrickConsoleQB.cpp +++ /dev/null @@ -1,240 +0,0 @@ -#include "ImgBrickConsoleQB.h" -#include "RebuildComponent.h" -#include "dZoneManager.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "MissionComponent.h" -#include "InventoryComponent.h" - -int32_t ImgBrickConsoleQB::ResetBricks = 30; -int32_t ImgBrickConsoleQB::ResetConsole = 60; -int32_t ImgBrickConsoleQB::ResetInteract = 45; - -void ImgBrickConsoleQB::OnStartup(Entity* self) { - self->SetNetworkVar(u"used", false); - - self->AddTimer("reset", ResetBricks); -} - -void ImgBrickConsoleQB::OnUse(Entity* self, Entity* user) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent->GetState() == REBUILD_COMPLETED) { - if (!self->GetNetworkVar<bool>(u"used")) { - const auto consoles = EntityManager::Instance()->GetEntitiesInGroup("Console"); - - auto bothBuilt = false; - - for (auto* console : consoles) { - auto* consoleRebuildComponent = console->GetComponent<RebuildComponent>(); - - if (consoleRebuildComponent->GetState() != REBUILD_COMPLETED) { - continue; - } - - console->CancelAllTimers(); - - if (console->GetNetworkVar<bool>(u"used")) { - bothBuilt = true; - } - } - - if (bothBuilt) { - SmashCanister(self); - } else { - SpawnBrick(self); - } - - self->AddTimer("Die", ResetInteract); - - auto onFX = 0; - - const auto location = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"console")); - - if (location == "Left") { - onFX = 2776; - } else { - onFX = 2779; - } - - const auto& facility = EntityManager::Instance()->GetEntitiesInGroup("FacilityPipes"); - - if (!facility.empty()) { - GameMessages::SendStopFXEffect(facility[0], true, location + "PipeEnergy"); - GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), onFX, u"create", location + "PipeOn"); - } - } - - auto* player = user; - - auto* missionComponent = player->GetComponent<MissionComponent>(); - auto* inventoryComponent = player->GetComponent<InventoryComponent>(); - - if (missionComponent != nullptr && inventoryComponent != nullptr) { - if (missionComponent->GetMissionState(1302) == MissionState::MISSION_STATE_ACTIVE) { - inventoryComponent->RemoveItem(13074, 1); - - missionComponent->ForceProgressTaskType(1302, 1, 1); - } - - if (missionComponent->GetMissionState(1926) == MissionState::MISSION_STATE_ACTIVE) { - inventoryComponent->RemoveItem(14472, 1); - - missionComponent->ForceProgressTaskType(1926, 1, 1); - } - } - - self->SetNetworkVar(u"used", true); - - GameMessages::SendTerminateInteraction(player->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); - } -} - -void ImgBrickConsoleQB::SpawnBrick(Entity* self) { - const auto netDevil = dZoneManager::Instance()->GetSpawnersByName("MaelstromBug"); - if (!netDevil.empty()) { - netDevil[0]->Reset(); - netDevil[0]->Deactivate(); - } - - const auto brick = dZoneManager::Instance()->GetSpawnersByName("Imagination"); - if (!brick.empty()) { - brick[0]->Activate(); - } -} - -void ImgBrickConsoleQB::SmashCanister(Entity* self) { - const auto brick = EntityManager::Instance()->GetEntitiesInGroup("Imagination"); - if (!brick.empty()) { - GameMessages::SendPlayFXEffect(brick[0]->GetObjectID(), 122, u"create", "bluebrick"); - GameMessages::SendPlayFXEffect(brick[0]->GetObjectID(), 1034, u"cast", "imaginationexplosion"); - } - - const auto canisters = EntityManager::Instance()->GetEntitiesInGroup("Canister"); - for (auto* canister : canisters) { - canister->Smash(canister->GetObjectID(), VIOLENT); - } - - const auto canister = dZoneManager::Instance()->GetSpawnersByName("BrickCanister"); - if (!canister.empty()) { - canister[0]->Reset(); - canister[0]->Deactivate(); - } -} - -void ImgBrickConsoleQB::OnRebuildComplete(Entity* self, Entity* target) { - auto energyFX = 0; - - const auto location = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"console")); - - if (location == "Left") { - energyFX = 2775; - } else { - energyFX = 2778; - } - - const auto& facility = EntityManager::Instance()->GetEntitiesInGroup("FacilityPipes"); - - if (!facility.empty()) { - GameMessages::SendStopFXEffect(facility[0], true, location + "PipeOff"); - GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), energyFX, u"create", location + "PipeEnergy"); - } - - const auto consoles = EntityManager::Instance()->GetEntitiesInGroup("Console"); - - for (auto* console : consoles) { - auto* consoleRebuildComponent = console->GetComponent<RebuildComponent>(); - - if (consoleRebuildComponent->GetState() != REBUILD_COMPLETED) { - continue; - } - - console->CancelAllTimers(); - } - - self->AddTimer("Die", ResetConsole); -} - -void ImgBrickConsoleQB::OnDie(Entity* self, Entity* killer) { - if (self->GetVar<bool>(u"Died")) { - return; - } - - self->CancelAllTimers(); - - self->SetVar(u"Died", true); - - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent->GetState() == REBUILD_COMPLETED) { - auto offFX = 0; - - const auto location = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"console")); - - if (location == "Left") { - offFX = 2774; - } else { - offFX = 2777; - } - - const auto& facility = EntityManager::Instance()->GetEntitiesInGroup("FacilityPipes"); - - if (!facility.empty()) { - GameMessages::SendStopFXEffect(facility[0], true, location + "PipeEnergy"); - GameMessages::SendStopFXEffect(facility[0], true, location + "PipeOn"); - GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), offFX, u"create", location + "PipeOff"); - GameMessages::SendPlayFXEffect(facility[0]->GetObjectID(), 2750, u"create", location + "imagination_canister"); - } - } - - const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); - - const auto pipeGroup = myGroup.substr(0, 10); - - const auto firstPipe = pipeGroup + "1"; - - const auto samePipeSpawner = dZoneManager::Instance()->GetSpawnersByName(myGroup); - if (!samePipeSpawner.empty()) { - samePipeSpawner[0]->Reset(); - samePipeSpawner[0]->Deactivate(); - } - - const auto firstPipeSpawner = dZoneManager::Instance()->GetSpawnersByName(firstPipe); - if (!firstPipeSpawner.empty()) { - firstPipeSpawner[0]->Activate(); - } - - const auto netdevil = dZoneManager::Instance()->GetSpawnersByName("Imagination"); - if (!netdevil.empty()) { - netdevil[0]->Reset(); - netdevil[0]->Deactivate(); - } - - const auto brick = dZoneManager::Instance()->GetSpawnersByName("MaelstromBug"); - if (!brick.empty()) { - brick[0]->Activate(); - } - - const auto canister = dZoneManager::Instance()->GetSpawnersByName("BrickCanister"); - if (!canister.empty()) { - canister[0]->Activate(); - } - - self->SetNetworkVar(u"used", false); -} - -void ImgBrickConsoleQB::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "reset") { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent->GetState() == REBUILD_OPEN) { - self->Smash(self->GetObjectID(), SILENT); - } - } else if (timerName == "Die") { - const auto consoles = EntityManager::Instance()->GetEntitiesInGroup("Console"); - - for (auto* console : consoles) { - console->Smash(console->GetObjectID(), VIOLENT); - } - } -} diff --git a/dScripts/InstanceExitTransferPlayerToLastNonInstance.cpp b/dScripts/InstanceExitTransferPlayerToLastNonInstance.cpp deleted file mode 100644 index 4e42e7fb..00000000 --- a/dScripts/InstanceExitTransferPlayerToLastNonInstance.cpp +++ /dev/null @@ -1,56 +0,0 @@ -#include "InstanceExitTransferPlayerToLastNonInstance.h" -#include "GameMessages.h" -#include "Player.h" -#include "Character.h" -#include "dServer.h" - -void InstanceExitTransferPlayerToLastNonInstance::OnUse(Entity* self, Entity* user) { - auto transferText = self->GetVar<std::u16string>(u"transferText"); - if (transferText.empty()) - transferText = u"DRAGON_EXIT_QUESTION"; - - GameMessages::SendDisplayMessageBox( - user->GetObjectID(), - true, - self->GetObjectID(), - u"Instance_Exit", - 1, - transferText, - u"", - user->GetSystemAddress() - ); -} - -void InstanceExitTransferPlayerToLastNonInstance::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - auto* player = dynamic_cast<Player*>(sender); - if (player == nullptr) - return; - - auto* character = sender->GetCharacter(); - if (character != nullptr) { - if (identifier == u"Instance_Exit" && button == 1) { - auto lastInstance = character->GetLastNonInstanceZoneID(); - - // Sanity check - if (lastInstance == 0) { - switch (Game::server->GetZoneID()) { - case 2001: - lastInstance = 2000; - break; - case 1402: - lastInstance = 1400; - break; - default: - lastInstance = 1100; - break; - } - } - - player->SendToZone(lastInstance); - } - } - - GameMessages::SendTerminateInteraction(sender->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); -} - - diff --git a/dScripts/LegoDieRoll.cpp b/dScripts/LegoDieRoll.cpp deleted file mode 100644 index e86550f9..00000000 --- a/dScripts/LegoDieRoll.cpp +++ /dev/null @@ -1,52 +0,0 @@ -#include "LegoDieRoll.h" -#include "Entity.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void LegoDieRoll::OnStartup(Entity* self) { - self->AddTimer("DoneRolling", 10.0f); - self->AddTimer("ThrowDice", LegoDieRoll::animTime); -} - -void LegoDieRoll::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "DoneRolling") { - self->Smash(self->GetObjectID(), SILENT); - } else if (timerName == "ThrowDice") { - int dieRoll = GeneralUtils::GenerateRandomNumber<int>(1, 6); - - switch (dieRoll) { - case 1: - GameMessages::SendPlayAnimation(self, u"roll-die-1"); - break; - case 2: - GameMessages::SendPlayAnimation(self, u"roll-die-2"); - break; - case 3: - GameMessages::SendPlayAnimation(self, u"roll-die-3"); - break; - case 4: - GameMessages::SendPlayAnimation(self, u"roll-die-4"); - break; - case 5: - GameMessages::SendPlayAnimation(self, u"roll-die-5"); - break; - case 6: - { - GameMessages::SendPlayAnimation(self, u"roll-die-6"); - // tracking the It's Truly Random Achievement - auto* owner = self->GetOwner(); - auto* missionComponent = owner->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - const auto rollMissionState = missionComponent->GetMissionState(756); - if (rollMissionState == MissionState::MISSION_STATE_ACTIVE) { - missionComponent->ForceProgress(756, 1103, 1); - } - } - break; - } - default: - break; - } - } -} diff --git a/dScripts/Lieutenant.cpp b/dScripts/Lieutenant.cpp deleted file mode 100644 index d3b0fc1f..00000000 --- a/dScripts/Lieutenant.cpp +++ /dev/null @@ -1,44 +0,0 @@ -#include "Lieutenant.h" -#include "SkillComponent.h" -#include "dZoneManager.h" - -void Lieutenant::OnStartup(Entity* self) { - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - skillComponent->CalculateBehavior(1127, 24812, self->GetObjectID(), true); -} - -void Lieutenant::OnDie(Entity* self, Entity* killer) { - const auto myLOT = self->GetLOT(); - - std::string spawnerName; - - switch (myLOT) { - case 16047: - spawnerName = "EarthShrine_ERail"; - break; - case 16050: - spawnerName = "IceShrine_QBBouncer"; - break; - case 16049: - spawnerName = "LightningShrine_LRail"; - break; - default: - return; - } - - const auto spawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName); - - if (spawners.empty()) { - return; - } - - for (auto* spawner : spawners) { - spawner->Reset(); - spawner->Activate(); - } -} diff --git a/dScripts/MaestromExtracticatorServer.cpp b/dScripts/MaestromExtracticatorServer.cpp deleted file mode 100644 index caaba28f..00000000 --- a/dScripts/MaestromExtracticatorServer.cpp +++ /dev/null @@ -1,50 +0,0 @@ -#include "MaestromExtracticatorServer.h" -#include "GameMessages.h" -#include "GeneralUtils.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void MaestromExtracticatorServer::OnStartup(Entity* self) { - //self:SetNetworkVar("current_anim", failAnim) - GameMessages::SendPlayAnimation(self, GeneralUtils::ASCIIToUTF16(failAnim)); - - self->AddTimer("PlayFail", defaultTime); - self->AddTimer("RemoveSample", destroyAfterNoSampleTime); -} - -void MaestromExtracticatorServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (sender == nullptr) - return; - - if (args == "attemptCollection") { - Entity* player = EntityManager::Instance()->GetEntity(self->GetSpawnerID()); - if (!player) return; - - auto missionComponent = player->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) return; - - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SMASH, 14718); - CollectSample(self, sender->GetObjectID()); - sender->ScheduleKillAfterUpdate(); - } -} - -void MaestromExtracticatorServer::CollectSample(Entity* self, LWOOBJID sampleObj) { - PlayAnimAndReturnTime(self, collectAnim); - self->AddTimer("RemoveSample", defaultTime); -} - -void MaestromExtracticatorServer::PlayAnimAndReturnTime(Entity* self, std::string animID) { - GameMessages::SendPlayAnimation(self, GeneralUtils::ASCIIToUTF16(animID)); -} - -void MaestromExtracticatorServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "RemoveSample") { - self->ScheduleKillAfterUpdate(); - } - - if (timerName == "PlayFail") { - GameMessages::SendPlayAnimation(self, GeneralUtils::ASCIIToUTF16(failAnim)); - } -} diff --git a/dScripts/MaestromExtracticatorServer.h b/dScripts/MaestromExtracticatorServer.h deleted file mode 100644 index c4adb51e..00000000 --- a/dScripts/MaestromExtracticatorServer.h +++ /dev/null @@ -1,18 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class MaestromExtracticatorServer : public CppScripts::Script { -public: - void OnStartup(Entity* self); - void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3); - void CollectSample(Entity* self, LWOOBJID sampleObj); - void PlayAnimAndReturnTime(Entity* self, std::string animID); - void OnTimerDone(Entity* self, std::string timerName); - -private: - const std::string failAnim = "idle_maelstrom"; - const std::string collectAnim = "collect_maelstrom"; - const float defaultTime = 4.0f; - const float destroyAfterNoSampleTime = 8.0f; -}; diff --git a/dScripts/MailBoxServer.cpp b/dScripts/MailBoxServer.cpp deleted file mode 100644 index c2534f3e..00000000 --- a/dScripts/MailBoxServer.cpp +++ /dev/null @@ -1,19 +0,0 @@ -#include "MailBoxServer.h" -#include "AMFFormat.h" -#include "GameMessages.h" - -void MailBoxServer::OnUse(Entity* self, Entity* user) { - AMFStringValue* value = new AMFStringValue(); - value->SetStringValue("Mail"); - AMFArrayValue args; - args.InsertValue("state", value); - GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", &args); -} - -void MailBoxServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - if (args == "toggleMail") { - AMFArrayValue args; - args.InsertValue("visible", new AMFFalseValue()); - GameMessages::SendUIMessageServerToSingleClient(sender, sender->GetSystemAddress(), "ToggleMail", &args); - } -} diff --git a/dScripts/MastTeleport.cpp b/dScripts/MastTeleport.cpp deleted file mode 100644 index ebde7b20..00000000 --- a/dScripts/MastTeleport.cpp +++ /dev/null @@ -1,88 +0,0 @@ -#include "MastTeleport.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "Preconditions.h" - -#ifdef _WIN32 -#define _USE_MATH_DEFINES -#include <math.h> -#endif - -void MastTeleport::OnStartup(Entity* self) { - self->SetNetworkVar<std::string>(u"hookPreconditions", "154;44", UNASSIGNED_SYSTEM_ADDRESS); -} - -void MastTeleport::OnRebuildComplete(Entity* self, Entity* target) { - if (Preconditions::Check(target, 154) && Preconditions::Check(target, 44)) { - self->SetVar<LWOOBJID>(u"userID", target->GetObjectID()); - - GameMessages::SendSetStunned(target->GetObjectID(), PUSH, target->GetSystemAddress(), - LWOOBJID_EMPTY, true, true, true, true, true, true, true - ); - - self->AddTimer("Start", 3); - } -} - -void MastTeleport::OnTimerDone(Entity* self, std::string timerName) { - const auto playerId = self->GetVar<LWOOBJID>(u"userID"); - - auto* player = EntityManager::Instance()->GetEntity(playerId); - - if (player == nullptr) return; - - if (timerName == "Start") { - auto position = self->GetPosition(); - auto rotation = self->GetRotation(); - - GameMessages::SendTeleport(playerId, position, rotation, player->GetSystemAddress(), true); - - // Hacky fix for odd rotations - if (self->GetVar<std::u16string>(u"MastName") != u"Jail") { - GameMessages::SendOrientToAngle(playerId, true, (M_PI / 180) * 140.0f, player->GetSystemAddress()); - } else { - GameMessages::SendOrientToAngle(playerId, true, (M_PI / 180) * 100.0f, player->GetSystemAddress()); - } - - const auto cinematic = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"Cinematic")); - const auto leanIn = self->GetVar<float>(u"LeanIn"); - - if (!cinematic.empty()) { - GameMessages::SendPlayCinematic(playerId, GeneralUtils::ASCIIToUTF16(cinematic), player->GetSystemAddress(), - true, true, false, false, 0, false, leanIn - ); - } - - GameMessages::SendPlayFXEffect(playerId, 6039, u"hook", "hook", LWOOBJID_EMPTY, 1, 1, true); - - GameMessages::SendPlayAnimation(player, u"crow-swing-no-equip"); - - GameMessages::SendPlayAnimation(self, u"swing"); - - self->AddTimer("PlayerAnimDone", 6.25f); - } else if (timerName == "PlayerAnimDone") { - GameMessages::SendStopFXEffect(player, true, "hook"); - - auto forward = self->GetRotation().GetForwardVector(); - - const auto degrees = -25.0f; - - const auto rads = degrees * (static_cast<float>(M_PI) / 180.0f); - - const Vector3 newPlayerRot = { 0, rads, 0 }; - - auto position = self->GetPosition(); - - position.x += (forward.x * 20.5f); - position.y += 12; - position.z += (forward.z * 20.5f); - - GameMessages::SendOrientToAngle(playerId, true, rads, player->GetSystemAddress()); - - GameMessages::SendTeleport(playerId, position, NiQuaternion::IDENTITY, player->GetSystemAddress()); - - GameMessages::SendSetStunned(playerId, POP, player->GetSystemAddress(), - LWOOBJID_EMPTY, true, true, true, true, true, true, true - ); - } -} diff --git a/dScripts/MinigameTreasureChestServer.cpp b/dScripts/MinigameTreasureChestServer.cpp deleted file mode 100644 index 66222d59..00000000 --- a/dScripts/MinigameTreasureChestServer.cpp +++ /dev/null @@ -1,64 +0,0 @@ -#include "MinigameTreasureChestServer.h" -#include "ScriptedActivityComponent.h" -#include "TeamManager.h" -#include "EntityManager.h" -#include "dZoneManager.h" - -void MinigameTreasureChestServer::OnUse(Entity* self, Entity* user) { - auto* sac = self->GetComponent<ScriptedActivityComponent>(); - if (sac == nullptr) - return; - - if (self->GetVar<bool>(u"used")) - return; - self->SetVar<bool>(u"used", true); - - if (!IsPlayerInActivity(self, user->GetObjectID())) - UpdatePlayer(self, user->GetObjectID()); - - auto* team = TeamManager::Instance()->GetTeam(user->GetObjectID()); - uint32_t activityRating = 0; - if (team != nullptr) { - for (const auto& teamMemberID : team->members) { - auto* teamMember = EntityManager::Instance()->GetEntity(teamMemberID); - if (teamMember != nullptr) { - activityRating = CalculateActivityRating(self, teamMemberID); - - if (self->GetLOT() == frakjawChestId) activityRating = team->members.size(); - - LootGenerator::Instance().DropActivityLoot(teamMember, self, sac->GetActivityID(), activityRating); - } - } - } else { - activityRating = CalculateActivityRating(self, user->GetObjectID()); - - if (self->GetLOT() == frakjawChestId) activityRating = 1; - - LootGenerator::Instance().DropActivityLoot(user, self, sac->GetActivityID(), activityRating); - } - - sac->PlayerRemove(user->GetObjectID()); - - auto* zoneControl = dZoneManager::Instance()->GetZoneControlObject(); - if (zoneControl != nullptr) { - zoneControl->OnFireEventServerSide(self, "Survival_Update", 0); - } - - self->Smash(self->GetObjectID()); -} - -uint32_t MinigameTreasureChestServer::CalculateActivityRating(Entity* self, LWOOBJID playerID) { - auto* team = TeamManager::Instance()->GetTeam(playerID); - return team != nullptr ? team->members.size() * 100 : ActivityManager::CalculateActivityRating(self, playerID) * 100; -} - -void MinigameTreasureChestServer::OnStartup(Entity* self) { - - // BONS treasure chest thinks it's on FV, causing it to start a lobby - if (dZoneManager::Instance()->GetZoneID().GetMapID() == 1204) { - auto* sac = self->GetComponent<ScriptedActivityComponent>(); - if (sac != nullptr) { - sac->SetInstanceMapID(1204); - } - } -} diff --git a/dScripts/MonCoreNookDoors.cpp b/dScripts/MonCoreNookDoors.cpp deleted file mode 100644 index dc759c50..00000000 --- a/dScripts/MonCoreNookDoors.cpp +++ /dev/null @@ -1,37 +0,0 @@ -#include "MonCoreNookDoors.h" -#include "dZoneManager.h" - -void MonCoreNookDoors::OnStartup(Entity* self) { - SpawnDoor(self); -} - -void MonCoreNookDoors::SpawnDoor(Entity* self) { - const auto doorNum = self->GetVarAsString(u"number"); - - if (doorNum.empty()) { - return; - } - - const auto spawners = dZoneManager::Instance()->GetSpawnersByName("MonCoreNookDoor0" + doorNum); - - if (spawners.empty()) { - return; - } - - auto* spawner = spawners[0]; - - spawner->Reset(); - spawner->Activate(); -} - -void MonCoreNookDoors::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - if (args == "DoorSmashed") { - self->AddTimer("RespawnDoor", 30); - } -} - -void MonCoreNookDoors::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "RespawnDoor") { - SpawnDoor(self); - } -} diff --git a/dScripts/MonCoreSmashableDoors.cpp b/dScripts/MonCoreSmashableDoors.cpp deleted file mode 100644 index 4ec11e56..00000000 --- a/dScripts/MonCoreSmashableDoors.cpp +++ /dev/null @@ -1,21 +0,0 @@ -#include "MonCoreSmashableDoors.h" -#include "EntityManager.h" - -void MonCoreSmashableDoors::OnDie(Entity* self, Entity* killer) { - auto myNum = self->GetVarAsString(u"spawner_name"); - - myNum = myNum.substr(myNum.length() - 1, 1); - - auto triggerGroup = "CoreNookTrig0" + myNum; - - // Get the trigger - auto triggers = EntityManager::Instance()->GetEntitiesInGroup(triggerGroup); - - if (triggers.empty()) { - return; - } - - for (auto trigger : triggers) { - trigger->OnFireEventServerSide(self, "DoorSmashed"); - } -} diff --git a/dScripts/NPCAddRemoveItem.cpp b/dScripts/NPCAddRemoveItem.cpp index ce47b12a..13677072 100644 --- a/dScripts/NPCAddRemoveItem.cpp +++ b/dScripts/NPCAddRemoveItem.cpp @@ -1,7 +1,8 @@ #include "NPCAddRemoveItem.h" #include "InventoryComponent.h" +#include "eMissionState.h" -void NPCAddRemoveItem::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { +void NPCAddRemoveItem::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { auto* inventory = target->GetComponent<InventoryComponent>(); if (inventory == nullptr) return; @@ -10,9 +11,9 @@ void NPCAddRemoveItem::OnMissionDialogueOK(Entity* self, Entity* target, int mis if (missionSetting.first == missionID) { for (const auto& itemSetting : missionSetting.second) { for (const auto& lot : itemSetting.items) { - if (itemSetting.add && (missionState == MissionState::MISSION_STATE_AVAILABLE || missionState == MissionState::MISSION_STATE_COMPLETE_AVAILABLE)) { - inventory->AddItem(lot, 1, eLootSourceType::LOOT_SOURCE_NONE); - } else if (itemSetting.remove && (missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE || missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE)) { + if (itemSetting.add && (missionState == eMissionState::AVAILABLE || missionState == eMissionState::COMPLETE_AVAILABLE)) { + inventory->AddItem(lot, 1, eLootSourceType::NONE); + } else if (itemSetting.remove && (missionState == eMissionState::READY_TO_COMPLETE || missionState == eMissionState::COMPLETE_READY_TO_COMPLETE)) { inventory->RemoveItem(lot, 1); } } diff --git a/dScripts/NPCAddRemoveItem.h b/dScripts/NPCAddRemoveItem.h index a266f817..73a7f2c9 100644 --- a/dScripts/NPCAddRemoveItem.h +++ b/dScripts/NPCAddRemoveItem.h @@ -12,7 +12,7 @@ struct ItemSetting { */ class NPCAddRemoveItem : public CppScripts::Script { protected: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; virtual std::map<uint32_t, std::vector<ItemSetting>> GetSettings(); private: void OnStartup(Entity* self) override; diff --git a/dScripts/NjColeNPC.cpp b/dScripts/NjColeNPC.cpp deleted file mode 100644 index f151db40..00000000 --- a/dScripts/NjColeNPC.cpp +++ /dev/null @@ -1,48 +0,0 @@ -#include "NjColeNPC.h" -#include "MissionComponent.h" -#include "InventoryComponent.h" - -void NjColeNPC::OnEmoteReceived(Entity* self, int32_t emote, Entity* target) { - if (emote != 393) { - return; - } - - auto* inventoryComponent = target->GetComponent<InventoryComponent>(); - - if (inventoryComponent == nullptr) { - return; - } - - if (!inventoryComponent->IsEquipped(14499) && !inventoryComponent->IsEquipped(16644)) { - return; - } - - auto* missionComponent = target->GetComponent<MissionComponent>(); - - if (missionComponent == nullptr) { - return; - } - - missionComponent->ForceProgressTaskType(1818, 1, 1); -} - -void NjColeNPC::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - NjNPCMissionSpinjitzuServer::OnMissionDialogueOK(self, target, missionID, missionState); - - if (missionID == 1818 && missionState >= MissionState::MISSION_STATE_READY_TO_COMPLETE) { - auto* missionComponent = target->GetComponent<MissionComponent>(); - auto* inventoryComponent = target->GetComponent<InventoryComponent>(); - - if (missionComponent == nullptr || inventoryComponent == nullptr) { - return; - } - - if (inventoryComponent->GetLotCount(14499) > 0) { - inventoryComponent->RemoveItem(14499, 1); - } else { - return; - } - - inventoryComponent->AddItem(16644, 1, eLootSourceType::LOOT_SOURCE_NONE); - } -} diff --git a/dScripts/NjColeNPC.h b/dScripts/NjColeNPC.h deleted file mode 100644 index cf8e67e1..00000000 --- a/dScripts/NjColeNPC.h +++ /dev/null @@ -1,7 +0,0 @@ -#include "NjNPCMissionSpinjitzuServer.h" - -class NjColeNPC : public NjNPCMissionSpinjitzuServer { - void OnEmoteReceived(Entity* self, int32_t emote, Entity* target) override; - - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -}; diff --git a/dScripts/NjDragonEmblemChestServer.cpp b/dScripts/NjDragonEmblemChestServer.cpp deleted file mode 100644 index fa4d8556..00000000 --- a/dScripts/NjDragonEmblemChestServer.cpp +++ /dev/null @@ -1,15 +0,0 @@ -#include "NjDragonEmblemChestServer.h" -#include "Character.h" -#include "DestroyableComponent.h" - -void NjDragonEmblemChestServer::OnUse(Entity* self, Entity* user) { - auto* character = user->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(NJ_WU_SHOW_DAILY_CHEST, false); - } - - auto* destroyable = self->GetComponent<DestroyableComponent>(); - if (destroyable != nullptr) { - LootGenerator::Instance().DropLoot(user, self, destroyable->GetLootMatrixID(), 0, 0); - } -} diff --git a/dScripts/NjGarmadonCelebration.cpp b/dScripts/NjGarmadonCelebration.cpp deleted file mode 100644 index c223c55c..00000000 --- a/dScripts/NjGarmadonCelebration.cpp +++ /dev/null @@ -1,17 +0,0 @@ -#include "NjGarmadonCelebration.h" -#include "Character.h" -#include "GameMessages.h" - -void NjGarmadonCelebration::OnCollisionPhantom(Entity* self, Entity* target) { - auto* character = target->GetCharacter(); - - if (character == nullptr) { - return; - } - - if (!character->GetPlayerFlag(ePlayerFlags::NJ_GARMADON_CINEMATIC_SEEN)) { - character->SetPlayerFlag(ePlayerFlags::NJ_GARMADON_CINEMATIC_SEEN, true); - - GameMessages::SendStartCelebrationEffect(target, target->GetSystemAddress(), GarmadonCelebrationID); - } -} diff --git a/dScripts/NjIceRailActivator.cpp b/dScripts/NjIceRailActivator.cpp deleted file mode 100644 index 2549cd0f..00000000 --- a/dScripts/NjIceRailActivator.cpp +++ /dev/null @@ -1,25 +0,0 @@ -#include "NjIceRailActivator.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void NjIceRailActivator::OnPlayerRailArrived(Entity* self, Entity* sender, const std::u16string& pathName, - int32_t waypoint) { - const auto breakPoint = self->GetVar<int32_t>(BreakpointVariable); - if (breakPoint == waypoint) { - const auto& blockGroup = self->GetVar<std::u16string>(BlockGroupVariable); - - for (auto* block : EntityManager::Instance()->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(blockGroup))) { - GameMessages::SendPlayAnimation(block, u"explode"); - - const auto blockID = block->GetObjectID(); - - self->AddCallbackTimer(1.0f, [self, blockID]() { - auto* block = EntityManager::Instance()->GetEntity(blockID); - - if (block != nullptr) { - block->Kill(self); - } - }); - } - } -} diff --git a/dScripts/NjJayMissionItems.cpp b/dScripts/NjJayMissionItems.cpp deleted file mode 100644 index cc871b86..00000000 --- a/dScripts/NjJayMissionItems.cpp +++ /dev/null @@ -1,13 +0,0 @@ -#include "NjJayMissionItems.h" - -void NjJayMissionItems::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - NjNPCMissionSpinjitzuServer::OnMissionDialogueOK(self, target, missionID, missionState); - NPCAddRemoveItem::OnMissionDialogueOK(self, target, missionID, missionState); -} - -std::map<uint32_t, std::vector<ItemSetting>> NjJayMissionItems::GetSettings() { - return { - {1789, {{{14474},false, true}}}, - {1927, {{{14493},false, true}}} - }; -} diff --git a/dScripts/NjJayMissionItems.h b/dScripts/NjJayMissionItems.h deleted file mode 100644 index c49f49ea..00000000 --- a/dScripts/NjJayMissionItems.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once -#include "NjNPCMissionSpinjitzuServer.h" -#include "NPCAddRemoveItem.h" -#include <vector> -#include <map> - -class NjJayMissionItems : public NjNPCMissionSpinjitzuServer, NPCAddRemoveItem { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - std::map<uint32_t, std::vector<ItemSetting>> GetSettings() override; -}; diff --git a/dScripts/NjMonastryBossInstance.cpp b/dScripts/NjMonastryBossInstance.cpp deleted file mode 100644 index dbab7365..00000000 --- a/dScripts/NjMonastryBossInstance.cpp +++ /dev/null @@ -1,523 +0,0 @@ -#include "NjMonastryBossInstance.h" -#include "RebuildComponent.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "dZoneManager.h" -#include "GameMessages.h" -#include "BaseCombatAIComponent.h" -#include "BuffComponent.h" -#include "SkillComponent.h" -#include "TeamManager.h" -#include <algorithm> - -// // // // // // // -// Event handling // -// // // // // // // - -void NjMonastryBossInstance::OnStartup(Entity* self) { - auto spawnerNames = std::vector<std::string>{ LedgeFrakjawSpawner, LowerFrakjawSpawner, BaseEnemiesSpawner + std::to_string(1), - BaseEnemiesSpawner + std::to_string(2), BaseEnemiesSpawner + std::to_string(3), - BaseEnemiesSpawner + std::to_string(4), CounterweightSpawner }; - - // Add a notification request for all the spawned entities, corresponds to notifySpawnedObjectLoaded - for (const auto& spawnerName : spawnerNames) { - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(spawnerName)) { - spawner->AddEntitySpawnedCallback([self, this](Entity* entity) { - const auto lot = entity->GetLOT(); - switch (lot) { - case LedgedFrakjawLOT: - NjMonastryBossInstance::HandleLedgedFrakjawSpawned(self, entity); - return; - case CounterWeightLOT: - NjMonastryBossInstance::HandleCounterWeightSpawned(self, entity); - return; - case LowerFrakjawLOT: - NjMonastryBossInstance::HandleLowerFrakjawSpawned(self, entity); - return; - default: - NjMonastryBossInstance::HandleWaveEnemySpawned(self, entity); - return; - } - }); - } - } -} - -void NjMonastryBossInstance::OnPlayerLoaded(Entity* self, Entity* player) { - ActivityTimerStop(self, WaitingForPlayersTimer); - - // Join the player in the activity - UpdatePlayer(self, player->GetObjectID()); - - // Buff the player - auto* destroyableComponent = player->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->SetHealth((int32_t)destroyableComponent->GetMaxHealth()); - destroyableComponent->SetArmor((int32_t)destroyableComponent->GetMaxArmor()); - destroyableComponent->SetImagination((int32_t)destroyableComponent->GetMaxImagination()); - } - - // Add player ID to instance - auto totalPlayersLoaded = self->GetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable); - totalPlayersLoaded.push_back(player->GetObjectID()); - - // Properly position the player - self->SetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable, totalPlayersLoaded); - // This was always spawning all players at position one before and other values cause players to be invisible. - TeleportPlayer(player, 1); - - // Large teams face a tougher challenge - if (totalPlayersLoaded.size() >= 3) - self->SetVar<bool>(LargeTeamVariable, true); - - // Start the game if all players in the team have loaded - auto* team = TeamManager::Instance()->GetTeam(player->GetObjectID()); - if (team == nullptr || totalPlayersLoaded.size() == team->members.size()) { - StartFight(self); - return; - } - - self->AddCallbackTimer(0.0f, [self, player]() { - if (player != nullptr) { - // If we don't have enough players yet, wait for the others to load and notify the client to play a cool cinematic - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayerLoaded", 0, 0, - player->GetObjectID(), "", player->GetSystemAddress()); - } - }); - - ActivityTimerStart(self, WaitingForPlayersTimer, 45.0f, 45.0f); -} - -void NjMonastryBossInstance::OnPlayerExit(Entity* self, Entity* player) { - UpdatePlayer(self, player->GetObjectID(), true); - // Fetch the total players loaded from the vars - auto totalPlayersLoaded = self->GetVar<std::vector<LWOOBJID> >(TotalPlayersLoadedVariable); - - // Find the player to remove - auto playerToRemove = std::find(totalPlayersLoaded.begin(), totalPlayersLoaded.end(), player->GetObjectID()); - - // If we found the player remove them from out list of players - if (playerToRemove != totalPlayersLoaded.end()) { - totalPlayersLoaded.erase(playerToRemove); - } else { - Game::logger->Log("NjMonastryBossInstance", "Failed to remove player at exit."); - } - - // Set the players loaded var back - self->SetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable, totalPlayersLoaded); - - // Since this is an exit method, check if enough players have left. If enough have left - // resize the instance to account for such. - if (totalPlayersLoaded.size() <= 2) self->SetVar<bool>(LargeTeamVariable, false); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayerLeft", 0, 0, player->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); -} - -void NjMonastryBossInstance::OnActivityTimerDone(Entity* self, const std::string& name) { - auto split = GeneralUtils::SplitString(name, TimerSplitChar); - auto timerName = split[0]; - auto objectID = split.size() > 1 ? (LWOOBJID)std::stoull(split[1]) : LWOOBJID_EMPTY; - - if (timerName == WaitingForPlayersTimer) { - StartFight(self); - } else if (timerName == SpawnNextWaveTimer) { - auto* frakjaw = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(LedgeFrakjawVariable)); - if (frakjaw != nullptr) { - SummonWave(self, frakjaw); - } - } else if (timerName == SpawnWaveTimer) { - auto wave = self->GetVar<uint32_t>(WaveNumberVariable); - self->SetVar<uint32_t>(WaveNumberVariable, wave + 1); - self->SetVar<uint32_t>(TotalAliveInWaveVariable, 0); - - if (wave < m_Waves.size()) { - auto waves = m_Waves.at(wave); - auto counter = 0; - - for (const auto& waveEnemy : waves) { - const auto numberToSpawn = self->GetVar<bool>(LargeTeamVariable) - ? waveEnemy.largeNumber : waveEnemy.smallNumber; - - auto spawnIndex = counter % 4 + 1; - SpawnOnNetwork(self, waveEnemy.lot, numberToSpawn, BaseEnemiesSpawner + std::to_string(spawnIndex)); - counter++; - } - } - } else if (timerName + TimerSplitChar == UnstunTimer) { - auto* entity = EntityManager::Instance()->GetEntity(objectID); - if (entity != nullptr) { - auto* combatAI = entity->GetComponent<BaseCombatAIComponent>(); - if (combatAI != nullptr) { - combatAI->SetDisabled(false); - } - } - } else if (timerName == SpawnCounterWeightTimer) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(CounterweightSpawner); - if (!spawners.empty()) { - // Spawn the counter weight at a specific waypoint, there's one for each round - auto* spawner = spawners.front(); - - spawner->Spawn({ - spawner->m_Info.nodes.at((self->GetVar<uint32_t>(WaveNumberVariable) - 1) % 3) - }, true); - } - } else if (timerName == LowerFrakjawCamTimer) { - // Destroy the frakjaw on the ledge - auto* ledgeFrakjaw = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(LedgeFrakjawVariable)); - if (ledgeFrakjaw != nullptr) { - ledgeFrakjaw->Kill(); - } - - ActivityTimerStart(self, SpawnLowerFrakjawTimer, 1.0f, 1.0f); - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, - LWOOBJID_EMPTY, BottomFrakSpawn, UNASSIGNED_SYSTEM_ADDRESS); - } else if (timerName == SpawnLowerFrakjawTimer) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(LowerFrakjawSpawner); - if (!spawners.empty()) { - auto* spawner = spawners.front(); - spawner->Activate(); - } - } else if (timerName == SpawnRailTimer) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, - LWOOBJID_EMPTY, FireRailSpawn, UNASSIGNED_SYSTEM_ADDRESS); - - auto spawners = dZoneManager::Instance()->GetSpawnersByName(FireRailSpawner); - if (!spawners.empty()) { - auto* spawner = spawners.front(); - spawner->Activate(); - } - } else if (timerName + TimerSplitChar == FrakjawSpawnInTimer) { - auto* lowerFrakjaw = EntityManager::Instance()->GetEntity(objectID); - if (lowerFrakjaw != nullptr) { - LowerFrakjawSummon(self, lowerFrakjaw); - } - } else if (timerName == WaveOverTimer) { - WaveOver(self); - } else if (timerName == FightOverTimer) { - FightOver(self); - } -} - -// // // // // // // // -// Custom functions // -// // // // // // // // - -void NjMonastryBossInstance::StartFight(Entity* self) { - if (self->GetVar<bool>(FightStartedVariable)) - return; - - self->SetVar<bool>(FightStartedVariable, true); - - // Activate the frakjaw spawner - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(LedgeFrakjawSpawner)) { - spawner->Activate(); - } -} - -void NjMonastryBossInstance::HandleLedgedFrakjawSpawned(Entity* self, Entity* ledgedFrakjaw) { - self->SetVar<LWOOBJID>(LedgeFrakjawVariable, ledgedFrakjaw->GetObjectID()); - SummonWave(self, ledgedFrakjaw); -} - -void NjMonastryBossInstance::HandleCounterWeightSpawned(Entity* self, Entity* counterWeight) { - auto* rebuildComponent = counterWeight->GetComponent<RebuildComponent>(); - if (rebuildComponent != nullptr) { - rebuildComponent->AddRebuildStateCallback([this, self, counterWeight](eRebuildState state) { - - switch (state) { - case REBUILD_BUILDING: - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, - 0, 0, counterWeight->GetObjectID(), - BaseCounterweightQB + std::to_string(self->GetVar<uint32_t>(WaveNumberVariable)), - UNASSIGNED_SYSTEM_ADDRESS); - return; - case REBUILD_INCOMPLETE: - GameMessages::SendNotifyClientObject(self->GetObjectID(), EndCinematicNotification, - 0, 0, LWOOBJID_EMPTY, "", - UNASSIGNED_SYSTEM_ADDRESS); - return; - case REBUILD_RESETTING: - ActivityTimerStart(self, SpawnCounterWeightTimer, 0.0f, 0.0f); - return; - case REBUILD_COMPLETED: { - // TODO: Move the platform? - - // The counterweight is actually a moving platform and we should listen to the last waypoint event here - // 0.5f is a rough estimate of that path, though, and results in less needed logic - self->AddCallbackTimer(0.5f, [this, self, counterWeight]() { - if (counterWeight != nullptr) { - counterWeight->Kill(); - } - - auto* frakjaw = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(LedgeFrakjawVariable)); - if (frakjaw == nullptr) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"LedgeFrakjawDead", 0, - 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - return; - } - - auto* skillComponent = frakjaw->GetComponent<SkillComponent>(); - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(1635, 39097, frakjaw->GetObjectID(), true, false); - } - - GameMessages::SendPlayAnimation(frakjaw, StunnedAnimation); - GameMessages::SendPlayNDAudioEmitter(frakjaw, UNASSIGNED_SYSTEM_ADDRESS, CounterSmashAudio); - - // Before wave 4 we should lower frakjaw from the ledge - if (self->GetVar<uint32_t>(WaveNumberVariable) == 3) { - LowerFrakjaw(self, frakjaw); - return; - } - - ActivityTimerStart(self, SpawnNextWaveTimer, 2.0f, 2.0f); - }); - } - default: - return; - } - }); - } -} - -void NjMonastryBossInstance::HandleLowerFrakjawSpawned(Entity* self, Entity* lowerFrakjaw) { - GameMessages::SendPlayAnimation(lowerFrakjaw, TeleportInAnimation); - self->SetVar<LWOOBJID>(LowerFrakjawVariable, lowerFrakjaw->GetObjectID()); - - auto* combatAI = lowerFrakjaw->GetComponent<BaseCombatAIComponent>(); - if (combatAI != nullptr) { - combatAI->SetDisabled(true); - } - - auto* destroyableComponent = lowerFrakjaw->GetComponent<DestroyableComponent>(); - if (destroyableComponent != nullptr) { - destroyableComponent->AddOnHitCallback([this, self, lowerFrakjaw](Entity* attacker) { - NjMonastryBossInstance::HandleLowerFrakjawHit(self, lowerFrakjaw, attacker); - }); - } - - lowerFrakjaw->AddDieCallback([this, self, lowerFrakjaw]() { - NjMonastryBossInstance::HandleLowerFrakjawDied(self, lowerFrakjaw); - }); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"LedgeFrakjawDead", 0, 0, - LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - - if (self->GetVar<bool>(LargeTeamVariable)) { - // Double frakjaws health for large teams - if (destroyableComponent != nullptr) { - const auto doubleHealth = destroyableComponent->GetHealth() * 2; - destroyableComponent->SetHealth(doubleHealth); - destroyableComponent->SetMaxHealth((float_t)doubleHealth); - } - - ActivityTimerStart(self, FrakjawSpawnInTimer + std::to_string(lowerFrakjaw->GetObjectID()), - 2.0f, 2.0f); - ActivityTimerStart(self, UnstunTimer + std::to_string(lowerFrakjaw->GetObjectID()), - 7.0f, 7.0f); - } else { - ActivityTimerStart(self, UnstunTimer + std::to_string(lowerFrakjaw->GetObjectID()), - 5.0f, 5.0f); - } -} - -void NjMonastryBossInstance::HandleLowerFrakjawHit(Entity* self, Entity* lowerFrakjaw, Entity* attacker) { - auto* destroyableComponent = lowerFrakjaw->GetComponent<DestroyableComponent>(); - if (destroyableComponent == nullptr) - return; - - // Progress the fight to the last wave if frakjaw has less than 50% of his health left - if (destroyableComponent->GetHealth() <= (uint32_t)destroyableComponent->GetMaxHealth() / 2 && !self->GetVar<bool>(OnLastWaveVarbiale)) { - self->SetVar<bool>(OnLastWaveVarbiale, true); - - // Stun frakjaw during the cinematic - auto* combatAI = lowerFrakjaw->GetComponent<BaseCombatAIComponent>(); - if (combatAI != nullptr) { - combatAI->SetDisabled(true); - } - ActivityTimerStart(self, UnstunTimer + std::to_string(lowerFrakjaw->GetObjectID()), 5.0f, 5.0f); - - const auto trashMobsAlive = self->GetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable); - std::vector<LWOOBJID> newTrashMobs = {}; - - for (const auto& trashMobID : trashMobsAlive) { - auto* trashMob = EntityManager::Instance()->GetEntity(trashMobID); - if (trashMob != nullptr) { - newTrashMobs.push_back(trashMobID); - - // Stun all the enemies until the cinematic is over - auto* trashMobCombatAI = trashMob->GetComponent<BaseCombatAIComponent>(); - if (trashMobCombatAI != nullptr) { - trashMobCombatAI->SetDisabled(true); - } - ActivityTimerStart(self, UnstunTimer + std::to_string(trashMobID), 5.0f, 5.0f); - } - } - - self->SetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable, newTrashMobs); - - LowerFrakjawSummon(self, lowerFrakjaw); - RemovePoison(self); - } -} - -void NjMonastryBossInstance::HandleLowerFrakjawDied(Entity* self, Entity* lowerFrakjaw) { - ActivityTimerStart(self, FightOverTimer, 2.0f, 2.0f); -} - -void NjMonastryBossInstance::HandleWaveEnemySpawned(Entity* self, Entity* waveEnemy) { - waveEnemy->AddDieCallback([this, self, waveEnemy]() { - NjMonastryBossInstance::HandleWaveEnemyDied(self, waveEnemy); - }); - - auto waveEnemies = self->GetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable); - waveEnemies.push_back(waveEnemy->GetObjectID()); - self->SetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable, waveEnemies); - - auto* combatAI = waveEnemy->GetComponent<BaseCombatAIComponent>(); - if (combatAI != nullptr) { - combatAI->SetDisabled(true); - ActivityTimerStart(self, UnstunTimer + std::to_string(waveEnemy->GetObjectID()), 3.0f, 3.0f); - } -} - -void NjMonastryBossInstance::HandleWaveEnemyDied(Entity* self, Entity* waveEnemy) { - auto waveEnemies = self->GetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable); - waveEnemies.erase(std::remove(waveEnemies.begin(), waveEnemies.end(), waveEnemy->GetObjectID()), waveEnemies.end()); - self->SetVar<std::vector<LWOOBJID>>(TrashMobsAliveVariable, waveEnemies); - - if (waveEnemies.empty()) { - ActivityTimerStart(self, WaveOverTimer, 2.0f, 2.0f); - } -} - -void NjMonastryBossInstance::TeleportPlayer(Entity* player, uint32_t position) { - for (const auto* spawnPoint : EntityManager::Instance()->GetEntitiesInGroup("SpawnPoint" + std::to_string(position))) { - GameMessages::SendTeleport(player->GetObjectID(), spawnPoint->GetPosition(), spawnPoint->GetRotation(), - player->GetSystemAddress(), true); - } -} - -void NjMonastryBossInstance::SummonWave(Entity* self, Entity* frakjaw) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, LWOOBJID_EMPTY, - LedgeFrakSummon, UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendPlayAnimation(frakjaw, SummonAnimation); - - // Stop the music for the first, fourth and fifth wave - const auto wave = self->GetVar<uint32_t>(WaveNumberVariable); - if (wave >= 1 || wave < (m_Waves.size() - 1)) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), StopMusicNotification, 0, 0, - LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(wave - 1), - UNASSIGNED_SYSTEM_ADDRESS); - } - - // After frakjaw moves down the music stays the same - if (wave < (m_Waves.size() - 1)) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), StartMusicNotification, 0, 0, - LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(wave), - UNASSIGNED_SYSTEM_ADDRESS); - } - - ActivityTimerStart(self, SpawnWaveTimer, 4.0f, 4.0f); -} - -void NjMonastryBossInstance::LowerFrakjawSummon(Entity* self, Entity* frakjaw) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, - LWOOBJID_EMPTY, BottomFrakSummon, UNASSIGNED_SYSTEM_ADDRESS); - ActivityTimerStart(self, SpawnWaveTimer, 2.0f, 2.0f); - GameMessages::SendPlayAnimation(frakjaw, SummonAnimation); -} - -void NjMonastryBossInstance::RemovePoison(Entity* self) { - const auto& totalPlayer = self->GetVar<std::vector<LWOOBJID>>(TotalPlayersLoadedVariable); - for (const auto& playerID : totalPlayer) { - - auto* player = EntityManager::Instance()->GetEntity(playerID); - if (player != nullptr) { - - auto* buffComponent = player->GetComponent<BuffComponent>(); - if (buffComponent != nullptr) { - buffComponent->RemoveBuff(PoisonBuff); - } - } - } -} - -void NjMonastryBossInstance::LowerFrakjaw(Entity* self, Entity* frakjaw) { - GameMessages::SendPlayAnimation(frakjaw, TeleportOutAnimation); - ActivityTimerStart(self, LowerFrakjawCamTimer, 2.0f, 2.0f); - - GameMessages::SendNotifyClientObject(frakjaw->GetObjectID(), StopMusicNotification, 0, 0, - LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(m_Waves.size() - 3), UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendNotifyClientObject(frakjaw->GetObjectID(), StartMusicNotification, 0, 0, - LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(m_Waves.size() - 2), UNASSIGNED_SYSTEM_ADDRESS); -} - -void NjMonastryBossInstance::SpawnOnNetwork(Entity* self, const LOT& toSpawn, const uint32_t& numberToSpawn, const std::string& spawnerName) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(spawnerName); - if (spawners.empty() || numberToSpawn <= 0) - return; - - auto* spawner = spawners.front(); - - // Spawn the lot N times - spawner->SetSpawnLot(toSpawn); - for (auto i = 0; i < numberToSpawn; i++) - spawner->Spawn({ spawner->m_Info.nodes.at(i % spawner->m_Info.nodes.size()) }, true); -} - -void NjMonastryBossInstance::WaveOver(Entity* self) { - auto wave = self->GetVar<uint32_t>(WaveNumberVariable); - if (wave >= m_Waves.size() - 1) - return; - - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, - LWOOBJID_EMPTY, BaseCounterweightSpawn + std::to_string(wave), - UNASSIGNED_SYSTEM_ADDRESS); - ActivityTimerStart(self, SpawnCounterWeightTimer, 1.5f, 1.5f); - RemovePoison(self); -} - -void NjMonastryBossInstance::FightOver(Entity* self) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"GroundFrakjawDead", 0, 0, - LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - - // Remove all the enemies from the battlefield - for (auto i = 1; i < 5; i++) { - auto spawners = dZoneManager::Instance()->GetSpawnersByName(BaseEnemiesSpawner + std::to_string(i)); - if (!spawners.empty()) { - auto* spawner = spawners.front(); - spawner->Deactivate(); - spawner->Reset(); - } - } - - RemovePoison(self); - ActivityTimerStart(self, SpawnRailTimer, 1.5f, 1.5f); - - // Set the music to play the victory music - GameMessages::SendNotifyClientObject(self->GetObjectID(), StopMusicNotification, 0, 0, - LWOOBJID_EMPTY, AudioWaveAudio + std::to_string(m_Waves.size() - 2), - UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendNotifyClientObject(self->GetObjectID(), FlashMusicNotification, 0, 0, - LWOOBJID_EMPTY, "Monastery_Frakjaw_Battle_Win", UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendNotifyClientObject(self->GetObjectID(), PlayCinematicNotification, 0, 0, - LWOOBJID_EMPTY, TreasureChestSpawning, UNASSIGNED_SYSTEM_ADDRESS); - - auto treasureChests = EntityManager::Instance()->GetEntitiesInGroup(ChestSpawnpointGroup); - for (auto* treasureChest : treasureChests) { - auto info = EntityInfo{}; - - info.lot = ChestLOT; - info.pos = treasureChest->GetPosition(); - info.rot = treasureChest->GetRotation(); - info.spawnerID = self->GetObjectID(); - info.settings = { - new LDFData<LWOOBJID>(u"parent_tag", self->GetObjectID()) - }; - - // Finally spawn a treasure chest at the correct spawn point - auto* chestObject = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(chestObject); - } -} diff --git a/dScripts/NjNPCMissionSpinjitzuServer.cpp b/dScripts/NjNPCMissionSpinjitzuServer.cpp deleted file mode 100644 index 19f5f42a..00000000 --- a/dScripts/NjNPCMissionSpinjitzuServer.cpp +++ /dev/null @@ -1,24 +0,0 @@ -#include "NjNPCMissionSpinjitzuServer.h" -#include "Character.h" -#include "EntityManager.h" - -void NjNPCMissionSpinjitzuServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, - MissionState missionState) { - - const auto& element = self->GetVar<std::u16string>(ElementVariable); - if (missionID == ElementMissions.at(element) && missionState >= MissionState::MISSION_STATE_READY_TO_COMPLETE) { - - const auto targetID = target->GetObjectID(); - - // Wait for an animation to complete and flag that the player has learned spinjitzu - self->AddCallbackTimer(5.0f, [targetID, element]() { - auto* target = EntityManager::Instance()->GetEntity(targetID); - if (target != nullptr) { - auto* character = target->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(ElementFlags.at(element), true); - } - } - }); - } -} diff --git a/dScripts/NjNPCMissionSpinjitzuServer.h b/dScripts/NjNPCMissionSpinjitzuServer.h deleted file mode 100644 index 8f25a86a..00000000 --- a/dScripts/NjNPCMissionSpinjitzuServer.h +++ /dev/null @@ -1,24 +0,0 @@ -#pragma once -#include "CppScripts.h" -#include <map> - -static std::map<std::u16string, uint32_t> ElementFlags = { - {u"earth", ePlayerFlags::NJ_EARTH_SPINJITZU}, - {u"lightning", ePlayerFlags::NJ_LIGHTNING_SPINJITZU}, - {u"ice", ePlayerFlags::NJ_ICE_SPINJITZU}, - {u"fire", ePlayerFlags::NJ_FIRE_SPINJITZU} -}; - -static std::map<std::u16string, uint32_t> ElementMissions = { - {u"earth", 1796}, - {u"lightning", 1952}, - {u"ice", 1959}, - {u"fire", 1962}, -}; - -class NjNPCMissionSpinjitzuServer : public CppScripts::Script { -public: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -private: - const std::u16string ElementVariable = u"element"; -}; diff --git a/dScripts/NjRailActivatorsServer.cpp b/dScripts/NjRailActivatorsServer.cpp deleted file mode 100644 index a07de24a..00000000 --- a/dScripts/NjRailActivatorsServer.cpp +++ /dev/null @@ -1,16 +0,0 @@ -#include "NjRailActivatorsServer.h" -#include "RebuildComponent.h" -#include "Character.h" - -void NjRailActivatorsServer::OnUse(Entity* self, Entity* user) { - const auto flag = self->GetVar<int32_t>(u"RailFlagNum"); - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - // Only allow use if this is not a quick build or the quick build is built - if (rebuildComponent == nullptr || rebuildComponent->GetState() == REBUILD_COMPLETED) { - auto* character = user->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(flag, true); - } - } -} diff --git a/dScripts/NjRailPostServer.cpp b/dScripts/NjRailPostServer.cpp deleted file mode 100644 index 23389a98..00000000 --- a/dScripts/NjRailPostServer.cpp +++ /dev/null @@ -1,51 +0,0 @@ -#include "NjRailPostServer.h" -#include "RebuildComponent.h" -#include "EntityManager.h" - -void NjRailPostServer::OnStartup(Entity* self) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - if (rebuildComponent != nullptr) { - self->SetNetworkVar<bool>(NetworkNotActiveVariable, true); - } -} - -void NjRailPostServer::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, - int32_t param2) { - if (name == "PostRebuilt") { - self->SetNetworkVar<bool>(NetworkNotActiveVariable, false); - } else if (name == "PostDied") { - self->SetNetworkVar<bool>(NetworkNotActiveVariable, true); - } -} - -void NjRailPostServer::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == REBUILD_COMPLETED) { - auto* relatedRail = GetRelatedRail(self); - if (relatedRail == nullptr) - return; - - relatedRail->NotifyObject(self, "PostRebuilt"); - - if (self->GetVar<bool>(NotActiveVariable)) - return; - - self->SetNetworkVar(NetworkNotActiveVariable, false); - } else if (state == REBUILD_RESETTING) { - auto* relatedRail = GetRelatedRail(self); - if (relatedRail == nullptr) - return; - - relatedRail->NotifyObject(self, "PostDied"); - } -} - -Entity* NjRailPostServer::GetRelatedRail(Entity* self) { - const auto& railGroup = self->GetVar<std::u16string>(RailGroupVariable); - if (!railGroup.empty()) { - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(railGroup))) { - return entity; - } - } - - return nullptr; -} diff --git a/dScripts/NjScrollChestServer.cpp b/dScripts/NjScrollChestServer.cpp deleted file mode 100644 index 4e1350b4..00000000 --- a/dScripts/NjScrollChestServer.cpp +++ /dev/null @@ -1,17 +0,0 @@ -#include "NjScrollChestServer.h" -#include "InventoryComponent.h" - -void NjScrollChestServer::OnUse(Entity* self, Entity* user) { - const auto keyLOT = self->GetVar<LOT>(u"KeyNum"); - const auto rewardItemLOT = self->GetVar<LOT>(u"openItemID"); - - auto* playerInventory = user->GetComponent<InventoryComponent>(); - if (playerInventory != nullptr && playerInventory->GetLotCount(keyLOT) == 1) { - - // Check for the key and remove - playerInventory->RemoveItem(keyLOT, 1); - - // Reward the player with the item set - playerInventory->AddItem(rewardItemLOT, 1, eLootSourceType::LOOT_SOURCE_NONE); - } -} diff --git a/dScripts/NjWuNPC.cpp b/dScripts/NjWuNPC.cpp deleted file mode 100644 index 855e4433..00000000 --- a/dScripts/NjWuNPC.cpp +++ /dev/null @@ -1,63 +0,0 @@ -#include "NjWuNPC.h" -#include "MissionComponent.h" -#include "Character.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void NjWuNPC::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - - // The Dragon statue daily mission - if (missionID == m_MainDragonMissionID) { - auto* character = target->GetCharacter(); - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (character == nullptr || missionComponent == nullptr) - return; - - switch (missionState) { - case MissionState::MISSION_STATE_AVAILABLE: - case MissionState::MISSION_STATE_COMPLETE_AVAILABLE: - { - // Reset the sub missions - for (const auto& subMissionID : m_SubDragonMissionIDs) { - missionComponent->RemoveMission(subMissionID); - missionComponent->AcceptMission(subMissionID); - } - - character->SetPlayerFlag(ePlayerFlags::NJ_WU_SHOW_DAILY_CHEST, false); - - // Hide the chest - for (auto* chest : EntityManager::Instance()->GetEntitiesInGroup(m_DragonChestGroup)) { - GameMessages::SendNotifyClientObject(chest->GetObjectID(), m_ShowChestNotification, 0, -1, - target->GetObjectID(), "", target->GetSystemAddress()); - } - - return; - } - case MissionState::MISSION_STATE_READY_TO_COMPLETE: - case MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE: - { - character->SetPlayerFlag(NJ_WU_SHOW_DAILY_CHEST, true); - - // Show the chest - for (auto* chest : EntityManager::Instance()->GetEntitiesInGroup(m_DragonChestGroup)) { - GameMessages::SendNotifyClientObject(chest->GetObjectID(), m_ShowChestNotification, 1, -1, - target->GetObjectID(), "", target->GetSystemAddress()); - } - - auto playerID = target->GetObjectID(); - self->AddCallbackTimer(5.0f, [this, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - if (player == nullptr) - return; - - // Stop the dragon effects - for (auto* dragon : EntityManager::Instance()->GetEntitiesInGroup(m_DragonStatueGroup)) { - GameMessages::SendStopFXEffect(dragon, true, "on"); - } - }); - } - default: - return; - } - } -} diff --git a/dScripts/NjWuNPC.h b/dScripts/NjWuNPC.h deleted file mode 100644 index d0cd750b..00000000 --- a/dScripts/NjWuNPC.h +++ /dev/null @@ -1,13 +0,0 @@ -#pragma once -#include "AmTemplateSkillVolume.h" - -class NjWuNPC : public AmTemplateSkillVolume { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - const uint32_t m_MainDragonMissionID = 2040; - const std::vector<uint32_t> m_SubDragonMissionIDs = { 2064, 2065, 2066, 2067 }; - - // Groups and variables - const std::string m_DragonChestGroup = "DragonEmblemChest"; - const std::string m_DragonStatueGroup = "Minidragons"; - const std::u16string m_ShowChestNotification = u"showChest"; -}; diff --git a/dScripts/NjhubLavaPlayerDeathTrigger.cpp b/dScripts/NjhubLavaPlayerDeathTrigger.cpp deleted file mode 100644 index c065f84a..00000000 --- a/dScripts/NjhubLavaPlayerDeathTrigger.cpp +++ /dev/null @@ -1,9 +0,0 @@ -#include "NjhubLavaPlayerDeathTrigger.h" -#include "Entity.h" - -void NjhubLavaPlayerDeathTrigger::OnCollisionPhantom(Entity* self, Entity* target) { - if (!target->IsPlayer()) - return; - - target->Smash(self->GetObjectID(), VIOLENT, u"drown"); -} diff --git a/dScripts/NpcAgCourseStarter.cpp b/dScripts/NpcAgCourseStarter.cpp deleted file mode 100644 index 503d966a..00000000 --- a/dScripts/NpcAgCourseStarter.cpp +++ /dev/null @@ -1,108 +0,0 @@ -#include "NpcAgCourseStarter.h" -#include "EntityManager.h" -#include "ScriptedActivityComponent.h" -#include "GameMessages.h" -#include "LeaderboardManager.h" -#include "MissionComponent.h" -#include <ctime> - -void NpcAgCourseStarter::OnStartup(Entity* self) { - -} - -void NpcAgCourseStarter::OnUse(Entity* self, Entity* user) { - auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); - - if (scriptedActivityComponent == nullptr) { - return; - } - - if (scriptedActivityComponent->GetActivityPlayerData(user->GetObjectID()) != nullptr) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"exit", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); - } else { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"start", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); - } -} - -void NpcAgCourseStarter::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); - - if (scriptedActivityComponent == nullptr) { - return; - } - - if (identifier == u"player_dialog_cancel_course" && button == 1) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stop_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"cancel_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - - scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); - - EntityManager::Instance()->SerializeEntity(self); - } else if (identifier == u"player_dialog_start_course" && button == 1) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"start_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - - GameMessages::SendActivityStart(self->GetObjectID(), sender->GetSystemAddress()); - - auto* data = scriptedActivityComponent->AddActivityPlayerData(sender->GetObjectID()); - - if (data->values[1] != 0) return; - - time_t startTime = std::time(0) + 4; // Offset for starting timer - - data->values[1] = *(float*)&startTime; - - EntityManager::Instance()->SerializeEntity(self); - } else if (identifier == u"FootRaceCancel") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stop_timer", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - - if (scriptedActivityComponent->GetActivityPlayerData(sender->GetObjectID()) != nullptr) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"exit", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - } else { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"start", 0, 0, LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - } - - scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); - } -} - -void NpcAgCourseStarter::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); - if (scriptedActivityComponent == nullptr) - return; - - auto* data = scriptedActivityComponent->GetActivityPlayerData(sender->GetObjectID()); - if (data == nullptr) - return; - - if (args == "course_cancel") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"cancel_timer", 0, 0, - LWOOBJID_EMPTY, "", sender->GetSystemAddress()); - scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); - } else if (args == "course_finish") { - time_t endTime = std::time(0); - time_t finish = (endTime - *(time_t*)&data->values[1]); - - data->values[2] = *(float*)&finish; - - auto* missionComponent = sender->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - missionComponent->ForceProgressTaskType(1884, 1, 1, false); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MINIGAME, -finish, self->GetObjectID(), - "performact_time"); - } - - EntityManager::Instance()->SerializeEntity(self); - LeaderboardManager::SaveScore(sender->GetObjectID(), scriptedActivityComponent->GetActivityID(), - 0, (uint32_t)finish); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"ToggleLeaderBoard", - scriptedActivityComponent->GetActivityID(), 0, sender->GetObjectID(), - "", sender->GetSystemAddress()); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stop_timer", 1, finish, LWOOBJID_EMPTY, "", - sender->GetSystemAddress()); - - scriptedActivityComponent->RemoveActivityPlayerData(sender->GetObjectID()); - } -} diff --git a/dScripts/NpcCowboyServer.cpp b/dScripts/NpcCowboyServer.cpp deleted file mode 100644 index c4940e4d..00000000 --- a/dScripts/NpcCowboyServer.cpp +++ /dev/null @@ -1,26 +0,0 @@ -#include "NpcCowboyServer.h" -#include "MissionState.h" -#include "InventoryComponent.h" - -void NpcCowboyServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID != 1880) { - return; - } - - auto* inventoryComponent = target->GetComponent<InventoryComponent>(); - - if (inventoryComponent == nullptr) { - return; - } - - if (missionState == MissionState::MISSION_STATE_COMPLETE_ACTIVE || - missionState == MissionState::MISSION_STATE_ACTIVE || - missionState == MissionState::MISSION_STATE_AVAILABLE || - missionState == MissionState::MISSION_STATE_COMPLETE_AVAILABLE) { - if (inventoryComponent->GetLotCount(14378) == 0) { - inventoryComponent->AddItem(14378, 1, eLootSourceType::LOOT_SOURCE_NONE); - } - } else if (missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE || missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { - inventoryComponent->RemoveItem(14378, 1); - } -} diff --git a/dScripts/NpcCowboyServer.h b/dScripts/NpcCowboyServer.h deleted file mode 100644 index e600d798..00000000 --- a/dScripts/NpcCowboyServer.h +++ /dev/null @@ -1,7 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NpcCowboyServer : public CppScripts::Script -{ - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -}; diff --git a/dScripts/NpcEpsilonServer.cpp b/dScripts/NpcEpsilonServer.cpp deleted file mode 100644 index f6b8f5c4..00000000 --- a/dScripts/NpcEpsilonServer.cpp +++ /dev/null @@ -1,10 +0,0 @@ -#include "NpcEpsilonServer.h" -#include "GameMessages.h" - -void NpcEpsilonServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - - //If we are completing the Nexus Force join mission, play the celebration for it: - if (missionID == 1851) { - GameMessages::SendStartCelebrationEffect(target, target->GetSystemAddress(), 22); - } -} diff --git a/dScripts/NpcEpsilonServer.h b/dScripts/NpcEpsilonServer.h deleted file mode 100644 index 798da33e..00000000 --- a/dScripts/NpcEpsilonServer.h +++ /dev/null @@ -1,7 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NpcEpsilonServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState); -}; - diff --git a/dScripts/NpcNjAssistantServer.cpp b/dScripts/NpcNjAssistantServer.cpp deleted file mode 100644 index a53d8ba3..00000000 --- a/dScripts/NpcNjAssistantServer.cpp +++ /dev/null @@ -1,27 +0,0 @@ -#include "NpcNjAssistantServer.h" -#include "GameMessages.h" -#include "InventoryComponent.h" -#include "MissionComponent.h" -#include "Item.h" - -void NpcNjAssistantServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID != mailMission) return; - - if (missionState == MissionState::MISSION_STATE_COMPLETE || missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"switch", 0, 0, LWOOBJID_EMPTY, "", target->GetSystemAddress()); - - auto* inv = static_cast<InventoryComponent*>(target->GetComponent(COMPONENT_TYPE_INVENTORY)); - - // If we are ready to complete our missions, we take the kit from you: - if (inv && missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE) { - auto* id = inv->FindItemByLot(14397); //the kit's lot - - if (id != nullptr) { - inv->RemoveItem(id->GetLot(), id->GetCount()); - } - } - } else if (missionState == MissionState::MISSION_STATE_AVAILABLE) { - auto* missionComponent = static_cast<MissionComponent*>(target->GetComponent(COMPONENT_TYPE_MISSION)); - missionComponent->CompleteMission(mailAchievement, true); - } -} diff --git a/dScripts/NpcNjAssistantServer.h b/dScripts/NpcNjAssistantServer.h deleted file mode 100644 index 7ba847d0..00000000 --- a/dScripts/NpcNjAssistantServer.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NpcNjAssistantServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState); - -private: - int mailMission = 1728; //mission to get the item out of your mailbox - int mailAchievement = 1729; // fun fact: spelled "Achivement" in the actual script -}; diff --git a/dScripts/NpcNpSpacemanBob.cpp b/dScripts/NpcNpSpacemanBob.cpp deleted file mode 100644 index 5157e488..00000000 --- a/dScripts/NpcNpSpacemanBob.cpp +++ /dev/null @@ -1,13 +0,0 @@ -#include "NpcNpSpacemanBob.h" -#include "DestroyableComponent.h" -#include "MissionComponent.h" - -void NpcNpSpacemanBob::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE && missionID == 173) { - DestroyableComponent* destroyable = static_cast<DestroyableComponent*>(target->GetComponent(COMPONENT_TYPE_DESTROYABLE)); - destroyable->SetImagination(6); - MissionComponent* mission = static_cast<MissionComponent*>(target->GetComponent(COMPONENT_TYPE_MISSION)); - - mission->CompleteMission(664); - } -} diff --git a/dScripts/NpcNpSpacemanBob.h b/dScripts/NpcNpSpacemanBob.h deleted file mode 100644 index 08cc850d..00000000 --- a/dScripts/NpcNpSpacemanBob.h +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NpcNpSpacemanBob : public CppScripts::Script -{ -public: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState); -}; - diff --git a/dScripts/NpcPirateServer.cpp b/dScripts/NpcPirateServer.cpp deleted file mode 100644 index 47571fa6..00000000 --- a/dScripts/NpcPirateServer.cpp +++ /dev/null @@ -1,17 +0,0 @@ -#include "NpcPirateServer.h" -#include "InventoryComponent.h" - -void NpcPirateServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - auto* inventory = target->GetComponent<InventoryComponent>(); - if (inventory != nullptr && missionID == 1881) { - auto* luckyShovel = inventory->FindItemByLot(14591); - - // Add or remove the lucky shovel based on whether the mission was completed or started - if ((missionState == MissionState::MISSION_STATE_AVAILABLE || missionState == MissionState::MISSION_STATE_COMPLETE_AVAILABLE) - && luckyShovel == nullptr) { - inventory->AddItem(14591, 1, eLootSourceType::LOOT_SOURCE_NONE); - } else if (missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE || missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { - inventory->RemoveItem(14591, 1); - } - } -} diff --git a/dScripts/NpcPirateServer.h b/dScripts/NpcPirateServer.h deleted file mode 100644 index dcb399c5..00000000 --- a/dScripts/NpcPirateServer.h +++ /dev/null @@ -1,6 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NpcPirateServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -}; diff --git a/dScripts/NpcWispServer.cpp b/dScripts/NpcWispServer.cpp deleted file mode 100644 index cd4a4d9c..00000000 --- a/dScripts/NpcWispServer.cpp +++ /dev/null @@ -1,43 +0,0 @@ -#include "NpcWispServer.h" -#include "InventoryComponent.h" -#include "EntityManager.h" -#include "Entity.h" -#include "GameMessages.h" - -void NpcWispServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID != 1849 && missionID != 1883) - return; - - auto* inventory = target->GetComponent<InventoryComponent>(); - if (inventory == nullptr) - return; - - LOT maelstromVacuumLot = 14592; - auto* maelstromVacuum = inventory->FindItemByLot(maelstromVacuumLot); - - // For the daily we add the maelstrom vacuum if the player doesn't have it yet - if (missionID == 1883 && (missionState == MissionState::MISSION_STATE_AVAILABLE || missionState == MissionState::MISSION_STATE_COMPLETE_AVAILABLE) - && maelstromVacuum == nullptr) { - inventory->AddItem(maelstromVacuumLot, 1, eLootSourceType::LOOT_SOURCE_NONE); - } else if (missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE || missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { - inventory->RemoveItem(maelstromVacuumLot, 1); - } - - // Next up hide or show the samples based on the mission state - auto visible = 1; - if (missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE || missionState == MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE) { - visible = 0; - } - - auto groups = missionID == 1849 - ? std::vector<std::string> { "MaelstromSamples" } - : std::vector<std::string>{ "MaelstromSamples", "MaelstromSamples2ndary1", "MaelstromSamples2ndary2" }; - - for (const auto& group : groups) { - auto samples = EntityManager::Instance()->GetEntitiesInGroup(group); - for (auto* sample : samples) { - GameMessages::SendNotifyClientObject(sample->GetObjectID(), u"SetVisibility", visible, 0, - target->GetObjectID(), "", target->GetSystemAddress()); - } - } -} diff --git a/dScripts/NpcWispServer.h b/dScripts/NpcWispServer.h deleted file mode 100644 index 86c9c33d..00000000 --- a/dScripts/NpcWispServer.h +++ /dev/null @@ -1,6 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NpcWispServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState); -}; diff --git a/dScripts/NsConcertChoiceBuildManager.cpp b/dScripts/NsConcertChoiceBuildManager.cpp deleted file mode 100644 index 33436525..00000000 --- a/dScripts/NsConcertChoiceBuildManager.cpp +++ /dev/null @@ -1,64 +0,0 @@ -#include "NsConcertChoiceBuildManager.h" -#include "EntityManager.h" - -const std::vector<Crate> NsConcertChoiceBuildManager::crates{ - { "laser", 11203, 5.0, "Concert_Laser_QB_" }, - { "rocket", 11204, 3.0, "Concert_Rocket_QB_" }, - { "speaker", 11205, 5.0, "Concert_Speaker_QB_" }, - { "spotlight", 11206, 5.0, "Concert_Spotlight_QB_" } -}; - -void NsConcertChoiceBuildManager::OnStartup(Entity* self) { - NsConcertChoiceBuildManager::SpawnCrate(self); -} - -void NsConcertChoiceBuildManager::SpawnCrate(Entity* self) { - const auto spawnNumber = self->GetVar<uint32_t>(u"spawnNumber") % crates.size(); - const auto crate = crates[spawnNumber]; - - const auto groups = self->GetGroups(); - if (groups.empty()) - return; - - // Groups are of the form CB_1, CB_2, etc. - auto group = groups.at(0); - const auto splitGroup = GeneralUtils::SplitString(group, '_'); - if (splitGroup.size() < 2) - return; - const auto groupNumber = std::stoi(splitGroup.at(1)); - - EntityInfo info{}; - info.lot = crate.lot; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.spawnerID = self->GetObjectID(); - info.settings = { - new LDFData<bool>(u"startsQBActivator", true), - new LDFData<std::string>(u"grpNameQBShowBricks", crate.group + std::to_string(groupNumber)), - new LDFData<std::u16string>(u"groupID", GeneralUtils::ASCIIToUTF16("Crate_" + group)), - new LDFData<float>(u"crateTime", crate.time), - }; - - auto* spawnedCrate = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(spawnedCrate); - - spawnedCrate->AddDieCallback([self]() { - self->CancelAllTimers(); // Don't switch if the crate was smashed - self->SetVar<LWOOBJID>(u"currentCrate", LWOOBJID_EMPTY); - }); - - self->SetVar<uint32_t>(u"spawnNumber", spawnNumber + 1); - self->SetVar<float>(u"currentTimer", crate.time); - self->SetVar<LWOOBJID>(u"currentCrate", spawnedCrate->GetObjectID()); - - // Timer that rotates the crates - self->AddCallbackTimer(crate.time, [self]() { - auto crateID = self->GetVar<LWOOBJID>(u"currentCrate"); - if (crateID != LWOOBJID_EMPTY) { - EntityManager::Instance()->DestroyEntity(crateID); - self->SetVar<LWOOBJID>(u"currentCrate", LWOOBJID_EMPTY); - } - - SpawnCrate(self); - }); -} diff --git a/dScripts/NsConcertInstrument.cpp b/dScripts/NsConcertInstrument.cpp deleted file mode 100644 index 508b7b5b..00000000 --- a/dScripts/NsConcertInstrument.cpp +++ /dev/null @@ -1,362 +0,0 @@ -#include "NsConcertInstrument.h" -#include "GameMessages.h" -#include "Item.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "RebuildComponent.h" -#include "SoundTriggerComponent.h" -#include "MissionComponent.h" - -// Constants are at the bottom - -void NsConcertInstrument::OnStartup(Entity* self) { - self->SetVar<bool>(u"beingPlayed", false); - self->SetVar<LWOOBJID>(u"activePlayer", LWOOBJID_EMPTY); - self->SetVar<LWOOBJID>(u"oldItemLeft", LWOOBJID_EMPTY); - self->SetVar<LWOOBJID>(u"oldItemRight", LWOOBJID_EMPTY); -} - -void NsConcertInstrument::OnRebuildNotifyState(Entity* self, eRebuildState state) { - if (state == REBUILD_RESETTING || state == REBUILD_OPEN) { - self->SetVar<LWOOBJID>(u"activePlayer", LWOOBJID_EMPTY); - } -} - -void NsConcertInstrument::OnRebuildComplete(Entity* self, Entity* target) { - if (!target->GetIsDead()) { - self->SetVar<LWOOBJID>(u"activePlayer", target->GetObjectID()); - - self->AddCallbackTimer(0.2f, [self, target]() { - RepositionPlayer(self, target); - if (hideInstrumentOnPlay.at(GetInstrumentLot(self))) - self->SetNetworkVar<bool>(u"Hide", true); - }); - - self->AddCallbackTimer(0.1f, [self, target]() { - StartPlayingInstrument(self, target); - }); - } -} - -void NsConcertInstrument::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (args == "stopPlaying") { - const auto activePlayerID = self->GetVar<LWOOBJID>(u"activePlayer"); - if (activePlayerID == LWOOBJID_EMPTY) - return; - - const auto activePlayer = EntityManager::Instance()->GetEntity(activePlayerID); - if (activePlayer == nullptr) - return; - - StopPlayingInstrument(self, activePlayer); - } -} - -void NsConcertInstrument::OnTimerDone(Entity* self, std::string name) { - const auto activePlayerID = self->GetVar<LWOOBJID>(u"activePlayer"); - if (activePlayerID == LWOOBJID_EMPTY) - return; - - // If for some reason the player becomes null (for example an unexpected leave), we need to clean up - const auto activePlayer = EntityManager::Instance()->GetEntity(activePlayerID); - if (activePlayer == nullptr && name != "cleanupAfterStop") { - StopPlayingInstrument(self, nullptr); - return; - } - - if (activePlayer != nullptr && name == "checkPlayer" && self->GetVar<bool>(u"beingPlayed")) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"checkMovement", 0, 0, - activePlayer->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); - auto* stats = activePlayer->GetComponent<DestroyableComponent>(); - if (stats) { - if (stats->GetImagination() > 0) { - self->AddTimer("checkPlayer", updateFrequency); - } else { - StopPlayingInstrument(self, activePlayer); - } - } - } else if (activePlayer != nullptr && name == "deductImagination" && self->GetVar<bool>(u"beingPlayed")) { - auto* stats = activePlayer->GetComponent<DestroyableComponent>(); - if (stats) - stats->SetImagination(stats->GetImagination() - instrumentImaginationCost); - - self->AddTimer("deductImagination", instrumentCostFrequency); - } else if (name == "cleanupAfterStop") { - if (activePlayer != nullptr) { - UnEquipInstruments(self, activePlayer); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stopPlaying", 0, 0, - activePlayer->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); - } - - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - if (rebuildComponent != nullptr) - rebuildComponent->ResetRebuild(false); - - self->Smash(self->GetObjectID(), VIOLENT); - self->SetVar<LWOOBJID>(u"activePlayer", LWOOBJID_EMPTY); - } else if (activePlayer != nullptr && name == "achievement") { - auto* missionComponent = activePlayer->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - missionComponent->ForceProgress(302, 462, self->GetLOT()); - } - self->AddTimer("achievement2", 10.0f); - } else if (activePlayer != nullptr && name == "achievement2") { - auto* missionComponent = activePlayer->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - missionComponent->ForceProgress(602, achievementTaskID.at(GetInstrumentLot(self)), self->GetLOT()); - } - } -} - -void NsConcertInstrument::StartPlayingInstrument(Entity* self, Entity* player) { - const auto instrumentLot = GetInstrumentLot(self); - self->SetVar<bool>(u"beingPlayed", true); - - // Stuff to notify the player - EquipInstruments(self, player); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"startPlaying", 0, 0, - player->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); - GameMessages::SendPlayCinematic(player->GetObjectID(), cinematics.at(instrumentLot), UNASSIGNED_SYSTEM_ADDRESS); - self->AddCallbackTimer(1.0f, [player, instrumentLot]() { - GameMessages::SendPlayAnimation(player, animations.at(instrumentLot), 2.0f); - }); - - for (auto* soundBox : EntityManager::Instance()->GetEntitiesInGroup("Audio-Concert")) { - auto* soundTrigger = soundBox->GetComponent<SoundTriggerComponent>(); - if (soundTrigger != nullptr) { - soundTrigger->ActivateMusicCue(music.at(instrumentLot)); - } - } - - // Add timers for deducting imagination and checking if the instruments can still be played - self->AddTimer("checkPlayer", updateFrequency); - self->AddTimer("deductImagination", instrumentCostFrequency); - self->AddTimer("achievement", 20.0f); -} - -void NsConcertInstrument::StopPlayingInstrument(Entity* self, Entity* player) { - // No use in stopping twice - if (!self->GetVar<bool>(u"beingPlayed")) - return; - - const auto instrumentLot = GetInstrumentLot(self); - - // Player might be null if they left - if (player != nullptr) { - auto* missions = player->GetComponent<MissionComponent>(); - if (missions != nullptr && missions->GetMissionState(176) == MissionState::MISSION_STATE_ACTIVE) { - missions->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - } - - GameMessages::SendEndCinematic(player->GetObjectID(), cinematics.at(instrumentLot), UNASSIGNED_SYSTEM_ADDRESS, 1.0f); - GameMessages::SendPlayAnimation(player, smashAnimations.at(instrumentLot), 2.0f); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stopCheckingMovement", 0, 0, - player->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); - } - - self->SetVar<bool>(u"beingPlayed", false); - - for (auto* soundBox : EntityManager::Instance()->GetEntitiesInGroup("Audio-Concert")) { - auto* soundTrigger = soundBox->GetComponent<SoundTriggerComponent>(); - if (soundTrigger != nullptr) { - soundTrigger->DeactivateMusicCue(music.at(instrumentLot)); - } - } - - self->CancelAllTimers(); - self->AddTimer("cleanupAfterStop", instrumentSmashAnimationTime.at(instrumentLot)); -} - -void NsConcertInstrument::EquipInstruments(Entity* self, Entity* player) { - auto* inventory = player->GetComponent<InventoryComponent>(); - if (inventory != nullptr) { - auto equippedItems = inventory->GetEquippedItems(); - - // Un equip the current left item - const auto equippedLeftItem = equippedItems.find("special_l"); - if (equippedLeftItem != equippedItems.end()) { - auto* leftItem = inventory->FindItemById(equippedLeftItem->second.id); - if (leftItem != nullptr) { - leftItem->UnEquip(); - self->SetVar<LWOOBJID>(u"oldItemLeft", leftItem->GetId()); - } - } - - // Un equip the current right item - const auto equippedRightItem = equippedItems.find("special_r"); - if (equippedRightItem != equippedItems.end()) { - auto* rightItem = inventory->FindItemById(equippedRightItem->second.id); - if (rightItem != nullptr) { - rightItem->UnEquip(); - self->SetVar<LWOOBJID>(u"oldItemRight", rightItem->GetId()); - } - } - - // Equip the left hand instrument - const auto leftInstrumentLot = instrumentLotLeft.find(GetInstrumentLot(self))->second; - if (leftInstrumentLot != LOT_NULL) { - inventory->AddItem(leftInstrumentLot, 1, eLootSourceType::LOOT_SOURCE_NONE, TEMP_ITEMS, {}, LWOOBJID_EMPTY, false); - auto* leftInstrument = inventory->FindItemByLot(leftInstrumentLot, TEMP_ITEMS); - leftInstrument->Equip(); - } - - // Equip the right hand instrument - const auto rightInstrumentLot = instrumentLotRight.find(GetInstrumentLot(self))->second; - if (rightInstrumentLot != LOT_NULL) { - inventory->AddItem(rightInstrumentLot, 1, eLootSourceType::LOOT_SOURCE_NONE, TEMP_ITEMS, {}, LWOOBJID_EMPTY, false); - auto* rightInstrument = inventory->FindItemByLot(rightInstrumentLot, TEMP_ITEMS); - rightInstrument->Equip(); - } - } -} - -void NsConcertInstrument::UnEquipInstruments(Entity* self, Entity* player) { - auto* inventory = player->GetComponent<InventoryComponent>(); - if (inventory != nullptr) { - auto equippedItems = inventory->GetEquippedItems(); - - // Un equip the current left instrument - const auto equippedInstrumentLeft = equippedItems.find("special_l"); - if (equippedInstrumentLeft != equippedItems.end()) { - auto* leftItem = inventory->FindItemById(equippedInstrumentLeft->second.id); - if (leftItem != nullptr) { - leftItem->UnEquip(); - inventory->RemoveItem(leftItem->GetLot(), 1, TEMP_ITEMS); - } - } - - // Un equip the current right instrument - const auto equippedInstrumentRight = equippedItems.find("special_r"); - if (equippedInstrumentRight != equippedItems.end()) { - auto* rightItem = inventory->FindItemById(equippedInstrumentRight->second.id); - if (rightItem != nullptr) { - rightItem->UnEquip(); - inventory->RemoveItem(rightItem->GetLot(), 1, TEMP_ITEMS); - } - } - - // Equip the old left hand item - const auto leftItemID = self->GetVar<LWOOBJID>(u"oldItemLeft"); - if (leftItemID != LWOOBJID_EMPTY) { - auto* item = inventory->FindItemById(leftItemID); - if (item != nullptr) - item->Equip(); - self->SetVar<LWOOBJID>(u"oldItemLeft", LWOOBJID_EMPTY); - } - - // Equip the old right hand item - const auto rightItemID = self->GetVar<LWOOBJID>(u"oldItemRight"); - if (rightItemID != LWOOBJID_EMPTY) { - auto* item = inventory->FindItemById(rightItemID); - if (item != nullptr) - item->Equip(); - self->SetVar<LWOOBJID>(u"oldItemRight", LWOOBJID_EMPTY); - } - } -} - -void NsConcertInstrument::RepositionPlayer(Entity* self, Entity* player) { - auto position = self->GetPosition(); - auto rotation = self->GetRotation(); - position.SetY(0.0f); - - switch (GetInstrumentLot(self)) { - case Bass: - case Guitar: - position.SetX(position.GetX() + 5.0f); - break; - case Keyboard: - position.SetX(position.GetX() - 0.45f); - position.SetZ(position.GetZ() + 0.75f); - rotation = NiQuaternion::CreateFromAxisAngle(position, -0.8f); // Slight rotation to make the animation sensible - break; - case Drum: - position.SetZ(position.GetZ() - 0.5f); - break; - } - - GameMessages::SendTeleport(player->GetObjectID(), position, rotation, player->GetSystemAddress()); -} - -InstrumentLot NsConcertInstrument::GetInstrumentLot(Entity* self) { - return static_cast<const InstrumentLot>(self->GetLOT()); -} - -// Static stuff needed for script execution - -const std::map<InstrumentLot, std::u16string> NsConcertInstrument::animations{ - { Guitar, u"guitar"}, - { Bass, u"bass"}, - { Keyboard, u"keyboard"}, - { Drum, u"drums"} -}; - -const std::map<InstrumentLot, std::u16string> NsConcertInstrument::smashAnimations{ - {Guitar, u"guitar-smash"}, - {Bass, u"bass-smash"}, - {Keyboard, u"keyboard-smash"}, - {Drum, u"keyboard-smash"} -}; - -const std::map<InstrumentLot, float> NsConcertInstrument::instrumentSmashAnimationTime{ - {Guitar, 2.167f}, - {Bass, 1.167f}, - {Keyboard, 1.0f}, - {Drum, 1.0f} -}; - -const std::map<InstrumentLot, std::string> NsConcertInstrument::music{ - {Guitar, "Concert_Guitar"}, - {Bass, "Concert_Bass"}, - {Keyboard, "Concert_Keys"}, - {Drum, "Concert_Drums"}, -}; - -const std::map<InstrumentLot, std::u16string> NsConcertInstrument::cinematics{ - {Guitar, u"Concert_Cam_G"}, - {Bass, u"Concert_Cam_B"}, - {Keyboard, u"Concert_Cam_K"}, - {Drum, u"Concert_Cam_D"}, -}; - -const std::map<InstrumentLot, LOT> NsConcertInstrument::instrumentLotLeft{ - {Guitar, 4991}, - {Bass, 4992}, - {Keyboard, LOT_NULL}, - {Drum, 4995}, -}; - -const std::map<InstrumentLot, LOT> NsConcertInstrument::instrumentLotRight{ - {Guitar, LOT_NULL}, - {Bass, LOT_NULL}, - {Keyboard, LOT_NULL}, - {Drum, 4996}, -}; - -const std::map<InstrumentLot, bool> NsConcertInstrument::hideInstrumentOnPlay{ - {Guitar, true}, - {Bass, true}, - {Keyboard, false}, - {Drum, false}, -}; - -const std::map<InstrumentLot, float> NsConcertInstrument::instrumentEquipTime{ - {Guitar, 1.033}, - {Bass, 0.75}, - {Keyboard, -1}, - {Drum, 0}, -}; - -const std::map<InstrumentLot, uint32_t> NsConcertInstrument::achievementTaskID{ - {Guitar, 911}, - {Bass, 912}, - {Keyboard, 913}, - {Drum, 914}, -}; - -const uint32_t NsConcertInstrument::instrumentImaginationCost = 2; - -const float NsConcertInstrument::instrumentCostFrequency = 4.0f; - -const float NsConcertInstrument::updateFrequency = 1.0f; diff --git a/dScripts/NsConcertQuickBuild.cpp b/dScripts/NsConcertQuickBuild.cpp deleted file mode 100644 index fcec4dc7..00000000 --- a/dScripts/NsConcertQuickBuild.cpp +++ /dev/null @@ -1,224 +0,0 @@ -#include "NsConcertQuickBuild.h" -#include "EntityManager.h" -#include "NsConcertChoiceBuildManager.h" -#include "DestroyableComponent.h" -#include "GameMessages.h" -#include "MovingPlatformComponent.h" -#include "MissionComponent.h" - -const float NsConcertQuickBuild::resetTime = 40.0f; -const float NsConcertQuickBuild::resetBlinkTime = 6.0f; -const float NsConcertQuickBuild::resetStageTime = 66.5f; -const float NsConcertQuickBuild::resetActivatorTime = 30.0f; -const std::map<LOT, QuickBuildSet> NsConcertQuickBuild::quickBuildSets{ - {5846, QuickBuildSet {"laser", {"discoball", "discofloor", "stagelights", "spotlight"}}}, - {5847, QuickBuildSet {"spotlight", {"spotlight", "stagelights"}}}, - {5848, QuickBuildSet {"rocket", {"flamethrower"}}}, - {5845, QuickBuildSet {"speaker", {"speaker", "speakerHill", "stagelights", "spotlight"}}} -}; - -const std::map<std::string, std::string> NsConcertQuickBuild::quickBuildFX{ - {"discoball", "effectsDiscoball"}, - {"speaker", "effectsShell"}, - {"speakerHill", "effectsHill"}, - {"spotlight", "effectsHill"}, - {"discofloor", "effectsShell"}, - {"flamethrower", "effectsShell"}, - {"stagelights", "effectsShell"} -}; - -std::vector<LWOOBJID> NsConcertQuickBuild::finishedQuickBuilds = {}; - -void NsConcertQuickBuild::OnStartup(Entity* self) { - const auto groups = self->GetGroups(); - if (groups.empty()) - return; - - // Groups are of the form Concert_Laser_QB_1, Concert_Laser_QB_2, etc. - auto group = groups.at(0); - const auto splitGroup = GeneralUtils::SplitString(group, '_'); - if (splitGroup.size() < 4) - return; - - // Get the manager of the crate of this quick build - const auto groupNumber = std::stoi(splitGroup.at(3)); - const auto managerObjects = EntityManager::Instance()->GetEntitiesInGroup("CB_" + std::to_string(groupNumber)); - if (managerObjects.empty()) - return; - - auto* managerObject = managerObjects.at(0); - self->SetVar<LWOOBJID>(u"managerObject", managerObject->GetObjectID()); - self->SetVar<int32_t>(u"groupNumber", groupNumber); - - // Makes the quick build blink after a certain amount of time - self->AddCallbackTimer(GetBlinkTime(resetActivatorTime), [self]() { - self->SetNetworkVar<float>(u"startEffect", NsConcertQuickBuild::GetBlinkTime(resetActivatorTime)); - }); - - // Destroys the quick build after a while if it wasn't built - self->AddCallbackTimer(resetActivatorTime, [self]() { - self->SetNetworkVar<float>(u"startEffect", -1.0f); - self->Smash(self->GetObjectID(), SILENT); - }); -} - -float NsConcertQuickBuild::GetBlinkTime(float time) { - return time <= NsConcertQuickBuild::resetBlinkTime ? 1.0f : time - NsConcertQuickBuild::resetBlinkTime; -} - -void NsConcertQuickBuild::OnDie(Entity* self, Entity* killer) { - auto* managerObject = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"managerObject")); - if (managerObject) { - managerObject->CancelAllTimers(); - managerObject->AddCallbackTimer(1.0f, [managerObject]() { - NsConcertChoiceBuildManager::SpawnCrate(managerObject); - }); - } - - auto position = std::find(finishedQuickBuilds.begin(), finishedQuickBuilds.end(), self->GetObjectID()); - if (position != finishedQuickBuilds.end()) - finishedQuickBuilds.erase(position); -} - -void NsConcertQuickBuild::OnRebuildComplete(Entity* self, Entity* target) { - const auto groupNumber = self->GetVar<int32_t>(u"groupNumber"); - finishedQuickBuilds.push_back(self->GetObjectID()); - self->SetNetworkVar<float>(u"startEffect", -1.0f); - - ProgressStageCraft(self, target); - - // Find all the quick build objects of the same lot - auto finishedQuickBuildObjects = std::vector<Entity*>(); - for (auto quickBuildID : finishedQuickBuilds) { - const auto quickBuildObject = EntityManager::Instance()->GetEntity(quickBuildID); - if (quickBuildObject && quickBuildObject->GetLOT() == self->GetLOT()) { - quickBuildObject->SetVar<LWOOBJID>(u"Player_" + (GeneralUtils::to_u16string(groupNumber)), target->GetObjectID()); - finishedQuickBuildObjects.push_back(quickBuildObject); - } - } - - // If all 4 sets were built, do cool stuff - if (finishedQuickBuildObjects.size() >= 4) { - - // Move all the platforms so the user can collect the imagination brick - const auto movingPlatforms = EntityManager::Instance()->GetEntitiesInGroup("ConcertPlatforms"); - for (auto* movingPlatform : movingPlatforms) { - auto* component = movingPlatform->GetComponent<MovingPlatformComponent>(); - if (component) { - component->WarpToWaypoint(component->GetLastWaypointIndex()); - - movingPlatform->AddCallbackTimer(resetStageTime, [movingPlatform, component]() { - component->WarpToWaypoint(0); - }); - } - } - - ProgressLicensedTechnician(self); - - // Reset all timers for the quickbuilds and make them indestructible - for (auto quickBuild : finishedQuickBuildObjects) { - quickBuild->SetNetworkVar<float>(u"startEffect", -1.0f); - quickBuild->CancelAllTimers(); - - // Indicate that the stage will reset - quickBuild->AddCallbackTimer(GetBlinkTime(resetStageTime), [quickBuild]() { - quickBuild->SetNetworkVar<float>(u"startEffect", GetBlinkTime(resetTime)); - }); - - // Reset the stage - quickBuild->AddCallbackTimer(resetStageTime, [quickBuild]() { - CancelEffects(quickBuild); - quickBuild->SetNetworkVar<float>(u"startEffect", -1); - quickBuild->Smash(); - }); - - auto* destroyableComponent = quickBuild->GetComponent<DestroyableComponent>(); - if (destroyableComponent) - destroyableComponent->SetFaction(-1); - } - - UpdateEffects(self); - return; - } - - // If not all 4 sets were built, reset the timers that were set on spawn - self->CancelAllTimers(); - - // Makes the quick build blink after a certain amount of time - self->AddCallbackTimer(GetBlinkTime(resetTime), [self]() { - self->SetNetworkVar<float>(u"startEffect", NsConcertQuickBuild::GetBlinkTime(resetActivatorTime)); - }); - - // Destroys the quick build after a while if it wasn't built - self->AddCallbackTimer(resetTime, [self]() { - self->SetNetworkVar<float>(u"startEffect", -1.0f); - self->Smash(self->GetObjectID()); - }); -} - -void NsConcertQuickBuild::ProgressStageCraft(Entity* self, Entity* player) { - auto* missionComponent = player->GetComponent<MissionComponent>(); - if (missionComponent) { - - // Has to be forced as to not accidentally trigger the licensed technician achievement - switch (self->GetLOT()) { - case 5845: - missionComponent->ForceProgress(283, 432, 5845); - break; - case 5846: - missionComponent->ForceProgress(283, 433, 5846); - break; - case 5847: - missionComponent->ForceProgress(283, 434, 5847); - break; - case 5848: - missionComponent->ForceProgress(283, 435, 5848); - break; - default: - break; - } - } -} - -void NsConcertQuickBuild::ProgressLicensedTechnician(Entity* self) { - for (auto i = 1; i < 5; i++) { - const auto playerID = self->GetVar<LWOOBJID>(u"Player_" + (GeneralUtils::to_u16string(i))); - if (playerID != LWOOBJID_EMPTY) { - const auto player = EntityManager::Instance()->GetEntity(playerID); - if (player) { - auto playerMissionComponent = player->GetComponent<MissionComponent>(); - if (playerMissionComponent) - playerMissionComponent->ForceProgress(598, 903, self->GetLOT()); - } - } - } -} - -void NsConcertQuickBuild::UpdateEffects(Entity* self) { - CancelEffects(self); - - auto setIterator = quickBuildSets.find(self->GetLOT()); - if (setIterator == quickBuildSets.end()) - return; - - for (const auto& effectName : setIterator->second.effects) { - const auto effectObjects = EntityManager::Instance()->GetEntitiesInGroup(quickBuildFX.at(effectName)); - for (auto* effectObject : effectObjects) { - GameMessages::SendPlayFXEffect(effectObject, 0, GeneralUtils::ASCIIToUTF16(effectName), - effectName + "Effect", LWOOBJID_EMPTY, 1, 1, true); - } - } -} - -void NsConcertQuickBuild::CancelEffects(Entity* self) { - auto setIterator = quickBuildSets.find(self->GetLOT()); - if (setIterator == quickBuildSets.end()) - return; - - for (const auto& effectName : setIterator->second.effects) { - const auto effectObjects = EntityManager::Instance()->GetEntitiesInGroup(quickBuildFX.at(effectName)); - for (auto* effectObject : effectObjects) { - GameMessages::SendStopFXEffect(effectObject, true, effectName + "Effect"); - } - } -} diff --git a/dScripts/NsGetFactionMissionServer.cpp b/dScripts/NsGetFactionMissionServer.cpp deleted file mode 100644 index d759e3ad..00000000 --- a/dScripts/NsGetFactionMissionServer.cpp +++ /dev/null @@ -1,54 +0,0 @@ -#include "NsGetFactionMissionServer.h" -#include "GameMessages.h" -#include "MissionComponent.h" -#include "Character.h" - -void NsGetFactionMissionServer::OnRespondToMission(Entity* self, int missionID, Entity* player, int reward) { - if (missionID != 474) return; - - if (reward != LOT_NULL) { - std::vector<int> factionMissions; - int celebrationID = -1; - int flagID = -1; - - if (reward == 6980) { - // Venture League - factionMissions = { 555, 556 }; - celebrationID = 14; - flagID = 46; - } else if (reward == 6979) { - // Assembly - factionMissions = { 544, 545 }; - celebrationID = 15; - flagID = 47; - } else if (reward == 6981) { - // Paradox - factionMissions = { 577, 578 }; - celebrationID = 16; - flagID = 48; - } else if (reward == 6978) { - // Sentinel - factionMissions = { 566, 567 }; - celebrationID = 17; - flagID = 49; - } - - factionMissions.push_back(778); - - if (celebrationID != -1) { - GameMessages::SendStartCelebrationEffect(player, player->GetSystemAddress(), celebrationID); - } - - if (flagID != -1) { - player->GetCharacter()->SetPlayerFlag(ePlayerFlags::JOINED_A_FACTION, true); - player->GetCharacter()->SetPlayerFlag(flagID, true); - } - - MissionComponent* mis = static_cast<MissionComponent*>(player->GetComponent(COMPONENT_TYPE_MISSION)); - - for (int mission : factionMissions) { - mis->AcceptMission(mission); - mis->CompleteMission(mission); - } - } -} diff --git a/dScripts/NsJohnnyMissionServer.cpp b/dScripts/NsJohnnyMissionServer.cpp deleted file mode 100644 index 2d9ae93c..00000000 --- a/dScripts/NsJohnnyMissionServer.cpp +++ /dev/null @@ -1,14 +0,0 @@ -#include "NsJohnnyMissionServer.h" -#include "MissionComponent.h" - -void NsJohnnyMissionServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID == 773 && missionState <= MissionState::MISSION_STATE_ACTIVE) { - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - missionComponent->AcceptMission(774); - missionComponent->AcceptMission(775); - missionComponent->AcceptMission(776); - missionComponent->AcceptMission(777); - } - } -} diff --git a/dScripts/NsJohnnyMissionServer.h b/dScripts/NsJohnnyMissionServer.h deleted file mode 100644 index 8de39afa..00000000 --- a/dScripts/NsJohnnyMissionServer.h +++ /dev/null @@ -1,6 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NsJohnnyMissionServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -}; diff --git a/dScripts/NsLegoClubDoor.cpp b/dScripts/NsLegoClubDoor.cpp deleted file mode 100644 index 56a213b8..00000000 --- a/dScripts/NsLegoClubDoor.cpp +++ /dev/null @@ -1,152 +0,0 @@ -#include "NsLegoClubDoor.h" -#include "dZoneManager.h" -#include "GameMessages.h" -#include "AMFFormat.h" - -void NsLegoClubDoor::OnStartup(Entity* self) { - self->SetVar(u"currentZone", (int32_t)dZoneManager::Instance()->GetZoneID().GetMapID()); - self->SetVar(u"choiceZone", m_ChoiceZoneID); - self->SetVar(u"teleportAnim", m_TeleportAnim); - self->SetVar(u"teleportString", m_TeleportString); - self->SetVar(u"spawnPoint", m_SpawnPoint); - - args = {}; - - AMFStringValue* callbackClient = new AMFStringValue(); - callbackClient->SetStringValue(std::to_string(self->GetObjectID())); - args.InsertValue("callbackClient", callbackClient); - - AMFStringValue* strIdentifier = new AMFStringValue(); - strIdentifier->SetStringValue("choiceDoor"); - args.InsertValue("strIdentifier", strIdentifier); - - AMFStringValue* title = new AMFStringValue(); - title->SetStringValue("%[UI_CHOICE_DESTINATION]"); - args.InsertValue("title", title); - - AMFArrayValue* choiceOptions = new AMFArrayValue(); - - { - AMFArrayValue* nsArgs = new AMFArrayValue(); - - AMFStringValue* image = new AMFStringValue(); - image->SetStringValue("textures/ui/zone_thumnails/Nimbus_Station.dds"); - nsArgs->InsertValue("image", image); - - AMFStringValue* caption = new AMFStringValue(); - caption->SetStringValue("%[UI_CHOICE_NS]"); - nsArgs->InsertValue("caption", caption); - - AMFStringValue* identifier = new AMFStringValue(); - identifier->SetStringValue("zoneID_1200"); - nsArgs->InsertValue("identifier", identifier); - - AMFStringValue* tooltipText = new AMFStringValue(); - tooltipText->SetStringValue("%[UI_CHOICE_NS_HOVER]"); - nsArgs->InsertValue("tooltipText", tooltipText); - - choiceOptions->PushBackValue(nsArgs); - } - - { - AMFArrayValue* ntArgs = new AMFArrayValue(); - - AMFStringValue* image = new AMFStringValue(); - image->SetStringValue("textures/ui/zone_thumnails/Nexus_Tower.dds"); - ntArgs->InsertValue("image", image); - - AMFStringValue* caption = new AMFStringValue(); - caption->SetStringValue("%[UI_CHOICE_NT]"); - ntArgs->InsertValue("caption", caption); - - AMFStringValue* identifier = new AMFStringValue(); - identifier->SetStringValue("zoneID_1900"); - ntArgs->InsertValue("identifier", identifier); - - AMFStringValue* tooltipText = new AMFStringValue(); - tooltipText->SetStringValue("%[UI_CHOICE_NT_HOVER]"); - ntArgs->InsertValue("tooltipText", tooltipText); - - choiceOptions->PushBackValue(ntArgs); - } - - options = choiceOptions; - - args.InsertValue("options", choiceOptions); -} - -void NsLegoClubDoor::OnUse(Entity* self, Entity* user) { - auto* player = user; - - if (CheckChoice(self, player)) { - AMFArrayValue* multiArgs = new AMFArrayValue(); - - AMFStringValue* callbackClient = new AMFStringValue(); - callbackClient->SetStringValue(std::to_string(self->GetObjectID())); - multiArgs->InsertValue("callbackClient", callbackClient); - - AMFStringValue* strIdentifier = new AMFStringValue(); - strIdentifier->SetStringValue("choiceDoor"); - multiArgs->InsertValue("strIdentifier", strIdentifier); - - AMFStringValue* title = new AMFStringValue(); - title->SetStringValue("%[UI_CHOICE_DESTINATION]"); - multiArgs->InsertValue("title", title); - - multiArgs->InsertValue("options", options); - - GameMessages::SendUIMessageServerToSingleClient(player, player->GetSystemAddress(), "QueueChoiceBox", multiArgs); - } else if (self->GetVar<int32_t>(u"currentZone") != m_ChoiceZoneID) { - AMFArrayValue* multiArgs = new AMFArrayValue(); - - AMFStringValue* state = new AMFStringValue(); - state->SetStringValue("Lobby"); - multiArgs->InsertValue("state", state); - - AMFArrayValue* context = new AMFArrayValue(); - - AMFStringValue* user = new AMFStringValue(); - user->SetStringValue(std::to_string(player->GetObjectID())); - context->InsertValue("user", user); - - AMFStringValue* callbackObj = new AMFStringValue(); - callbackObj->SetStringValue(std::to_string(self->GetObjectID())); - context->InsertValue("callbackObj", callbackObj); - - AMFStringValue* helpVisible = new AMFStringValue(); - helpVisible->SetStringValue("show"); - context->InsertValue("HelpVisible", helpVisible); - - AMFStringValue* type = new AMFStringValue(); - type->SetStringValue("Lego_Club_Valid"); - context->InsertValue("type", type); - - multiArgs->InsertValue("context", context); - - GameMessages::SendUIMessageServerToSingleClient(player, player->GetSystemAddress(), "pushGameState", multiArgs); - } else { - BaseOnUse(self, player); - } -} - -void NsLegoClubDoor::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - std::u16string strIdentifier = identifier; - - if (strIdentifier == u"PlayButton" || strIdentifier == u"CloseButton") { - strIdentifier = u"TransferBox"; - } - - BaseOnMessageBoxResponse(self, sender, button, strIdentifier, userData); -} - -void NsLegoClubDoor::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { - BaseChoiceBoxRespond(self, sender, button, buttonIdentifier, identifier); -} - -void NsLegoClubDoor::OnTimerDone(Entity* self, std::string timerName) { - BaseOnTimerDone(self, timerName); -} - -void NsLegoClubDoor::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); -} diff --git a/dScripts/NsLegoClubDoor.h b/dScripts/NsLegoClubDoor.h deleted file mode 100644 index 0a7a6ee0..00000000 --- a/dScripts/NsLegoClubDoor.h +++ /dev/null @@ -1,23 +0,0 @@ -#pragma once -#include "CppScripts.h" -#include "ChooseYourDestinationNsToNt.h" -#include "BaseConsoleTeleportServer.h" - -class NsLegoClubDoor : public CppScripts::Script, ChooseYourDestinationNsToNt, BaseConsoleTeleportServer -{ -public: - void OnStartup(Entity* self) override; - void OnUse(Entity* self, Entity* user) override; - void OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) override; - void OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) override; - void OnTimerDone(Entity* self, std::string timerName) override; - void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; - -private: - int32_t m_ChoiceZoneID = 1700; - std::string m_SpawnPoint = "NS_LEGO_Club"; - std::u16string m_TeleportAnim = u"lup-teleport"; - std::u16string m_TeleportString = u"ROCKET_TOOLTIP_USE_THE_GATEWAY_TO_TRAVEL_TO_LUP_WORLD"; - AMFArrayValue args = {}; - AMFArrayValue* options = {}; -}; diff --git a/dScripts/NsLupTeleport.cpp b/dScripts/NsLupTeleport.cpp deleted file mode 100644 index 9cd4359b..00000000 --- a/dScripts/NsLupTeleport.cpp +++ /dev/null @@ -1,100 +0,0 @@ -#include "NsLupTeleport.h" -#include "dZoneManager.h" -#include "GameMessages.h" -#include "AMFFormat.h" - -void NsLupTeleport::OnStartup(Entity* self) { - self->SetVar(u"currentZone", (int32_t)dZoneManager::Instance()->GetZoneID().GetMapID()); - self->SetVar(u"choiceZone", m_ChoiceZoneID); - self->SetVar(u"teleportAnim", m_TeleportAnim); - self->SetVar(u"teleportString", m_TeleportString); - self->SetVar(u"spawnPoint", m_SpawnPoint); - - args = {}; - - AMFStringValue* callbackClient = new AMFStringValue(); - callbackClient->SetStringValue(std::to_string(self->GetObjectID())); - args.InsertValue("callbackClient", callbackClient); - - AMFStringValue* strIdentifier = new AMFStringValue(); - strIdentifier->SetStringValue("choiceDoor"); - args.InsertValue("strIdentifier", strIdentifier); - - AMFStringValue* title = new AMFStringValue(); - title->SetStringValue("%[UI_CHOICE_DESTINATION]"); - args.InsertValue("title", title); - - AMFArrayValue* choiceOptions = new AMFArrayValue(); - - { - AMFArrayValue* nsArgs = new AMFArrayValue(); - - AMFStringValue* image = new AMFStringValue(); - image->SetStringValue("textures/ui/zone_thumnails/Nimbus_Station.dds"); - nsArgs->InsertValue("image", image); - - AMFStringValue* caption = new AMFStringValue(); - caption->SetStringValue("%[UI_CHOICE_NS]"); - nsArgs->InsertValue("caption", caption); - - AMFStringValue* identifier = new AMFStringValue(); - identifier->SetStringValue("zoneID_1200"); - nsArgs->InsertValue("identifier", identifier); - - AMFStringValue* tooltipText = new AMFStringValue(); - tooltipText->SetStringValue("%[UI_CHOICE_NS_HOVER]"); - nsArgs->InsertValue("tooltipText", tooltipText); - - choiceOptions->PushBackValue(nsArgs); - } - - { - AMFArrayValue* ntArgs = new AMFArrayValue(); - - AMFStringValue* image = new AMFStringValue(); - image->SetStringValue("textures/ui/zone_thumnails/Nexus_Tower.dds"); - ntArgs->InsertValue("image", image); - - AMFStringValue* caption = new AMFStringValue(); - caption->SetStringValue("%[UI_CHOICE_NT]"); - ntArgs->InsertValue("caption", caption); - - AMFStringValue* identifier = new AMFStringValue(); - identifier->SetStringValue("zoneID_1900"); - ntArgs->InsertValue("identifier", identifier); - - AMFStringValue* tooltipText = new AMFStringValue(); - tooltipText->SetStringValue("%[UI_CHOICE_NT_HOVER]"); - ntArgs->InsertValue("tooltipText", tooltipText); - - choiceOptions->PushBackValue(ntArgs); - } - - args.InsertValue("options", choiceOptions); -} - -void NsLupTeleport::OnUse(Entity* self, Entity* user) { - auto* player = user; - - if (CheckChoice(self, player)) { - GameMessages::SendUIMessageServerToSingleClient(player, player->GetSystemAddress(), "QueueChoiceBox", &args); - } else { - BaseOnUse(self, player); - } -} - -void NsLupTeleport::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - BaseOnMessageBoxResponse(self, sender, button, identifier, userData); -} - -void NsLupTeleport::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { - BaseChoiceBoxRespond(self, sender, button, buttonIdentifier, identifier); -} - -void NsLupTeleport::OnTimerDone(Entity* self, std::string timerName) { - BaseOnTimerDone(self, timerName); -} - -void NsLupTeleport::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); -} diff --git a/dScripts/NsLupTeleport.h b/dScripts/NsLupTeleport.h deleted file mode 100644 index 35edf0bc..00000000 --- a/dScripts/NsLupTeleport.h +++ /dev/null @@ -1,22 +0,0 @@ -#pragma once -#include "CppScripts.h" -#include "ChooseYourDestinationNsToNt.h" -#include "BaseConsoleTeleportServer.h" - -class NsLupTeleport : public CppScripts::Script, ChooseYourDestinationNsToNt, BaseConsoleTeleportServer -{ -public: - void OnStartup(Entity* self) override; - void OnUse(Entity* self, Entity* user) override; - void OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) override; - void OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) override; - void OnTimerDone(Entity* self, std::string timerName) override; - void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; - -private: - int32_t m_ChoiceZoneID = 1600; - std::string m_SpawnPoint = "NS_LW"; - std::u16string m_TeleportAnim = u"lup-teleport"; - std::u16string m_TeleportString = u"UI_TRAVEL_TO_LUP_STATION"; - AMFArrayValue args = {}; -}; diff --git a/dScripts/NsModularBuild.cpp b/dScripts/NsModularBuild.cpp deleted file mode 100644 index 065d061e..00000000 --- a/dScripts/NsModularBuild.cpp +++ /dev/null @@ -1,16 +0,0 @@ -#include "NsModularBuild.h" -#include "MissionComponent.h" - -void NsModularBuild::OnModularBuildExit(Entity* self, Entity* player, bool bCompleted, std::vector<LOT> modules) { - if (bCompleted) { - MissionComponent* mission = static_cast<MissionComponent*>(player->GetComponent(COMPONENT_TYPE_MISSION)); - - if (mission->GetMissionState(m_MissionNum) == MissionState::MISSION_STATE_ACTIVE) { - for (LOT mod : modules) { - if (mod == 9516 || mod == 9517 || mod == 9518) { - mission->ForceProgress(m_MissionNum, 1178, 1); - } - } - } - } -} diff --git a/dScripts/NsQbImaginationStatue.cpp b/dScripts/NsQbImaginationStatue.cpp deleted file mode 100644 index a2e335b7..00000000 --- a/dScripts/NsQbImaginationStatue.cpp +++ /dev/null @@ -1,40 +0,0 @@ -#include "NsQbImaginationStatue.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void NsQbImaginationStatue::OnStartup(Entity* self) { - -} - -void NsQbImaginationStatue::OnRebuildComplete(Entity* self, Entity* target) { - if (target == nullptr) return; - - self->SetVar(u"Player", target->GetObjectID()); - - SpawnLoot(self); - - self->AddTimer("SpawnDelay", 1.5f); - - self->AddTimer("StopSpawner", 10.0f); -} - -void NsQbImaginationStatue::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "SpawnDelay") { - SpawnLoot(self); - - self->AddTimer("SpawnDelay", 1.5f); - } else if (timerName == "StopSpawner") { - self->CancelAllTimers(); - } -} - -void NsQbImaginationStatue::SpawnLoot(Entity* self) { - const auto playerId = self->GetVar<LWOOBJID>(u"Player"); - - auto* player = EntityManager::Instance()->GetEntity(playerId); - - if (player == nullptr) return; - - GameMessages::SendDropClientLoot(player, self->GetObjectID(), 935, 0); - GameMessages::SendDropClientLoot(player, self->GetObjectID(), 935, 0); -} diff --git a/dScripts/NsTokenConsoleServer.cpp b/dScripts/NsTokenConsoleServer.cpp deleted file mode 100644 index a62c165c..00000000 --- a/dScripts/NsTokenConsoleServer.cpp +++ /dev/null @@ -1,61 +0,0 @@ -#include "NsTokenConsoleServer.h" -#include "InventoryComponent.h" -#include "GameMessages.h" -#include "Character.h" -#include "MissionComponent.h" -#include "RebuildComponent.h" - -void NsTokenConsoleServer::OnStartup(Entity* self) { - -} - -void NsTokenConsoleServer::OnUse(Entity* self, Entity* user) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - - if (rebuildComponent == nullptr) { - return; - } - - if (rebuildComponent->GetState() != REBUILD_COMPLETED) { - return; - } - - auto* inventoryComponent = user->GetComponent<InventoryComponent>(); - auto* missionComponent = user->GetComponent<MissionComponent>(); - auto* character = user->GetCharacter(); - - if (inventoryComponent == nullptr || missionComponent == nullptr || character == nullptr) { - return; - } - - if (inventoryComponent->GetLotCount(6194) < 25) { - return; - } - - inventoryComponent->RemoveItem(6194, 25); - - const auto useSound = self->GetVar<std::string>(u"sound1"); - - if (!useSound.empty()) { - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, useSound); - } - - // Player must be in faction to interact with this entity. - LOT tokenLOT = 0; - - if (character->GetPlayerFlag(46)) { - tokenLOT = 8321; - } else if (character->GetPlayerFlag(47)) { - tokenLOT = 8318; - } else if (character->GetPlayerFlag(48)) { - tokenLOT = 8320; - } else if (character->GetPlayerFlag(49)) { - tokenLOT = 8319; - } - - inventoryComponent->AddItem(tokenLOT, 5, eLootSourceType::LOOT_SOURCE_NONE); - - missionComponent->ForceProgressTaskType(863, 1, 1, false); - - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} diff --git a/dScripts/NtAssemblyTubeServer.cpp b/dScripts/NtAssemblyTubeServer.cpp deleted file mode 100644 index f30f4fc7..00000000 --- a/dScripts/NtAssemblyTubeServer.cpp +++ /dev/null @@ -1,114 +0,0 @@ -#include "NtAssemblyTubeServer.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void NtAssemblyTubeServer::OnStartup(Entity* self) { - self->SetProximityRadius(5, "teleport"); -} - -void NtAssemblyTubeServer::OnPlayerLoaded(Entity* self, Entity* player) { - -} - -void NtAssemblyTubeServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (status != "ENTER" || !entering->IsPlayer() || name != "teleport") return; - - auto* player = entering; - const auto playerID = player->GetObjectID(); - - RunAssemblyTube(self, player); - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - } -} - -void NtAssemblyTubeServer::RunAssemblyTube(Entity* self, Entity* player) { - const auto playerID = player->GetObjectID(); - - const auto iter = m_TeleportingPlayerTable.find(playerID); - if (iter == m_TeleportingPlayerTable.end()) m_TeleportingPlayerTable[playerID] = false; - const auto bPlayerBeingTeleported = m_TeleportingPlayerTable[playerID]; - - if (player->IsPlayer() && !bPlayerBeingTeleported) { - auto teleCinematic = self->GetVar<std::u16string>(u"Cinematic"); - - GameMessages::SendSetStunned(playerID, PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true - ); - - if (!teleCinematic.empty()) { - const auto teleCinematicUname = teleCinematic; - GameMessages::SendPlayCinematic(player->GetObjectID(), teleCinematicUname, player->GetSystemAddress(), - true, true, true, false, 0, false, -1, false, true - ); - } - - GameMessages::SendPlayAnimation(player, u"tube-sucker", 4.0f); - - const auto animTime = 3; - - self->AddCallbackTimer(animTime, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - TeleportPlayer(self, player); - }); - } -} - -void NtAssemblyTubeServer::TeleportPlayer(Entity* self, Entity* player) { - auto destinationGroup = self->GetVar<std::u16string>(u"teleGroup"); - auto* destination = self; - - if (!destinationGroup.empty()) { - const auto& groupObjs = EntityManager::Instance()->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(destinationGroup)); - - if (!groupObjs.empty()) { - destination = groupObjs[0]; - } - } - - const auto destPosition = destination->GetPosition(); - const auto destRotation = destination->GetRotation(); - - GameMessages::SendTeleport(player->GetObjectID(), destPosition, destRotation, player->GetSystemAddress(), true); - - GameMessages::SendPlayAnimation(player, u"tube-resurrect", 4.0f); - - const auto animTime = 2; - - const auto playerID = player->GetObjectID(); - - self->AddCallbackTimer(animTime, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - UnlockPlayer(self, player); - }); - - const auto useSound = self->GetVar<std::string>(u"sound1"); - - if (!useSound.empty()) { - GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), useSound); - } -} - -void NtAssemblyTubeServer::UnlockPlayer(Entity* self, Entity* player) { - const auto playerID = player->GetObjectID(); - - m_TeleportingPlayerTable[playerID] = false; - - GameMessages::SendSetStunned(playerID, POP, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true - ); -} diff --git a/dScripts/NtCombatChallengeDummy.cpp b/dScripts/NtCombatChallengeDummy.cpp deleted file mode 100644 index c9cd8f0a..00000000 --- a/dScripts/NtCombatChallengeDummy.cpp +++ /dev/null @@ -1,26 +0,0 @@ -#include "NtCombatChallengeDummy.h" -#include "EntityManager.h" - -void NtCombatChallengeDummy::OnDie(Entity* self, Entity* killer) { - const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); - - auto* challengeObject = EntityManager::Instance()->GetEntity(challengeObjectID); - - if (challengeObject != nullptr) { - for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { - script->OnDie(challengeObject, killer); - } - } -} - -void NtCombatChallengeDummy::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); - - auto* challengeObject = EntityManager::Instance()->GetEntity(challengeObjectID); - - if (challengeObject != nullptr) { - for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { - script->OnHitOrHealResult(challengeObject, attacker, damage); - } - } -} diff --git a/dScripts/NtCombatChallengeExplodingDummy.cpp b/dScripts/NtCombatChallengeExplodingDummy.cpp deleted file mode 100644 index c117c63a..00000000 --- a/dScripts/NtCombatChallengeExplodingDummy.cpp +++ /dev/null @@ -1,32 +0,0 @@ -#include "NtCombatChallengeExplodingDummy.h" -#include "EntityManager.h" -#include "SkillComponent.h" - -void NtCombatChallengeExplodingDummy::OnDie(Entity* self, Entity* killer) { - const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); - - auto* challengeObject = EntityManager::Instance()->GetEntity(challengeObjectID); - - if (challengeObject != nullptr) { - for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { - script->OnDie(challengeObject, killer); - } - } -} - -void NtCombatChallengeExplodingDummy::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - const auto challengeObjectID = self->GetVar<LWOOBJID>(u"challengeObjectID"); - - auto* challengeObject = EntityManager::Instance()->GetEntity(challengeObjectID); - - if (challengeObject != nullptr) { - for (CppScripts::Script* script : CppScripts::GetEntityScripts(challengeObject)) { - script->OnHitOrHealResult(challengeObject, attacker, damage); - } - } - auto skillComponent = self->GetComponent<SkillComponent>(); - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(1338, 30875, attacker->GetObjectID()); - } - self->Kill(attacker); -} diff --git a/dScripts/NtCombatChallengeServer.cpp b/dScripts/NtCombatChallengeServer.cpp deleted file mode 100644 index 2b88ccf8..00000000 --- a/dScripts/NtCombatChallengeServer.cpp +++ /dev/null @@ -1,208 +0,0 @@ -#include "NtCombatChallengeServer.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "InventoryComponent.h" -#include "MissionComponent.h" - -void NtCombatChallengeServer::OnUse(Entity* self, Entity* user) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UI_Open", 0, 0, user->GetObjectID(), "", user->GetSystemAddress()); -} - -void NtCombatChallengeServer::OnDie(Entity* self, Entity* killer) { - if (killer != self && killer != nullptr) { - SpawnTargetDummy(self); - } -} - - -void NtCombatChallengeServer::OnHitOrHealResult(Entity* self, Entity* attacker, int32_t damage) { - const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); - - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - auto totalDmg = self->GetVar<int32_t>(u"totalDmg"); - - totalDmg += damage; - - self->SetVar(u"totalDmg", totalDmg); - self->SetNetworkVar(u"totalDmg", totalDmg); - - GameMessages::SendPlayNDAudioEmitter(self, attacker->GetSystemAddress(), scoreSound); -} - - -void NtCombatChallengeServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UI_Close", 0, 0, sender->GetObjectID(), "", sender->GetSystemAddress()); -} - - -void NtCombatChallengeServer::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - if (identifier == u"PlayButton" && button == 1) { - self->SetNetworkVar(u"bInUse", true); - - self->SetVar(u"playerID", sender->GetObjectID()); - - auto* inventoryComponent = sender->GetComponent<InventoryComponent>(); - - if (inventoryComponent != nullptr) { - inventoryComponent->RemoveItem(3039, 1); - } - - GameMessages::SendPlayNDAudioEmitter(self, sender->GetSystemAddress(), startSound); - - self->AddTimer("start_delay", 2.0f); - - GameMessages::SendShowActivityCountdown(self->GetObjectID(), false, false, u"", 0, sender->GetSystemAddress()); - - self->SetNetworkVar(u"toggle", true); - } else if (identifier == u"CloseButton") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UI_Close", 1, 0, sender->GetObjectID(), "", sender->GetSystemAddress()); - } -} - -void NtCombatChallengeServer::SpawnTargetDummy(Entity* self) { - const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); - - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - auto targetNumber = self->GetVar<int32_t>(u"TargetNumber"); - if (targetNumber == 0) targetNumber = 1; - - if (targetNumber > tTargets.size()) targetNumber = tTargets.size(); - - self->SetVar<int32_t>(u"TargetNumber", targetNumber + 1); - - const auto dummyLOT = tTargets[targetNumber - 1]; - - EntityInfo info{}; - info.lot = dummyLOT; - info.spawnerID = self->GetObjectID(); - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.settings = { new LDFData<std::string>(u"custom_script_server", "scripts\\02_server\\Map\\NT\\L_NT_COMBAT_CHALLENGE_DUMMY.lua") }; - - auto* dummy = EntityManager::Instance()->CreateEntity(info); - - dummy->SetVar(u"challengeObjectID", self->GetObjectID()); - - EntityManager::Instance()->ConstructEntity(dummy); - - self->SetVar(u"currentDummy", dummy->GetObjectID()); -} - -void NtCombatChallengeServer::SetAttackImmunity(LWOOBJID objID, bool bTurnOn) { - -} - -void NtCombatChallengeServer::OnChildLoaded(Entity* self, Entity* child) { - auto targetNumber = self->GetVar<int32_t>(u"TargetNumber"); - if (targetNumber == 0) targetNumber = 1; - self->SetVar(u"TargetNumber", targetNumber + 1); - - const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); - - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - child->SetRotation(NiQuaternion::LookAt(child->GetPosition(), player->GetPosition())); - - self->SetVar(u"currentTargetID", child->GetObjectID()); - - EntityManager::Instance()->SerializeEntity(child); - - child->GetGroups().push_back("targets_" + std::to_string(self->GetObjectID())); -} - -void NtCombatChallengeServer::ResetGame(Entity* self) { - const auto totalDmg = self->GetVar<int32_t>(u"totalDmg"); - const auto playerID = self->GetVar<LWOOBJID>(u"playerID"); - - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player != nullptr) { - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - for (const auto& mission : tMissions) { - if (totalDmg >= mission.damage) { - missionComponent->ForceProgressTaskType(mission.mission, 1, 1); - } - } - } - } - - self->SetVar(u"TargetNumber", 1); - self->SetVar(u"playerID", LWOOBJID_EMPTY); - self->SetVar(u"totalDmg", 0); - self->SetNetworkVar(u"totalDmg", false); - self->SetNetworkVar(u"update_time", 0); - - const auto& targetObjs = EntityManager::Instance()->GetEntitiesInGroup("targets_" + std::to_string(self->GetObjectID())); - - for (auto* target : targetObjs) { - target->Smash(self->GetObjectID()); - } - - const auto currentID = self->GetVar<LWOOBJID>(u"currentDummy"); - - auto* current = EntityManager::Instance()->GetEntity(currentID); - - if (current != nullptr) { - current->Smash(self->GetObjectID()); - } -} - -void NtCombatChallengeServer::OnActivityTimerUpdate(Entity* self, float timeRemaining) { - self->SetNetworkVar(u"update_time", std::ceil(timeRemaining)); - - if (timeRemaining <= 3) { - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, timerLowSound); - } else { - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, timerSound); - } -} - -void NtCombatChallengeServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "start_delay") { - self->SetVar(u"game_tick", gameTime); - - SpawnTargetDummy(self); - - self->AddTimer("game_tick", 1); - - self->SetNetworkVar(u"totalTime", gameTime); - } else if (timerName == "game_tick") { - auto gameTick = self->GetVar<float>(u"game_tick"); - - gameTick -= 1; - - self->SetVar(u"game_tick", gameTick); - - if (gameTick <= 0) { - ResetGame(self); - - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, stopSound); - - self->AddTimer("reset_tick", 5); - } else { - self->AddTimer("game_tick", 1); - - OnActivityTimerUpdate(self, gameTick); - } - } else if (timerName == "reset_tick") { - self->SetNetworkVar(u"toggle", false); - self->SetNetworkVar(u"bInUse", false); - } -} diff --git a/dScripts/NtConsoleTeleportServer.cpp b/dScripts/NtConsoleTeleportServer.cpp deleted file mode 100644 index 324a2fc0..00000000 --- a/dScripts/NtConsoleTeleportServer.cpp +++ /dev/null @@ -1,28 +0,0 @@ -#include "NtConsoleTeleportServer.h" -#include "Entity.h" -#include "AMFFormat.h" - -void NtConsoleTeleportServer::OnStartup(Entity* self) { - self->SetVar(u"teleportAnim", m_TeleportAnim); - self->SetVar(u"teleportString", m_TeleportString); -} - -void NtConsoleTeleportServer::OnUse(Entity* self, Entity* user) { - BaseOnUse(self, user); -} - -void NtConsoleTeleportServer::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { - BaseOnMessageBoxResponse(self, sender, button, identifier, userData); -} - -void NtConsoleTeleportServer::OnChoiceBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& buttonIdentifier, const std::u16string& identifier) { - -} - -void NtConsoleTeleportServer::OnTimerDone(Entity* self, std::string timerName) { - BaseOnTimerDone(self, timerName); -} - -void NtConsoleTeleportServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - BaseOnFireEventServerSide(self, sender, args, param1, param2, param3); -} diff --git a/dScripts/NtDirtCloudServer.cpp b/dScripts/NtDirtCloudServer.cpp deleted file mode 100644 index a9a70c5c..00000000 --- a/dScripts/NtDirtCloudServer.cpp +++ /dev/null @@ -1,46 +0,0 @@ -#include "NtDirtCloudServer.h" -#include "MissionComponent.h" - -std::map<std::string, std::vector<int32_t>> NtDirtCloudServer::m_Missions = -{ - {"Dirt_Clouds_Sent", {1333,1253}}, - {"Dirt_Clouds_Assem", {1333,1276}}, - {"Dirt_Clouds_Para", {1333,1277}}, - {"Dirt_Clouds_Halls", {1333,1283}} -}; - -void NtDirtCloudServer::OnStartup(Entity* self) { - self->SetVar(u"CloudOn", true); -} - -void NtDirtCloudServer::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (message != "soapspray") { - return; - } - - if (!self->GetVar<bool>(u"CloudOn")) { - return; - } - - const auto mySpawner = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); - - if (m_Missions.count(mySpawner) == 0) { - return; - } - - const auto& myMis = m_Missions[mySpawner]; - - auto* missionComponent = caster->GetComponent<MissionComponent>(); - - if (missionComponent == nullptr) { - return; - } - - for (const auto missionID : myMis) { - missionComponent->ForceProgressTaskType(missionID, 1, 1); - } - - self->SetVar(u"CloudOn", false); - - self->Smash(self->GetObjectID(), VIOLENT); -} diff --git a/dScripts/NtDukeServer.cpp b/dScripts/NtDukeServer.cpp deleted file mode 100644 index 327d3290..00000000 --- a/dScripts/NtDukeServer.cpp +++ /dev/null @@ -1,38 +0,0 @@ -#include "NtDukeServer.h" -#include "InventoryComponent.h" -#include "MissionComponent.h" - -void NtDukeServer::SetVariables(Entity* self) { - self->SetVar<float_t>(m_SpyProximityVariable, 35.0f); - - self->SetVar<SpyData>(m_SpyDataVariable, { - NT_FACTION_SPY_DUKE, 13548, 1319 - }); - - self->SetVar<std::vector<SpyDialogue>>(m_SpyDialogueTableVariable, { - { "DUKE_NT_CONVO_1", 0 }, - { "DUKE_NT_CONVO_2", 0 }, - { "DUKE_NT_CONVO_3", 0 }, - }); - - // If there's an alternating conversation, indices should be provided using the conversationID variables - self->SetVar<std::vector<LWOOBJID>>(m_SpyCinematicObjectsVariable, { self->GetObjectID() }); -} - -void NtDukeServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - - // Handles adding and removing the sword for the Crux Prime Sword mission - auto* missionComponent = target->GetComponent<MissionComponent>(); - auto* inventoryComponent = target->GetComponent<InventoryComponent>(); - - if (missionComponent != nullptr && inventoryComponent != nullptr) { - auto state = missionComponent->GetMissionState(m_SwordMissionID); - auto lotCount = inventoryComponent->GetLotCount(m_SwordLot); - - if ((state == MissionState::MISSION_STATE_AVAILABLE || state == MissionState::MISSION_STATE_ACTIVE) && lotCount < 1) { - inventoryComponent->AddItem(m_SwordLot, 1, eLootSourceType::LOOT_SOURCE_NONE); - } else if (state == MissionState::MISSION_STATE_READY_TO_COMPLETE) { - inventoryComponent->RemoveItem(m_SwordLot, lotCount); - } - } -} diff --git a/dScripts/NtDukeServer.h b/dScripts/NtDukeServer.h deleted file mode 100644 index 0878e86c..00000000 --- a/dScripts/NtDukeServer.h +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once -#include "NtFactionSpyServer.h" - -class NtDukeServer : public NtFactionSpyServer { - void SetVariables(Entity* self) override; - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - const uint32_t m_SwordMissionID = 1448; - const LOT m_SwordLot = 13777; -}; diff --git a/dScripts/NtFactionSpyServer.cpp b/dScripts/NtFactionSpyServer.cpp index ac068a32..a1161880 100644 --- a/dScripts/NtFactionSpyServer.cpp +++ b/dScripts/NtFactionSpyServer.cpp @@ -4,6 +4,10 @@ #include "InventoryComponent.h" #include "GameMessages.h" #include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" +#include "eCinematicEvent.h" +#include "ePlayerFlag.h" void NtFactionSpyServer::OnStartup(Entity* self) { SetVariables(self); @@ -12,7 +16,7 @@ void NtFactionSpyServer::OnStartup(Entity* self) { auto* proximityMonitor = self->GetComponent<ProximityMonitorComponent>(); if (proximityMonitor == nullptr) { proximityMonitor = new ProximityMonitorComponent(self, -1, -1); - self->AddComponent(COMPONENT_TYPE_PROXIMITY_MONITOR, proximityMonitor); + self->AddComponent(eReplicaComponentType::PROXIMITY_MONITOR, proximityMonitor); } proximityMonitor->SetProximityRadius(self->GetVar<float_t>(m_SpyProximityVariable), m_ProximityName); @@ -54,7 +58,7 @@ bool NtFactionSpyServer::IsSpy(Entity* self, Entity* possibleSpy) { auto* character = possibleSpy->GetCharacter(); // A player is a spy if they have the spy mission, have the spy equipment equipped and don't have the spy flag set yet - return missionComponent != nullptr && missionComponent->GetMissionState(spyData.missionID) == MissionState::MISSION_STATE_ACTIVE + return missionComponent != nullptr && missionComponent->GetMissionState(spyData.missionID) == eMissionState::ACTIVE && inventoryComponent != nullptr && inventoryComponent->IsEquipped(spyData.itemID) && character != nullptr && !character->GetPlayerFlag(spyData.flagID); } @@ -75,14 +79,14 @@ void NtFactionSpyServer::OnCinematicUpdate(Entity* self, Entity* sender, eCinema // Make sure we're listening to the root we're interested in if (pathRoot == cinematicRoot) { - if (event == STARTED && pathIndex >= 0 && pathIndex < dialogueTable.size()) { + if (event == eCinematicEvent::STARTED && pathIndex >= 0 && pathIndex < dialogueTable.size()) { // If the cinematic started, show part of the conversation GameMessages::SendNotifyClientObject(self->GetObjectID(), m_SpyDialogueNotification, 0, 0, ParamObjectForConversationID(self, dialogueTable.at(pathIndex).conversationID), dialogueTable.at(pathIndex).token, sender->GetSystemAddress()); - } else if (event == ENDED && pathIndex >= dialogueTable.size() - 1) { + } else if (event == eCinematicEvent::ENDED && pathIndex >= dialogueTable.size() - 1) { auto spyData = self->GetVar<SpyData>(m_SpyDataVariable); auto* character = sender->GetCharacter(); if (character != nullptr) { diff --git a/dScripts/NtFactionSpyServer.h b/dScripts/NtFactionSpyServer.h index 67955dd4..4406859c 100644 --- a/dScripts/NtFactionSpyServer.h +++ b/dScripts/NtFactionSpyServer.h @@ -7,7 +7,7 @@ struct SpyDialogue { }; struct SpyData { - uint32_t flagID; + int32_t flagID; LOT itemID; uint32_t missionID; }; diff --git a/dScripts/NtHaelServer.cpp b/dScripts/NtHaelServer.cpp deleted file mode 100644 index fbad421f..00000000 --- a/dScripts/NtHaelServer.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "NtHaelServer.h" -#include "Entity.h" - -void NtHaelServer::SetVariables(Entity* self) { - self->SetVar<float_t>(m_SpyProximityVariable, 25.0f); - - self->SetVar<SpyData>(m_SpyDataVariable, { - NT_FACTION_SPY_HAEL, 13892, 1321 - }); - - self->SetVar<std::vector<SpyDialogue>>(m_SpyDialogueTableVariable, { - { "HAEL_NT_CONVO_1", 0 }, - { "HAEL_NT_CONVO_2", 0 }, - { "HAEL_NT_CONVO_3", 0 }, - { "HAEL_NT_CONVO_4", 0 }, - }); - - // If there's an alternating conversation, indices should be provided using the conversationID variables - self->SetVar<std::vector<LWOOBJID>>(m_SpyCinematicObjectsVariable, { self->GetObjectID() }); -} diff --git a/dScripts/NtImagBeamBuffer.cpp b/dScripts/NtImagBeamBuffer.cpp deleted file mode 100644 index d98a7403..00000000 --- a/dScripts/NtImagBeamBuffer.cpp +++ /dev/null @@ -1,53 +0,0 @@ -#include "NtImagBeamBuffer.h" -#include "EntityManager.h" -#include "SkillComponent.h" - -void NtImagBeamBuffer::OnStartup(Entity* self) { - self->SetProximityRadius(100, "ImagZone"); - - self->AddTimer("BuffImag", 2.0f); -} - -void NtImagBeamBuffer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name != "ImagZone" || !entering->IsPlayer()) { - return; - } - - if (status == "ENTER") { - const auto& iter = std::find(m_EntitiesInProximity.begin(), m_EntitiesInProximity.end(), entering->GetObjectID()); - - if (iter == m_EntitiesInProximity.end()) { - m_EntitiesInProximity.push_back(entering->GetObjectID()); - } - } else if (status == "LEAVE") { - const auto& iter = std::find(m_EntitiesInProximity.begin(), m_EntitiesInProximity.end(), entering->GetObjectID()); - - if (iter != m_EntitiesInProximity.end()) { - m_EntitiesInProximity.erase(iter); - } - } -} - -void NtImagBeamBuffer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName != "BuffImag") { - return; - } - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - for (const auto entityID : m_EntitiesInProximity) { - auto* entity = EntityManager::Instance()->GetEntity(entityID); - - if (entity == nullptr) { - continue; - } - - skillComponent->CalculateBehavior(1311, 30235, entityID, true); - } - - self->AddTimer("BuffImag", 2.0f); -} diff --git a/dScripts/NtImagimeterVisibility.cpp b/dScripts/NtImagimeterVisibility.cpp deleted file mode 100644 index c0f9bf51..00000000 --- a/dScripts/NtImagimeterVisibility.cpp +++ /dev/null @@ -1,11 +0,0 @@ -#include "NtImagimeterVisibility.h" -#include "GameMessages.h" -#include "Entity.h" -#include "Character.h" - -void NTImagimeterVisibility::OnRebuildComplete(Entity* self, Entity* target) { - auto* character = target->GetCharacter(); - if (character) character->SetPlayerFlag(ePlayerFlags::NT_PLINTH_REBUILD, true); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlinthBuilt", 0, 0, LWOOBJID_EMPTY, "", target->GetSystemAddress()); -} diff --git a/dScripts/NtOverbuildServer.cpp b/dScripts/NtOverbuildServer.cpp deleted file mode 100644 index 8411e55b..00000000 --- a/dScripts/NtOverbuildServer.cpp +++ /dev/null @@ -1,30 +0,0 @@ -#include "NtOverbuildServer.h" -#include "EntityManager.h" - -void NtOverbuildServer::SetVariables(Entity* self) { - self->SetVar<float_t>(m_SpyProximityVariable, 30.0f); - - self->SetVar<SpyData>(m_SpyDataVariable, { - NT_FACTION_SPY_OVERBUILD, 13891, 1320 - }); - - self->SetVar<std::vector<SpyDialogue>>(m_SpyDialogueTableVariable, { - { "OVERBUILD_NT_CONVO_1", 0 }, - { "OVERBUILD_NT_CONVO_2", 1 }, - { "OVERBUILD_NT_CONVO_3", 0 }, - { "OVERBUILD_NT_CONVO_4", 1 }, - { "OVERBUILD_NT_CONVO_5", 0 }, - { "OVERBUILD_NT_CONVO_6", 1 }, - { "OVERBUILD_NT_CONVO_7", 0 }, - }); - - // Find the second object Dr. Overbuild interacts with - LWOOBJID otherConvoObjectID = LWOOBJID_EMPTY; - for (auto* otherConvoObject : EntityManager::Instance()->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(m_OtherEntitiesGroupVariable)))) { - otherConvoObjectID = otherConvoObject->GetObjectID(); - break; - } - - // If there's an alternating conversation, indices should be provided using the conversationID variables - self->SetVar<std::vector<LWOOBJID>>(m_SpyCinematicObjectsVariable, { self->GetObjectID(), otherConvoObjectID }); -} diff --git a/dScripts/NtParadoxPanelServer.cpp b/dScripts/NtParadoxPanelServer.cpp deleted file mode 100644 index 06482c82..00000000 --- a/dScripts/NtParadoxPanelServer.cpp +++ /dev/null @@ -1,66 +0,0 @@ -#include "NtParadoxPanelServer.h" -#include "GameMessages.h" -#include "MissionComponent.h" -#include "EntityManager.h" -#include "Character.h" - -void NtParadoxPanelServer::OnUse(Entity* self, Entity* user) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"bActive", 1, 0, user->GetObjectID(), "", user->GetSystemAddress()); - - GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); - - self->SetVar(u"bActive", true); - - auto* missionComponent = user->GetComponent<MissionComponent>(); - - const auto playerID = user->GetObjectID(); - - for (const auto mission : tPlayerOnMissions) { - if (missionComponent->GetMissionState(mission) != MissionState::MISSION_STATE_ACTIVE) { - continue; - } - - self->AddCallbackTimer(2, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - const auto flag = self->GetVar<int32_t>(u"flag"); - - player->GetCharacter()->SetPlayerFlag(flag, true); - - GameMessages::SendPlayAnimation(player, u"rebuild-celebrate"); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SparkStop", 0, 0, player->GetObjectID(), "", player->GetSystemAddress()); - GameMessages::SendSetStunned(player->GetObjectID(), eStunState::POP, player->GetSystemAddress(), LWOOBJID_EMPTY, false, false, true, false, true, true, false, false, true); - self->SetVar(u"bActive", false); - }); - GameMessages::SendPlayAnimation(user, u"nexus-powerpanel", 6.0f); - GameMessages::SendSetStunned(user->GetObjectID(), eStunState::PUSH, user->GetSystemAddress(), LWOOBJID_EMPTY, false, false, true, false, true, true, false, false, true); - return; - } - - GameMessages::SendPlayAnimation(user, shockAnim); - - const auto dir = self->GetRotation().GetRightVector(); - - GameMessages::SendKnockback(user->GetObjectID(), self->GetObjectID(), self->GetObjectID(), 0, { dir.x * 15, 5, dir.z * 15 }); - - GameMessages::SendPlayFXEffect(self, 6432, u"create", "console_sparks", LWOOBJID_EMPTY, 1.0, 1.0, true); - - self->AddCallbackTimer(2, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"bActive", 0, 0, player->GetObjectID(), "", player->GetSystemAddress()); - - GameMessages::SendStopFXEffect(self, true, "console_sparks"); - - self->SetVar(u"bActive", false); - }); -} diff --git a/dScripts/NtParadoxTeleServer.cpp b/dScripts/NtParadoxTeleServer.cpp deleted file mode 100644 index 8003e93f..00000000 --- a/dScripts/NtParadoxTeleServer.cpp +++ /dev/null @@ -1,113 +0,0 @@ -#include "NtParadoxTeleServer.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void NtParadoxTeleServer::OnStartup(Entity* self) { - self->SetProximityRadius(5, "teleport"); -} - -void NtParadoxTeleServer::OnPlayerLoaded(Entity* self, Entity* player) { - -} - -void NtParadoxTeleServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (status != "ENTER" || !entering->IsPlayer() || name != "teleport") return; - - auto* player = entering; - const auto playerID = player->GetObjectID(); - - const auto iter = m_TeleportingPlayerTable.find(playerID); - if (iter == m_TeleportingPlayerTable.end()) m_TeleportingPlayerTable[playerID] = false; - const auto bPlayerBeingTeleported = m_TeleportingPlayerTable[playerID]; - - if (player->IsPlayer() && !bPlayerBeingTeleported) { - GameMessages::SendSetStunned(playerID, PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true - ); - - GameMessages::SendPlayAnimation(player, u"teledeath", 4.0f); - - const auto animTime = 2; - - self->AddCallbackTimer(animTime, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - TeleportPlayer(self, player); - }); - } - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - } -} - -void NtParadoxTeleServer::TeleportPlayer(Entity* self, Entity* player) { - auto destinationGroup = self->GetVar<std::u16string>(u"teleGroup"); - auto* destination = self; - - if (!destinationGroup.empty()) { - const auto& groupObjs = EntityManager::Instance()->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(destinationGroup)); - - if (!groupObjs.empty()) { - destination = groupObjs[0]; - } - } - - const auto destPosition = destination->GetPosition(); - const auto destRotation = destination->GetRotation(); - - auto teleCinematic = self->GetVar<std::u16string>(u"Cinematic"); - - if (!teleCinematic.empty()) { - const auto teleCinematicUname = teleCinematic; - GameMessages::SendPlayCinematic(player->GetObjectID(), teleCinematicUname, player->GetSystemAddress()); - } - - GameMessages::SendTeleport(player->GetObjectID(), destPosition, destRotation, player->GetSystemAddress(), true); - - GameMessages::SendPlayAnimation(player, u"paradox-teleport-in", 4.0f); - - const auto animTime = 2; - - const auto playerID = player->GetObjectID(); - - self->AddCallbackTimer(animTime, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - UnlockPlayer(self, player); - }); - - const auto useSound = self->GetVar<std::string>(u"sound1"); - - if (!useSound.empty()) { - GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), useSound); - } -} - -void NtParadoxTeleServer::UnlockPlayer(Entity* self, Entity* player) { - const auto playerID = player->GetObjectID(); - - m_TeleportingPlayerTable[playerID] = false; - - GameMessages::SendSetStunned(playerID, POP, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true - ); - - auto teleCinematic = self->GetVar<std::u16string>(u"Cinematic"); - - if (!teleCinematic.empty()) { - const auto teleCinematicUname = teleCinematic; - GameMessages::SendEndCinematic(player->GetObjectID(), teleCinematicUname, player->GetSystemAddress()); - } -} diff --git a/dScripts/NtSentinelWalkwayServer.cpp b/dScripts/NtSentinelWalkwayServer.cpp deleted file mode 100644 index 7b943172..00000000 --- a/dScripts/NtSentinelWalkwayServer.cpp +++ /dev/null @@ -1,43 +0,0 @@ -#include "NtSentinelWalkwayServer.h" -#include "PhantomPhysicsComponent.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void NtSentinelWalkwayServer::OnStartup(Entity* self) { - auto* phantomPhysicsComponent = self->GetComponent<PhantomPhysicsComponent>(); - - if (phantomPhysicsComponent == nullptr) { - return; - } - - auto force = self->GetVar<int32_t>(u"force"); - - if (force == 0) { - force = 115; - } - - const auto forward = self->GetRotation().GetRightVector() * -1; - - phantomPhysicsComponent->SetEffectType(0); // PUSH - phantomPhysicsComponent->SetDirectionalMultiplier(force); - phantomPhysicsComponent->SetDirection(forward); - phantomPhysicsComponent->SetPhysicsEffectActive(true); - - EntityManager::Instance()->SerializeEntity(self); - - self->SetProximityRadius(3, "speedboost"); -} - -void NtSentinelWalkwayServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name != "speedboost" || !entering->IsPlayer() || status != "ENTER") { - return; - } - - auto* player = entering; - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - } -} diff --git a/dScripts/NtSleepingGuard.cpp b/dScripts/NtSleepingGuard.cpp deleted file mode 100644 index 145df6c8..00000000 --- a/dScripts/NtSleepingGuard.cpp +++ /dev/null @@ -1,35 +0,0 @@ -#include "NtSleepingGuard.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void NtSleepingGuard::OnStartup(Entity* self) { - self->SetNetworkVar<bool>(u"asleep", true); -} - -void NtSleepingGuard::OnEmoteReceived(Entity* self, const int32_t emote, Entity* target) { - if (!self->GetNetworkVar<bool>(u"asleep")) - return; - - // Check if emote is in m_ValidEmotes - if (std::find(m_ValidEmotes.begin(), m_ValidEmotes.end(), emote) == m_ValidEmotes.end()) - return; - - // Set asleep to false - self->SetNetworkVar<bool>(u"asleep", false); - - GameMessages::SendPlayAnimation(self, u"greet"); - - auto* missionComponent = target->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->CompleteMission(1346); - } - - self->AddTimer("AsleepAgain", 5.0f); -} - -void NtSleepingGuard::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "AsleepAgain") { - self->SetNetworkVar<bool>(u"asleep", true); - } -} diff --git a/dScripts/NtVandaServer.cpp b/dScripts/NtVandaServer.cpp deleted file mode 100644 index bfc35203..00000000 --- a/dScripts/NtVandaServer.cpp +++ /dev/null @@ -1,13 +0,0 @@ -#include "NtVandaServer.h" -#include "InventoryComponent.h" - -void NtVandaServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - - // Removes the alien parts after completing the mission - if (missionID == m_AlienPartMissionID && missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE) { - auto* inventoryComponent = target->GetComponent<InventoryComponent>(); - for (const auto& alienPartLot : m_AlienPartLots) { - inventoryComponent->RemoveItem(alienPartLot, 1); - } - } -} diff --git a/dScripts/NtVandaServer.h b/dScripts/NtVandaServer.h deleted file mode 100644 index 69e868f5..00000000 --- a/dScripts/NtVandaServer.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class NtVandaServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - const uint32_t m_AlienPartMissionID = 1183; - const std::vector<LOT> m_AlienPartLots = { 12479, 12480, 12481 }; -}; diff --git a/dScripts/NtVentureCannonServer.cpp b/dScripts/NtVentureCannonServer.cpp deleted file mode 100644 index d7f4cb99..00000000 --- a/dScripts/NtVentureCannonServer.cpp +++ /dev/null @@ -1,122 +0,0 @@ -#include "NtVentureCannonServer.h" -#include "GameMessages.h" -#include "EntityManager.h" - -void NtVentureCannonServer::OnUse(Entity* self, Entity* user) { - auto* player = user; - const auto playerID = player->GetObjectID(); - - auto enterCinematic = self->GetVar<std::u16string>(u"EnterCinematic"); - - if (enterCinematic.empty()) { - return; - } - - self->SetNetworkVar(u"bIsInUse", true); - - GameMessages::SendSetStunned(playerID, PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true - ); - - auto destPosition = self->GetPosition(); - - destPosition.y += 5 - 1.57f; - - auto destRotation = self->GetRotation(); - - GameMessages::SendTeleport(playerID, destPosition, destRotation, player->GetSystemAddress(), true); - - GameMessages::SendPlayAnimation(player, u"scale-down", 4.0f); - - const auto enterCinematicUname = enterCinematic; - GameMessages::SendPlayCinematic(player->GetObjectID(), enterCinematicUname, player->GetSystemAddress()); - - GameMessages::SendPlayNDAudioEmitter(player, player->GetSystemAddress(), "{e8bf79ce-7453-4a7d-b872-fee65e97ff15}"); - - self->AddCallbackTimer(3, [this, self]() { - self->SetNetworkVar(u"bIsInUse", false); - }); - - self->AddCallbackTimer(1.5f, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - EnterCannonEnded(self, player); - }); -} - -void NtVentureCannonServer::EnterCannonEnded(Entity* self, Entity* player) { - const auto playerID = player->GetObjectID(); - - const auto& cannonEffectGroup = EntityManager::Instance()->GetEntitiesInGroup("cannonEffect"); - - if (!cannonEffectGroup.empty()) { - auto* cannonEffect = cannonEffectGroup[0]; - - GameMessages::SendPlayFXEffect(cannonEffect, 6036, u"create", "cannon_blast", LWOOBJID_EMPTY, 1, 1, true); - - GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(cannonEffect, u"camshake-bridge", cannonEffect->GetObjectID(), 100); - } - - FirePlayer(self, player); - - auto exitCinematic = self->GetVar<std::u16string>(u"ExitCinematic"); - - if (exitCinematic.empty()) { - UnlockCannonPlayer(self, player); - - return; - } - - const auto exitCinematicUname = exitCinematic; - GameMessages::SendPlayCinematic(player->GetObjectID(), exitCinematicUname, player->GetSystemAddress(), - true, true, true, false, 0, false, 0, false, false - ); - - self->AddCallbackTimer(1.5f, [this, self, playerID]() { - auto* player = EntityManager::Instance()->GetEntity(playerID); - - if (player == nullptr) { - return; - } - - ExitCannonEnded(self, player); - }); -} - -void NtVentureCannonServer::ExitCannonEnded(Entity* self, Entity* player) { - UnlockCannonPlayer(self, player); -} - -void NtVentureCannonServer::UnlockCannonPlayer(Entity* self, Entity* player) { - GameMessages::SendSetStunned(player->GetObjectID(), POP, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true - ); - - self->SetNetworkVar(u"bIsInUse", false); - - GameMessages::SendTerminateInteraction(player->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} - -void NtVentureCannonServer::FirePlayer(Entity* self, Entity* player) { - auto destinationGroup = self->GetVar<std::u16string>(u"teleGroup"); - auto* destination = self; - - if (!destinationGroup.empty()) { - const auto& groupObjs = EntityManager::Instance()->GetEntitiesInGroup(GeneralUtils::UTF16ToWTF8(destinationGroup)); - - if (!groupObjs.empty()) { - destination = groupObjs[0]; - } - } - - const auto destPosition = destination->GetPosition(); - const auto destRotation = destination->GetRotation(); - - GameMessages::SendTeleport(player->GetObjectID(), destPosition, destRotation, player->GetSystemAddress(), true); - - GameMessages::SendPlayAnimation(player, u"venture-cannon-out", 4.0f); -} diff --git a/dScripts/NtVentureSpeedPadServer.cpp b/dScripts/NtVentureSpeedPadServer.cpp deleted file mode 100644 index 0995d1df..00000000 --- a/dScripts/NtVentureSpeedPadServer.cpp +++ /dev/null @@ -1,28 +0,0 @@ -#include "NtVentureSpeedPadServer.h" -#include "SkillComponent.h" -#include "MissionComponent.h" - -void NtVentureSpeedPadServer::OnStartup(Entity* self) { - self->SetProximityRadius(3, "speedboost"); -} - - -void NtVentureSpeedPadServer::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name != "speedboost" || !entering->IsPlayer() || status != "ENTER") { - return; - } - - auto* player = entering; - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SCRIPT, self->GetLOT()); - } - - auto* skillComponent = player->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - skillComponent->CalculateBehavior(927, 18913, player->GetObjectID(), true); - } -} diff --git a/dScripts/PersonalFortress.cpp b/dScripts/PersonalFortress.cpp deleted file mode 100644 index e7e89e80..00000000 --- a/dScripts/PersonalFortress.cpp +++ /dev/null @@ -1,50 +0,0 @@ -#include "PersonalFortress.h" -#include "GameMessages.h" -#include "SkillComponent.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" - -void PersonalFortress::OnStartup(Entity* self) { - auto* owner = self->GetOwner(); - self->AddTimer("FireSkill", 1.5); - GameMessages::SendSetStunned(owner->GetObjectID(), PUSH, owner->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true, true, true - ); - - auto* destroyableComponent = owner->GetComponent<DestroyableComponent>(); - - if (destroyableComponent != nullptr) { - destroyableComponent->PushImmunity(); - } - - EntityManager::Instance()->SerializeEntity(owner); -} - -void PersonalFortress::OnDie(Entity* self, Entity* killer) { - auto* owner = self->GetOwner(); - GameMessages::SendSetStunned(owner->GetObjectID(), POP, owner->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true, true, true - ); - - auto* destroyableComponent = owner->GetComponent<DestroyableComponent>(); - - if (destroyableComponent != nullptr) { - destroyableComponent->PopImmunity(); - } - - EntityManager::Instance()->SerializeEntity(owner); -} - -void PersonalFortress::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "FireSkill") { - auto* owner = self->GetOwner(); - - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - skillComponent->CalculateBehavior(650, 13364, LWOOBJID_EMPTY, true, false); - } -} diff --git a/dScripts/PetDigBuild.cpp b/dScripts/PetDigBuild.cpp deleted file mode 100644 index ae159575..00000000 --- a/dScripts/PetDigBuild.cpp +++ /dev/null @@ -1,49 +0,0 @@ -#include "PetDigBuild.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void PetDigBuild::OnRebuildComplete(Entity* self, Entity* target) { - auto flagNumber = self->GetVar<std::u16string>(u"flagNum"); - - EntityInfo info{}; - auto pos = self->GetPosition(); - pos.SetY(pos.GetY() + 0.5f); - info.pos = pos; - info.rot = self->GetRotation(); - info.spawnerID = self->GetSpawnerID(); - info.settings = { - new LDFData<LWOOBJID>(u"builder", target->GetObjectID()), - new LDFData<LWOOBJID>(u"X", self->GetObjectID()) - }; - - if (!flagNumber.empty()) { - info.lot = 7410; // Normal GF treasure - info.settings.push_back(new LDFData<std::u16string>(u"groupID", u"Flag" + flagNumber)); - } else { - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (missionComponent != nullptr && missionComponent->GetMissionState(746) == MissionState::MISSION_STATE_ACTIVE) { - info.lot = 9307; // Special Captain Jack treasure that drops a mission item - } else { - info.lot = 3495; // Normal AG treasure - } - } - - auto* treasure = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(treasure); - self->SetVar<LWOOBJID>(u"chestObj", treasure->GetObjectID()); -} - -void PetDigBuild::OnDie(Entity* self, Entity* killer) { - auto treasureID = self->GetVar<LWOOBJID>(u"chestObj"); - if (treasureID == LWOOBJID_EMPTY) - return; - - auto treasure = EntityManager::Instance()->GetEntity(treasureID); - if (treasure == nullptr) - return; - - // If the quick build expired and the treasure was not collected, hide the treasure - if (!treasure->GetIsDead()) { - treasure->Smash(self->GetObjectID(), SILENT); - } -} diff --git a/dScripts/PetDigServer.cpp b/dScripts/PetDigServer.cpp deleted file mode 100644 index e26b079a..00000000 --- a/dScripts/PetDigServer.cpp +++ /dev/null @@ -1,234 +0,0 @@ -#include "dZoneManager.h" -#include "PetDigServer.h" -#include "MissionComponent.h" -#include "EntityManager.h" -#include "Character.h" -#include "PetComponent.h" - -std::vector<LWOOBJID> PetDigServer::treasures{}; - -const DigInfo PetDigServer::defaultDigInfo = DigInfo{ 3495, -1, -1, false, false, false, false }; - -/** - * Summary of all the special treasure behaviors, indexed by their lot - */ -const std::map<LOT, DigInfo> PetDigServer::digInfoMap{ - // Regular treasures - {3495, defaultDigInfo}, - - // Pet cove treasure - {7612, DigInfo { 7612, -1, -1, false, false, false, false }}, - - // Gnarled Forest flag treasure - {7410, DigInfo { 7410, -1, -1, false, true, false, false }}, - - // Gnarled Forest crab treasure - {9308, DigInfo { 9308, 7694, -1, false, false, false, false }}, - - // Avant Gardens mission treasure - {9307, DigInfo { 9307, -1, -1, false, true, false, true }}, - - // Avant Gardens bouncer treasure - {7559, DigInfo { 7559, -1, -1, false, false, true, false }}, - - // Crux Prime dragon treasure - {13098, DigInfo { 13098, 13067, 1298, false, false, false, false }}, - - // Bone treasure (can only be digged using the dragon) - {12192, DigInfo { 12192, -1, -1, true, false, false, false }}, -}; - -void PetDigServer::OnStartup(Entity* self) { - treasures.push_back(self->GetObjectID()); - const auto digInfoIterator = digInfoMap.find(self->GetLOT()); - const auto digInfo = digInfoIterator != digInfoMap.end() ? digInfoIterator->second : defaultDigInfo; - - // Reset any bouncers that might've been created by the previous dig - if (digInfo.bouncer) { - auto bounceNumber = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"BouncerNumber")); - auto bouncerSpawners = dZoneManager::Instance()->GetSpawnersByName("PetBouncer" + bounceNumber); - auto switchSpawners = dZoneManager::Instance()->GetSpawnersByName("PetBouncerSwitch" + bounceNumber); - - for (auto* bouncerSpawner : bouncerSpawners) { - for (auto* bouncer : bouncerSpawner->m_Info.nodes) - bouncerSpawner->Deactivate(); - bouncerSpawner->Reset(); - } - - for (auto* switchSpawner : switchSpawners) { - switchSpawner->Deactivate(); - switchSpawner->Reset(); - } - } -} - -void PetDigServer::OnDie(Entity* self, Entity* killer) { - const auto iterator = std::find(treasures.begin(), treasures.end(), self->GetObjectID()); - if (iterator != treasures.end()) { - treasures.erase(iterator); - } - - auto* owner = killer->GetOwner(); - const auto digInfoIterator = digInfoMap.find(self->GetLOT()); - const auto digInfo = digInfoIterator != digInfoMap.end() ? digInfoIterator->second : defaultDigInfo; - - if (digInfo.spawnLot >= 0) { - PetDigServer::SpawnPet(self, owner, digInfo); - } else if (digInfo.builderOnly) { - - // Some treasures may only be retrieved by the player that built the diggable - auto builder = self->GetVar<LWOOBJID>(u"builder"); // Set by the pet dig build script - if (builder != owner->GetObjectID()) - return; - } else if (digInfo.xBuild) { - PetDigServer::HandleXBuildDig(self, owner, killer); - return; - } else if (digInfo.bouncer) { - PetDigServer::HandleBouncerDig(self, owner); - } - - PetDigServer::ProgressPetDigMissions(owner, self); - - self->SetNetworkVar<bool>(u"treasure_dug", true); - // TODO: Reset other pets - - // Handles smashing leftovers (edge case for the AG X) - auto* xObject = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"X")); - if (xObject != nullptr) { - xObject->Smash(xObject->GetObjectID(), VIOLENT); - } -} - -void PetDigServer::HandleXBuildDig(const Entity* self, Entity* owner, Entity* pet) { - auto playerID = self->GetVar<LWOOBJID>(u"builder"); - if (playerID == LWOOBJID_EMPTY || playerID != owner->GetObjectID()) - return; - - auto* playerEntity = EntityManager::Instance()->GetEntity(playerID); - if (!playerEntity || !playerEntity->GetParentUser() || !playerEntity->GetParentUser()->GetLastUsedChar()) - return; - - auto* player = playerEntity->GetCharacter(); - const auto groupID = self->GetVar<std::u16string>(u"groupID"); - auto playerFlag = 0; - - // The flag that the player dug up - if (groupID == u"Flag1") { - playerFlag = 61; - } else if (groupID == u"Flag2") { - playerFlag = 62; - } else if (groupID == u"Flag3") { - playerFlag = 63; - } - - // If the player doesn't have the flag yet - if (playerFlag != 0 && !player->GetPlayerFlag(playerFlag)) { - auto* petComponent = pet->GetComponent<PetComponent>(); - if (petComponent != nullptr) { - // TODO: Pet state = 9 ?? - } - - // Shows the flag object to the player - player->SetPlayerFlag(playerFlag, true); - } - - auto* xObject = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"X")); - if (xObject != nullptr) { - xObject->Smash(xObject->GetObjectID(), VIOLENT); - } -} - -void PetDigServer::HandleBouncerDig(const Entity* self, const Entity* owner) { - auto bounceNumber = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"BouncerNumber")); - auto bouncerSpawners = dZoneManager::Instance()->GetSpawnersByName("PetBouncer" + bounceNumber); - auto switchSpawners = dZoneManager::Instance()->GetSpawnersByName("PetBouncerSwitch" + bounceNumber); - - for (auto* bouncerSpawner : bouncerSpawners) { - bouncerSpawner->Activate(); - } - - for (auto* switchSpawner : switchSpawners) { - switchSpawner->Activate(); - } -} - -/** - * Progresses the Can You Dig It mission and the Pet Excavator Achievement if the player has never completed it yet - * \param owner the owner that just made a pet dig something up - */ -void PetDigServer::ProgressPetDigMissions(const Entity* owner, const Entity* chest) { - auto* missionComponent = owner->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - // Can You Dig It progress - const auto digMissionState = missionComponent->GetMissionState(843); - if (digMissionState == MissionState::MISSION_STATE_ACTIVE) { - missionComponent->ForceProgress(843, 1216, 1); - } - - // Pet Excavator progress - const auto excavatorMissionState = missionComponent->GetMissionState(505); - if (excavatorMissionState == MissionState::MISSION_STATE_ACTIVE) { - if (chest->HasVar(u"PetDig")) { - int32_t playerFlag = 1260 + chest->GetVarAs<int32_t>(u"PetDig"); - Character* player = owner->GetCharacter(); - - // check if player flag is set - if (!player->GetPlayerFlag(playerFlag)) { - missionComponent->ForceProgress(505, 767, 1); - player->SetPlayerFlag(playerFlag, 1); - } - } - } - } -} - -/** - * Some treasures spawn special pets, this handles that case - * \param owner the owner that just made a pet dig something up - * \param digInfo information regarding the treasure, will also contain info about the pet to spawn - */ -void PetDigServer::SpawnPet(Entity* self, const Entity* owner, const DigInfo digInfo) { - // Some treasures require a mission to be active - if (digInfo.requiredMission >= 0) { - auto* missionComponent = owner->GetComponent<MissionComponent>(); - if (missionComponent != nullptr && missionComponent->GetMissionState(digInfo.requiredMission) < MissionState::MISSION_STATE_ACTIVE) { - return; - } - } - - EntityInfo info{}; - info.lot = digInfo.spawnLot; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.spawnerID = self->GetSpawnerID(); - info.settings = { - new LDFData<LWOOBJID>(u"tamer", owner->GetObjectID()), - new LDFData<std::string>(u"group", "pet" + std::to_string(owner->GetObjectID())), - new LDFData<std::string>(u"spawnAnim", "spawn-pet"), - new LDFData<float>(u"spawnTimer", 1.0) - }; - - auto* spawnedPet = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(spawnedPet); -} - -Entity* PetDigServer::GetClosestTresure(NiPoint3 position) { - float closestDistance = 0; - Entity* closest = nullptr; - - for (const auto tresureId : treasures) { - auto* tresure = EntityManager::Instance()->GetEntity(tresureId); - - if (tresure == nullptr) continue; - - float distance = Vector3::DistanceSquared(tresure->GetPosition(), position); - - if (closest == nullptr || distance < closestDistance) { - closestDistance = distance; - closest = tresure; - } - } - - return closest; -} diff --git a/dScripts/PetFromDigServer.cpp b/dScripts/PetFromDigServer.cpp deleted file mode 100644 index 5aca7486..00000000 --- a/dScripts/PetFromDigServer.cpp +++ /dev/null @@ -1,44 +0,0 @@ -#include "PetFromDigServer.h" -#include "PetComponent.h" - -void PetFromDigServer::OnStartup(Entity* self) { - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr || petComponent->GetOwner() != nullptr) - return; - - // Triggers the local dig pet script for taming etc. - auto tamer = self->GetVar<LWOOBJID>(u"tamer"); - - // Client compares this with player:GetID() which is a string, so we'll have to give it a string - self->SetNetworkVar(u"pettamer", std::to_string(tamer)); - - // Kill if the player decides that the dig pet is not worthy - self->AddTimer("killself", 45.0f); -} - -void PetFromDigServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "killself") { - - // Don't accidentally kill a pet that is already owned - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr || petComponent->GetOwner() != nullptr) - return; - - self->Smash(self->GetObjectID(), SILENT); - } -} - -void PetFromDigServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) { - if (type == NOTIFY_TYPE_BEGIN) { - self->CancelTimer("killself"); - } else if (type == NOTIFY_TYPE_QUIT || type == NOTIFY_TYPE_FAILED) { - self->Smash(self->GetObjectID(), SILENT); - } else if (type == NOTIFY_TYPE_SUCCESS) { - auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr) - return; - // TODO: Remove custom group? - // Command the pet to the player as it may otherwise go to its spawn point which is non existant - // petComponent->Command(NiPoint3::ZERO, LWOOBJID_EMPTY, 6, 202, true); - } -} diff --git a/dScripts/PetFromDigServer.h b/dScripts/PetFromDigServer.h deleted file mode 100644 index 3faf248d..00000000 --- a/dScripts/PetFromDigServer.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class PetFromDigServer : public CppScripts::Script -{ -public: - void OnStartup(Entity* self) override; - void OnTimerDone(Entity* self, std::string timerName) override; - void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) override; -}; diff --git a/dScripts/PetFromObjectServer.cpp b/dScripts/PetFromObjectServer.cpp deleted file mode 100644 index 899b394a..00000000 --- a/dScripts/PetFromObjectServer.cpp +++ /dev/null @@ -1,34 +0,0 @@ -#include "PetFromObjectServer.h" -#include "PetComponent.h" - -void PetFromObjectServer::OnStartup(Entity* self) { - self->SetNetworkVar(u"pettamer", std::to_string(self->GetVar<LWOOBJID>(u"tamer"))); - self->AddTimer("killSelf", 45.0f); -} - -void PetFromObjectServer::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "killSelf") { - const auto* petComponent = self->GetComponent<PetComponent>(); - if (petComponent == nullptr || petComponent->GetOwner() != nullptr) - return; - self->Smash(self->GetObjectID(), SILENT); - } -} - -void PetFromObjectServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) { - switch (type) { - case NOTIFY_TYPE_BEGIN: - self->CancelAllTimers(); - break; - case NOTIFY_TYPE_QUIT: - case NOTIFY_TYPE_FAILED: - self->Smash(self->GetObjectID(), SILENT); - break; - case NOTIFY_TYPE_SUCCESS: - // TODO: Remove from groups? - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"UpdateSuccessPicking", 0, - 0, tamer->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); - default: - break; - } -} diff --git a/dScripts/PetFromObjectServer.h b/dScripts/PetFromObjectServer.h deleted file mode 100644 index 1a7c244b..00000000 --- a/dScripts/PetFromObjectServer.h +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class PetFromObjectServer : public CppScripts::Script { -public: - void OnStartup(Entity* self) override; - void OnTimerDone(Entity* self, std::string timerName) override; - void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, eNotifyType type) override; -}; diff --git a/dScripts/PirateRep.cpp b/dScripts/PirateRep.cpp deleted file mode 100644 index eb4cf510..00000000 --- a/dScripts/PirateRep.cpp +++ /dev/null @@ -1,11 +0,0 @@ -#include "PirateRep.h" -#include "Character.h" - -void PirateRep::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID == m_PirateRepMissionID && missionState >= MissionState::MISSION_STATE_READY_TO_COMPLETE) { - auto* character = target->GetCharacter(); - if (character) { - character->SetPlayerFlag(ePlayerFlags::GF_PIRATE_REP, true); - } - } -} diff --git a/dScripts/PirateRep.h b/dScripts/PirateRep.h deleted file mode 100644 index 8fc82c5e..00000000 --- a/dScripts/PirateRep.h +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class PirateRep : public CppScripts::Script { -public: - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; -private: - const int m_PirateRepMissionID = 301; -}; diff --git a/dScripts/PropertyBankInteract.cpp b/dScripts/PropertyBankInteract.cpp deleted file mode 100644 index 788768ca..00000000 --- a/dScripts/PropertyBankInteract.cpp +++ /dev/null @@ -1,43 +0,0 @@ -#include "PropertyBankInteract.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void PropertyBankInteract::OnStartup(Entity* self) { - auto* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); - if (zoneControl != nullptr) { - zoneControl->OnFireEventServerSide(self, "CheckForPropertyOwner"); - } -} - -void PropertyBankInteract::OnPlayerLoaded(Entity* self, Entity* player) { - auto* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); - if (zoneControl != nullptr) { - zoneControl->OnFireEventServerSide(self, "CheckForPropertyOwner"); - } -} - -void PropertyBankInteract::OnUse(Entity* self, Entity* user) { - - AMFArrayValue args; - auto* value = new AMFStringValue(); - value->SetStringValue("bank"); - args.InsertValue("state", value); - - GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", &args); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"OpenBank", 0, 0, LWOOBJID_EMPTY, - "", user->GetSystemAddress()); -} - -void PropertyBankInteract::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (args == "ToggleBank") { - AMFArrayValue amfArgs; - amfArgs.InsertValue("visible", new AMFFalseValue()); - - GameMessages::SendUIMessageServerToSingleClient(sender, sender->GetSystemAddress(), "ToggleBank", &amfArgs); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"CloseBank", 0, 0, LWOOBJID_EMPTY, - "", sender->GetSystemAddress()); - } -} diff --git a/dScripts/PropertyDeathPlane.cpp b/dScripts/PropertyDeathPlane.cpp deleted file mode 100644 index ab659d8a..00000000 --- a/dScripts/PropertyDeathPlane.cpp +++ /dev/null @@ -1,16 +0,0 @@ -#include "PropertyDeathPlane.h" -#include "Entity.h" -#include "GameMessages.h" -#include "EntityManager.h" - -void PropertyDeathPlane::OnCollisionPhantom(Entity* self, Entity* target) { - const auto teleportGroup = EntityManager::Instance()->GetEntitiesInGroup("Teleport"); - - if (teleportGroup.size() == 0) { - return; - } - - auto* teleport = teleportGroup[0]; - - GameMessages::SendTeleport(target->GetObjectID(), teleport->GetPosition(), teleport->GetRotation(), target->GetSystemAddress()); -} diff --git a/dScripts/PropertyDevice.cpp b/dScripts/PropertyDevice.cpp deleted file mode 100644 index 64771de1..00000000 --- a/dScripts/PropertyDevice.cpp +++ /dev/null @@ -1,25 +0,0 @@ -#include "PropertyDevice.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "MissionComponent.h" - -void PropertyDevice::OnStartup(Entity* self) { - auto* zoneControl = EntityManager::Instance()->GetZoneControlEntity(); - if (zoneControl != nullptr) { - zoneControl->OnFireEventServerSide(self, "CheckForPropertyOwner"); - } -} - -void PropertyDevice::OnRebuildComplete(Entity* self, Entity* target) { - auto propertyOwnerID = self->GetNetworkVar<std::string>(m_PropertyOwnerVariable); - if (propertyOwnerID == std::to_string(LWOOBJID_EMPTY)) - return; - - auto* missionComponent = target->GetComponent<MissionComponent>(); - if (missionComponent != nullptr) { - if (missionComponent->GetMissionState(m_PropertyMissionID) == MissionState::MISSION_STATE_ACTIVE) { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 641, u"create", "callhome"); - missionComponent->ForceProgress(m_PropertyMissionID, 1793, self->GetLOT()); - } - } -} diff --git a/dScripts/PropertyFXDamage.cpp b/dScripts/PropertyFXDamage.cpp deleted file mode 100644 index d12cc9e1..00000000 --- a/dScripts/PropertyFXDamage.cpp +++ /dev/null @@ -1,18 +0,0 @@ -#include "PropertyFXDamage.h" -#include "DestroyableComponent.h" -#include "SkillComponent.h" - -void PropertyFXDamage::OnCollisionPhantom(Entity* self, Entity* target) { - if (target == nullptr) - return; - - auto* skills = self->GetComponent<SkillComponent>(); - auto* targetStats = target->GetComponent<DestroyableComponent>(); - - if (skills != nullptr && targetStats != nullptr) { - auto targetFactions = targetStats->GetFactionIDs(); - if (std::find(targetFactions.begin(), targetFactions.end(), 1) != targetFactions.end()) { - skills->CalculateBehavior(11386, 692, target->GetObjectID()); - } - } -} diff --git a/dScripts/PropertyPlatform.cpp b/dScripts/PropertyPlatform.cpp deleted file mode 100644 index 89687ac3..00000000 --- a/dScripts/PropertyPlatform.cpp +++ /dev/null @@ -1,32 +0,0 @@ -#include "PropertyPlatform.h" -#include "RebuildComponent.h" -#include "GameMessages.h" - -void PropertyPlatform::OnRebuildComplete(Entity* self, Entity* target) { - // auto* movingPlatform = self->GetComponent<MovingPlatformComponent>(); - // if (movingPlatform != nullptr) { - // movingPlatform->StopPathing(); - // movingPlatform->SetNoAutoStart(true); - // } - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, - 0, 0, MovementPlatformState::Stationary); -} - -void PropertyPlatform::OnUse(Entity* self, Entity* user) { - auto* rebuildComponent = self->GetComponent<RebuildComponent>(); - if (rebuildComponent != nullptr && rebuildComponent->GetState() == REBUILD_COMPLETED) { - // auto* movingPlatform = self->GetComponent<MovingPlatformComponent>(); - // if (movingPlatform != nullptr) { - // movingPlatform->GotoWaypoint(1); - // } - GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, - 1, 1, MovementPlatformState::Moving); - - self->AddCallbackTimer(movementDelay + effectDelay, [self, this]() { - self->SetNetworkVar<float_t>(u"startEffect", dieDelay); - self->AddCallbackTimer(dieDelay, [self]() { - self->Smash(); - }); - }); - } -} diff --git a/dScripts/QbEnemyStunner.cpp b/dScripts/QbEnemyStunner.cpp deleted file mode 100644 index 5d31a788..00000000 --- a/dScripts/QbEnemyStunner.cpp +++ /dev/null @@ -1,65 +0,0 @@ -#include "QbEnemyStunner.h" -#include "SkillComponent.h" -#include "DestroyableComponent.h" - -void QbEnemyStunner::OnRebuildComplete(Entity* self, Entity* target) { - auto* destroyable = self->GetComponent<DestroyableComponent>(); - - if (destroyable != nullptr) { - destroyable->SetFaction(115); - } - - auto skillComponent = self->GetComponent<SkillComponent>(); - if (!skillComponent) return; - - // Get the skill IDs of this object. - CDObjectSkillsTable* skillsTable = CDClientManager::Instance()->GetTable<CDObjectSkillsTable>("ObjectSkills"); - auto skills = skillsTable->Query([=](CDObjectSkills entry) {return (entry.objectTemplate == self->GetLOT()); }); - std::map<uint32_t, uint32_t> skillBehaviorMap; - // For each skill, cast it with the associated behavior ID. - for (auto skill : skills) { - CDSkillBehaviorTable* skillBehaviorTable = CDClientManager::Instance()->GetTable<CDSkillBehaviorTable>("SkillBehavior"); - CDSkillBehavior behaviorData = skillBehaviorTable->GetSkillByID(skill.skillID); - - skillBehaviorMap.insert(std::make_pair(skill.skillID, behaviorData.behaviorID)); - } - - // If there are no skills found, insert a default skill to use. - if (skillBehaviorMap.size() == 0) { - skillBehaviorMap.insert(std::make_pair(499U, 6095U)); - } - - // Start all skills associated with the object next tick - self->AddTimer("TickTime", 0); - - self->AddTimer("PlayEffect", 20); - - self->SetVar<std::map<uint32_t, uint32_t>>(u"skillBehaviorMap", skillBehaviorMap); -} - -void QbEnemyStunner::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "DieTime") { - self->Smash(); - - self->CancelAllTimers(); - } else if (timerName == "PlayEffect") { - self->SetNetworkVar(u"startEffect", 5.0f, UNASSIGNED_SYSTEM_ADDRESS); - - self->AddTimer("DieTime", 5.0f); - } else if (timerName == "TickTime") { - auto* skillComponent = self->GetComponent<SkillComponent>(); - - if (skillComponent != nullptr) { - auto skillBehaviorMap = self->GetVar<std::map<uint32_t, uint32_t>>(u"skillBehaviorMap"); - if (skillBehaviorMap.size() == 0) { - // Should no skills have been found, default to the mermaid stunner - skillComponent->CalculateBehavior(499U, 6095U, LWOOBJID_EMPTY); - } else { - for (auto pair : skillBehaviorMap) { - skillComponent->CalculateBehavior(pair.first, pair.second, LWOOBJID_EMPTY); - } - } - } - self->AddTimer("TickTime", 1); - } -} diff --git a/dScripts/QbSpawner.cpp b/dScripts/QbSpawner.cpp deleted file mode 100644 index edee6148..00000000 --- a/dScripts/QbSpawner.cpp +++ /dev/null @@ -1,136 +0,0 @@ -#include "QbSpawner.h" -#include "BaseCombatAIComponent.h" -#include "MovementAIComponent.h" - -void QbSpawner::OnStartup(Entity* self) { - auto mobNum = self->GetVar<int>(u"mobNum"); - auto spawnDist = self->GetVar<float>(u"spawnDist"); - auto mobTemplate = self->GetVar<LWOOBJID>(u"mobTemplate"); - auto spawnTime = self->GetVar<float>(u"spawnTime"); - - if (!mobNum) self->SetVar<int>(u"mobNum", m_DefaultMobNum); - if (!spawnDist) self->SetVar<float>(u"spawnDist", m_DefaultSpawnDist); - if (!mobTemplate) self->SetVar<LWOOBJID>(u"mobTemplate", m_DefaultMobTemplate); - if (!spawnTime) self->SetVar<float>(u"spawnTime", m_DefaultSpawnTime); - - // go ahead and setup the mob table here - std::vector<LWOOBJID> mobTable; - mobTable.assign(self->GetVar<int>(u"mobNum"), LWOOBJID_EMPTY); - - self->SetVar<std::vector<LWOOBJID>>(u"mobTable", mobTable); -} - -void QbSpawner::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { - auto gateObjID = sender->GetObjectID(); - if (!gateObjID) return; - if (args == "spawnMobs") { - self->SetVar(u"gateObj", gateObjID); - auto spawnTime = self->GetVar<float>(u"spawnTime"); - self->AddTimer("SpawnMobEnemies", spawnTime); - } -} - -void QbSpawner::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "SpawnMobEnemies") { - auto mobTable = self->GetVar<std::vector<LWOOBJID>>(u"mobTable"); - - auto spawnDist = self->GetVar<float>(u"spawnDist"); - auto mobTemplate = self->GetVar<LWOOBJID>(u"mobTemplate"); - - auto gateObjID = self->GetVar<LWOOBJID>(u"gateObj"); - if (!gateObjID) return; - - auto* gate = EntityManager::Instance()->GetEntity(gateObjID); - if (!gate) return; - - auto oPos = gate->GetPosition(); - auto oDir = gate->GetRotation().GetForwardVector(); - NiPoint3 newPos( - oPos.x + (oDir.x * spawnDist), - oPos.y, - oPos.z + (oDir.z * spawnDist) - ); - auto newRot = NiQuaternion::LookAt(newPos, oPos); - - for (int i = 0; i < mobTable.size(); i++) { - int posOffset = -10; - if (mobTable[i] == LWOOBJID_EMPTY) { - posOffset = posOffset + 5 * i; - auto newOffset = newPos; - newOffset.z = newOffset.z + posOffset; - - EntityInfo info{}; - info.lot = mobTemplate; - info.pos = newOffset; - info.rot = newRot; - info.spawnerID = self->GetObjectID(); - info.spawnerNodeID = 0; - info.settings = { - new LDFData<bool>(u"no_timed_spawn", true), - new LDFData<float>(u"aggroRadius", 70), - new LDFData<float>(u"softtetherRadius", 80), - new LDFData<float>(u"tetherRadius", 90), - new LDFData<float>(u"wanderRadius", 5), - new LDFData<int>(u"mobTableLoc", i) - }; - - auto* child = EntityManager::Instance()->CreateEntity(info, nullptr, self); - EntityManager::Instance()->ConstructEntity(child); - - OnChildLoaded(self, child); - } else { - auto* mob = EntityManager::Instance()->GetEntity(mobTable[i]); - AggroTargetObject(self, mob); - } - } - - } -} - -void QbSpawner::OnChildLoaded(Entity* self, Entity* child) { - auto mobTable = self->GetVar<std::vector<LWOOBJID>>(u"mobTable"); - auto tableLoc = child->GetVar<int>(u"mobTableLoc"); - - mobTable[tableLoc] = child->GetObjectID(); - self->SetVar<std::vector<LWOOBJID>>(u"mobTable", mobTable); - - AggroTargetObject(self, child); - - const auto selfID = self->GetObjectID(); - - child->AddDieCallback([this, selfID, child]() { - auto* self = EntityManager::Instance()->GetEntity(selfID); - OnChildRemoved(self, child); - } - ); -} - -void QbSpawner::OnChildRemoved(Entity* self, Entity* child) { - auto mobTable = self->GetVar<std::vector<LWOOBJID>>(u"mobTable"); - auto tableLoc = child->GetVar<int>(u"mobTableLoc"); - - mobTable[tableLoc] = LWOOBJID_EMPTY; - self->SetVar<std::vector<LWOOBJID>>(u"mobTable", mobTable); -} - -void QbSpawner::AggroTargetObject(Entity* self, Entity* enemy) { - auto* baseCombatAIComponent = enemy->GetComponent<BaseCombatAIComponent>(); - if (!baseCombatAIComponent) return; - - auto gateObjID = self->GetVar<LWOOBJID>(u"gateObj"); - if (gateObjID) { - auto* gate = EntityManager::Instance()->GetEntity(gateObjID); - if (gate) { - auto* movementAIComponent = enemy->GetComponent<MovementAIComponent>(); - if (movementAIComponent) movementAIComponent->SetDestination(gate->GetPosition()); - baseCombatAIComponent->Taunt(gateObjID, 1000); - } - } - - auto playerObjID = self->GetVar<LWOOBJID>(u"player"); - if (playerObjID) { - baseCombatAIComponent->Taunt(playerObjID, 100); - } - -} - diff --git a/dScripts/RaceImagineCrateServer.cpp b/dScripts/RaceImagineCrateServer.cpp deleted file mode 100644 index a35007b4..00000000 --- a/dScripts/RaceImagineCrateServer.cpp +++ /dev/null @@ -1,55 +0,0 @@ -#include "CharacterComponent.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "PossessableComponent.h" -#include "RaceImagineCrateServer.h" -#include "RacingTaskParam.h" -#include "MissionComponent.h" -#include "SkillComponent.h" - -void RaceImagineCrateServer::OnDie(Entity* self, Entity* killer) { - if (self->GetVar<bool>(u"bIsDead")) { - return; - } - - self->SetVar<bool>(u"bIsDead", true); - - if (killer == nullptr) { - return; - } - - auto* skillComponent = killer->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - auto* destroyableComponent = killer->GetComponent<DestroyableComponent>(); - - if (destroyableComponent != nullptr) { - destroyableComponent->SetImagination(60); - - EntityManager::Instance()->SerializeEntity(killer); - } - - // Find possessor of race car to progress missions and update stats. - auto* possessableComponent = killer->GetComponent<PossessableComponent>(); - if (possessableComponent != nullptr) { - - auto* possessor = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); - if (possessor != nullptr) { - - auto* missionComponent = possessor->GetComponent<MissionComponent>(); - auto* characterComponent = possessor->GetComponent<CharacterComponent>(); - - if (characterComponent != nullptr) { - characterComponent->UpdatePlayerStatistic(RacingImaginationCratesSmashed); - characterComponent->UpdatePlayerStatistic(RacingSmashablesSmashed); - } - - // Progress racing smashable missions - if (missionComponent == nullptr) return; - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, 0, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_SMASHABLES); - } - } -} diff --git a/dScripts/RaceImaginePowerup.cpp b/dScripts/RaceImaginePowerup.cpp deleted file mode 100644 index 9bdd0813..00000000 --- a/dScripts/RaceImaginePowerup.cpp +++ /dev/null @@ -1,36 +0,0 @@ -#include "DestroyableComponent.h" -#include "EntityManager.h" -#include "PossessorComponent.h" -#include "RaceImaginePowerup.h" -#include "RacingTaskParam.h" -#include "MissionComponent.h" - -void RaceImaginePowerup::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, - int32_t param2, int32_t param3) { - if (sender->IsPlayer() && args == "powerup") { - auto* possessorComponent = sender->GetComponent<PossessorComponent>(); - - if (possessorComponent == nullptr) { - return; - } - - auto* vehicle = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); - - if (vehicle == nullptr) { - return; - } - - auto* destroyableComponent = vehicle->GetComponent<DestroyableComponent>(); - - if (destroyableComponent == nullptr) { - return; - } - - destroyableComponent->Imagine(10); - - auto* missionComponent = sender->GetComponent<MissionComponent>(); - - if (missionComponent == nullptr) return; - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, self->GetLOT(), (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_COLLECT_IMAGINATION); - } -} diff --git a/dScripts/RaceMaelstromGeiser.cpp b/dScripts/RaceMaelstromGeiser.cpp deleted file mode 100644 index b737d449..00000000 --- a/dScripts/RaceMaelstromGeiser.cpp +++ /dev/null @@ -1,85 +0,0 @@ -#include "RaceMaelstromGeiser.h" -#include "GameMessages.h" -#include "PossessableComponent.h" -#include "PossessorComponent.h" -#include "EntityManager.h" -#include "RacingControlComponent.h" -#include "dZoneManager.h" - -void RaceMaelstromGeiser::OnStartup(Entity* self) { - self->SetVar(u"AmFiring", false); - - self->AddTimer("downTime", self->GetVar<int32_t>(u"startTime")); - - self->SetProximityRadius(15, "deathZone"); -} - -void RaceMaelstromGeiser::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (!entering->IsPlayer() || name != "deathZone" || status != "ENTER") { - return; - } - - if (!self->GetVar<bool>(u"AmFiring")) { - return; - } - - auto* possessableComponent = entering->GetComponent<PossessableComponent>(); - - Entity* vehicle; - Entity* player; - - if (possessableComponent != nullptr) { - player = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); - - if (player == nullptr) { - return; - } - - vehicle = entering; - } else if (entering->IsPlayer()) { - auto* possessorComponent = entering->GetComponent<PossessorComponent>(); - - if (possessorComponent == nullptr) { - return; - } - - vehicle = EntityManager::Instance()->GetEntity(possessorComponent->GetPossessable()); - - if (vehicle == nullptr) { - return; - } - - player = entering; - } else { - return; - } - - - GameMessages::SendDie(vehicle, self->GetObjectID(), LWOOBJID_EMPTY, true, VIOLENT, u"", 0, 0, 0, true, false, 0); - - auto* zoneController = dZoneManager::Instance()->GetZoneControlObject(); - - auto* racingControlComponent = zoneController->GetComponent<RacingControlComponent>(); - - if (racingControlComponent != nullptr) { - racingControlComponent->OnRequestDie(player); - } -} - -void RaceMaelstromGeiser::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "downTime") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 4048, u"rebuild_medium", "geiser", LWOOBJID_EMPTY, 1, 1, true); - - self->AddTimer("buildUpTime", 1); - } else if (timerName == "buildUpTime") { - self->SetVar(u"AmFiring", true); - - self->AddTimer("killTime", 1.5f); - } else if (timerName == "killTime") { - GameMessages::SendStopFXEffect(self, true, "geiser"); - - self->SetVar(u"AmFiring", false); - - self->AddTimer("downTime", 3.0); - } -} diff --git a/dScripts/RaceSmashServer.cpp b/dScripts/RaceSmashServer.cpp deleted file mode 100644 index d0dc3d78..00000000 --- a/dScripts/RaceSmashServer.cpp +++ /dev/null @@ -1,30 +0,0 @@ -#include "CharacterComponent.h" -#include "EntityManager.h" -#include "PossessableComponent.h" -#include "RaceSmashServer.h" -#include "RacingTaskParam.h" -#include "MissionComponent.h" - -void RaceSmashServer::OnDie(Entity* self, Entity* killer) { - // Crate is smashed by the car - auto* possessableComponent = killer->GetComponent<PossessableComponent>(); - if (possessableComponent != nullptr) { - - auto* possessor = EntityManager::Instance()->GetEntity(possessableComponent->GetPossessor()); - if (possessor != nullptr) { - - auto* missionComponent = possessor->GetComponent<MissionComponent>(); - auto* characterComponent = possessor->GetComponent<CharacterComponent>(); - - if (characterComponent != nullptr) { - characterComponent->UpdatePlayerStatistic(RacingSmashablesSmashed); - } - - // Progress racing smashable missions - if (missionComponent == nullptr) return; - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, 0, (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_SMASHABLES); - // Progress missions that ask us to smash a specific smashable. - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_RACING, self->GetLOT(), (LWOOBJID)RacingTaskParam::RACING_TASK_PARAM_SMASH_SPECIFIC_SMASHABLE); - } - } -} diff --git a/dScripts/RainOfArrows.cpp b/dScripts/RainOfArrows.cpp deleted file mode 100644 index 141cd3ab..00000000 --- a/dScripts/RainOfArrows.cpp +++ /dev/null @@ -1,70 +0,0 @@ -#include "RainOfArrows.h" -#include "EntityManager.h" -#include "SkillComponent.h" -#include "GameMessages.h" - -void RainOfArrows::OnStartup(Entity* self) { - -} - -void RainOfArrows::OnRebuildComplete(Entity* self, Entity* target) { - auto myPos = self->GetPosition(); - auto myRot = self->GetRotation(); - - EntityInfo info; - info.lot = m_ArrowFXObject; - info.pos = myPos; - info.rot = myRot; - info.spawnerID = self->GetObjectID(); - - auto* entity = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(entity); - - self->SetVar<LWOOBJID>(u"ChildFX", entity->GetObjectID()); - self->SetVar<LWOOBJID>(u"playerID", target->GetObjectID()); - - self->AddTimer("ArrowsIncoming", m_ArrowDelay); - self->AddTimer("PlayArrowSound", m_ArrowDelay - 4); -} - -void RainOfArrows::OnTimerDone(Entity* self, std::string timerName) { - auto* child = EntityManager::Instance()->GetEntity( - self->GetVar<LWOOBJID>(u"ChildFX") - ); - - auto* player = EntityManager::Instance()->GetEntity( - self->GetVar<LWOOBJID>(u"playerID") - ); - - if (timerName == "ArrowsIncoming") { - if (child == nullptr) { - return; - } - - auto* skillComponent = child->GetComponent<SkillComponent>(); - - if (skillComponent == nullptr) { - return; - } - - skillComponent->CalculateBehavior( - m_ArrowSkill, - m_ArrowBehavior, - LWOOBJID_EMPTY, - true, - false, - player != nullptr ? player->GetObjectID() : LWOOBJID_EMPTY - ); - - self->AddTimer("FireSkill", 0.7f); - } else if (timerName == "PlayArrowSound") { - GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, m_ArrowsGUID); - } else if (timerName == "FireSkill") { - if (child != nullptr) { - child->Smash(); - } - - self->Smash(player != nullptr ? player->GetObjectID() : LWOOBJID_EMPTY); - } -} diff --git a/dScripts/RemoveRentalGear.cpp b/dScripts/RemoveRentalGear.cpp deleted file mode 100644 index 06d964b9..00000000 --- a/dScripts/RemoveRentalGear.cpp +++ /dev/null @@ -1,39 +0,0 @@ -#include "RemoveRentalGear.h" -#include "InventoryComponent.h" -#include "Item.h" -#include "Character.h" - -/* --------------------------------------------------------------- ---Removes the rental gear from the player on mission turn in --- ---created mrb ... 5 / 25 / 11 ---updated abeechler 6 / 27 / 11 ... Add session flag resetting for set equips ---ported Max 21/07/2020 --------------------------------------------------------------- ---add missionID configData to the object in HF to remove this ---gear what the specified mission is completed --------------------------------------------------------------- -*/ - -void RemoveRentalGear::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - if (missionID != defaultMission && missionID != 313) return; - - if (missionState == MissionState::MISSION_STATE_COMPLETE || missionState == MissionState::MISSION_STATE_READY_TO_COMPLETE) { - auto inv = static_cast<InventoryComponent*>(target->GetComponent(COMPONENT_TYPE_INVENTORY)); - if (!inv) return; - - //remove the inventory items - for (int item : gearSets) { - auto* id = inv->FindItemByLot(item); - if (id) { - inv->UnEquipItem(id); - inv->RemoveItem(id->GetLot(), id->GetCount()); - } - } - - //reset the equipment flag - auto character = target->GetCharacter(); - if (character) character->SetPlayerFlag(equipFlag, false); - } -} diff --git a/dScripts/RemoveRentalGear.h b/dScripts/RemoveRentalGear.h deleted file mode 100644 index 49ca0860..00000000 --- a/dScripts/RemoveRentalGear.h +++ /dev/null @@ -1,12 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class RemoveRentalGear : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState); - -private: - int defaultMission = 768; //mission to remove gearSets on completion - std::vector<int> gearSets = { 14359,14321,14353,14315 }; //inventory items to remove - int equipFlag = 126; //Set upon wearing trial faction armor for the first time in a session -}; - diff --git a/dScripts/RockHydrantBroken.cpp b/dScripts/RockHydrantBroken.cpp deleted file mode 100644 index 835d52f6..00000000 --- a/dScripts/RockHydrantBroken.cpp +++ /dev/null @@ -1,38 +0,0 @@ -#include "RockHydrantBroken.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void RockHydrantBroken::OnStartup(Entity* self) { - self->AddTimer("playEffect", 1); - - const auto hydrant = "hydrant" + self->GetVar<std::string>(u"hydrant"); - - const auto bouncers = EntityManager::Instance()->GetEntitiesInGroup(hydrant); - - for (auto* bouncer : bouncers) { - self->SetVar<LWOOBJID>(u"bouncer", bouncer->GetObjectID()); - - - GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); - - GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"enableCollision", UNASSIGNED_SYSTEM_ADDRESS); - } - - self->AddTimer("KillBroken", 10); -} - -void RockHydrantBroken::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "KillBroken") { - auto* bouncer = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"bouncer")); - - if (bouncer != nullptr) { - GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); - - GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"disableCollision", UNASSIGNED_SYSTEM_ADDRESS); - } - - self->Kill(); - } else if (timerName == "playEffect") { - GameMessages::SendPlayFXEffect(self->GetObjectID(), 4737, u"water", "water", LWOOBJID_EMPTY, 1, 1, true); - } -} diff --git a/dScripts/RockHydrantSmashable.cpp b/dScripts/RockHydrantSmashable.cpp deleted file mode 100644 index d76eca8d..00000000 --- a/dScripts/RockHydrantSmashable.cpp +++ /dev/null @@ -1,20 +0,0 @@ -#include "RockHydrantSmashable.h" -#include "EntityManager.h" -#include "GeneralUtils.h" - -void RockHydrantSmashable::OnDie(Entity* self, Entity* killer) { - const auto hydrantName = self->GetVar<std::u16string>(u"hydrant"); - - LDFBaseData* data = new LDFData<std::string>(u"hydrant", GeneralUtils::UTF16ToWTF8(hydrantName)); - - EntityInfo info{}; - info.lot = ROCK_HYDRANT_BROKEN; - info.pos = self->GetPosition(); - info.rot = self->GetRotation(); - info.settings = { data }; - info.spawnerID = self->GetSpawnerID(); - - auto* hydrant = EntityManager::Instance()->CreateEntity(info); - - EntityManager::Instance()->ConstructEntity(hydrant); -} diff --git a/dScripts/SGCannon.cpp b/dScripts/SGCannon.cpp deleted file mode 100644 index 049837a4..00000000 --- a/dScripts/SGCannon.cpp +++ /dev/null @@ -1,1069 +0,0 @@ -#include "SGCannon.h" -#include "EntityManager.h" -#include "GameMessages.h" -#include "dZoneManager.h" -#include "Player.h" -#include "Character.h" -#include "ShootingGalleryComponent.h" -#include "PossessorComponent.h" -#include "CharacterComponent.h" -#include "SimplePhysicsComponent.h" -#include "MovementAIComponent.h" -#include "../dWorldServer/ObjectIDManager.h" -#include "MissionComponent.h" - -void SGCannon::OnStartup(Entity* self) { - Game::logger->Log("SGCannon", "OnStartup"); - - m_Waves = GetWaves(); - constants = GetConstants(); - - ResetVars(self); - - self->SetVar<bool>(GameStartedVariable, false); - self->SetVar<Vector3>(InitialVelocityVariable, {}); - self->SetVar<uint32_t>(ImpactSkillVariale, constants.impactSkillID); - - auto* shootingGalleryComponent = self->GetComponent<ShootingGalleryComponent>(); - if (shootingGalleryComponent != nullptr) { - shootingGalleryComponent->SetStaticParams({ - Vector3 { -327.8609924316406, 256.8999938964844, 1.6482199430465698 }, - Vector3 { -181.4320068359375, 212.39999389648438, 2.5182199478149414 } - }); - - shootingGalleryComponent->SetDynamicParams({ - Vector3 { 0.0, 4.3, 9.0 }, - Vector3 { }, - 129.0, - 800.0, - 30.0, - 0.0, - -1.0, - 58.6 - }); - } - - self->SetVar<uint32_t>(TimeLimitVariable, 30); - self->SetVar<std::vector<LOT>>(ValidActorsVariable, { 3109, 3110, 3111, 3112, 3125, 3126 }); - self->SetVar<std::vector<LOT>>(ValidEffectsVariable, { 3122 }); - self->SetVar<std::vector<uint32_t>>(StreakBonusVariable, { 1, 2, 5, 10 }); - self->SetVar<bool>(SuperChargeActiveVariable, false); - self->SetVar<uint32_t>(MatrixVariable, 1); - self->SetVar<bool>(InitVariable, true); - - auto* simplePhysicsComponent = self->GetComponent<SimplePhysicsComponent>(); - - if (simplePhysicsComponent != nullptr) { - simplePhysicsComponent->SetPhysicsMotionState(5); - } -} - -void SGCannon::OnPlayerLoaded(Entity* self, Entity* player) { - Game::logger->Log("SGCannon", "Player loaded"); - self->SetVar<LWOOBJID>(PlayerIDVariable, player->GetObjectID()); -} - -void SGCannon::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - Script::OnFireEventServerSide(self, sender, args, param1, param2, param3); -} - -void SGCannon::OnActivityStateChangeRequest(Entity* self, LWOOBJID senderID, int32_t value1, int32_t value2, - const std::u16string& stringValue) { - Game::logger->Log("SGCannon", "Got activity state change request: %s", GeneralUtils::UTF16ToWTF8(stringValue).c_str()); - if (stringValue == u"clientready") { - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - Game::logger->Log("SGCannon", "Player is ready"); - /*GameMessages::SendSetStunned(player->GetObjectID(), PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, - true, true, true, true, true, true, true);*/ - - Game::logger->Log("SGCannon", "Sending ActivityEnter"); - - GameMessages::SendActivityEnter(self->GetObjectID(), player->GetSystemAddress()); - - auto* shootingGalleryComponent = self->GetComponent<ShootingGalleryComponent>(); - - if (shootingGalleryComponent != nullptr) { - shootingGalleryComponent->SetCurrentPlayerID(player->GetObjectID()); - - Game::logger->Log("SGCannon", "Setting player ID"); - - EntityManager::Instance()->SerializeEntity(self); - } else { - Game::logger->Log("SGCannon", "Shooting gallery component is null"); - } - - auto* characterComponent = player->GetComponent<CharacterComponent>(); - - if (characterComponent != nullptr) { - characterComponent->SetIsRacing(true); - characterComponent->SetCurrentActivity(2); - auto possessor = player->GetComponent<PossessorComponent>(); - if (possessor) { - possessor->SetPossessable(self->GetObjectID()); - possessor->SetPossessableType(ePossessionType::NO_POSSESSION); - } - - EntityManager::Instance()->SerializeEntity(player); - } - - self->SetNetworkVar<bool>(HideScoreBoardVariable, true); - self->SetNetworkVar<bool>(ReSetSuperChargeVariable, true); - self->SetNetworkVar<bool>(ShowLoadingUI, true); - - /* - GameMessages::SendTeleport( - player->GetObjectID(), - {-292.6415710449219, 230.20237731933594, -3.9090466499328613}, - {0.7067984342575073, -6.527870573336259e-05, 0.707414984703064, 0.00021762956748716533}, - player->GetSystemAddress(), true - ); - */ - - //GameMessages::SendRequestActivityEnter(self->GetObjectID(), player->GetSystemAddress(), false, player->GetObjectID()); - } else { - Game::logger->Log("SGCannon", "Player not found"); - } - } else if (value1 == 1200) { - StartGame(self); - } -} - -void SGCannon::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, - const std::u16string& userData) { - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - if (button == 1 && identifier == u"Shooting_Gallery_Stop") { - UpdatePlayer(self, player->GetObjectID(), true); - RemovePlayer(player->GetObjectID()); - StopGame(self, true); - return; - } - - if (identifier == u"Scoreboardinfo") { - GameMessages::SendDisplayMessageBox(player->GetObjectID(), true, - dZoneManager::Instance()->GetZoneControlObject()->GetObjectID(), - u"Shooting_Gallery_Retry?", 2, u"Retry?", - u"", player->GetSystemAddress()); - } else { - if ((button == 1 && (identifier == u"Shooting_Gallery_Retry" || identifier == u"RePlay")) - || identifier == u"SG1" || button == 0) { - - if (identifier == u"RePlay") { - static_cast<Player*>(player)->SendToZone(1300); - - return; - } - - self->SetNetworkVar<bool>(ClearVariable, true); - StartGame(self); - } else if (button == 1 && identifier == u"Shooting_Gallery_Exit") { - UpdatePlayer(self, player->GetObjectID(), true); - RemovePlayer(player->GetObjectID()); - } - } - } -} - -void SGCannon::OnActivityTimerDone(Entity* self, const std::string& name) { - if (name == SuperChargeTimer && !self->GetVar<bool>(SuperChargePausedVariable)) { - if (self->GetVar<bool>(WaveStatusVariable) || self->GetVar<uint32_t>(CurrentSuperChargedTimeVariable) < 1) { - self->SetNetworkVar<uint32_t>(ChargeCountingVariable, 99); - self->SetNetworkVar<uint32_t>(SuperChargeBarVariable, 0); - ToggleSuperCharge(self, false); - } - } else if (name == SpawnWaveTimer) { - if (self->GetVar<bool>(GameStartedVariable)) { - self->SetVar<bool>(WaveStatusVariable, true); - const auto wave = (int32_t)self->GetVar<uint32_t>(ThisWaveVariable); - - if (wave != 0 && self->GetVar<bool>(SuperChargePausedVariable)) { - StartChargedCannon(self, self->GetVar<uint32_t>(CurrentSuperChargedTimeVariable)); - self->SetVar<uint32_t>(CurrentSuperChargedTimeVariable, 0); - } - - TimerToggle(self, true); - - for (const auto& enemyToSpawn : m_Waves.at(self->GetVar<uint32_t>(ThisWaveVariable))) { - SpawnObject(self, enemyToSpawn, true); - } - - Game::logger->Log("SGCannon", "Current wave spawn: %i/%i", wave, m_Waves.size()); - - // All waves completed - const auto timeLimit = (float_t)self->GetVar<uint32_t>(TimeLimitVariable); - if (wave >= m_Waves.size()) { - ActivityTimerStart(self, GameOverTimer, timeLimit, timeLimit); - } else { - ActivityTimerStart(self, EndWaveTimer, timeLimit, timeLimit); - } - - const auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - GameMessages::SendPlayFXEffect(player->GetObjectID(), -1, u"SG-start", ""); - - GameMessages::SendStartActivityTime(self->GetObjectID(), timeLimit, player->GetSystemAddress()); - Game::logger->Log("SGCannon", "Sending ActivityPause false"); - - GameMessages::SendActivityPause(self->GetObjectID(), false, player->GetSystemAddress()); - } - } - } else if (name == EndWaveTimer) { - self->SetVar<bool>(WaveStatusVariable, false); - TimerToggle(self); - RecordPlayerScore(self); - - if (self->GetVar<uint32_t>(ThisWaveVariable) >= 2) { - GameMessages::SendActivityPause(self->GetObjectID(), true); - ActivityTimerStart(self, GameOverTimer, 0.1, 0.1); - return; - } - - self->SetVar<uint32_t>(ThisWaveVariable, self->GetVar<uint32_t>(ThisWaveVariable) + 1); - PlaySceneAnimation(self, u"wave" + GeneralUtils::to_u16string(self->GetVar<uint32_t>(ThisWaveVariable)), true, true, 1.7f); - self->SetNetworkVar<uint32_t>(WaveNumVariable, self->GetVar<uint32_t>(ThisWaveVariable) + 1); - self->SetNetworkVar<uint32_t>(WaveStrVariable, self->GetVar<uint32_t>(TimeLimitVariable)); - - Game::logger->Log("SGCannon", "Current wave: %i/%i", self->GetVar<uint32_t>(ThisWaveVariable), m_Waves.size()); - - if (self->GetVar<uint32_t>(ThisWaveVariable) >= m_Waves.size()) { - ActivityTimerStart(self, GameOverTimer, 0.1, 0.1); - } else { - ActivityTimerStart(self, SpawnWaveTimer, constants.inBetweenWavePause, constants.inBetweenWavePause); - } - - Game::logger->Log("SGCannon", "Sending ActivityPause true"); - - GameMessages::SendActivityPause(self->GetObjectID(), true); - if (self->GetVar<bool>(SuperChargeActiveVariable) && !self->GetVar<bool>(SuperChargePausedVariable)) { - PauseChargeCannon(self); - } - } else if (name == GameOverTimer) { - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - Game::logger->Log("SGCannon", "Sending ActivityPause true"); - - GameMessages::SendActivityPause(self->GetObjectID(), true, player->GetSystemAddress()); - - /*const auto leftoverCannonballs = EntityManager::Instance()->GetEntitiesInGroup("cannonball"); - if (leftoverCannonballs.empty()) { - RecordPlayerScore(self); - - } else { - ActivityTimerStart(self, EndGameBufferTimer, 1, leftoverCannonballs.size()); - }*/ - - ActivityTimerStart(self, EndGameBufferTimer, 1, 1); - - TimerToggle(self); - } - } else if (name.rfind(DoSpawnTimer, 0) == 0) { - if (self->GetVar<bool>(GameStartedVariable)) { - const auto spawnNumber = (uint32_t)std::stoi(name.substr(7)); - const auto& activeSpawns = self->GetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable); - const auto& toSpawn = activeSpawns.at(spawnNumber); - - const auto pathIndex = GeneralUtils::GenerateRandomNumber<float_t>(0, toSpawn.spawnPaths.size() - 1); - - const auto* path = dZoneManager::Instance()->GetZone()->GetPath( - toSpawn.spawnPaths.at(pathIndex) - ); - - auto info = EntityInfo{}; - info.lot = toSpawn.lot; - info.spawnerID = self->GetObjectID(); - info.pos = path->pathWaypoints.at(0).position; - - info.settings = { - new LDFData<SGEnemy>(u"SpawnData", toSpawn), - new LDFData<std::string>(u"custom_script_server", "scripts/ai/ACT/SG_TARGET.lua"), - new LDFData<std::string>(u"custom_script_client", "scripts/client/ai/SG_TARGET_CLIENT.lua"), - new LDFData<std::string>(u"attached_path", path->pathName), - new LDFData<uint32_t>(u"attached_path_start", 0), - new LDFData<std::u16string>(u"groupID", u"SGEnemy") - }; - - Game::logger->Log("SGCannon", "Spawning enemy %i on path %s", toSpawn.lot, path->pathName.c_str()); - - auto* enemy = EntityManager::Instance()->CreateEntity(info, nullptr, self); - EntityManager::Instance()->ConstructEntity(enemy); - - if (true) { - auto* movementAI = new MovementAIComponent(enemy, {}); - - enemy->AddComponent(COMPONENT_TYPE_MOVEMENT_AI, movementAI); - - movementAI->SetSpeed(toSpawn.initialSpeed); - movementAI->SetCurrentSpeed(toSpawn.initialSpeed); - movementAI->SetHaltDistance(0.0f); - - std::vector<NiPoint3> pathWaypoints; - - for (const auto& waypoint : path->pathWaypoints) { - pathWaypoints.push_back(waypoint.position); - } - - if (GeneralUtils::GenerateRandomNumber<float_t>(0, 1) < 0.5f) { - std::reverse(pathWaypoints.begin(), pathWaypoints.end()); - } - - movementAI->SetPath(pathWaypoints); - - enemy->AddDieCallback([this, self, enemy, name]() { - RegisterHit(self, enemy, name); - }); - } - - // Save the enemy and tell it to start pathing - if (enemy != nullptr) { - const_cast<std::vector<LWOOBJID>&>(self->GetVar<std::vector<LWOOBJID>>(SpawnedObjects)).push_back(enemy->GetObjectID()); - GameMessages::SendPlatformResync(enemy, UNASSIGNED_SYSTEM_ADDRESS); - } - } - } else if (name == EndGameBufferTimer) { - RecordPlayerScore(self); - StopGame(self, false); - } -} - -void -SGCannon::OnActivityTimerUpdate(Entity* self, const std::string& name, float_t timeRemaining, float_t elapsedTime) { - ActivityManager::OnActivityTimerUpdate(self, name, timeRemaining, elapsedTime); -} - -void SGCannon::StartGame(Entity* self) { - self->SetNetworkVar<uint32_t>(TimeLimitVariable, self->GetVar<uint32_t>(TimeLimitVariable)); - self->SetNetworkVar<bool>(AudioStartIntroVariable, true); - self->SetVar<LOT>(CurrentRewardVariable, LOT_NULL); - - auto rewardObjects = EntityManager::Instance()->GetEntitiesInGroup(constants.rewardModelGroup); - for (auto* reward : rewardObjects) { - reward->OnFireEventServerSide(self, ModelToBuildEvent); - } - - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - GetLeaderboardData(self, player->GetObjectID(), GetActivityID(self)); - Game::logger->Log("SGCannon", "Sending ActivityStart"); - GameMessages::SendActivityStart(self->GetObjectID(), player->GetSystemAddress()); - - GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"start", ""); - - self->SetNetworkVar<bool>(ClearVariable, true); - DoGameStartup(self); - - if (!self->GetVar<bool>(FirstTimeDoneVariable)) { - TakeActivityCost(self, player->GetObjectID()); - } - - self->SetVar<bool>(FirstTimeDoneVariable, true); - } - - SpawnNewModel(self); -} - -void SGCannon::DoGameStartup(Entity* self) { - ResetVars(self); - self->SetVar<bool>(GameStartedVariable, true); - self->SetNetworkVar<bool>(ClearVariable, true); - self->SetVar<uint32_t>(ThisWaveVariable, 0); - - if (constants.firstWaveStartTime < 1) { - constants.firstWaveStartTime = 1; - } - - ActivityTimerStart(self, SpawnWaveTimer, constants.firstWaveStartTime, - constants.firstWaveStartTime); -} - -void SGCannon::SpawnNewModel(Entity* self) { - - // Add a new reward to the existing rewards - const auto currentReward = self->GetVar<LOT>(CurrentRewardVariable); - if (currentReward != -1) { - auto rewards = self->GetVar<std::vector<LOT>>(RewardsVariable); - rewards.push_back(currentReward); - self->SetNetworkVar<int32_t>(RewardAddedVariable, currentReward); - } - - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - for (auto* rewardModel : EntityManager::Instance()->GetEntitiesInGroup(constants.rewardModelGroup)) { - uint32_t lootMatrix; - switch (self->GetVar<uint32_t>(MatrixVariable)) { - case 1: - lootMatrix = constants.scoreLootMatrix1; - break; - case 2: - lootMatrix = constants.scoreLootMatrix2; - break; - case 3: - lootMatrix = constants.scoreLootMatrix3; - break; - case 4: - lootMatrix = constants.scoreLootMatrix4; - break; - case 5: - lootMatrix = constants.scoreLootMatrix5; - break; - default: - lootMatrix = 0; - } - - if (lootMatrix != 0) { - std::unordered_map<LOT, int32_t> toDrop = {}; - toDrop = LootGenerator::Instance().RollLootMatrix(player, lootMatrix); - - for (auto drop : toDrop) { - rewardModel->OnFireEventServerSide(self, ModelToBuildEvent, drop.first); - self->SetVar<LOT>(CurrentRewardVariable, drop.first); - } - } - } - } -} - -void SGCannon::RemovePlayer(LWOOBJID playerID) { - auto* player = EntityManager::Instance()->GetEntity(playerID); - if (player == nullptr) - return; - - auto* playerObject = dynamic_cast<Player*>(player); - if (playerObject == nullptr) - return; - - auto* character = playerObject->GetCharacter(); - if (character != nullptr) { - playerObject->SendToZone(character->GetLastNonInstanceZoneID()); - } -} - -void SGCannon::StartChargedCannon(Entity* self, uint32_t optionalTime) { - optionalTime = optionalTime == 0 ? constants.chargedTime : optionalTime; - self->SetVar<bool>(SuperChargePausedVariable, false); - ToggleSuperCharge(self, true); - ActivityTimerStart(self, SuperChargeTimer, 1, optionalTime); - - if (!self->GetVar<bool>(WaveStatusVariable)) { - PauseChargeCannon(self); - } -} - -void SGCannon::TimerToggle(Entity* self, bool start) { - if (start) { - self->SetNetworkVar<uint32_t>(CountVariable, self->GetVar<uint32_t>(TimeLimitVariable)); - self->SetVar<bool>(GameStartedVariable, true); - } else { - self->SetNetworkVar<bool>(StopVariable, true); - } -} - -void SGCannon::SpawnObject(Entity* self, const SGEnemy& toSpawn, bool spawnNow) { - auto activeSpawns = self->GetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable); - activeSpawns.push_back(toSpawn); - self->SetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable, activeSpawns); - - self->SetVar(SpawnNumberVariable, activeSpawns.size() - 1); - const auto timerName = DoSpawnTimer + std::to_string(activeSpawns.size() - 1); - - if (spawnNow) { - if (toSpawn.minSpawnTime > 0 && toSpawn.maxSpawnTime > 0) { - const auto spawnTime = GeneralUtils::GenerateRandomNumber<float_t>(toSpawn.minSpawnTime, toSpawn.maxSpawnTime); - - ActivityTimerStart(self, timerName, spawnTime, spawnTime); - } else { - ActivityTimerStart(self, timerName, 1, 1); - } - } else if (toSpawn.respawns) { - const auto spawnTime = GeneralUtils::GenerateRandomNumber<float_t>(toSpawn.minRespawnTime, toSpawn.maxRespawnTime); - - ActivityTimerStart(self, timerName, spawnTime, spawnTime); - } -} - -void SGCannon::RecordPlayerScore(Entity* self) { - const auto totalScore = self->GetVar<uint32_t>(TotalScoreVariable); - const auto currentWave = self->GetVar<uint32_t>(ThisWaveVariable); - - if (currentWave > 0) { - auto totalWaveScore = 0; - auto playerScores = self->GetVar<std::vector<int32_t>>(PlayerScoresVariable); - - for (const auto& waveScore : playerScores) { - totalWaveScore += waveScore; - } - - if (currentWave >= playerScores.size()) { - playerScores.push_back(totalWaveScore); - } else { - playerScores[currentWave] = totalWaveScore; - } - } -} - -void SGCannon::PlaySceneAnimation(Entity* self, const std::u16string& animationName, bool onCannon, bool onPlayer, float_t priority) { - for (auto* cannon : EntityManager::Instance()->GetEntitiesInGroup("cannongroup")) { - GameMessages::SendPlayAnimation(cannon, animationName, priority); - } - - if (onCannon) { - GameMessages::SendPlayAnimation(self, animationName, priority); - } - - if (onPlayer) { - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player != nullptr) { - GameMessages::SendPlayAnimation(player, animationName, priority); - } - } -} - -void SGCannon::PauseChargeCannon(Entity* self) { - const auto time = std::max((uint32_t)std::ceil(ActivityTimerGetCurrentTime(self, SuperChargeTimer)), (uint32_t)1); - - self->SetVar<bool>(SuperChargePausedVariable, true); - self->SetVar<uint32_t>(CurrentSuperChargedTimeVariable, time); - self->SetNetworkVar<uint32_t>(ChargeCountingVariable, time); - - ActivityTimerStop(self, SuperChargeTimer); -} - -void SGCannon::StopGame(Entity* self, bool cancel) { - self->SetNetworkVar<bool>(ReSetSuperChargeVariable, true); - self->SetNetworkVar<bool>(HideSuperChargeVariable, true); - - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player == nullptr) - return; - - ToggleSuperCharge(self, false); - - // The player won, store all the score and send rewards - if (!cancel) { - auto percentage = 0; - auto misses = self->GetVar<uint32_t>(MissesVariable); - auto fired = self->GetVar<uint32_t>(ShotsFiredVariable); - - if (fired > 0) { - percentage = misses / fired; - } - - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (missionComponent != nullptr) { - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MINIGAME, self->GetVar<uint32_t>(TotalScoreVariable), self->GetObjectID(), "performact_score"); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_MINIGAME, self->GetVar<uint32_t>(MaxStreakVariable), self->GetObjectID(), "performact_streak"); - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_ACTIVITY, m_CannonLot, 0, "", self->GetVar<uint32_t>(TotalScoreVariable)); - } - - LootGenerator::Instance().GiveActivityLoot(player, self, GetGameID(self), self->GetVar<uint32_t>(TotalScoreVariable)); - - StopActivity(self, player->GetObjectID(), self->GetVar<uint32_t>(TotalScoreVariable), self->GetVar<uint32_t>(MaxStreakVariable), percentage); - self->SetNetworkVar<bool>(AudioFinalWaveDoneVariable, true); - - // Give the player the model rewards they earned - auto* inventory = player->GetComponent<InventoryComponent>(); - if (inventory != nullptr) { - for (const auto rewardLot : self->GetVar<std::vector<LOT>>(RewardsVariable)) { - inventory->AddItem(rewardLot, 1, eLootSourceType::LOOT_SOURCE_ACTIVITY, eInventoryType::MODELS); - } - } - - self->SetNetworkVar<std::u16string>(u"UI_Rewards", - GeneralUtils::to_u16string(self->GetVar<uint32_t>(TotalScoreVariable)) + u"_0_0_0_0_0_0" - ); - - GameMessages::SendRequestActivitySummaryLeaderboardData( - player->GetObjectID(), - self->GetObjectID(), - player->GetSystemAddress(), - GetGameID(self), - 1, - 10, - 0, - false - ); - } - - GameMessages::SendActivityStop(self->GetObjectID(), false, cancel, player->GetSystemAddress()); - self->SetVar<bool>(GameStartedVariable, false); - ActivityTimerStopAllTimers(self); - - // Destroy all spawners - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup("SGEnemy")) { - entity->Kill(); - } - - ResetVars(self); -} - -void SGCannon::RegisterHit(Entity* self, Entity* target, const std::string& timerName) { - const auto& spawnInfo = target->GetVar<SGEnemy>(u"SpawnData"); - - if (spawnInfo.respawns) { - const auto respawnTime = GeneralUtils::GenerateRandomNumber<float_t>(spawnInfo.minRespawnTime, spawnInfo.maxRespawnTime); - - ActivityTimerStart(self, timerName, respawnTime, respawnTime); - } - - int score = spawnInfo.score; - - if (score > 0) { - score += score * GetCurrentBonus(self); - - if (!self->GetVar<bool>(SuperChargeActiveVariable)) { - self->SetVar<uint32_t>(u"m_curStreak", self->GetVar<uint32_t>(u"m_curStreak") + 1); - } - } else { - if (!self->GetVar<bool>(SuperChargeActiveVariable)) { - self->SetVar<uint32_t>(u"m_curStreak", 0); - } - - self->SetNetworkVar<bool>(u"hitFriend", true); - } - - auto lastSuperTotal = self->GetVar<uint32_t>(u"LastSuperTotal"); - - auto scScore = self->GetVar<uint32_t>(TotalScoreVariable) - lastSuperTotal; - - Game::logger->Log("SGCannon", "LastSuperTotal: %i, scScore: %i, constants.chargedPoints: %i", - lastSuperTotal, scScore, constants.chargedPoints - ); - - if (!self->GetVar<bool>(SuperChargeActiveVariable) && scScore >= constants.chargedPoints && score >= 0) { - StartChargedCannon(self); - self->SetNetworkVar<float>(u"SuperChargeBar", 100.0f); - self->SetVar<uint32_t>(u"LastSuperTotal", self->GetVar<uint32_t>(TotalScoreVariable)); - } - - UpdateStreak(self); - - GameMessages::SendNotifyClientShootingGalleryScore(self->GetObjectID(), UNASSIGNED_SYSTEM_ADDRESS, - 0.0f, - score, - target->GetObjectID(), - target->GetPosition() - ); - - auto newScore = (int)self->GetVar<uint32_t>(TotalScoreVariable) + score; - - if (newScore < 0) { - newScore = 0; - } - - self->SetVar<uint32_t>(TotalScoreVariable, newScore); - - self->SetNetworkVar<uint32_t>(u"updateScore", newScore); - - self->SetNetworkVar<std::u16string>(u"beatHighScore", GeneralUtils::to_u16string(newScore)); - - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - if (player == nullptr) return; - - auto missionComponent = player->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) return; - - missionComponent->Progress(MissionTaskType::MISSION_TASK_TYPE_SMASH, spawnInfo.lot, self->GetObjectID()); -} - -void SGCannon::UpdateStreak(Entity* self) { - const auto streakBonus = GetCurrentBonus(self); - - const auto curStreak = self->GetVar<uint32_t>(u"m_curStreak"); - - const auto marks = curStreak % 3; - - self->SetNetworkVar<uint32_t>(u"cStreak", curStreak); - - if (curStreak >= 0 && curStreak < 13) { - if (marks == 1) { - self->SetNetworkVar<bool>(u"Mark1", true); - } else if (marks == 2) { - self->SetNetworkVar<bool>(u"Mark2", true); - } else if (marks == 0 && curStreak > 0) { - self->SetVar<float_t>(u"StreakBonus", streakBonus); - self->SetNetworkVar<bool>(u"ShowStreak", streakBonus + 1); - self->SetNetworkVar<bool>(u"Mark3", true); - } else { - self->SetVar<float_t>(u"StreakBonus", streakBonus); - self->SetNetworkVar<bool>(u"UnMarkAll", true); - } - } - auto maxStreak = self->GetVar<uint32_t>(MaxStreakVariable); - if (maxStreak < curStreak) self->SetVar<uint32_t>(MaxStreakVariable, curStreak); -} - -float_t SGCannon::GetCurrentBonus(Entity* self) { - auto streak = self->GetVar<uint32_t>(u"m_curStreak"); - - if (streak > 12) { - streak = 12; - } - - return streak / 3; -} - -void SGCannon::ToggleSuperCharge(Entity* self, bool enable) { - if (enable && self->GetVar<bool>(SuperChargeActiveVariable)) - return; - - auto* player = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); - - if (player == nullptr) { - Game::logger->Log("SGCannon", "Player not found in toggle super charge"); - return; - } - - auto* inventoryComponent = player->GetComponent<InventoryComponent>(); - - auto equippedItems = inventoryComponent->GetEquippedItems(); - - Game::logger->Log("SGCannon", "Player has %d equipped items", equippedItems.size()); - - auto skillID = constants.cannonSkill; - auto coolDown = constants.cannonRefireRate; - - auto* selfInventoryComponent = self->GetComponent<InventoryComponent>(); - - if (inventoryComponent == nullptr) { - Game::logger->Log("SGCannon", "Inventory component not found"); - return; - } - - if (enable) { - Game::logger->Log("SGCannon", "Player is activating super charge"); - selfInventoryComponent->UpdateSlot("greeble_r", { ObjectIDManager::GenerateRandomObjectID(), 6505, 1, 0 }); - selfInventoryComponent->UpdateSlot("greeble_l", { ObjectIDManager::GenerateRandomObjectID(), 6506, 1, 0 }); - - // TODO: Equip items - skillID = constants.cannonSuperChargeSkill; - coolDown = 400; - } else { - selfInventoryComponent->UpdateSlot("greeble_r", { ObjectIDManager::GenerateRandomObjectID(), 0, 0, 0 }); - selfInventoryComponent->UpdateSlot("greeble_l", { ObjectIDManager::GenerateRandomObjectID(), 0, 0, 0 }); - - self->SetNetworkVar<float>(u"SuperChargeBar", 0); - - Game::logger->Log("SGCannon", "Player disables super charge"); - - // TODO: Unequip items - for (const auto& equipped : equippedItems) { - if (equipped.first == "special_r" || equipped.first == "special_l") { - Game::logger->Log("SGCannon", "Trying to unequip a weapon, %i", equipped.second.lot); - - auto* item = inventoryComponent->FindItemById(equipped.second.id); - - if (item != nullptr) { - inventoryComponent->UnEquipItem(item); - } else { - Game::logger->Log("SGCannon", "Item not found, %i", equipped.second.lot); - } - } - } - - self->SetVar<uint32_t>(NumberOfChargesVariable, 0); - } - - const auto& constants = GetConstants(); - - auto* shootingGalleryComponent = self->GetComponent<ShootingGalleryComponent>(); - - if (shootingGalleryComponent == nullptr) { - return; - } - - DynamicShootingGalleryParams properties = shootingGalleryComponent->GetDynamicParams(); - - properties.cannonFOV = 58.6f; - properties.cannonVelocity = 129.0; - properties.cannonRefireRate = 800; - properties.cannonMinDistance = 30; - properties.cannonTimeout = -1; - - shootingGalleryComponent->SetDynamicParams(properties); - - EntityManager::Instance()->SerializeEntity(self); - EntityManager::Instance()->SerializeEntity(player); - - self->SetNetworkVar<uint64_t>(CannonBallSkillIDVariable, skillID); - self->SetVar<bool>(SuperChargeActiveVariable, enable); -} - -std::vector<std::vector<SGEnemy>> SGCannon::GetWaves() { - return { - // Wave 1 - { - // Ship 1 - { - std::vector<std::string> { "Wave_1_Ship_1", "Wave_1_Ship_3" }, - 6015, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 2 - { - std::vector<std::string> { "Wave_1_Ship_2", "Wave_1_Ship_4" }, - 6300, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Sub 1 - { - std::vector<std::string> { "Wave_1_Sub_1", "Wave_1_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 10.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Sub 2 - { - std::vector<std::string> { "Wave_2_Sub_1", "Wave_2_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Friendly - { - std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, - 2168,0.0,5.0,true, 2.0, 5.0, - 1.0, -1000, false, 0.0, 1.0, - 1.0, false,true - } - }, - - // Wave 2 - { - // Ship 1 - { - std::vector<std::string> { "Wave_1_Ship_1", "Wave_1_Ship_3" }, - 6015, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 2 - { - std::vector<std::string> { "Wave_1_Ship_2", "Wave_1_Ship_4" }, - 6300, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 3 - { - std::vector<std::string> { "Wave_2_Ship_1" }, - 6300, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 4 - { - std::vector<std::string> { "Wave_2_Ship_2" }, - 6015, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Sub 1 - { - std::vector<std::string> { "Wave_1_Sub_1", "Wave_1_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Sub 2 - { - std::vector<std::string> { "Wave_2_Sub_1", "Wave_2_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Duck - { - std::vector<std::string> { "Wave_1_Duck_1", "Wave_1_Duck_2" }, - 5946, 5.0, 10.0, true, 5.0, 10.0, - 4.0, 5000, false, 0.0, 1.0, - 1.0, false, true - }, - - // Friendly - { - std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, - 2168,0.0,5.0,true, 2.0, 5.0, - 1.0, -1000, false, 0.0, 1.0, - 1.0, false,true - } - }, - - // Wave 3 - { - // Ship 1 - { - std::vector<std::string> { "Wave_1_Ship_1", "Wave_1_Ship_3" }, - 6015, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 2 - { - std::vector<std::string> { "Wave_1_Ship_2", "Wave_1_Ship_4" }, - 6300, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 3 - { - std::vector<std::string> { "Wave_2_Ship_1", "Wave_2_Ship_2" }, - 6015, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ship 4 - { - std::vector<std::string> { "Wave_3_Ship_1", "Wave_3_Ship_2" }, - 6300, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1500, false, 0.0, 1.0, - 1.0, false, true - }, - - // Sub 1 - { - std::vector<std::string> { "Wave_1_Sub_1", "Wave_1_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Sub 2 - { - std::vector<std::string> { "Wave_2_Sub_1", "Wave_2_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Sub 3 - { - std::vector<std::string> { "Wave_3_Sub_1", "Wave_3_Sub_2" }, - 6016, 0.0, 2.0, true, 0.0, 2.0, - 2.0, 1000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Duck - { - std::vector<std::string> { "Wave_1_Duck_1", "Wave_1_Duck_2" }, - 5946, 5.0, 10.0, true, 5.0, 10.0, - 4.0, 5000, false, 0.0, 1.0, - 1.0, false, true - }, - - // Ness - { - std::vector<std::string> { "Wave_1_Ness_1", "Wave_1_Ness_2", "Wave_2_Ness_1" }, - 2565, 10.0, 15.0, true, 10.0, 15.0, - 2.0, 10000, false, 0.0, 1.0, - 1.0, true, true - }, - - // Friendly 1 - { - std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, - 2168,0.0,5.0,true, 2.0, 5.0, - 1.0, -1000, false, 0.0, 1.0, - 1.0, false,true - }, - - // Friendly 2 - { - std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, - 2168,0.0,5.0,true, 2.0, 5.0, - 1.0, -1000, false, 0.0, 1.0, - 1.0, false,true - } - } - }; -} - -void SGCannon::ResetVars(Entity* self) { - self->SetVar<uint32_t>(SpawnNumberVariable, 0); - self->SetVar<uint32_t>(CurrentSpawnNumberVariable, 0); - self->SetVar<uint32_t>(ThisWaveVariable, 0); - self->SetVar<uint32_t>(GameScoreVariable, 0); - self->SetVar<uint32_t>(GameTimeVariable, 0); - self->SetVar<bool>(GameStartedVariable, false); - self->SetVar<uint32_t>(ShotsFiredVariable, 0); - self->SetVar<uint32_t>(MaxStreakVariable, 0); - self->SetVar<uint32_t>(MissesVariable, 0); - self->SetVar<uint32_t>(CurrentStreakVariable, 0); - self->SetVar<uint32_t>(CurrentSuperChargedTimeVariable, 0); - self->SetVar<std::vector<uint32_t>>(StreakBonusVariable, {}); - self->SetVar<uint32_t>(LastSuperTotalVariable, 0); - self->SetVar<LOT>(CurrentRewardVariable, LOT_NULL); - self->SetVar<std::vector<LOT>>(RewardsVariable, {}); - self->SetVar<uint32_t>(TotalScoreVariable, 0); - - const_cast<std::vector<SGEnemy>&>(self->GetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable)).clear(); - self->SetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable, {}); - - const_cast<std::vector<LWOOBJID>&>(self->GetVar<std::vector<LWOOBJID>>(SpawnedObjects)).clear(); - self->SetVar<std::vector<LWOOBJID>>(SpawnedObjects, {}); - - if (self->GetVar<bool>(InitVariable)) { - ToggleSuperCharge(self, false); - } - - self->SetVar<uint32_t>(ImpactSkillVariale, constants.impactSkillID); - self->SetVar<std::vector<int32_t>>(PlayerScoresVariable, {}); - ActivityTimerStopAllTimers(self); -} - -SGConstants SGCannon::GetConstants() { - return { - Vector3 { -908.542480, 229.773178, -908.542480 }, - Quaternion { 0.91913521289825, 0, 0.39394217729568, 0 }, - 1864, - 34, - 1822, - Vector3 { 6.652, -2, 1.5 }, - 157, - 129.0, - 30.0, - 800.0, - Vector3 { 0, 4.3, 9 }, - 6297, - 1822, - 249, - 228, - -1, - 58.6, - true, - 2, - 10, - 25000, - "QBRewardGroup", - 1864, - 50000, - 157, - 100000, - 187, - 200000, - 188, - 400000, - 189, - 800000, - 190, - 4.0, - 7.0 - }; -} diff --git a/dScripts/SGCannon.h b/dScripts/SGCannon.h deleted file mode 100644 index df9831ad..00000000 --- a/dScripts/SGCannon.h +++ /dev/null @@ -1,153 +0,0 @@ -#pragma once -#include <regex> -#include "ActivityManager.h" - -struct SGEnemy { - std::vector<std::string> spawnPaths{}; - LOT lot; - float_t minSpawnTime; - float_t maxSpawnTime; - bool respawns; - float_t minRespawnTime; - float_t maxRespawnTime; - float_t initialSpeed; - int32_t score; - bool changeSpeedAtWaypoint; - float_t speedChangeChance; - float_t minSpeed; - float_t maxSpeed; - bool isMovingPlatform; - bool despawnOnLastWaypoint; -}; - -struct SGConstants { - Vector3 playerStartPosition; - Quaternion playerStartRotation; - LOT cannonLot; - uint32_t impactSkillID; - LOT projectileLot; - Vector3 playerOffset; - uint32_t rewardModelMatrix; - float_t cannonVelocity; - float_t cannonMinDistance; - float_t cannonRefireRate; - Vector3 cannonBarrelOffset; - LOT cannonSuperchargedProjectileLot; - LOT cannonProjectileLot; - uint32_t cannonSuperChargeSkill; - uint32_t cannonSkill; - int32_t cannonTimeout; - float_t cannonFOV; - bool useLeaderboards; - uint32_t streakModifier; - uint32_t chargedTime; - uint32_t chargedPoints; - std::string rewardModelGroup; - uint32_t activityID; - uint32_t scoreReward1; - uint32_t scoreLootMatrix1; - uint32_t scoreReward2; - uint32_t scoreLootMatrix2; - uint32_t scoreReward3; - uint32_t scoreLootMatrix3; - uint32_t scoreReward4; - uint32_t scoreLootMatrix4; - uint32_t scoreReward5; - uint32_t scoreLootMatrix5; - float_t firstWaveStartTime; - float_t inBetweenWavePause; -}; - -class SGCannon : public ActivityManager { -public: - void OnStartup(Entity* self) override; - void OnPlayerLoaded(Entity* self, Entity* player) override; - void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; - void OnActivityStateChangeRequest(Entity* self, LWOOBJID senderID, int32_t value1, - int32_t value2, const std::u16string& stringValue) override; - void OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, - const std::u16string& userData) override; - void OnActivityTimerDone(Entity* self, const std::string& name) override; - void OnActivityTimerUpdate(Entity* self, const std::string& name, float_t timeRemaining, float_t elapsedTime) override; -private: - static std::vector<std::vector<SGEnemy>> GetWaves(); - static SGConstants GetConstants(); - void ResetVars(Entity* self); - void StartGame(Entity* self); - void DoGameStartup(Entity* self); - void SpawnNewModel(Entity* self); - void TimerToggle(Entity* self, bool start = false); - void SpawnObject(Entity* self, const SGEnemy& toSpawn, bool spawnNow = false); - void StartChargedCannon(Entity* self, uint32_t optionalTime = 0); - void ToggleSuperCharge(Entity* self, bool enable); - void RecordPlayerScore(Entity* self); - void PlaySceneAnimation(Entity* self, const std::u16string& animationName, bool onCannon, bool onPlayer, float_t priority); - static void RemovePlayer(LWOOBJID playerID); - void PauseChargeCannon(Entity* self); - void StopGame(Entity* self, bool cancel = false); - void RegisterHit(Entity* self, Entity* target, const std::string& timerName); - void UpdateStreak(Entity* self); - float_t GetCurrentBonus(Entity* self); - - LOT m_CannonLot = 1864; - std::u16string PlayerIDVariable = u"PlayerID"; - std::u16string HideScoreBoardVariable = u"HideScoreBoard"; - std::u16string ReSetSuperChargeVariable = u"ReSetSuperCharge"; - std::u16string ShowLoadingUI = u"showLoadingUI"; - std::u16string SpawnNumberVariable = u"SpawnNum"; - std::u16string CurrentSpawnNumberVariable = u"CurSpawnNum"; - std::u16string ThisWaveVariable = u"ThisWave"; - std::u16string GameScoreVariable = u"GameScore"; - std::u16string GameTimeVariable = u"GameTime"; - std::u16string GameStartedVariable = u"GameStarted"; - std::u16string ShotsFiredVariable = u"ShotsFired"; - std::u16string MaxStreakVariable = u"MaxStreak"; - std::u16string MissesVariable = u"Misses"; - std::u16string CurrentStreakVariable = u"CurrentStreak"; - std::u16string CurrentSuperChargedTimeVariable = u"CurrentSuperChargedTime"; - std::u16string StreakBonusVariable = u"StreakBonus"; - std::u16string LastSuperTotalVariable = u"LastSuperTotal"; - std::u16string CurrentRewardVariable = u"CurrentReward"; - std::u16string RewardsVariable = u"Rewards"; - std::u16string TotalScoreVariable = u"TotalScore"; - std::u16string InitVariable = u"Init"; - std::u16string ImpactSkillVariale = u"ImpactSkill"; - std::u16string PlayerScoresVariable = u"PlayerScores"; - std::u16string InitialVelocityVariable = u"InitialVelocity"; - std::u16string ValidActorsVariable = u"ValidActors"; - std::u16string ValidEffectsVariable = u"ValidEffects"; - std::u16string SuperChargeActiveVariable = u"SuperChargeActive"; - std::u16string MatrixVariable = u"Matrix"; - std::u16string TimeLimitVariable = u"game_timelimit"; - std::u16string AudioStartIntroVariable = u"Audio_Start_Intro"; - std::u16string ClearVariable = u"Clear"; - std::u16string FirstTimeDoneVariable = u"FirstTimeDone"; - std::u16string RewardAddedVariable = u"rewardAdded"; - std::u16string SuperChargePausedVariable = u"Super_Charge_Paused"; - std::u16string WaveStatusVariable = u"WaveStatus"; - std::u16string CountVariable = u"count"; - std::u16string StopVariable = u"Stop"; - std::u16string ActiveSpawnsVariable = u"ActiveSpawns"; - std::u16string SpawnedObjects = u"SpawnedObjects"; - std::u16string WaveNumVariable = u"wave.waveNum"; - std::u16string WaveStrVariable = u"wave.waveStr"; - std::u16string ChargeCountingVariable = u"charge_counting"; - std::u16string SuperChargeBarVariable = u"SuperChargeBar"; - std::u16string NumberOfChargesVariable = u"NumberOfCharges"; - std::u16string CannonBallSkillIDVariable = u"cbskill"; - std::u16string HideSuperChargeVariable = u"HideSuper"; - std::u16string AudioFinalWaveDoneVariable = u"Audio_Final_Wave_Done"; - - std::string SpawnWaveTimer = "SpawnWave"; - std::string EndWaveTimer = "EndWave"; - std::string GameOverTimer = "GameOver"; - std::string DoSpawnTimer = "DoSpawn"; - std::string EndGameBufferTimer = "endGameBuffer"; - std::string SuperChargeTimer = "SuperChargeTimer"; - - std::string ModelToBuildEvent = "modelToBuild"; - - std::regex DoSpawnRegex = std::regex("\\d*"); - SGConstants constants{}; - std::vector<std::vector<SGEnemy>> m_Waves{}; -}; diff --git a/dScripts/ScriptComponent.h b/dScripts/ScriptComponent.h index 94226627..77dff5bf 100644 --- a/dScripts/ScriptComponent.h +++ b/dScripts/ScriptComponent.h @@ -9,6 +9,7 @@ #include "CppScripts.h" #include "Component.h" #include <string> +#include "eReplicaComponentType.h" class Entity; @@ -18,7 +19,7 @@ class Entity; */ class ScriptComponent : public Component { public: - static const uint32_t ComponentType = COMPONENT_TYPE_SCRIPT; + static const eReplicaComponentType ComponentType = eReplicaComponentType::SCRIPT; ScriptComponent(Entity* parent, std::string scriptName, bool serialized, bool client = false); ~ScriptComponent() override; diff --git a/dScripts/ScriptedPowerupSpawner.cpp b/dScripts/ScriptedPowerupSpawner.cpp index 730e65ba..2c502f26 100644 --- a/dScripts/ScriptedPowerupSpawner.cpp +++ b/dScripts/ScriptedPowerupSpawner.cpp @@ -1,6 +1,7 @@ #include "ScriptedPowerupSpawner.h" #include "RenderComponent.h" #include "EntityManager.h" +#include "Loot.h" void ScriptedPowerupSpawner::OnTemplateStartup(Entity* self) { self->SetVar<uint32_t>(u"currentCycle", 1); @@ -20,7 +21,7 @@ void ScriptedPowerupSpawner::OnTimerDone(Entity* self, std::string message) { drops.emplace(itemLOT, 1); // Spawn the required number of powerups - auto* owner = EntityManager::Instance()->GetEntity(self->GetSpawnerID()); + auto* owner = Game::entityManager->GetEntity(self->GetSpawnerID()); if (owner != nullptr) { auto* renderComponent = self->GetComponent<RenderComponent>(); for (auto i = 0; i < self->GetVar<uint32_t>(u"numberOfPowerups"); i++) { diff --git a/dScripts/SpawnGryphonServer.cpp b/dScripts/SpawnGryphonServer.cpp deleted file mode 100644 index 22e5ff59..00000000 --- a/dScripts/SpawnGryphonServer.cpp +++ /dev/null @@ -1,27 +0,0 @@ -#include "SpawnGryphonServer.h" -#include "InventoryComponent.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void SpawnGryphonServer::SetVariables(Entity* self) { - self->SetVar<LOT>(u"petLOT", 12433); - self->SetVar<std::string>(u"petType", "gryphon"); - self->SetVar<uint32_t>(u"maxPets", 2); - self->SetVar<std::u16string>(u"spawnAnim", u"spawn"); - self->SetVar<std::u16string>(u"spawnCinematic", u"SentinelPet"); -} - -void SpawnGryphonServer::OnUse(Entity* self, Entity* user) { - auto* missionComponent = user->GetComponent<MissionComponent>(); - auto* inventoryComponent = user->GetComponent<InventoryComponent>(); - - // Little extra for handling the case of the egg being placed the first time - if (missionComponent != nullptr && inventoryComponent != nullptr - && missionComponent->GetMissionState(1391) == MissionState::MISSION_STATE_ACTIVE) { - inventoryComponent->RemoveItem(12483, inventoryComponent->GetLotCount(12483)); - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); - return; - } - - SpawnPetBaseServer::OnUse(self, user); -} diff --git a/dScripts/SpawnPetBaseServer.cpp b/dScripts/SpawnPetBaseServer.cpp index 1d73a5b8..395a20c2 100644 --- a/dScripts/SpawnPetBaseServer.cpp +++ b/dScripts/SpawnPetBaseServer.cpp @@ -2,6 +2,8 @@ #include "GameMessages.h" #include "EntityManager.h" #include "PetComponent.h" +#include "EntityInfo.h" +#include "eTerminateType.h" void SpawnPetBaseServer::OnStartup(Entity* self) { SetVariables(self); @@ -9,7 +11,7 @@ void SpawnPetBaseServer::OnStartup(Entity* self) { } void SpawnPetBaseServer::OnUse(Entity* self, Entity* user) { - auto possibleSpawners = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(u"petType") + "Spawner"); + auto possibleSpawners = Game::entityManager->GetEntitiesInGroup(self->GetVar<std::string>(u"petType") + "Spawner"); if (possibleSpawners.empty()) return; @@ -31,8 +33,8 @@ void SpawnPetBaseServer::OnUse(Entity* self, Entity* user) { new LDFData<float_t>(u"spawnTimer", 1.0f) }; - auto* pet = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(pet); + auto* pet = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(pet); self->SetVar<std::string>(u"spawnedPets", self->GetVar<std::string>(u"spawnedPets") + "," + std::to_string(pet->GetObjectID())); @@ -42,7 +44,7 @@ void SpawnPetBaseServer::OnUse(Entity* self, Entity* user) { GameMessages::SendPlayCinematic(user->GetObjectID(), spawnCinematic, UNASSIGNED_SYSTEM_ADDRESS); } - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); } bool SpawnPetBaseServer::CheckNumberOfPets(Entity* self, Entity* user) { @@ -55,7 +57,7 @@ bool SpawnPetBaseServer::CheckNumberOfPets(Entity* self, Entity* user) { if (petID.empty()) continue; - const auto* spawnedPet = EntityManager::Instance()->GetEntity(std::stoull(petID)); + const auto* spawnedPet = Game::entityManager->GetEntity(std::stoull(petID)); if (spawnedPet == nullptr) continue; diff --git a/dScripts/SpecialImaginePowerupSpawner.cpp b/dScripts/SpecialImaginePowerupSpawner.cpp deleted file mode 100644 index c6cab294..00000000 --- a/dScripts/SpecialImaginePowerupSpawner.cpp +++ /dev/null @@ -1,47 +0,0 @@ -#include "SpecialImaginePowerupSpawner.h" - -#include "GameMessages.h" -#include "SkillComponent.h" -#include "DestroyableComponent.h" -#include "EntityManager.h" - -void SpecialImaginePowerupSpawner::OnStartup(Entity* self) { - self->SetProximityRadius(1.5f, "powerupEnter"); - self->SetVar(u"bIsDead", false); -} - -void SpecialImaginePowerupSpawner::OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) { - if (name != "powerupEnter" && status != "ENTER") { - return; - } - - if (entering->GetLOT() != 1) { - return; - } - - if (self->GetVar<bool>(u"bIsDead")) { - return; - } - - GameMessages::SendPlayFXEffect(self, -1, u"pickup", "", LWOOBJID_EMPTY, 1, 1, true); - - SkillComponent* skillComponent; - if (!self->TryGetComponent(COMPONENT_TYPE_SKILL, skillComponent)) { - return; - } - - const auto source = entering->GetObjectID(); - - skillComponent->CalculateBehavior(13, 20, source); - - DestroyableComponent* destroyableComponent; - if (!self->TryGetComponent(COMPONENT_TYPE_DESTROYABLE, destroyableComponent)) { - return; - } - - self->SetVar(u"bIsDead", true); - - self->AddCallbackTimer(1.0f, [self]() { - EntityManager::Instance()->ScheduleForKill(self); - }); -} diff --git a/dScripts/SpecialImaginePowerupSpawner.h b/dScripts/SpecialImaginePowerupSpawner.h deleted file mode 100644 index eb628951..00000000 --- a/dScripts/SpecialImaginePowerupSpawner.h +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class SpecialImaginePowerupSpawner final : public CppScripts::Script -{ -public: - void OnStartup(Entity* self) override; - void OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) override; -}; diff --git a/dScripts/SsModularBuildServer.cpp b/dScripts/SsModularBuildServer.cpp deleted file mode 100644 index d5b6da50..00000000 --- a/dScripts/SsModularBuildServer.cpp +++ /dev/null @@ -1,15 +0,0 @@ -#include "SsModularBuildServer.h" -#include "MissionComponent.h" - -void SsModularBuildServer::OnModularBuildExit(Entity* self, Entity* player, bool bCompleted, std::vector<LOT> modules) { - int missionNum = 1732; - - if (bCompleted) { - MissionComponent* mission = static_cast<MissionComponent*>(player->GetComponent(COMPONENT_TYPE_MISSION)); - Mission* rocketMission = mission->GetMission(missionNum); - - if (rocketMission->GetMissionState() == MissionState::MISSION_STATE_ACTIVE) { - mission->ForceProgress(missionNum, 2478, 1); - } - } -} diff --git a/dScripts/StinkyFishTarget.cpp b/dScripts/StinkyFishTarget.cpp deleted file mode 100644 index 19dbce88..00000000 --- a/dScripts/StinkyFishTarget.cpp +++ /dev/null @@ -1,42 +0,0 @@ -#include "StinkyFishTarget.h" -#include "EntityManager.h" - -void StinkyFishTarget::OnStartup(Entity* self) { - auto position = self->GetPosition(); - position.SetY(position.GetY() - 0.5f); - self->SetPosition(position); -} - -void StinkyFishTarget::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { - if (message != "stinkfish" || self->GetVar<bool>(u"used")) - return; - - self->SetVar<bool>(u"used", true); - self->SetVar<LWOOBJID>(u"player", caster->GetObjectID()); - - EntityInfo entityInfo{}; - entityInfo.pos = self->GetPosition(); - entityInfo.rot = self->GetRotation(); - entityInfo.spawnerID = self->GetObjectID(); - entityInfo.settings = { - new LDFData<bool>(u"no_timed_spawn", true) - }; - - auto* fish = EntityManager::Instance()->CreateEntity(entityInfo); - EntityManager::Instance()->ConstructEntity(fish); - - self->SetVar<LWOOBJID>(u"fish", fish->GetObjectID()); - self->AddTimer("smash", 5.0f); -} - -void StinkyFishTarget::OnTimerDone(Entity* self, std::string timerName) { - if (timerName == "smash") { - const auto playerID = self->GetVar<LWOOBJID>(u"player"); - auto* fish = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(u"fish")); - - if (fish != nullptr) { - fish->Smash(playerID); - self->Smash(playerID); - } - } -} diff --git a/dScripts/StinkyFishTarget.h b/dScripts/StinkyFishTarget.h deleted file mode 100644 index 6c52171d..00000000 --- a/dScripts/StinkyFishTarget.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class StinkyFishTarget : public CppScripts::Script { - void OnStartup(Entity* self) override; - void OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) override; - void OnTimerDone(Entity* self, std::string timerName) override; -}; diff --git a/dScripts/StoryBoxInteractServer.cpp b/dScripts/StoryBoxInteractServer.cpp deleted file mode 100644 index 2683ddd4..00000000 --- a/dScripts/StoryBoxInteractServer.cpp +++ /dev/null @@ -1,48 +0,0 @@ -#include "StoryBoxInteractServer.h" -#include "Character.h" -#include "GameMessages.h" -#include "dServer.h" -#include "AMFFormat.h" - -void StoryBoxInteractServer::OnUse(Entity* self, Entity* user) { - if (self->GetVar<bool>(u"hasCustomText")) { - const auto& customText = self->GetVar<std::string>(u"customText"); - - { - AMFArrayValue args; - - auto* state = new AMFStringValue(); - state->SetStringValue("Story"); - - args.InsertValue("state", state); - - GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "pushGameState", &args); - } - - user->AddCallbackTimer(0.1f, [user, customText]() { - AMFArrayValue args; - - auto* text = new AMFStringValue(); - text->SetStringValue(customText); - - args.InsertValue("visible", new AMFTrueValue()); - args.InsertValue("text", text); - - GameMessages::SendUIMessageServerToSingleClient(user, user->GetSystemAddress(), "ToggleStoryBox", &args); - }); - - return; - } - - const auto storyText = self->GetVarAsString(u"storyText"); - - int32_t boxFlag = self->GetVar<int32_t>(u"altFlagID"); - if (boxFlag <= 0) { - boxFlag = (10000 + Game::server->GetZoneID() + std::stoi(storyText.substr(storyText.length() - 2))); - } - - if (user->GetCharacter()->GetPlayerFlag(boxFlag) == false) { - user->GetCharacter()->SetPlayerFlag(boxFlag, true); - GameMessages::SendFireEventClientSide(self->GetObjectID(), user->GetSystemAddress(), u"achieve", LWOOBJID_EMPTY, 0, -1, LWOOBJID_EMPTY); - } -} diff --git a/dScripts/TokenConsoleServer.cpp b/dScripts/TokenConsoleServer.cpp deleted file mode 100644 index f21938d1..00000000 --- a/dScripts/TokenConsoleServer.cpp +++ /dev/null @@ -1,36 +0,0 @@ -#include "TokenConsoleServer.h" -#include "InventoryComponent.h" -#include "GameMessages.h" -#include "Character.h" - -//2021-05-03 - max - added script, omitted some parts related to inheritance in lua which we don't need - -void TokenConsoleServer::OnUse(Entity* self, Entity* user) { - auto* inv = static_cast<InventoryComponent*>(user->GetComponent(COMPONENT_TYPE_INVENTORY)); - - //make sure the user has the required amount of infected bricks - if (inv && inv->GetLotCount(6194) >= bricksToTake) { - //yeet the bricks - inv->RemoveItem(6194, bricksToTake); - - //play sound - GameMessages::SendPlayNDAudioEmitter(self, user->GetSystemAddress(), "947d0d52-c7f8-4516-8dee-e1593a7fd1d1"); - - //figure out which faction the player belongs to: - auto character = user->GetCharacter(); - if (!character) return; - // At this point the player has to be in a faction. - LOT tokenLOT = 0; - if (character->GetPlayerFlag(ePlayerFlags::VENTURE_FACTION)) //venture - tokenLOT = 8321; - else if (character->GetPlayerFlag(ePlayerFlags::ASSEMBLY_FACTION)) //assembly - tokenLOT = 8318; - else if (character->GetPlayerFlag(ePlayerFlags::PARADOX_FACTION)) //paradox - tokenLOT = 8320; - else if (character->GetPlayerFlag(ePlayerFlags::SENTINEL_FACTION)) //sentinel - tokenLOT = 8319; - inv->AddItem(tokenLOT, tokensToGive, eLootSourceType::LOOT_SOURCE_NONE); - } - - GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); -} diff --git a/dScripts/TouchMissionUpdateServer.cpp b/dScripts/TouchMissionUpdateServer.cpp deleted file mode 100644 index f732305e..00000000 --- a/dScripts/TouchMissionUpdateServer.cpp +++ /dev/null @@ -1,51 +0,0 @@ -#include "TouchMissionUpdateServer.h" - -#include "Entity.h" -#include "GameMessages.h" -#include "MissionComponent.h" - -void TouchMissionUpdateServer::OnStartup(Entity* self) { - self->SetProximityRadius(20, "touchCheck"); // Those does not have a collider for some reason? -} - -void TouchMissionUpdateServer::OnCollisionPhantom(Entity* self, Entity* target) { - int32_t missionId = self->GetVar<int32_t>(u"TouchCompleteID"); - - if (missionId == 0) { - return; - } - - auto* missionComponent = static_cast<MissionComponent*>(target->GetComponent(COMPONENT_TYPE_MISSION)); - - if (missionComponent == nullptr) { - return; - } - - auto* mission = missionComponent->GetMission(missionId); - - if (mission == nullptr) { - return; - } - - const auto state = mission->GetMissionState(); - - if (state >= MissionState::MISSION_STATE_COMPLETE || mission->GetCompletions() > 1) { - return; - } - - for (auto* task : mission->GetTasks()) { - if (!task->IsComplete()) { - task->Complete(); - } - } - - mission->CheckCompletion(); -} - -void TouchMissionUpdateServer::OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) { - if (name != "touchCheck" || status != "ENTER") { - return; - } - - OnCollisionPhantom(self, entering); -} diff --git a/dScripts/TreasureChestDragonServer.cpp b/dScripts/TreasureChestDragonServer.cpp deleted file mode 100644 index 15f0709c..00000000 --- a/dScripts/TreasureChestDragonServer.cpp +++ /dev/null @@ -1,42 +0,0 @@ -#include "TreasureChestDragonServer.h" -#include "ScriptedActivityComponent.h" -#include "TeamManager.h" -#include "EntityManager.h" - -void TreasureChestDragonServer::OnStartup(Entity* self) { - -} - -void TreasureChestDragonServer::OnUse(Entity* self, Entity* user) { - if (self->GetVar<bool>(u"bUsed")) { - return; - } - - self->SetVar<bool>(u"bUsed", true); - - auto* scriptedActivityComponent = self->GetComponent<ScriptedActivityComponent>(); - - if (scriptedActivityComponent == nullptr) { - return; - } - - auto rating = 1; - - auto* team = TeamManager::Instance()->GetTeam(user->GetObjectID()); - - if (team != nullptr) { - rating = team->members.size(); - - for (const auto member : team->members) { - auto* memberObject = EntityManager::Instance()->GetEntity(member); - - if (memberObject == nullptr) continue; - - LootGenerator::Instance().DropActivityLoot(memberObject, self, scriptedActivityComponent->GetActivityID(), rating); - } - } else { - LootGenerator::Instance().DropActivityLoot(user, self, scriptedActivityComponent->GetActivityID(), rating); - } - - self->Smash(self->GetObjectID()); -} diff --git a/dScripts/TriggerAmbush.cpp b/dScripts/TriggerAmbush.cpp deleted file mode 100644 index 726b45d7..00000000 --- a/dScripts/TriggerAmbush.cpp +++ /dev/null @@ -1,37 +0,0 @@ -#include "TriggerAmbush.h" - -#include "dZoneManager.h" - -void TriggerAmbush::OnStartup(Entity* self) { - self->SetProximityRadius(20, "ambush"); -} - -void TriggerAmbush::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { - if (name != "ambush" || status != "ENTER" || !entering->IsPlayer()) return; - - if (self->GetVar<bool>(u"triggered")) return; - - self->SetVar(u"triggered", true); - - const auto spawners = dZoneManager::Instance()->GetSpawnersByName("Ambush"); - - for (auto* spawner : spawners) { - spawner->Activate(); - } - - self->AddTimer("TriggeredTimer", 45); -} - -void TriggerAmbush::OnTimerDone(Entity* self, std::string timerName) { - if (timerName != "TriggeredTimer") return; - - self->SetVar(u"triggered", false); - - const auto spawners = dZoneManager::Instance()->GetSpawnersByName("Ambush"); - - for (auto* spawner : spawners) { - spawner->Reset(); - - spawner->Deactivate(); - } -} diff --git a/dScripts/VeBricksampleServer.cpp b/dScripts/VeBricksampleServer.cpp deleted file mode 100644 index 5166d0c3..00000000 --- a/dScripts/VeBricksampleServer.cpp +++ /dev/null @@ -1,21 +0,0 @@ -#include "VeBricksampleServer.h" -#include "InventoryComponent.h" -#include "EntityManager.h" -#include "MissionComponent.h" -#include "GameMessages.h" - -void VeBricksampleServer::OnUse(Entity* self, Entity* user) { - auto* missionComponent = user->GetComponent<MissionComponent>(); - if (missionComponent != nullptr && missionComponent->GetMissionState(1183) == MissionState::MISSION_STATE_ACTIVE) { - const auto loot = self->GetVar<int32_t>(m_LootVariable); - auto* inventoryComponent = user->GetComponent<InventoryComponent>(); - - if (loot && inventoryComponent != nullptr && inventoryComponent->GetLotCount(loot) == 0) { - inventoryComponent->AddItem(loot, 1, eLootSourceType::LOOT_SOURCE_ACTIVITY); - - for (auto* brickEntity : EntityManager::Instance()->GetEntitiesInGroup("Bricks")) { - GameMessages::SendNotifyClientObject(brickEntity->GetObjectID(), u"Pickedup"); - } - } - } -} diff --git a/dScripts/VeEpsilonServer.cpp b/dScripts/VeEpsilonServer.cpp deleted file mode 100644 index cc6ecafe..00000000 --- a/dScripts/VeEpsilonServer.cpp +++ /dev/null @@ -1,26 +0,0 @@ -#include "VeEpsilonServer.h" -#include "Character.h" -#include "EntityManager.h" -#include "GameMessages.h" - -void VeEpsilonServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) { - auto* character = target->GetCharacter(); - if (character == nullptr) - return; - - // Resets the player flags that track which consoles they've used - if ((missionID == m_ConsoleMissionID || missionID == m_ConsoleRepeatMissionID) - && (missionState == MissionState::MISSION_STATE_AVAILABLE || missionState == MissionState::MISSION_STATE_COMPLETE_AVAILABLE)) { - - for (auto i = 0; i < 10; i++) { - character->SetPlayerFlag(m_ConsoleBaseFlag + i, false); - } - } - - // Notify the client that all objects have updated - self->AddCallbackTimer(3.0f, [this]() { - for (const auto* console : EntityManager::Instance()->GetEntitiesInGroup(m_ConsoleGroup)) { - GameMessages::SendNotifyClientObject(console->GetObjectID(), u""); - } - }); -} diff --git a/dScripts/VeEpsilonServer.h b/dScripts/VeEpsilonServer.h deleted file mode 100644 index d1236e96..00000000 --- a/dScripts/VeEpsilonServer.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once -#include "CppScripts.h" - -class VeEpsilonServer : public CppScripts::Script { - void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, MissionState missionState) override; - const uint32_t m_ConsoleMissionID = 1220; - const uint32_t m_ConsoleRepeatMissionID = 1225; - const uint32_t m_ConsoleBaseFlag = 1010; - const std::string m_ConsoleGroup = "Consoles"; -}; diff --git a/dScripts/VeMissionConsole.cpp b/dScripts/VeMissionConsole.cpp deleted file mode 100644 index 2a424c4d..00000000 --- a/dScripts/VeMissionConsole.cpp +++ /dev/null @@ -1,25 +0,0 @@ -#include "VeMissionConsole.h" -#include "InventoryComponent.h" -#include "Character.h" -#include "GameMessages.h" - -void VeMissionConsole::OnUse(Entity* self, Entity* user) { - LootGenerator::Instance().DropActivityLoot(user, self, 12551); - - auto* inventoryComponent = user->GetComponent<InventoryComponent>(); - if (inventoryComponent != nullptr) { - inventoryComponent->AddItem(12547, 1, eLootSourceType::LOOT_SOURCE_ACTIVITY); // Add the panel required for pickup - } - - // The flag to set is 101<number> - const auto flagNumber = self->GetVar<std::u16string>(m_NumberVariable); - const auto flag = std::stoi("101" + GeneralUtils::UTF16ToWTF8(flagNumber)); - - auto* character = user->GetCharacter(); - if (character != nullptr) { - character->SetPlayerFlag(flag, true); - } - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u""); - GameMessages::SendTerminateInteraction(user->GetObjectID(), FROM_INTERACTION, self->GetObjectID()); -} diff --git a/dScripts/WhFans.cpp b/dScripts/WhFans.cpp deleted file mode 100644 index 44354127..00000000 --- a/dScripts/WhFans.cpp +++ /dev/null @@ -1,71 +0,0 @@ -#include "WhFans.h" - -#include "RenderComponent.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "PhantomPhysicsComponent.h" - -void WhFans::OnStartup(Entity* self) { - self->SetVar<bool>(u"alive", true); - self->SetVar<bool>(u"on", false); - - ToggleFX(self, false); -} - -void WhFans::ToggleFX(Entity* self, bool hit) { - std::string fanGroup; - const auto& groups = self->GetGroups(); - if (!groups.empty()) { - fanGroup = groups[0]; - } else { - fanGroup = ""; - } - - std::vector<Entity*> fanVolumes = EntityManager::Instance()->GetEntitiesInGroup(fanGroup); - - auto* renderComponent = self->GetComponent<RenderComponent>(); - - if (renderComponent == nullptr) return; - - if (fanVolumes.size() == 0 || !self->GetVar<bool>(u"alive")) return; - - if (self->GetVar<bool>(u"on")) { - GameMessages::SendPlayAnimation(self, u"fan-off"); - - renderComponent->StopEffect("fanOn"); - self->SetVar<bool>(u"on", false); - - for (Entity* volume : fanVolumes) { - auto volumePhys = volume->GetComponent<PhantomPhysicsComponent>(); - if (!volumePhys) continue; - volumePhys->SetPhysicsEffectActive(false); - EntityManager::Instance()->SerializeEntity(volume); - } - } else if (!self->GetVar<bool>(u"on") && self->GetVar<bool>(u"alive")) { - GameMessages::SendPlayAnimation(self, u"fan-on"); - - self->SetVar<bool>(u"on", true); - - for (Entity* volume : fanVolumes) { - auto volumePhys = volume->GetComponent<PhantomPhysicsComponent>(); - if (!volumePhys) continue; - volumePhys->SetPhysicsEffectActive(true); - EntityManager::Instance()->SerializeEntity(volume); - } - } -} - -void WhFans::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args.length() == 0 || !self->GetVar<bool>(u"alive")) return; - - if ((args == "turnOn" && self->GetVar<bool>(u"on")) || (args == "turnOff" && !self->GetVar<bool>(u"on"))) return; - ToggleFX(self, false); -} - -void WhFans::OnDie(Entity* self, Entity* killer) { - if (self->GetVar<bool>(u"on")) { - ToggleFX(self, true); - } - self->SetVar<bool>(u"alive", false); -} diff --git a/dScripts/WildAmbients.cpp b/dScripts/WildAmbients.cpp deleted file mode 100644 index 16dfa043..00000000 --- a/dScripts/WildAmbients.cpp +++ /dev/null @@ -1,6 +0,0 @@ -#include "WildAmbients.h" -#include "GameMessages.h" - -void WildAmbients::OnUse(Entity* self, Entity* user) { - GameMessages::SendPlayAnimation(self, u"interact"); -} diff --git a/dScripts/WishingWellServer.cpp b/dScripts/WishingWellServer.cpp deleted file mode 100644 index 8ac9e0b2..00000000 --- a/dScripts/WishingWellServer.cpp +++ /dev/null @@ -1,44 +0,0 @@ -#include "WishingWellServer.h" -#include "ScriptedActivityComponent.h" -#include "GameMessages.h" - -void WishingWellServer::OnStartup(Entity* self) { -} - -void WishingWellServer::OnUse(Entity* self, Entity* user) { - auto* scriptedActivity = self->GetComponent<ScriptedActivityComponent>(); - - if (!scriptedActivity->TakeCost(user)) { - return; - } - - const auto audio = self->GetVar<std::string>(u"sound1"); - - if (!audio.empty()) { - GameMessages::SendPlayNDAudioEmitter(self, user->GetSystemAddress(), audio); - } - - LootGenerator::Instance().DropActivityLoot( - user, - self, - static_cast<uint32_t>(scriptedActivity->GetActivityID()), - GeneralUtils::GenerateRandomNumber<int32_t>(1, 1000) - ); - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"StartCooldown", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); - - const auto userID = user->GetObjectID(); - - self->AddCallbackTimer(10, [self, userID]() { - auto* user = EntityManager::Instance()->GetEntity(userID); - - if (user == nullptr) return; - - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"StopCooldown", 0, 0, LWOOBJID_EMPTY, "", user->GetSystemAddress()); - }); - - GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); -} - -void WishingWellServer::OnTimerDone(Entity* self, std::string timerName) { -} diff --git a/dScripts/ZoneAgMedProperty.cpp b/dScripts/ZoneAgMedProperty.cpp deleted file mode 100644 index db1c4c4b..00000000 --- a/dScripts/ZoneAgMedProperty.cpp +++ /dev/null @@ -1,42 +0,0 @@ -#include "ZoneAgMedProperty.h" -#include "Entity.h" - -void ZoneAgMedProperty::SetGameVariables(Entity* self) { - - self->SetVar<std::string>(ClaimMarkerGroup, "ClaimMarker"); - self->SetVar<std::string>(GeneratorGroup, "Generator"); - self->SetVar<std::string>(GuardGroup, "Guard"); - self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); - self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); - self->SetVar<std::string>(SpotsGroup, "Spots"); - self->SetVar<std::string>(MSCloudsGroup, "maelstrom"); - self->SetVar<std::string>(EnemiesGroup, "Enemies"); - self->SetVar<std::string>(FXManagerGroup, "FXObject"); - self->SetVar<std::string>(ImagOrbGroup, "Orb"); - self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); - - self->SetVar<std::vector<std::string>>(EnemiesSpawner, { - "StrombieWander", "Strombies", "Mechs", "OtherEnemy" - }); - self->SetVar<std::string>(ClaimMarkerSpawner, "ClaimMarker"); - self->SetVar<std::string>(GeneratorSpawner, "Generator"); - self->SetVar<std::string>(DamageFXSpawner, "MaelstromFX"); - self->SetVar<std::string>(FXSpotsSpawner, "MaelstromSpots"); - self->SetVar<std::string>(PropertyMGSpawner, "PropertyGuard"); - self->SetVar<std::string>(ImageOrbSpawner, "Orb"); - self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); - self->SetVar<std::string>(SmashablesSpawner, "Smashables"); - self->SetVar<std::string>(FXManagerSpawner, "FXObject"); - self->SetVar<std::string>(PropObjsSpawner, "BankObj"); - self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "BirdFX", "SunBeam" }); - self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, {}); - - self->SetVar<uint32_t>(defeatedProperyFlag, 118); - self->SetVar<uint32_t>(placedModelFlag, 119); - self->SetVar<uint32_t>(guardMissionFlag, 1293); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 1294); - self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); - self->SetVar<LOT>(generatorIdFlag, 10118); - self->SetVar<LOT>(orbIDFlag, 10226); - self->SetVar<LOT>(behaviorQBID, 10445); -} diff --git a/dScripts/ZoneAgProperty.cpp b/dScripts/ZoneAgProperty.cpp deleted file mode 100644 index db2bb78a..00000000 --- a/dScripts/ZoneAgProperty.cpp +++ /dev/null @@ -1,424 +0,0 @@ -#include "ZoneAgProperty.h" -#include "EntityManager.h" -#include "Character.h" -#include "Entity.h" -#include "GameMessages.h" -#include "dZoneManager.h" -#include "RenderComponent.h" -#include "MissionComponent.h" - -void ZoneAgProperty::SetGameVariables(Entity* self) { - self->SetVar<std::string>(GuardGroup, "Guard"); - self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); - self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); - self->SetVar<std::string>(PropertyBorderGroup, "PropertyBorder"); - self->SetVar<std::string>(LandTargetGroup, "Land_Target"); - self->SetVar<std::string>(SpiderScreamGroup, "Spider_Scream"); - self->SetVar<std::vector<std::string>>(ROFTargetsGroup, { "ROF_Targets_00", "ROF_Targets_01", "ROF_Targets_02", "ROF_Targets_03", "ROF_Targets_04" }); - self->SetVar<std::string>(SpiderEggsGroup, "SpiderEggs"); - self->SetVar<std::string>(RocksGroup, "Rocks"); - self->SetVar<std::string>(EnemiesGroup, "SpiderBoss"); - self->SetVar<std::vector<std::string>>(ZoneVolumesGroup, { "Zone1Vol", "Zone2Vol", "Zone3Vol", "Zone4Vol", "Zone5Vol", "Zone6Vol", "Zone7Vol", "Zone8Vol", "AggroVol", "TeleVol" }); - self->SetVar<std::string>(FXManagerGroup, "FXObject"); - - self->SetVar<std::string>(EnemiesSpawner, "SpiderBoss"); - self->SetVar<std::vector<std::string>>(BossSensorSpawner, { "Zone1Vol", "Zone2Vol", "Zone3Vol", "Zone4Vol", "Zone5Vol", "Zone6Vol", "Zone7Vol", "Zone8Vol", "RFS_Targets", "AggroVol", "TeleVol" }); - self->SetVar<std::string>(LandTargetSpawner, "Land_Target"); - self->SetVar<std::string>(SpiderScreamSpawner, "Spider_Scream"); - self->SetVar<std::vector<std::string>>(ROFTargetsSpawner, { "ROF_Targets_00", "ROF_Targets_01", "ROF_Targets_02", "ROF_Targets_03", "ROF_Targets_04" }); - self->SetVar<std::string>(PropertyMGSpawner, "PropertyGuard"); - self->SetVar<std::string>(FXManagerSpawner, "FXObject"); - self->SetVar<std::string>(PropObjsSpawner, "BankObj"); - self->SetVar<std::string>(SpiderEggsSpawner, "SpiderEggs"); - self->SetVar<std::string>(RocksSpawner, "Rocks"); - self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "BirdFX", "SunBeam" }); - self->SetVar<std::vector<std::string>>(SpiderRocketSpawner, { "SpiderRocket_Bot", "SpiderRocket_Mid", "SpiderRocket_Top" }); - self->SetVar<std::string>(MailboxSpawner, "Mailbox"); - self->SetVar<std::string>(LauncherSpawner, "Launcher"); - self->SetVar<std::string>(InstancerSpawner, "Instancer"); - - self->SetVar<uint32_t>(defeatedProperyFlag, 71); - self->SetVar<uint32_t>(placedModelFlag, 73); - self->SetVar<uint32_t>(guardFirstMissionFlag, 891); - self->SetVar<uint32_t>(guardMissionFlag, 320); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 951); -} - -void ZoneAgProperty::OnStartup(Entity* self) { - LoadProperty(self); -} - -void ZoneAgProperty::OnPlayerLoaded(Entity* self, Entity* player) { - CheckForOwner(self); - - auto rented = self->GetVar<LWOOBJID>(u"PropertyOwner") == LWOOBJID_EMPTY; - self->SetVar<bool>(u"rented", rented); - - if (!rented) { - const auto numberOfPlayers = self->GetVar<int32_t>(u"numberOfPlayers"); - self->SetVar<int32_t>(u"numberOfPlayers", numberOfPlayers + 1); - } - - if (dZoneManager::Instance()->GetZone()->GetZoneID().GetMapID() == 1102) { - GameMessages::SendPlay2DAmbientSound(player, GUIDMaelstrom); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"maelstromSkyOn", 0, 0, - LWOOBJID_EMPTY, "", player->GetSystemAddress()); - - self->SetNetworkVar(u"unclaimed", true); - - return; - } - - BasePlayerLoaded(self, player); -} - -void ZoneAgProperty::PropGuardCheck(Entity* self, Entity* player) { - auto* missionComponent = player->GetComponent<MissionComponent>(); - if (missionComponent == nullptr) - return; - - const auto state = missionComponent->GetMissionState(self->GetVar<uint32_t>(guardMissionFlag)); - const auto firstState = missionComponent->GetMissionState(self->GetVar<uint32_t>(guardFirstMissionFlag)); - - if (firstState < MissionState::MISSION_STATE_COMPLETE || (state != MissionState::MISSION_STATE_COMPLETE && state != MissionState::MISSION_STATE_COMPLETE_READY_TO_COMPLETE)) - ActivateSpawner(self->GetVar<std::string>(PropertyMGSpawner)); -} - -void ZoneAgProperty::OnZoneLoadedInfo(Entity* self) { - LoadProperty(self); -} - -void ZoneAgProperty::LoadInstance(Entity* self) { - SetGameVariables(self); - - for (auto* spawner : dZoneManager::Instance()->GetSpawnersByName(self->GetVar<std::string>(InstancerSpawner))) { - for (auto* spawnerNode : spawner->m_Info.nodes) { - spawnerNode->config.push_back( - new LDFData<std::string>(u"custom_script_server", - R"(scripts\ai\GENERAL\L_INSTANCE_EXIT_TRANSFER_PLAYER_TO_LAST_NON_INSTANCE.lua)")); - spawnerNode->config.push_back(new LDFData<std::u16string>(u"transferText", u"SPIDER_QUEEN_EXIT_QUESTION")); - } - } - - ActivateSpawner(self->GetVar<std::string>(InstancerSpawner)); -} - -void ZoneAgProperty::LoadProperty(Entity* self) { - SetGameVariables(self); - ActivateSpawner(self->GetVar<std::string>(LauncherSpawner)); - ActivateSpawner(self->GetVar<std::string>(MailboxSpawner)); -} - -void ZoneAgProperty::ProcessGroupObjects(Entity* self, std::string group) { -} - -void ZoneAgProperty::SpawnSpots(Entity* self) { - for (const auto& spot : self->GetVar<std::vector<std::string>>(ROFTargetsSpawner)) { - ActivateSpawner(spot); - } - - ActivateSpawner(self->GetVar<std::string>(LandTargetSpawner)); -} - -void ZoneAgProperty::KillSpots(Entity* self) { - for (const auto& spot : self->GetVar<std::vector<std::string>>(ROFTargetsSpawner)) { - DeactivateSpawner(spot); - } - - for (const auto& groupName : self->GetVar<std::vector<std::string>>(ROFTargetsGroup)) { - for (auto* spot : EntityManager::Instance()->GetEntitiesInGroup(groupName)) { - spot->Kill(); - } - } - - DeactivateSpawner(self->GetVar<std::string>(LandTargetSpawner)); - for (auto* landTarget : EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(LandTargetSpawner))) { - landTarget->Kill(); - } -} - -void ZoneAgProperty::SpawnCrashedRocket(Entity* self) { - for (const auto& rocket : self->GetVar<std::vector<std::string>>(SpiderRocketSpawner)) { - ActivateSpawner(rocket); - } -} - -void ZoneAgProperty::KillCrashedRocket(Entity* self) { - for (const auto& rocket : self->GetVar<std::vector<std::string>>(SpiderRocketSpawner)) { - DeactivateSpawner(rocket); - DestroySpawner(rocket); - } -} - -void ZoneAgProperty::StartMaelstrom(Entity* self, Entity* player) { - ActivateSpawner(self->GetVar<std::string>(EnemiesSpawner)); - for (const auto& sensor : self->GetVar<std::vector<std::string>>(BossSensorSpawner)) { - ActivateSpawner(sensor); - } - - ActivateSpawner(self->GetVar<std::string>(FXManagerSpawner)); - ActivateSpawner(self->GetVar<std::string>(SpiderScreamSpawner)); - ActivateSpawner(self->GetVar<std::string>(SpiderEggsSpawner)); - ActivateSpawner(self->GetVar<std::string>(RocksSpawner)); - - SpawnCrashedRocket(self); - - for (const auto& ambient : self->GetVar<std::vector<std::string>>(AmbientFXSpawner)) { - DeactivateSpawner(ambient); - DestroySpawner(ambient); - ResetSpawner(ambient); - } - - StartTornadoFx(self); - - if (player != nullptr) { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"maelstromSkyOn", 0, 0, LWOOBJID_EMPTY, - "", player->GetSystemAddress()); - } -} - -uint32_t ZoneAgProperty::RetrieveSpawnerId(Entity* self, const std::string& spawner) { - auto spawnerIDs = dZoneManager::Instance()->GetSpawnersByName(spawner); - if (spawnerIDs.empty()) - return 0; - - return spawnerIDs[0]->m_Info.spawnerID; -} - -void ZoneAgProperty::OnTimerDone(Entity* self, std::string timerName) { - BaseTimerDone(self, timerName); -} - -void ZoneAgProperty::BaseTimerDone(Entity* self, const std::string& timerName) { - if (timerName == "GuardFlyAway") { - const auto zoneId = dZoneManager::Instance()->GetZone()->GetWorldID(); - if (zoneId != 1150) - return; - - const auto entities = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(GuardGroup)); - if (entities.empty()) - return; - - auto* entity = entities[0]; - - GameMessages::SendNotifyClientObject(EntityManager::Instance()->GetZoneControlEntity()->GetObjectID(), u"GuardChat", 0, 0, entity->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); - LoadProperty(self); - - self->AddTimer("KillGuard", 5); - } else if (timerName == "KillGuard") { - KillGuard(self); - } else if (timerName == "tornadoOff") { - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup))) { - auto* renderComponent = entity->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - renderComponent->StopEffect("TornadoDebris", false); - renderComponent->StopEffect("TornadoVortex", false); - renderComponent->StopEffect("silhouette", false); - } - } - - self->AddTimer("ShowVendor", 1.2f); - self->AddTimer("ShowClearEffects", 2); - } else if (timerName == "ShowClearEffects") { - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup))) { - auto* renderComponent = entity->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - renderComponent->PlayEffect(-1, u"beamOn", "beam"); - } - } - - self->AddTimer("killSpider", 2); - self->AddTimer("turnSkyOff", 1.5f); - self->AddTimer("killFXObject", 8); - } else if (timerName == "turnSkyOff") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"SkyOff", 0, 0, LWOOBJID_EMPTY, - "", UNASSIGNED_SYSTEM_ADDRESS); - } else if (timerName == "killSpider") { - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(EnemiesGroup))) { - entity->Kill(); - } - - for (const auto& sensor : self->GetVar<std::vector<std::string>>(BossSensorSpawner)) { - DeactivateSpawner(sensor); - DestroySpawner(sensor); - } - - DeactivateSpawner(self->GetVar<std::string>(SpiderEggsSpawner)); - DestroySpawner(self->GetVar<std::string>(SpiderEggsSpawner)); - - DeactivateSpawner(self->GetVar<std::string>(RocksSpawner)); - DestroySpawner(self->GetVar<std::string>(RocksSpawner)); - - KillSpots(self); - KillCrashedRocket(self); - - DeactivateSpawner(self->GetVar<std::string>(SpiderScreamSpawner)); - DestroySpawner(self->GetVar<std::string>(SpiderScreamSpawner)); - - for (auto* player : EntityManager::Instance()->GetEntitiesByComponent(COMPONENT_TYPE_CHARACTER)) { - GameMessages::SendStop2DAmbientSound(player, true, GUIDMaelstrom); - GameMessages::SendPlay2DAmbientSound(player, GUIDPeaceful); - } - } else if (timerName == "ShowVendor") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"vendorOn", 0, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - for (const auto& ambient : self->GetVar<std::vector<std::string>>(AmbientFXSpawner)) { - ActivateSpawner(ambient); - } - } else if (timerName == "BoundsVisOn") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"boundsAnim", 0, 0, - LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS); - } else if (timerName == "runPlayerLoadedAgain") { - CheckForOwner(self); - } else if (timerName == "pollTornadoFX") { - StartTornadoFx(self); - } else if (timerName == "killFXObject") { - for (auto* entity : EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(FXManagerGroup))) { - auto* renderComponent = entity->GetComponent<RenderComponent>(); - if (renderComponent != nullptr) { - renderComponent->StopEffect("beam"); - } - } - - DestroySpawner(self->GetVar<std::string>(FXManagerSpawner)); - - self->SetVar<bool>(u"FXObjectGone", true); - } else if (timerName == "ProcessGroupObj") { - // TODO - } -} - -void ZoneAgProperty::OnZonePropertyRented(Entity* self, Entity* player) { - BaseZonePropertyRented(self, player); - - auto* character = player->GetCharacter(); - if (character == nullptr) - return; - - character->SetPlayerFlag(108, true); -} - -void ZoneAgProperty::OnZonePropertyModelPlaced(Entity* self, Entity* player) { - auto* character = player->GetCharacter(); - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (!character->GetPlayerFlag(101)) { - BaseZonePropertyModelPlaced(self, player); - character->SetPlayerFlag(101, true); - if (missionComponent->GetMissionState(871) == MissionState::MISSION_STATE_ACTIVE) { - self->SetNetworkVar<std::u16string>(u"Tooltip", u"AnotherModel"); - } - - } else if (!character->GetPlayerFlag(102)) { - character->SetPlayerFlag(102, true); - if (missionComponent->GetMissionState(871) == MissionState::MISSION_STATE_ACTIVE) { - self->SetNetworkVar<std::u16string>(u"Tooltip", u"TwoMoreModels"); - } - - } else if (!character->GetPlayerFlag(103)) { - character->SetPlayerFlag(103, true); - } else if (!character->GetPlayerFlag(104)) { - character->SetPlayerFlag(104, true); - self->SetNetworkVar<std::u16string>(u"Tooltip", u"TwoMoreModelsOff"); - } else if (self->GetVar<std::string>(u"tutorial") == "place_model") { - self->SetVar<std::string>(u"tutorial", ""); - self->SetNetworkVar<std::u16string>(u"Tooltip", u"PutAway"); - } -} - -void ZoneAgProperty::OnZonePropertyModelPickedUp(Entity* self, Entity* player) { - auto* character = player->GetCharacter(); - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (!character->GetPlayerFlag(109)) { - character->SetPlayerFlag(109, true); - if (missionComponent->GetMissionState(891) == MissionState::MISSION_STATE_ACTIVE && !character->GetPlayerFlag(110)) { - self->SetNetworkVar<std::u16string>(u"Tooltip", u"Rotate"); - } - } -} - -void ZoneAgProperty::OnZonePropertyModelRemoved(Entity* self, Entity* player) { - auto* character = player->GetCharacter(); - character->SetPlayerFlag(111, true); -} - -void ZoneAgProperty::OnZonePropertyModelRemovedWhileEquipped(Entity* self, Entity* player) { - ZoneAgProperty::OnZonePropertyModelRemoved(self, player); -} - -void ZoneAgProperty::OnZonePropertyModelRotated(Entity* self, Entity* player) { - auto* character = player->GetCharacter(); - auto* missionComponent = player->GetComponent<MissionComponent>(); - - if (!character->GetPlayerFlag(110)) { - character->SetPlayerFlag(110, true); - - if (missionComponent->GetMissionState(891) == MissionState::MISSION_STATE_ACTIVE) { - self->SetNetworkVar<std::u16string>(u"Tooltip", u"PlaceModel"); - self->SetVar<std::string>(u"tutorial", "place_model"); - } - } -} - -void ZoneAgProperty::OnZonePropertyModelEquipped(Entity* self) { - self->SetNetworkVar<std::u16string>(u"PlayerAction", u"ModelEquipped"); -} - -void ZoneAgProperty::OnZonePropertyEditBegin(Entity* self) { - self->SetNetworkVar<std::u16string>(u"PlayerAction", u"Enter"); -} - -void ZoneAgProperty::OnZonePropertyEditEnd(Entity* self) { - self->SetNetworkVar<std::u16string>(u"PlayerAction", u"Exit"); -} - -void ZoneAgProperty::OnPlayerExit(Entity* self) { - // TODO: Destroy stuff -} - -void ZoneAgProperty::RemovePlayerRef(Entity* self) { - // TODO: Destroy stuff -} - -void ZoneAgProperty::BaseOnFireEventServerSide(Entity* self, Entity* sender, std::string args) { - if (args == "propertyRented") { - const auto playerId = self->GetVar<LWOOBJID>(u"playerID"); - auto* player = EntityManager::Instance()->GetEntity(playerId); - if (player == nullptr) - return; - - OnZonePropertyRented(self, player); - } else if (args == "RetrieveZoneData") { - self->SetVar<LWOOBJID>(u"SpiderBossID", sender->GetObjectID()); - sender->SetVar<int32_t>(u"SpiderEggNetworkID", RetrieveSpawnerId(self, self->GetVar<std::string>(SpiderEggsSpawner))); - - std::vector<uint32_t> table; - - for (const auto& target : self->GetVar<std::vector<std::string>>(ROFTargetsSpawner)) { - table.push_back(RetrieveSpawnerId(self, target)); - } - - ROFTargetGroupIdTable = table; - - ProcessGroupObjects(self, self->GetVar<std::string>(LandTargetGroup)); - ProcessGroupObjects(self, self->GetVar<std::string>(SpiderScreamGroup)); - // ProcessGroupObjects(self, groups.ZoneVolumes); - } else if (args == "CheckForPropertyOwner") { - sender->SetNetworkVar<std::string>(u"PropertyOwnerID", std::to_string(self->GetVar<LWOOBJID>(u"PropertyOwner"))); - } else if (args == "ClearProperty") { - const auto playerId = self->GetVar<LWOOBJID>(u"playerID"); - auto* player = EntityManager::Instance()->GetEntity(playerId); - if (player == nullptr) - return; - - player->GetCharacter()->SetPlayerFlag(self->GetVar<uint32_t>(defeatedProperyFlag), true); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayCinematic", 0, 0, - LWOOBJID_EMPTY, destroyedCinematic, UNASSIGNED_SYSTEM_ADDRESS); - - self->AddTimer("tornadoOff", 0.5f); - } -} - -void ZoneAgProperty::NotifyDie(Entity* self) { - // TODO -} diff --git a/dScripts/ZoneAgSpiderQueen.cpp b/dScripts/ZoneAgSpiderQueen.cpp deleted file mode 100644 index ea517448..00000000 --- a/dScripts/ZoneAgSpiderQueen.cpp +++ /dev/null @@ -1,83 +0,0 @@ -#include "ZoneAgSpiderQueen.h" -#include "GameMessages.h" -#include "EntityManager.h" -#include "ZoneAgProperty.h" -#include "DestroyableComponent.h" - -void ZoneAgSpiderQueen::SetGameVariables(Entity* self) { - ZoneAgProperty::SetGameVariables(self); - - // Disable property flags - self->SetVar<uint32_t>(defeatedProperyFlag, 0); - self->SetVar<uint32_t>(placedModelFlag, 0); - self->SetVar<uint32_t>(guardFirstMissionFlag, 0); - self->SetVar<uint32_t>(guardMissionFlag, 0); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 0); -} - -void ZoneAgSpiderQueen::OnStartup(Entity* self) { - LoadInstance(self); - - SpawnSpots(self); - StartMaelstrom(self, nullptr); -} - -void ZoneAgSpiderQueen::BasePlayerLoaded(Entity* self, Entity* player) { - ActivityManager::UpdatePlayer(self, player->GetObjectID()); - ActivityManager::TakeActivityCost(self, player->GetObjectID()); - - // Make sure the player has full stats when they join - auto* playerDestroyableComponent = player->GetComponent<DestroyableComponent>(); - if (playerDestroyableComponent != nullptr) { - playerDestroyableComponent->SetImagination(playerDestroyableComponent->GetMaxImagination()); - playerDestroyableComponent->SetArmor(playerDestroyableComponent->GetMaxArmor()); - playerDestroyableComponent->SetHealth(playerDestroyableComponent->GetMaxHealth()); - } - - self->SetNetworkVar(u"unclaimed", true); - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"maelstromSkyOn", 0, 0, LWOOBJID_EMPTY, - "", player->GetSystemAddress()); -} - -void -ZoneAgSpiderQueen::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - if (args == "ClearProperty") { - GameMessages::SendNotifyClientObject(self->GetObjectID(), u"PlayCinematic", 0, 0, - LWOOBJID_EMPTY, destroyedCinematic, UNASSIGNED_SYSTEM_ADDRESS); - self->AddTimer("tornadoOff", 0.5f); - } else { - ZoneAgProperty::BaseOnFireEventServerSide(self, sender, args); - } -} - -void ZoneAgSpiderQueen::OnPlayerExit(Entity* self, Entity* player) { - UpdatePlayer(self, player->GetObjectID(), true); -} - -void ZoneAgSpiderQueen::OnTimerDone(Entity* self, std::string timerName) { - - // Disable some stuff from the regular property - if (timerName == "BoundsVisOn" || timerName == "GuardFlyAway" || timerName == "ShowVendor") - return; - - if (timerName == "killSpider") { - auto spawnTargets = EntityManager::Instance()->GetEntitiesInGroup(self->GetVar<std::string>(LandTargetGroup)); - for (auto* spawnTarget : spawnTargets) { - EntityInfo info{}; - - info.spawnerID = spawnTarget->GetObjectID(); - info.pos = spawnTarget->GetPosition(); - info.rot = spawnTarget->GetRotation(); - info.lot = chestObject; - info.settings = { - new LDFData<LWOOBJID>(u"parent_tag", self->GetObjectID()) - }; - - auto* chest = EntityManager::Instance()->CreateEntity(info); - EntityManager::Instance()->ConstructEntity(chest); - } - } - - ZoneAgProperty::BaseTimerDone(self, timerName); -} diff --git a/dScripts/ZoneFvProperty.cpp b/dScripts/ZoneFvProperty.cpp deleted file mode 100644 index 67ada039..00000000 --- a/dScripts/ZoneFvProperty.cpp +++ /dev/null @@ -1,40 +0,0 @@ -#include "ZoneFvProperty.h" -#include "Entity.h" - -void ZoneFvProperty::SetGameVariables(Entity* self) { - self->SetVar<std::string>(ClaimMarkerGroup, "Platform"); - self->SetVar<std::string>(GeneratorGroup, "Generator"); - self->SetVar<std::string>(GuardGroup, "Guard"); - self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); - self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); - self->SetVar<std::string>(SpotsGroup, "Spots"); - self->SetVar<std::string>(MSCloudsGroup, "Clouds"); - self->SetVar<std::string>(EnemiesGroup, "Enemies"); - self->SetVar<std::string>(FXManagerGroup, "FXManager"); - self->SetVar<std::string>(ImagOrbGroup, "Orb"); - self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); - - self->SetVar<std::vector<std::string>>(EnemiesSpawner, - { "RoninWander", "RoninGen", "HorsemenGen" }); - self->SetVar<std::string>(ClaimMarkerSpawner, "Platform"); - self->SetVar<std::string>(GeneratorSpawner, "Generator"); - self->SetVar<std::string>(DamageFXSpawner, "Clouds"); - self->SetVar<std::string>(FXSpotsSpawner, "Spots"); - self->SetVar<std::string>(PropertyMGSpawner, "Guard"); - self->SetVar<std::string>(ImageOrbSpawner, "Orb"); - self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); - self->SetVar<std::string>(SmashablesSpawner, "Smashables"); - self->SetVar<std::string>(FXManagerSpawner, "FXManager"); - self->SetVar<std::string>(PropObjsSpawner, "BankObj"); - self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Ash", "FX", "Fog" }); - self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, {}); - - self->SetVar<uint32_t>(defeatedProperyFlag, 99); - self->SetVar<uint32_t>(placedModelFlag, 107); - self->SetVar<uint32_t>(guardMissionFlag, 874); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 950); - self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); - self->SetVar<LOT>(generatorIdFlag, 11023); - self->SetVar<LOT>(orbIDFlag, 10226); - self->SetVar<LOT>(behaviorQBID, 11011); -} diff --git a/dScripts/ZoneGfProperty.cpp b/dScripts/ZoneGfProperty.cpp deleted file mode 100644 index 565d8cd6..00000000 --- a/dScripts/ZoneGfProperty.cpp +++ /dev/null @@ -1,40 +0,0 @@ -#include "ZoneGfProperty.h" -#include "Entity.h" - -void ZoneGfProperty::SetGameVariables(Entity* self) { - self->SetVar<std::string>(ClaimMarkerGroup, "BehavQB"); - self->SetVar<std::string>(GeneratorGroup, "Generator"); - self->SetVar<std::string>(GuardGroup, "Guard"); - self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); - self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); - self->SetVar<std::string>(SpotsGroup, "Spots"); - self->SetVar<std::string>(MSCloudsGroup, "Clouds"); - self->SetVar<std::string>(EnemiesGroup, "Enemies"); - self->SetVar<std::string>(FXManagerGroup, "FXManager"); - self->SetVar<std::string>(ImagOrbGroup, "Orb"); - self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); - - self->SetVar<std::vector<std::string>>(EnemiesSpawner, - { "PiratesWander", "PiratesGen", "AdmiralsWander", "AdmiralsGen" }); - self->SetVar<std::string>(ClaimMarkerSpawner, "BehavPlat"); - self->SetVar<std::string>(GeneratorSpawner, "Generator"); - self->SetVar<std::string>(DamageFXSpawner, "Clouds"); - self->SetVar<std::string>(FXSpotsSpawner, "Spots"); - self->SetVar<std::string>(PropertyMGSpawner, "Guard"); - self->SetVar<std::string>(ImageOrbSpawner, "Orb"); - self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); - self->SetVar<std::string>(SmashablesSpawner, "Smashables"); - self->SetVar<std::string>(FXManagerSpawner, "FXManager"); - self->SetVar<std::string>(PropObjsSpawner, "BankObj"); - self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Birds", "Falls", "Sunbeam" }); - self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, { "TrappedPlatform", "IceBarrier", "FireBeast" }); - - self->SetVar<uint32_t>(defeatedProperyFlag, 98); - self->SetVar<uint32_t>(placedModelFlag, 106); - self->SetVar<uint32_t>(guardMissionFlag, 873); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 949); - self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); - self->SetVar<LOT>(generatorIdFlag, 11109); - self->SetVar<LOT>(orbIDFlag, 10226); - self->SetVar<LOT>(behaviorQBID, 11001); -} diff --git a/dScripts/ZoneNsMedProperty.cpp b/dScripts/ZoneNsMedProperty.cpp deleted file mode 100644 index 30e597ee..00000000 --- a/dScripts/ZoneNsMedProperty.cpp +++ /dev/null @@ -1,40 +0,0 @@ -#include "ZoneNsMedProperty.h" -#include "Entity.h" - -void ZoneNsMedProperty::SetGameVariables(Entity* self) { - self->SetVar<std::string>(ClaimMarkerGroup, "ClaimMarker"); - self->SetVar<std::string>(GeneratorGroup, "Generator"); - self->SetVar<std::string>(GuardGroup, "Guard"); - self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); - self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); - self->SetVar<std::string>(SpotsGroup, "Spots"); - self->SetVar<std::string>(MSCloudsGroup, "maelstrom"); - self->SetVar<std::string>(EnemiesGroup, "Enemies"); - self->SetVar<std::string>(FXManagerGroup, "FXObject"); - self->SetVar<std::string>(ImagOrbGroup, "Orb"); - self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); - - self->SetVar<std::vector<std::string>>(EnemiesSpawner, - { "Admirals", "AdmiralsWander", "Mechs", "Ronin", "RoninWander" }); - self->SetVar<std::string>(ClaimMarkerSpawner, "ClaimMarker"); - self->SetVar<std::string>(GeneratorSpawner, "Generator"); - self->SetVar<std::string>(DamageFXSpawner, "MaelstromFX"); - self->SetVar<std::string>(FXSpotsSpawner, "MaelstromSpots"); - self->SetVar<std::string>(PropertyMGSpawner, "PropertyGuard"); - self->SetVar<std::string>(ImageOrbSpawner, "Orb"); - self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); - self->SetVar<std::string>(SmashablesSpawner, "Smashables"); - self->SetVar<std::string>(FXManagerSpawner, "FXObject"); - self->SetVar<std::string>(PropObjsSpawner, "BankObj"); - self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Rockets" }); - self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, { }); - - self->SetVar<uint32_t>(defeatedProperyFlag, 122); - self->SetVar<uint32_t>(placedModelFlag, 123); - self->SetVar<uint32_t>(guardMissionFlag, 1322); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 1294); - self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); - self->SetVar<LOT>(generatorIdFlag, 11031); - self->SetVar<LOT>(orbIDFlag, 10226); - self->SetVar<LOT>(behaviorQBID, 10445); -} diff --git a/dScripts/ZoneNsProperty.cpp b/dScripts/ZoneNsProperty.cpp deleted file mode 100644 index 49364ee8..00000000 --- a/dScripts/ZoneNsProperty.cpp +++ /dev/null @@ -1,41 +0,0 @@ -#include "ZoneNsProperty.h" -#include "Entity.h" - -void ZoneNsProperty::SetGameVariables(Entity* self) { - self->SetVar<std::string>(ClaimMarkerGroup, "Rhino"); - self->SetVar<std::string>(GeneratorGroup, "Generator"); - self->SetVar<std::string>(GuardGroup, "Guard"); - self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); - self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); - self->SetVar<std::string>(SpotsGroup, "Spots"); - self->SetVar<std::string>(MSCloudsGroup, "Clouds"); - self->SetVar<std::string>(EnemiesGroup, "Enemies"); - self->SetVar<std::string>(FXManagerGroup, "FXManager"); - self->SetVar<std::string>(ImagOrbGroup, "Orb"); - self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); - - self->SetVar<std::vector<std::string>>(EnemiesSpawner, { - "StrombieWander", "StrombieGen", "PirateWander", "PirateGen", "RoninGen" - }); - self->SetVar<std::string>(ClaimMarkerSpawner, "ClaimMarker"); - self->SetVar<std::string>(GeneratorSpawner, "Generator"); - self->SetVar<std::string>(DamageFXSpawner, "MSClouds"); - self->SetVar<std::string>(FXSpotsSpawner, "Spots"); - self->SetVar<std::string>(PropertyMGSpawner, "Guard"); - self->SetVar<std::string>(ImageOrbSpawner, "Orb"); - self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); - self->SetVar<std::string>(SmashablesSpawner, "Smashables"); - self->SetVar<std::string>(FXManagerSpawner, "FXManager"); - self->SetVar<std::string>(PropObjsSpawner, "BankObj"); - self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Rockets" }); - self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, { "Cage", "Platform", "Door" }); - - self->SetVar<uint32_t>(defeatedProperyFlag, 97); - self->SetVar<uint32_t>(placedModelFlag, 105); - self->SetVar<uint32_t>(guardMissionFlag, 872); - self->SetVar<uint32_t>(brickLinkMissionIDFlag, 948); - self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); - self->SetVar<LOT>(generatorIdFlag, 11031); - self->SetVar<LOT>(orbIDFlag, 10226); - self->SetVar<LOT>(behaviorQBID, 11009); -} diff --git a/dScripts/ZoneNsWaves.cpp b/dScripts/ZoneNsWaves.cpp deleted file mode 100644 index 6f78acb6..00000000 --- a/dScripts/ZoneNsWaves.cpp +++ /dev/null @@ -1,500 +0,0 @@ -#include "ZoneNsWaves.h" - -WaveConstants ZoneNsWaves::GetConstants() { - return { - 60, - 2, - 6, - 2, - "surprise", - "intro" - }; -} - -std::vector<std::string> ZoneNsWaves::GetSpawnerNames() { - return { - "Base_MobA", - "Base_MobB", - "Base_MobC", - "MobA_01", - "MobB_01", - "MobC_01", - "MobA_02", - "MobB_02", - "MobC_02", - "MobA_03", - "MobB_03", - "MobC_03", - "Reward_01", - "Base_Reward", - "Obstacle_01", - "Boss", - "Ape_Boss", - "Geyser_01", - "Treasure_01", - "Cavalry_Boss", - "Horseman_01", - "Horseman_02", - "Horseman_03", - "Horseman_04" - }; -} - -std::vector<WaveMission> ZoneNsWaves::GetWaveMissions() { - return { - {190, 7, 1242}, - {240, 7, 1226}, - {450, 15, 1243}, - {600, 15, 1227}, - {720, 22, 1244}, - {840, 22, 1228}, - {1080, 29, 1245}, - {1200, 29, 1229}, - }; -} - -std::vector<Wave> ZoneNsWaves::GetWaves() { - return { - // Wave 1 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::stromling_minifig, 8, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::stromling_minifig, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::stromling_minifig, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::stromling_minifig, 2, GetSpawnerName(SpawnerName::gf_A) }, - } - }, - - // Wave 2 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::stromling, 8, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::gf_A) }, - } - }, - - // Wave 3 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::stromling, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::gf_A) }, - }, - }, - - // Wave 4 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::stromling, 3, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 5 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::stromling, 1, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::stromling, 1, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 6 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::stromling, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 7 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::stromling_boss, 1, GetSpawnerName(SpawnerName::Boss) }, - }, - {1885}, - {}, - "Stromling_Boss", - 5.0f - }, - - // Wave 8 - Wave { - std::vector<MobDefinition> { - {SpawnLOTS::mushroom, 6, GetSpawnerName(SpawnerName::Reward_01) }, - {SpawnLOTS::mushroom, 3, GetSpawnerName(SpawnerName::interior_Reward) }, - }, {}, {}, "", -1.0f, - 25, - }, - - // Wave 9 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 10 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 11 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::concert_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::gf_C) }, - } - }, - - // Wave 12 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_C) }, - } - }, - - // Wave 13 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 3, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 14 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::mech, 2, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::concert_C) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_B) }, - { SpawnLOTS::mech, 1, GetSpawnerName(SpawnerName::gf_C) }, - } - }, - - // Wave 15 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::ape_boss, 1, GetSpawnerName(SpawnerName::Ape_Boss) }, - - }, - {1886}, - {}, - "Gorilla_Boss", - 5.0f - }, - - // Wave 16 - Wave { - std::vector<MobDefinition> { - {SpawnLOTS::outhouse, 3, GetSpawnerName(SpawnerName::interior_Reward) }, - {SpawnLOTS::mushroom, 6, GetSpawnerName(SpawnerName::Reward_01) }, - }, {}, {}, "", -1.0f, - 25, - }, - - // Wave 17 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::hammerling_melee, 1, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 18 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::hammerling_melee, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::hammerling_melee, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 19 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::hammerling, 4, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::sentry, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::hammerling, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::gf_B) }, - } - }, - - // Wave 20 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::sentry, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::concert_C) }, - { SpawnLOTS::hammerling, 1, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::sentry, 1, GetSpawnerName(SpawnerName::gf_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::gf_C) }, - } - }, - - // Wave 21 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::admiral, 2, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::ronin, 2, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::spiderling_ve, 2, GetSpawnerName(SpawnerName::interior_C) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::ronin, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::ronin, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::concert_C) }, - { SpawnLOTS::admiral, 1, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::ronin, 1, GetSpawnerName(SpawnerName::gf_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::gf_C) }, - } - }, - - // Wave 22 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::spiderling_boss, 1, GetSpawnerName(SpawnerName::Cavalry_Boss) }, - }, - {1887}, - {}, - "Spiderling_Boss", - 5.0f - }, - - // Wave 23 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::outhouse, 6, GetSpawnerName(SpawnerName::Reward_01) }, - { SpawnLOTS::outhouse, 3, GetSpawnerName(SpawnerName::interior_Reward) }, - { SpawnLOTS::maelstrom_chest, 4, GetSpawnerName(SpawnerName::Obstacle) }, - }, {}, {}, "", -1.0f, - 25, - }, - - // Wave 24 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::pirate, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::pirate, 3, GetSpawnerName(SpawnerName::ag_A) }, - { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::ronin, 2, GetSpawnerName(SpawnerName::interior_B) }, - } - }, - - // Wave 25 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::cavalry, 2, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::gf_B) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::gf_A) }, - { SpawnLOTS::spiderling, 2, GetSpawnerName(SpawnerName::concert_A) }, - { SpawnLOTS::spiderling, 1, GetSpawnerName(SpawnerName::ag_A) }, - } - }, - - // Wave 26 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::ronin, 3, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::gf_B) }, - { SpawnLOTS::spiderling_ve, 1, GetSpawnerName(SpawnerName::concert_B) }, - { SpawnLOTS::admiral_cp, 2, GetSpawnerName(SpawnerName::gf_C) }, - { SpawnLOTS::admiral_cp, 2, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::concert_C) }, - } - }, - - // Wave 27 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::ronin, 5, GetSpawnerName(SpawnerName::interior_A) }, - { SpawnLOTS::ronin, 4, GetSpawnerName(SpawnerName::interior_B) }, - { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::ag_C) }, - { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::gf_C) }, - { SpawnLOTS::cavalry, 1, GetSpawnerName(SpawnerName::concert_C) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::ag_B) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::gf_B) }, - { SpawnLOTS::admiral_cp, 1, GetSpawnerName(SpawnerName::concert_B) }, - } - }, - - // Wave 28 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::dragon_statue, 12, GetSpawnerName(SpawnerName::Reward_01) }, - }, {}, {}, "", -1.0f, - 30, - }, - - // Wave 29 - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::horseman_boss01, 1, GetSpawnerName(SpawnerName::Horseman_01) }, - { SpawnLOTS::horseman_boss02, 1, GetSpawnerName(SpawnerName::Horseman_02) }, - { SpawnLOTS::horseman_boss03, 1, GetSpawnerName(SpawnerName::Horseman_03) }, - { SpawnLOTS::horseman_boss04, 1, GetSpawnerName(SpawnerName::Horseman_04) }, - }, - {1888}, - {1236, 1237, 1249}, - "Horsemen_Boss", - 5.0f - }, - - // Wave 30 (treasure) - Wave { - std::vector<MobDefinition> { - { SpawnLOTS::treasure_chest, 1, GetSpawnerName(SpawnerName::Treasure_01) }, - }, {}, {}, - "Treasure_Camera", - 5.0f, - (uint32_t)-1, - true, - 30, - }, - }; -} - -std::string ZoneNsWaves::GetSpawnerName(SpawnerName spawnerName) { - switch (spawnerName) { - case interior_A: - return "Base_MobA"; - case interior_B: - return "Base_MobB"; - case interior_C: - return "Base_MobC"; - case gf_A: - return "MobA_01"; - case gf_B: - return "MobB_01"; - case gf_C: - return "MobC_01"; - case concert_A: - return "MobA_02"; - case concert_B: - return "MobB_02"; - case concert_C: - return "MobC_02"; - case ag_A: - return "MobA_03"; - case ag_B: - return "MobB_03"; - case ag_C: - return "MobC_03"; - case Reward_01: - return "Reward_01"; - case interior_Reward: - return "Base_Reward"; - case Obstacle: - return "Obstacle_01"; - case Boss: - return "Boss"; - case Ape_Boss: - return "Ape_Boss"; - case Geyser: - return "Geyser_01"; - case Treasure_01: - return "Treasure_01"; - case Cavalry_Boss: - return "Cavalry_Boss"; - case Horseman_01: - return "Horseman_01"; - case Horseman_02: - return "Horseman_02"; - case Horseman_03: - return "Horseman_03"; - case Horseman_04: - return "Horseman_04"; - default: - return ""; - } -} diff --git a/dScripts/ZoneNsWaves.h b/dScripts/ZoneNsWaves.h deleted file mode 100644 index 06503d19..00000000 --- a/dScripts/ZoneNsWaves.h +++ /dev/null @@ -1,69 +0,0 @@ -#pragma once -#include "BaseWavesServer.h" - -enum SpawnerName { - interior_A, - interior_B, - interior_C, - gf_A, - gf_B, - gf_C, - concert_A, - concert_B, - concert_C, - ag_A, - ag_B, - ag_C, - Reward_01, - interior_Reward, - Obstacle, - Boss, - Ape_Boss, - Geyser, - Treasure_01, - Cavalry_Boss, - Horseman_01, - Horseman_02, - Horseman_03, - Horseman_04, -}; - -enum SpawnLOTS : LOT { - stromling = 12586, - mech = 12587, - spiderling = 12588, - pirate = 12589, - admiral = 12590, - ape_boss = 12591, - stromling_boss = 12600, - hammerling = 12602, - sentry = 12604, - spiderling_ve = 12605, - spiderling_boss = 12609, - ronin = 12610, - cavalry = 12611, - dragon_boss = 12612, - stromling_minifig = 12586, - mushroom = 12614, - maelstrom_chest = 4894, - outhouse = 12616, - dragon_statue = 12617, - treasure_chest = 12423, - hammerling_melee = 12653, - maelstrom_geyser = 10314, - ronin_statue = 12611, - horseman_boss01 = 11999, - horseman_boss02 = 12467, - horseman_boss03 = 12468, - horseman_boss04 = 12469, - admiral_cp = 13523, -}; - -class ZoneNsWaves : public BaseWavesServer { - WaveConstants GetConstants() override; - std::vector<std::string> GetSpawnerNames() override; - std::vector<WaveMission> GetWaveMissions() override; - std::vector<Wave> GetWaves() override; -private: - static std::string GetSpawnerName(SpawnerName spawnerName); -}; diff --git a/dScripts/ZoneSGServer.cpp b/dScripts/ZoneSGServer.cpp deleted file mode 100644 index 6822abda..00000000 --- a/dScripts/ZoneSGServer.cpp +++ /dev/null @@ -1,26 +0,0 @@ -#include "ZoneSGServer.h" -#include "EntityManager.h" - -void ZoneSGServer::OnStartup(Entity* self) { - const auto cannons = EntityManager::Instance()->GetEntitiesByLOT(1864); - for (const auto& cannon : cannons) - self->SetVar<LWOOBJID>(CannonIDVariable, cannon->GetObjectID()); -} - -void ZoneSGServer::OnActivityStateChangeRequest(Entity* self, const LWOOBJID senderID, const int32_t value1, - const int32_t value2, const std::u16string& stringValue) { - - auto* cannon = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(CannonIDVariable)); - if (cannon != nullptr) { - cannon->OnActivityStateChangeRequest(senderID, value1, value2, stringValue); - } -} - -void ZoneSGServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, - int32_t param3) { - - auto* cannon = EntityManager::Instance()->GetEntity(self->GetVar<LWOOBJID>(CannonIDVariable)); - if (cannon != nullptr) { - cannon->OnFireEventServerSide(sender, args, param1, param2, param3); - } -} diff --git a/dScripts/ai/ACT/ActMine.cpp b/dScripts/ai/ACT/ActMine.cpp new file mode 100644 index 00000000..9651e13d --- /dev/null +++ b/dScripts/ai/ACT/ActMine.cpp @@ -0,0 +1,52 @@ +#include "ActMine.h" +#include "SkillComponent.h" +#include "DestroyableComponent.h" +#include "RebuildComponent.h" + +void ActMine::OnStartup(Entity* self) { + self->SetVar(u"RebuildComplete", false); + self->SetProximityRadius(MINE_RADIUS, "mineRadius"); +} + +void ActMine::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::COMPLETED) { + auto* rebuild = self->GetComponent<RebuildComponent>(); + if (rebuild) { + auto* builder = rebuild->GetBuilder(); + self->SetVar(u"Builder", builder->GetObjectID()); + } + + self->SetVar(u"RebuildComplete", true); + self->SetVar(u"NumWarnings", 0); + self->AddToGroup("reset"); + } + +} + +void ActMine::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + auto* detroyable = self->GetComponent<DestroyableComponent>(); + if (!detroyable) return; + if (status == "ENTER" && self->GetVar<bool>(u"RebuildComplete") == true && detroyable->IsEnemy(entering)) { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 242, u"orange", "sirenlight_B"); + self->AddTimer("Tick", TICK_TIME); + } +} + +void ActMine::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "Tick") { + if (self->GetVar<int>(u"NumWarnings") >= MAX_WARNINGS) { + auto* skill = self->GetComponent<SkillComponent>(); + if (!skill) return; + skill->CalculateBehavior(SKILL_ID, BEHAVIOR_ID, LWOOBJID_EMPTY); + self->AddTimer("BlowedUp", BLOWED_UP_TIME); + } else { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 242, u"orange", "sirenlight_B"); + self->AddTimer("Tick", TICK_TIME); + self->SetVar(u"NumWarnings", self->GetVar<int>(u"NumWarnings") + 1); + } + } + + if (timerName == "BlowedUp") { + self->Kill(self); + } +} diff --git a/dScripts/ActMine.h b/dScripts/ai/ACT/ActMine.h similarity index 100% rename from dScripts/ActMine.h rename to dScripts/ai/ACT/ActMine.h diff --git a/dScripts/ai/ACT/ActPlayerDeathTrigger.cpp b/dScripts/ai/ACT/ActPlayerDeathTrigger.cpp new file mode 100644 index 00000000..0674f0dd --- /dev/null +++ b/dScripts/ai/ACT/ActPlayerDeathTrigger.cpp @@ -0,0 +1,9 @@ +#include "ActPlayerDeathTrigger.h" + +#include "Entity.h" + +void ActPlayerDeathTrigger::OnCollisionPhantom(Entity* self, Entity* target) { + if (!target->IsPlayer() || target->GetIsDead() || !target->GetPlayerReadyForUpdates()) return; //Don't kill already dead players or players not ready + + target->Smash(self->GetObjectID(), eKillType::SILENT); +} diff --git a/dScripts/ActPlayerDeathTrigger.h b/dScripts/ai/ACT/ActPlayerDeathTrigger.h similarity index 100% rename from dScripts/ActPlayerDeathTrigger.h rename to dScripts/ai/ACT/ActPlayerDeathTrigger.h diff --git a/dScripts/ai/ACT/ActVehicleDeathTrigger.cpp b/dScripts/ai/ACT/ActVehicleDeathTrigger.cpp new file mode 100644 index 00000000..90333063 --- /dev/null +++ b/dScripts/ai/ACT/ActVehicleDeathTrigger.cpp @@ -0,0 +1,52 @@ +#include "ActVehicleDeathTrigger.h" +#include "PossessableComponent.h" +#include "GameMessages.h" +#include "RacingControlComponent.h" +#include "dZoneManager.h" +#include "EntityManager.h" +#include "PossessorComponent.h" + + +void ActVehicleDeathTrigger::OnCollisionPhantom(Entity* self, Entity* target) { + auto* possessableComponent = target->GetComponent<PossessableComponent>(); + + Entity* vehicle; + Entity* player; + + if (possessableComponent != nullptr) { + auto* player = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); + + if (player == nullptr) { + return; + } + + return; + } else if (target->IsPlayer()) { + auto* possessorComponent = target->GetComponent<PossessorComponent>(); + + if (possessorComponent == nullptr) { + return; + } + + vehicle = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); + + if (vehicle == nullptr) { + return; + } + + player = target; + } else { + return; + } + + + GameMessages::SendDie(vehicle, self->GetObjectID(), LWOOBJID_EMPTY, true, eKillType::VIOLENT, u"", 0, 0, 0, true, false, 0); + + auto* zoneController = Game::zoneManager->GetZoneControlObject(); + + auto* racingControlComponent = zoneController->GetComponent<RacingControlComponent>(); + + if (racingControlComponent != nullptr) { + racingControlComponent->OnRequestDie(player); + } +} diff --git a/dScripts/ActVehicleDeathTrigger.h b/dScripts/ai/ACT/ActVehicleDeathTrigger.h similarity index 100% rename from dScripts/ActVehicleDeathTrigger.h rename to dScripts/ai/ACT/ActVehicleDeathTrigger.h diff --git a/dScripts/ai/ACT/CMakeLists.txt b/dScripts/ai/ACT/CMakeLists.txt new file mode 100644 index 00000000..79deeded --- /dev/null +++ b/dScripts/ai/ACT/CMakeLists.txt @@ -0,0 +1,12 @@ +set(DSCRIPTS_SOURCES_AI_ACT + "ActMine.cpp" + "ActPlayerDeathTrigger.cpp" + "ActVehicleDeathTrigger.cpp") + +add_subdirectory(FootRace) + +foreach(file ${DSCRIPTS_SOURCES_AI_ACT_FOOTRACE}) + set(DSCRIPTS_SOURCES_AI_ACT ${DSCRIPTS_SOURCES_AI_ACT} "FootRace/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI_ACT ${DSCRIPTS_SOURCES_AI_ACT} PARENT_SCOPE) diff --git a/dScripts/ai/ACT/FootRace/BaseFootRaceManager.cpp b/dScripts/ai/ACT/FootRace/BaseFootRaceManager.cpp new file mode 100644 index 00000000..c02bf565 --- /dev/null +++ b/dScripts/ai/ACT/FootRace/BaseFootRaceManager.cpp @@ -0,0 +1,49 @@ +#include "BaseFootRaceManager.h" +#include "EntityManager.h" +#include "Character.h" +#include "Entity.h" + +void BaseFootRaceManager::OnStartup(Entity* self) { + // TODO: Add to FootRaceStarter group +} + +void BaseFootRaceManager::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + const auto splitArguments = GeneralUtils::SplitString(args, '_'); + if (splitArguments.size() > 1) { + + const auto eventName = splitArguments[0]; + const auto player = Game::entityManager->GetEntity(std::stoull(splitArguments[1])); + + if (player != nullptr) { + if (eventName == "updatePlayer") { + UpdatePlayer(self, player->GetObjectID()); + } else if (IsPlayerInActivity(self, player->GetObjectID())) { + if (eventName == "initialActivityScore") { + auto* character = player->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(115, true); + } + + SetActivityScore(self, player->GetObjectID(), 1); + } else if (eventName == "updatePlayerTrue") { + auto* character = player->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(115, false); + } + + UpdatePlayer(self, player->GetObjectID(), true); + } else if (eventName == "PlayerWon") { + auto* character = player->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(115, false); + if (param2 != -1) // Certain footraces set a flag + character->SetPlayerFlag(param2, true); + } + + StopActivity(self, player->GetObjectID(), 0, param1); + SaveScore(self, player->GetObjectID(), static_cast<float>(param1), static_cast<float>(param2), static_cast<float>(param3)); + } + } + } + } +} diff --git a/dScripts/BaseFootRaceManager.h b/dScripts/ai/ACT/FootRace/BaseFootRaceManager.h similarity index 100% rename from dScripts/BaseFootRaceManager.h rename to dScripts/ai/ACT/FootRace/BaseFootRaceManager.h diff --git a/dScripts/ai/ACT/FootRace/CMakeLists.txt b/dScripts/ai/ACT/FootRace/CMakeLists.txt new file mode 100644 index 00000000..c56986ff --- /dev/null +++ b/dScripts/ai/ACT/FootRace/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_AI_ACT_FOOTRACE + "BaseFootRaceManager.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/AG/ActSharkPlayerDeathTrigger.cpp b/dScripts/ai/AG/ActSharkPlayerDeathTrigger.cpp new file mode 100644 index 00000000..49fb6b87 --- /dev/null +++ b/dScripts/ai/AG/ActSharkPlayerDeathTrigger.cpp @@ -0,0 +1,20 @@ +#include "ActSharkPlayerDeathTrigger.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" +#include "Entity.h" + +void ActSharkPlayerDeathTrigger::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + if (args == "achieve") { + auto missionComponent = sender->GetComponent<MissionComponent>(); + if (!missionComponent) return; + + missionComponent->Progress(eMissionTaskType::SCRIPT, 8419); + + if (sender->GetIsDead() || !sender->GetPlayerReadyForUpdates()) return; //Don't kill already dead players or players not ready + + if (sender->GetCharacter()) { + sender->Smash(self->GetObjectID(), eKillType::VIOLENT, u"big-shark-death"); + } + } +} diff --git a/dScripts/ActSharkPlayerDeathTrigger.h b/dScripts/ai/AG/ActSharkPlayerDeathTrigger.h similarity index 100% rename from dScripts/ActSharkPlayerDeathTrigger.h rename to dScripts/ai/AG/ActSharkPlayerDeathTrigger.h diff --git a/dScripts/ai/AG/AgBusDoor.cpp b/dScripts/ai/AG/AgBusDoor.cpp new file mode 100644 index 00000000..fd6c272e --- /dev/null +++ b/dScripts/ai/AG/AgBusDoor.cpp @@ -0,0 +1,65 @@ +#include "AgBusDoor.h" +#include "Entity.h" +#include "GameMessages.h" +#include "ProximityMonitorComponent.h" + +void AgBusDoor::OnStartup(Entity* self) { + m_Counter = 0; + m_OuterCounter = 0; + self->SetProximityRadius(75, "busDoor"); + self->SetProximityRadius(85, "busDoorOuter"); +} + +void AgBusDoor::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name != "busDoor" && name != "busDoorOuter") return; + + // Make sure only humans are taken into account + if (!entering->GetCharacter()) return; + + auto* proximityMonitorComponent = self->GetComponent<ProximityMonitorComponent>(); + + if (proximityMonitorComponent == nullptr) return; + + m_Counter = 0; + m_OuterCounter = 0; + + for (const auto& pair : proximityMonitorComponent->GetProximityObjects("busDoor")) { + auto* entity = Game::entityManager->GetEntity(pair.first); + if (entity != nullptr && entity->IsPlayer()) m_Counter++; + } + + for (const auto& pair : proximityMonitorComponent->GetProximityObjects("busDoorOuter")) { + auto* entity = Game::entityManager->GetEntity(pair.first); + if (entity != nullptr && entity->IsPlayer()) m_OuterCounter++; + } + + if (status == "ENTER") { + // move up when a player is inside both radii + if (m_Counter > 0) { + MoveDoor(self, true); + } + } else if (status == "LEAVE") { + // move down when no players are inside either radii + if (m_Counter <= 0) { + MoveDoor(self, false); + } + } +} + +void AgBusDoor::MoveDoor(Entity* self, bool bOpen) { + if (bOpen) { + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 1, 0); + } else { + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, 1); + self->AddTimer("dustTimer", 2.0f); + } + + //This is currently commented out because it might be the reason that people's audio is cutting out. + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, "{9a24f1fa-3177-4745-a2df-fbd996d6e1e3}"); +} + +void AgBusDoor::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "dustTimer") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 642, u"create", "busDust", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + } +} diff --git a/dScripts/AgBusDoor.h b/dScripts/ai/AG/AgBusDoor.h similarity index 100% rename from dScripts/AgBusDoor.h rename to dScripts/ai/AG/AgBusDoor.h diff --git a/dScripts/AgDarkSpiderling.cpp b/dScripts/ai/AG/AgDarkSpiderling.cpp similarity index 100% rename from dScripts/AgDarkSpiderling.cpp rename to dScripts/ai/AG/AgDarkSpiderling.cpp diff --git a/dScripts/AgDarkSpiderling.h b/dScripts/ai/AG/AgDarkSpiderling.h similarity index 100% rename from dScripts/AgDarkSpiderling.h rename to dScripts/ai/AG/AgDarkSpiderling.h diff --git a/dScripts/ai/AG/AgFans.cpp b/dScripts/ai/AG/AgFans.cpp new file mode 100644 index 00000000..15f2ed0f --- /dev/null +++ b/dScripts/ai/AG/AgFans.cpp @@ -0,0 +1,86 @@ +#include "AgFans.h" + +#include "EntityManager.h" +#include "GameMessages.h" +#include "PhantomPhysicsComponent.h" +#include "RenderComponent.h" +#include "eReplicaComponentType.h" +#include "RenderComponent.h" +#include "Entity.h" + +void AgFans::OnStartup(Entity* self) { + self->SetVar<bool>(u"alive", true); + self->SetVar<bool>(u"on", false); + + ToggleFX(self, false); + + auto* renderComponent = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + + if (renderComponent == nullptr) { + return; + } + + renderComponent->PlayEffect(495, u"fanOn", "fanOn"); +} + +void AgFans::ToggleFX(Entity* self, bool hit) { + std::string fanGroup = self->GetGroups()[0]; + std::vector<Entity*> fanVolumes = Game::entityManager->GetEntitiesInGroup(fanGroup); + + auto* renderComponent = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + + if (renderComponent == nullptr) { + return; + } + + if (fanVolumes.size() == 0 || !self->GetVar<bool>(u"alive")) return; + + if (self->GetVar<bool>(u"on")) { + RenderComponent::PlayAnimation(self, u"fan-off"); + + renderComponent->StopEffect("fanOn"); + self->SetVar<bool>(u"on", false); + + for (Entity* volume : fanVolumes) { + PhantomPhysicsComponent* volumePhys = static_cast<PhantomPhysicsComponent*>(volume->GetComponent(eReplicaComponentType::PHANTOM_PHYSICS)); + if (!volumePhys) continue; + volumePhys->SetPhysicsEffectActive(false); + Game::entityManager->SerializeEntity(volume); + if (!hit) { + Entity* fxObj = Game::entityManager->GetEntitiesInGroup(fanGroup + "fx")[0]; + RenderComponent::PlayAnimation(fxObj, u"trigger"); + } + } + } else if (!self->GetVar<bool>(u"on") && self->GetVar<bool>(u"alive")) { + RenderComponent::PlayAnimation(self, u"fan-on"); + + renderComponent->PlayEffect(495, u"fanOn", "fanOn"); + self->SetVar<bool>(u"on", true); + + for (Entity* volume : fanVolumes) { + PhantomPhysicsComponent* volumePhys = static_cast<PhantomPhysicsComponent*>(volume->GetComponent(eReplicaComponentType::PHANTOM_PHYSICS)); + if (!volumePhys) continue; + volumePhys->SetPhysicsEffectActive(true); + Game::entityManager->SerializeEntity(volume); + if (!hit) { + Entity* fxObj = Game::entityManager->GetEntitiesInGroup(fanGroup + "fx")[0]; + RenderComponent::PlayAnimation(fxObj, u"idle"); + } + } + } +} + +void AgFans::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args.length() == 0 || !self->GetVar<bool>(u"alive")) return; + + if ((args == "turnOn" && self->GetVar<bool>(u"on")) || (args == "turnOff" && !self->GetVar<bool>(u"on"))) return; + ToggleFX(self, false); +} + +void AgFans::OnDie(Entity* self, Entity* killer) { + if (self->GetVar<bool>(u"on")) { + ToggleFX(self, true); + } + self->SetVar<bool>(u"alive", false); +} diff --git a/dScripts/AgFans.h b/dScripts/ai/AG/AgFans.h similarity index 100% rename from dScripts/AgFans.h rename to dScripts/ai/AG/AgFans.h diff --git a/dScripts/ai/AG/AgImagSmashable.cpp b/dScripts/ai/AG/AgImagSmashable.cpp new file mode 100644 index 00000000..2a35ed9a --- /dev/null +++ b/dScripts/ai/AG/AgImagSmashable.cpp @@ -0,0 +1,44 @@ +#include "AgImagSmashable.h" +#include "EntityManager.h" +#include "GeneralUtils.h" +#include "GameMessages.h" +#include "EntityInfo.h" +#include "DestroyableComponent.h" +#include "eReplicaComponentType.h" + +void AgImagSmashable::OnDie(Entity* self, Entity* killer) { + bool maxImagGreaterThanZero = false; + + if (killer) { + DestroyableComponent* dest = static_cast<DestroyableComponent*>(killer->GetComponent(eReplicaComponentType::DESTROYABLE)); + if (dest) { + maxImagGreaterThanZero = dest->GetMaxImagination() > 0; + } + + if (maxImagGreaterThanZero) { + int amount = GeneralUtils::GenerateRandomNumber<int>(0, 3); + for (int i = 0; i < amount; ++i) { + GameMessages::SendDropClientLoot(killer, self->GetObjectID(), 935, 0, self->GetPosition()); + } + } + } + + CrateAnimal(self); +} + +void AgImagSmashable::CrateAnimal(Entity* self) { + int funnychance = GeneralUtils::GenerateRandomNumber<int>(0, 26); + if (funnychance == 1) { + EntityInfo info; + info.lot = 8114; + info.pos = self->GetPosition(); + info.spawner = nullptr; + info.spawnerID = self->GetSpawnerID(); + info.spawnerNodeID = 0; + + Entity* newEntity = Game::entityManager->CreateEntity(info, nullptr); + if (newEntity) { + Game::entityManager->ConstructEntity(newEntity); + } + } +} diff --git a/dScripts/AgImagSmashable.h b/dScripts/ai/AG/AgImagSmashable.h similarity index 100% rename from dScripts/AgImagSmashable.h rename to dScripts/ai/AG/AgImagSmashable.h diff --git a/dScripts/ai/AG/AgJetEffectServer.cpp b/dScripts/ai/AG/AgJetEffectServer.cpp new file mode 100644 index 00000000..5d25c9e0 --- /dev/null +++ b/dScripts/ai/AG/AgJetEffectServer.cpp @@ -0,0 +1,60 @@ +#include "AgJetEffectServer.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "SkillComponent.h" +#include "eReplicaComponentType.h" +#include "RenderComponent.h" + +void AgJetEffectServer::OnUse(Entity* self, Entity* user) { + if (inUse || self->GetLOT() != 6859) return; + GameMessages::SendNotifyClientObject( + self->GetObjectID(), u"toggleInUse", 1, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS + ); + inUse = true; + + auto entities = Game::entityManager->GetEntitiesInGroup("Jet_FX"); + if (entities.empty()) return; + GameMessages::SendPlayFXEffect(entities.at(0), 641, u"create", "radarDish", LWOOBJID_EMPTY, 1, 1, true); + self->AddTimer("radarDish", 2.0f); + self->AddTimer("PlayEffect", 2.5f); + self->AddTimer("CineDone", 7.5f + 5.0f); // 7.5f is time the cinematic takes to play +} + +void AgJetEffectServer::OnRebuildComplete(Entity* self, Entity* target) { + if (self->GetLOT() != 6209) return; + auto entities = Game::entityManager->GetEntitiesInGroup("Jet_FX"); + if (entities.empty()) return; + RenderComponent::PlayAnimation(entities.at(0), u"jetFX"); + + // So we can give kill credit to person who build this + builder = target->GetObjectID(); + + auto groups = self->GetGroups(); + if (!groups.empty() && groups.at(0) == "Base_Radar") { + self->AddTimer("PlayEffect", 2.5f); + self->AddTimer("CineDone", 7.5f + 5.0f); // 7.5f is time the cinematic takes to play + } +} + +void AgJetEffectServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "radarDish") { + GameMessages::SendStopFXEffect(self, true, "radarDish"); + } else if (timerName == "PlayEffect") { + auto entities = Game::entityManager->GetEntitiesInGroup("mortarMain"); + if (entities.empty()) return; + + const auto selected = GeneralUtils::GenerateRandomNumber<int>(0, entities.size() - 1); + auto* mortar = entities.at(selected); + + // so we give proper credit to the builder for the kills from this skill + mortar->SetOwnerOverride(builder); + + auto* skillComponent = mortar->GetComponent<SkillComponent>(); + if (skillComponent) skillComponent->CastSkill(318); + } else if (timerName == "CineDone") { + GameMessages::SendNotifyClientObject( + self->GetObjectID(), u"toggleInUse", -1, 0, LWOOBJID_EMPTY, "", UNASSIGNED_SYSTEM_ADDRESS + ); + inUse = false; + } +} diff --git a/dScripts/AgJetEffectServer.h b/dScripts/ai/AG/AgJetEffectServer.h similarity index 100% rename from dScripts/AgJetEffectServer.h rename to dScripts/ai/AG/AgJetEffectServer.h diff --git a/dScripts/ai/AG/AgPicnicBlanket.cpp b/dScripts/ai/AG/AgPicnicBlanket.cpp new file mode 100644 index 00000000..bec5577c --- /dev/null +++ b/dScripts/ai/AG/AgPicnicBlanket.cpp @@ -0,0 +1,19 @@ +#include "AgPicnicBlanket.h" +#include "Loot.h" +#include "GameMessages.h" +#include "Entity.h" +#include "eTerminateType.h" + +void AgPicnicBlanket::OnUse(Entity* self, Entity* user) { + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); + if (self->GetVar<bool>(u"active")) + return; + self->SetVar<bool>(u"active", true); + + auto lootTable = std::unordered_map<LOT, int32_t>{ {935, 3} }; + LootGenerator::Instance().DropLoot(user, self, lootTable, 0, 0); + + self->AddCallbackTimer(5.0f, [self]() { + self->SetVar<bool>(u"active", false); + }); +} diff --git a/dScripts/AgPicnicBlanket.h b/dScripts/ai/AG/AgPicnicBlanket.h similarity index 100% rename from dScripts/AgPicnicBlanket.h rename to dScripts/ai/AG/AgPicnicBlanket.h diff --git a/dScripts/ai/AG/AgQbElevator.cpp b/dScripts/ai/AG/AgQbElevator.cpp new file mode 100644 index 00000000..5a535347 --- /dev/null +++ b/dScripts/ai/AG/AgQbElevator.cpp @@ -0,0 +1,60 @@ +#include "AgQbElevator.h" +#include "EntityManager.h" +#include "GameMessages.h" + +void AgQbElevator::OnStartup(Entity* self) { + +} + +//when the QB is finished being built by a player +void AgQbElevator::OnRebuildComplete(Entity* self, Entity* target) { + self->SetProximityRadius(proxRadius, "elevatorProx"); + self->SetI64(u"qbPlayer", target->GetObjectID()); + + float delayTime = killTime - endTime; + if (delayTime < 1) delayTime = 1; + + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, + 0, 0, eMovementPlatformState::Stationary); + + //add a timer that will kill the QB if no players get on in the killTime + self->AddTimer("startKillTimer", killTime); +} + +void AgQbElevator::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + //make sure we haven't already started pathing. + if (self->GetBoolean(u"qbPlayerRdy")) return; + + if (status == "ENTER") { + Entity* builder = Game::entityManager->GetEntity(self->GetI64(u"qbPlayer")); + if (builder && builder == entering) { + //the builder has entered so cancel the start timer and just start moving + self->SetBoolean(u"qbPlayerRdy", true); + self->CancelTimer("StartElevator"); + + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, + 1, 1, eMovementPlatformState::Moving); + } else if (!self->GetBoolean(u"StartTimer")) { + self->SetBoolean(u"StartTimer", true); + self->AddTimer("StartElevator", startTime); + } + } +} + +void AgQbElevator::OnTimerDone(Entity* self, std::string timerName) { + + if (timerName == "StartElevator") { + GameMessages::SendPlatformResync(self, UNASSIGNED_SYSTEM_ADDRESS, true, 0, + 1, 1, eMovementPlatformState::Moving); + } else if (timerName == "startKillTimer") { + killTimerStartup(self); + } else if (timerName == "KillTimer") { + self->Smash(self->GetObjectID(), eKillType::VIOLENT); + } +} + +void AgQbElevator::killTimerStartup(Entity* self) const { + self->CancelAllTimers(); + self->AddTimer("KillTimer", endTime); + self->SetNetworkVar<float>(u"startEffect", endTime); // Blinking effect +} diff --git a/dScripts/AgQbElevator.h b/dScripts/ai/AG/AgQbElevator.h similarity index 100% rename from dScripts/AgQbElevator.h rename to dScripts/ai/AG/AgQbElevator.h diff --git a/dScripts/ai/AG/AgQbWall.cpp b/dScripts/ai/AG/AgQbWall.cpp new file mode 100644 index 00000000..3e7d9c2b --- /dev/null +++ b/dScripts/ai/AG/AgQbWall.cpp @@ -0,0 +1,15 @@ +#include "AgQbWall.h" + +void AgQbWall::OnRebuildComplete(Entity* self, Entity* player) { + self->SetVar(u"player", player->GetObjectID()); + auto targetWallSpawners = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner")); + if (targetWallSpawners != "") { + auto groupObjs = Game::entityManager->GetEntitiesInGroup(targetWallSpawners); + for (auto* obj : groupObjs) { + if (obj) { + obj->SetVar(u"player", player->GetObjectID()); + obj->OnFireEventServerSide(self, "spawnMobs"); + } + } + } +} diff --git a/dScripts/AgQbWall.h b/dScripts/ai/AG/AgQbWall.h similarity index 100% rename from dScripts/AgQbWall.h rename to dScripts/ai/AG/AgQbWall.h diff --git a/dScripts/ai/AG/AgSalutingNpcs.cpp b/dScripts/ai/AG/AgSalutingNpcs.cpp new file mode 100644 index 00000000..d3bbe9ab --- /dev/null +++ b/dScripts/ai/AG/AgSalutingNpcs.cpp @@ -0,0 +1,10 @@ +#include "AgSalutingNpcs.h" +#include "RenderComponent.h" + +void AgSalutingNpcs::OnEmoteReceived(Entity* self, const int32_t emote, Entity* target) { + if (emote != 356) { + return; + } + + RenderComponent::PlayAnimation(self, u"salutePlayer"); +} diff --git a/dScripts/AgSalutingNpcs.h b/dScripts/ai/AG/AgSalutingNpcs.h similarity index 100% rename from dScripts/AgSalutingNpcs.h rename to dScripts/ai/AG/AgSalutingNpcs.h diff --git a/dScripts/AgShipPlayerDeathTrigger.cpp b/dScripts/ai/AG/AgShipPlayerDeathTrigger.cpp similarity index 100% rename from dScripts/AgShipPlayerDeathTrigger.cpp rename to dScripts/ai/AG/AgShipPlayerDeathTrigger.cpp diff --git a/dScripts/AgShipPlayerDeathTrigger.h b/dScripts/ai/AG/AgShipPlayerDeathTrigger.h similarity index 100% rename from dScripts/AgShipPlayerDeathTrigger.h rename to dScripts/ai/AG/AgShipPlayerDeathTrigger.h diff --git a/dScripts/ai/AG/AgShipPlayerShockServer.cpp b/dScripts/ai/AG/AgShipPlayerShockServer.cpp new file mode 100644 index 00000000..e14d48ae --- /dev/null +++ b/dScripts/ai/AG/AgShipPlayerShockServer.cpp @@ -0,0 +1,23 @@ +#include "AgShipPlayerShockServer.h" +#include "GameMessages.h" +#include "RenderComponent.h" +#include "Entity.h" +#include "eTerminateType.h" + +void AgShipPlayerShockServer::OnUse(Entity* self, Entity* user) { + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); + if (active) { + return; + } + active = true; + RenderComponent::PlayAnimation(user, shockAnim); + GameMessages::SendKnockback(user->GetObjectID(), self->GetObjectID(), self->GetObjectID(), 0, NiPoint3(-20, 10, -20)); + + GameMessages::SendPlayFXEffect(self, 1430, u"create", "console_sparks", LWOOBJID_EMPTY, 1.0, 1.0, true); + self->AddTimer("FXTime", fxTime); +} + +void AgShipPlayerShockServer::OnTimerDone(Entity* self, std::string timerName) { + GameMessages::SendStopFXEffect(self, true, "console_sparks"); + active = false; +} diff --git a/dScripts/AgShipPlayerShockServer.h b/dScripts/ai/AG/AgShipPlayerShockServer.h similarity index 100% rename from dScripts/AgShipPlayerShockServer.h rename to dScripts/ai/AG/AgShipPlayerShockServer.h diff --git a/dScripts/ai/AG/AgSpaceStuff.cpp b/dScripts/ai/AG/AgSpaceStuff.cpp new file mode 100644 index 00000000..130d4354 --- /dev/null +++ b/dScripts/ai/AG/AgSpaceStuff.cpp @@ -0,0 +1,106 @@ +#include "AgSpaceStuff.h" +#include "EntityInfo.h" +#include "GeneralUtils.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "RenderComponent.h" +#include "Entity.h" + +void AgSpaceStuff::OnStartup(Entity* self) { + self->AddTimer("FloaterScale", 5.0f); + + EntityInfo info{}; + + info.pos = { -418, 585, -30 }; + info.lot = 33; + info.spawnerID = self->GetObjectID(); + + auto* ref = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(ref); + + self->SetVar(u"ShakeObject", ref->GetObjectID()); + + self->AddTimer("ShipShakeIdle", 2.0f); + self->SetVar(u"RandomTime", 10); +} + +void AgSpaceStuff::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "FloaterScale") { + int scaleType = GeneralUtils::GenerateRandomNumber<int>(1, 5); + + RenderComponent::PlayAnimation(self, u"scale_0" + GeneralUtils::to_u16string(scaleType)); + self->AddTimer("FloaterPath", 0.4); + } else if (timerName == "FloaterPath") { + int pathType = GeneralUtils::GenerateRandomNumber<int>(1, 4); + int randTime = GeneralUtils::GenerateRandomNumber<int>(20, 25); + + RenderComponent::PlayAnimation(self, u"path_0" + (GeneralUtils::to_u16string(pathType))); + self->AddTimer("FloaterScale", randTime); + } else if (timerName == "ShipShakeExplode") { + DoShake(self, true); + } else if (timerName == "ShipShakeIdle") { + DoShake(self, false); + } +} + +void AgSpaceStuff::DoShake(Entity* self, bool explodeIdle) { + + if (!explodeIdle) { + auto* ref = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"ShakeObject")); + + const auto randomTime = self->GetVar<int>(u"RandomTime"); + auto time = GeneralUtils::GenerateRandomNumber<int>(0, randomTime + 1); + + if (time < randomTime / 2) { + time += randomTime / 2; + } + + self->AddTimer("ShipShakeIdle", static_cast<float>(time)); + + if (ref) + GameMessages::SendPlayEmbeddedEffectOnAllClientsNearObject(ref, FXName, ref->GetObjectID(), 500.0f); + + auto* debrisObject = GetEntityInGroup(DebrisFX); + + if (debrisObject) + GameMessages::SendPlayFXEffect(debrisObject, -1, u"DebrisFall", "Debris", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + + const auto randomFx = GeneralUtils::GenerateRandomNumber<int>(0, 3); + + auto* shipFxObject = GetEntityInGroup(ShipFX); + if (shipFxObject) { + std::string effectType = "shipboom" + std::to_string(randomFx); + GameMessages::SendPlayFXEffect(shipFxObject, 559, GeneralUtils::ASCIIToUTF16(effectType), "FX", LWOOBJID_EMPTY, 1.0f, 1.0f, true); + } + + self->AddTimer("ShipShakeExplode", 5.0f); + + auto* shipFxObject2 = GetEntityInGroup(ShipFX2); + if (shipFxObject2) + RenderComponent::PlayAnimation(shipFxObject2, u"explosion"); + } else { + auto* shipFxObject = GetEntityInGroup(ShipFX); + auto* shipFxObject2 = GetEntityInGroup(ShipFX2); + + if (shipFxObject) + RenderComponent::PlayAnimation(shipFxObject, u"idle"); + + if (shipFxObject2) + RenderComponent::PlayAnimation(shipFxObject2, u"idle"); + } +} + +Entity* AgSpaceStuff::GetEntityInGroup(const std::string& group) { + auto entities = Game::entityManager->GetEntitiesInGroup(group); + Entity* en = nullptr; + + for (auto entity : entities) { + if (entity) { + en = entity; + break; + } + } + + return en; +} diff --git a/dScripts/AgSpaceStuff.h b/dScripts/ai/AG/AgSpaceStuff.h similarity index 100% rename from dScripts/AgSpaceStuff.h rename to dScripts/ai/AG/AgSpaceStuff.h diff --git a/dScripts/AgStagePlatforms.cpp b/dScripts/ai/AG/AgStagePlatforms.cpp similarity index 100% rename from dScripts/AgStagePlatforms.cpp rename to dScripts/ai/AG/AgStagePlatforms.cpp diff --git a/dScripts/AgStagePlatforms.h b/dScripts/ai/AG/AgStagePlatforms.h similarity index 100% rename from dScripts/AgStagePlatforms.h rename to dScripts/ai/AG/AgStagePlatforms.h diff --git a/dScripts/ai/AG/AgStromlingProperty.cpp b/dScripts/ai/AG/AgStromlingProperty.cpp new file mode 100644 index 00000000..9a9ae33b --- /dev/null +++ b/dScripts/ai/AG/AgStromlingProperty.cpp @@ -0,0 +1,17 @@ +#include "AgStromlingProperty.h" +#include "MovementAIComponent.h" +#include "eReplicaComponentType.h" + +void AgStromlingProperty::OnStartup(Entity* self) { + auto movementInfo = MovementAIInfo{ + "Wander", + 71, + 3, + 100, + 1, + 4 + }; + + auto* movementAIComponent = new MovementAIComponent(self, movementInfo); + self->AddComponent(eReplicaComponentType::MOVEMENT_AI, movementAIComponent); +} diff --git a/dScripts/AgStromlingProperty.h b/dScripts/ai/AG/AgStromlingProperty.h similarity index 100% rename from dScripts/AgStromlingProperty.h rename to dScripts/ai/AG/AgStromlingProperty.h diff --git a/dScripts/AgTurret.cpp b/dScripts/ai/AG/AgTurret.cpp similarity index 100% rename from dScripts/AgTurret.cpp rename to dScripts/ai/AG/AgTurret.cpp diff --git a/dScripts/AgTurret.h b/dScripts/ai/AG/AgTurret.h similarity index 100% rename from dScripts/AgTurret.h rename to dScripts/ai/AG/AgTurret.h diff --git a/dScripts/ai/AG/CMakeLists.txt b/dScripts/ai/AG/CMakeLists.txt new file mode 100644 index 00000000..092b8de7 --- /dev/null +++ b/dScripts/ai/AG/CMakeLists.txt @@ -0,0 +1,18 @@ +set(DSCRIPTS_SOURCES_AI_AG + "AgShipPlayerDeathTrigger.cpp" + "AgSpaceStuff.cpp" + "AgShipPlayerShockServer.cpp" + "AgImagSmashable.cpp" + "ActSharkPlayerDeathTrigger.cpp" + "AgBusDoor.cpp" + "AgTurret.cpp" + "AgFans.cpp" + "AgSalutingNpcs.cpp" + "AgJetEffectServer.cpp" + "AgQbElevator.cpp" + "AgStromlingProperty.cpp" + "AgDarkSpiderling.cpp" + "AgPicnicBlanket.cpp" + "AgStagePlatforms.cpp" + "AgQbWall.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/CMakeLists.txt b/dScripts/ai/CMakeLists.txt new file mode 100644 index 00000000..44944b90 --- /dev/null +++ b/dScripts/ai/CMakeLists.txt @@ -0,0 +1,81 @@ +set(DSCRIPTS_SOURCES_AI) + +add_subdirectory(ACT) + +foreach(file ${DSCRIPTS_SOURCES_AI_ACT}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "ACT/${file}") +endforeach() + +add_subdirectory(AG) + +foreach(file ${DSCRIPTS_SOURCES_AI_AG}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "AG/${file}") +endforeach() + +add_subdirectory(FV) + +foreach(file ${DSCRIPTS_SOURCES_AI_FV}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "FV/${file}") +endforeach() + +add_subdirectory(GENERAL) + +foreach(file ${DSCRIPTS_SOURCES_AI_GENERAL}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "GENERAL/${file}") +endforeach() + +add_subdirectory(GF) + +foreach(file ${DSCRIPTS_SOURCES_AI_GF}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "GF/${file}") +endforeach() + +add_subdirectory(MINIGAME) + +foreach(file ${DSCRIPTS_SOURCES_AI_MINIGAME}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "MINIGAME/${file}") +endforeach() + +add_subdirectory(NP) + +foreach(file ${DSCRIPTS_SOURCES_AI_NP}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "NP/${file}") +endforeach() + +add_subdirectory(NS) + +foreach(file ${DSCRIPTS_SOURCES_AI_NS}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "NS/${file}") +endforeach() + +add_subdirectory(PETS) + +foreach(file ${DSCRIPTS_SOURCES_AI_PETS}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "PETS/${file}") +endforeach() + +add_subdirectory(PROPERTY) + +foreach(file ${DSCRIPTS_SOURCES_AI_PROPERTY}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "PROPERTY/${file}") +endforeach() + +add_subdirectory(RACING) + +foreach(file ${DSCRIPTS_SOURCES_AI_RACING}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "RACING/${file}") +endforeach() + +add_subdirectory(SPEC) + +foreach(file ${DSCRIPTS_SOURCES_AI_SPEC}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "SPEC/${file}") +endforeach() + +add_subdirectory(WILD) + +foreach(file ${DSCRIPTS_SOURCES_AI_WILD}) + set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} "WILD/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI ${DSCRIPTS_SOURCES_AI} PARENT_SCOPE) diff --git a/dScripts/ai/FV/ActNinjaSensei.cpp b/dScripts/ai/FV/ActNinjaSensei.cpp new file mode 100644 index 00000000..6a7dc893 --- /dev/null +++ b/dScripts/ai/FV/ActNinjaSensei.cpp @@ -0,0 +1,78 @@ +#include "ActNinjaSensei.h" +#include "Entity.h" +#include "EntityManager.h" +#include "GameMessages.h" + +void ActNinjaSensei::OnStartup(Entity* self) { + auto students = Game::entityManager->GetEntitiesInGroup(this->m_StudentGroup); + std::vector<Entity*> validStudents = {}; + for (auto* student : students) { + if (student && student->GetLOT() == this->m_StudentLOT) validStudents.push_back(student); + } + self->SetVar(u"students", validStudents); + self->AddTimer("crane", 5); +} + +void ActNinjaSensei::OnTimerDone(Entity* self, std::string timerName) { + auto students = self->GetVar<std::vector<Entity*>>(u"students"); + if (students.empty()) return; + + if (timerName == "crane") { + for (auto student : students) { + if (student) GameMessages::SendPlayAnimation(student, u"crane"); + } + GameMessages::SendPlayAnimation(self, u"crane"); + self->AddTimer("bow", 15.33); + } + + if (timerName == "bow") { + GameMessages::SendPlayAnimation(self, u"bow"); + for (auto student : students) { + if (student) GameMessages::SendPlayAnimation(student, u"bow"); + } + GameMessages::SendPlayAnimation(self, u"bow"); + self->AddTimer("tiger", 5); + } + + if (timerName == "tiger") { + GameMessages::SendPlayAnimation(self, u"tiger"); + for (auto student : students) { + if (student) GameMessages::SendPlayAnimation(student, u"tiger"); + } + GameMessages::SendPlayAnimation(self, u"tiger"); + self->AddTimer("bow2", 15.33); + } + + if (timerName == "bow2") { + GameMessages::SendPlayAnimation(self, u"bow"); + for (auto student : students) { + if (student) GameMessages::SendPlayAnimation(student, u"bow"); + } + GameMessages::SendPlayAnimation(self, u"bow"); + self->AddTimer("mantis", 5); + } + + if (timerName == "mantis") { + GameMessages::SendPlayAnimation(self, u"mantis"); + for (auto student : students) { + if (student) GameMessages::SendPlayAnimation(student, u"mantis"); + } + GameMessages::SendPlayAnimation(self, u"mantis"); + self->AddTimer("bow3", 15.3); + } + + if (timerName == "bow3") { + GameMessages::SendPlayAnimation(self, u"bow"); + for (auto student : students) { + if (student) GameMessages::SendPlayAnimation(student, u"bow"); + } + GameMessages::SendPlayAnimation(self, u"bow"); + self->AddTimer("repeat", 5); + } + + if (timerName == "repeat") { + self->CancelAllTimers(); + self->AddTimer("crane", 5); + } +} + diff --git a/dScripts/ai/FV/ActNinjaSensei.h b/dScripts/ai/FV/ActNinjaSensei.h new file mode 100644 index 00000000..c35ede12 --- /dev/null +++ b/dScripts/ai/FV/ActNinjaSensei.h @@ -0,0 +1,10 @@ +#pragma once +#include "CppScripts.h" + +class ActNinjaSensei : public CppScripts::Script { + void OnStartup(Entity* self) override; + void OnTimerDone(Entity* self, std::string timerName) override; +private: + std::string m_StudentGroup = "Sensei_kids"; + LOT m_StudentLOT = 2497; +}; diff --git a/dScripts/ai/FV/ActNinjaTurret.cpp b/dScripts/ai/FV/ActNinjaTurret.cpp new file mode 100644 index 00000000..ea6e2278 --- /dev/null +++ b/dScripts/ai/FV/ActNinjaTurret.cpp @@ -0,0 +1,18 @@ +#include "ActNinjaTurret.h" +#include "eRebuildState.h" + +void ActNinjaTurret::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::COMPLETED) { + self->SetVar(u"AmBuilt", true); + } else if (state == eRebuildState::RESETTING) { + self->SetVar(u"AmBuilt", false); + } +} + +void +ActNinjaTurret::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args == "ISpawned" && self->GetVar<bool>(u"AmBuilt")) { + sender->Smash(); + } +} diff --git a/dScripts/ActNinjaTurret.h b/dScripts/ai/FV/ActNinjaTurret.h similarity index 100% rename from dScripts/ActNinjaTurret.h rename to dScripts/ai/FV/ActNinjaTurret.h diff --git a/dScripts/ai/FV/ActParadoxPipeFix.cpp b/dScripts/ai/FV/ActParadoxPipeFix.cpp new file mode 100644 index 00000000..aad74dec --- /dev/null +++ b/dScripts/ai/FV/ActParadoxPipeFix.cpp @@ -0,0 +1,62 @@ +#include "ActParadoxPipeFix.h" +#include "EntityManager.h" +#include "RebuildComponent.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "eEndBehavior.h" + +void ActParadoxPipeFix::OnRebuildComplete(Entity* self, Entity* target) { + const auto myGroup = "AllPipes"; + + const auto groupObjs = Game::entityManager->GetEntitiesInGroup(myGroup); + + auto indexCount = 0; + + self->SetVar(u"PlayerID", target->GetObjectID()); + + for (auto* object : groupObjs) { + if (object == self) { + continue; + } + + auto* rebuildComponent = object->GetComponent<RebuildComponent>(); + + if (rebuildComponent->GetState() == eRebuildState::COMPLETED) { + indexCount++; + } + } + + if (indexCount >= 2) { + const auto refinery = Game::entityManager->GetEntitiesInGroup("Paradox"); + + if (!refinery.empty()) { + GameMessages::SendPlayFXEffect(refinery[0]->GetObjectID(), 3999, u"create", "pipeFX"); + } + + for (auto* object : groupObjs) { + auto* player = Game::entityManager->GetEntity(object->GetVar<LWOOBJID>(u"PlayerID")); + + if (player != nullptr) { + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->ForceProgressTaskType(769, 1, 1, false); + } + + GameMessages::SendPlayCinematic(player->GetObjectID(), u"ParadoxPipeFinish", player->GetSystemAddress(), true, true, false, false, eEndBehavior::RETURN, false, 2.0f); + } + + object->SetVar(u"PlayerID", LWOOBJID_EMPTY); + } + } +} + +void ActParadoxPipeFix::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::RESETTING) { + const auto refinery = Game::entityManager->GetEntitiesInGroup("Paradox"); + + if (!refinery.empty()) { + GameMessages::SendStopFXEffect(refinery[0], true, "pipeFX"); + } + } +} diff --git a/dScripts/ActParadoxPipeFix.h b/dScripts/ai/FV/ActParadoxPipeFix.h similarity index 100% rename from dScripts/ActParadoxPipeFix.h rename to dScripts/ai/FV/ActParadoxPipeFix.h diff --git a/dScripts/ai/FV/CMakeLists.txt b/dScripts/ai/FV/CMakeLists.txt new file mode 100644 index 00000000..56418706 --- /dev/null +++ b/dScripts/ai/FV/CMakeLists.txt @@ -0,0 +1,20 @@ +set(DSCRIPTS_SOURCES_AI_FV + "ActNinjaSensei.cpp" + "ActNinjaTurret.cpp" + "FvFlyingCreviceDragon.cpp" + "FvDragonSmashingGolemQb.cpp" + "FvFreeGfNinjas.cpp" + "FvPandaSpawnerServer.cpp" + "FvPandaServer.cpp" + "FvBrickPuzzleServer.cpp" + "FvConsoleLeftQuickbuild.cpp" + "FvConsoleRightQuickbuild.cpp" + "FvFacilityBrick.cpp" + "FvFacilityPipes.cpp" + "ActParadoxPipeFix.cpp" + "FvNinjaGuard.cpp" + "FvPassThroughWall.cpp" + "FvBounceOverWall.cpp" + "FvMaelstromGeyser.cpp" + "TriggerGas.cpp" + PARENT_SCOPE) diff --git a/dScripts/FvBounceOverWall.cpp b/dScripts/ai/FV/FvBounceOverWall.cpp similarity index 100% rename from dScripts/FvBounceOverWall.cpp rename to dScripts/ai/FV/FvBounceOverWall.cpp diff --git a/dScripts/FvBounceOverWall.h b/dScripts/ai/FV/FvBounceOverWall.h similarity index 100% rename from dScripts/FvBounceOverWall.h rename to dScripts/ai/FV/FvBounceOverWall.h diff --git a/dScripts/ai/FV/FvBrickPuzzleServer.cpp b/dScripts/ai/FV/FvBrickPuzzleServer.cpp new file mode 100644 index 00000000..ce12d76e --- /dev/null +++ b/dScripts/ai/FV/FvBrickPuzzleServer.cpp @@ -0,0 +1,68 @@ +#include "FvBrickPuzzleServer.h" +#include "GeneralUtils.h" +#include "dZoneManager.h" +#include "Spawner.h" +#include "RebuildComponent.h" + +void FvBrickPuzzleServer::OnStartup(Entity* self) { + const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); + + int32_t pipeNum = 0; + if (!GeneralUtils::TryParse<int32_t>(myGroup.substr(10, 1), pipeNum)) { + return; + } + + if (pipeNum != 1) { + self->AddTimer("reset", 30); + } +} + +void FvBrickPuzzleServer::OnDie(Entity* self, Entity* killer) { + const auto myGroup = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"spawner_name")); + + int32_t pipeNum = 0; + if (!GeneralUtils::TryParse<int32_t>(myGroup.substr(10, 1), pipeNum)) { + return; + } + + const auto pipeGroup = myGroup.substr(0, 10); + + const auto nextPipeNum = pipeNum + 1; + + const auto samePipeSpawners = Game::zoneManager->GetSpawnersByName(myGroup); + + if (!samePipeSpawners.empty()) { + samePipeSpawners[0]->SoftReset(); + + samePipeSpawners[0]->Deactivate(); + } + + if (killer != nullptr && killer->IsPlayer()) { + const auto nextPipe = pipeGroup + std::to_string(nextPipeNum); + + const auto nextPipeSpawners = Game::zoneManager->GetSpawnersByName(nextPipe); + + if (!nextPipeSpawners.empty()) { + nextPipeSpawners[0]->Activate(); + } + } else { + const auto nextPipe = pipeGroup + "1"; + + const auto firstPipeSpawners = Game::zoneManager->GetSpawnersByName(nextPipe); + + if (!firstPipeSpawners.empty()) { + firstPipeSpawners[0]->Activate(); + } + } + +} + +void FvBrickPuzzleServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "reset") { + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + + if (rebuildComponent != nullptr && rebuildComponent->GetState() == eRebuildState::OPEN) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } + } +} diff --git a/dScripts/FvBrickPuzzleServer.h b/dScripts/ai/FV/FvBrickPuzzleServer.h similarity index 100% rename from dScripts/FvBrickPuzzleServer.h rename to dScripts/ai/FV/FvBrickPuzzleServer.h diff --git a/dScripts/ai/FV/FvConsoleLeftQuickbuild.cpp b/dScripts/ai/FV/FvConsoleLeftQuickbuild.cpp new file mode 100644 index 00000000..29de1fe5 --- /dev/null +++ b/dScripts/ai/FV/FvConsoleLeftQuickbuild.cpp @@ -0,0 +1,49 @@ +#include "FvConsoleLeftQuickbuild.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "eTerminateType.h" +#include "eRebuildState.h" + +void FvConsoleLeftQuickbuild::OnStartup(Entity* self) { + self->SetVar(u"IAmBuilt", false); + self->SetVar(u"AmActive", false); +} + +void FvConsoleLeftQuickbuild::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::COMPLETED) { + self->SetVar(u"IAmBuilt", true); + + const auto objects = Game::entityManager->GetEntitiesInGroup("Facility"); + + if (!objects.empty()) { + objects[0]->NotifyObject(self, "ConsoleLeftUp"); + } + } else if (state == eRebuildState::RESETTING) { + self->SetVar(u"IAmBuilt", false); + self->SetVar(u"AmActive", false); + + const auto objects = Game::entityManager->GetEntitiesInGroup("Facility"); + + if (!objects.empty()) { + objects[0]->NotifyObject(self, "ConsoleLeftDown"); + } + } +} + +void FvConsoleLeftQuickbuild::OnUse(Entity* self, Entity* user) { + if (self->GetVar<bool>(u"AmActive")) { + return; + } + + if (self->GetVar<bool>(u"IAmBuilt")) { + self->SetVar(u"AmActive", true); + + const auto objects = Game::entityManager->GetEntitiesInGroup("Facility"); + + if (!objects.empty()) { + objects[0]->NotifyObject(self, "ConsoleLeftActive"); + } + } + + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} diff --git a/dScripts/FvConsoleLeftQuickbuild.h b/dScripts/ai/FV/FvConsoleLeftQuickbuild.h similarity index 100% rename from dScripts/FvConsoleLeftQuickbuild.h rename to dScripts/ai/FV/FvConsoleLeftQuickbuild.h diff --git a/dScripts/ai/FV/FvConsoleRightQuickbuild.cpp b/dScripts/ai/FV/FvConsoleRightQuickbuild.cpp new file mode 100644 index 00000000..20194f8d --- /dev/null +++ b/dScripts/ai/FV/FvConsoleRightQuickbuild.cpp @@ -0,0 +1,49 @@ +#include "FvConsoleRightQuickbuild.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "eTerminateType.h" +#include "eRebuildState.h" + +void FvConsoleRightQuickbuild::OnStartup(Entity* self) { + self->SetVar(u"IAmBuilt", false); + self->SetVar(u"AmActive", false); +} + +void FvConsoleRightQuickbuild::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::COMPLETED) { + self->SetVar(u"IAmBuilt", true); + + const auto objects = Game::entityManager->GetEntitiesInGroup("Facility"); + + if (!objects.empty()) { + objects[0]->NotifyObject(self, "ConsoleRightUp"); + } + } else if (state == eRebuildState::RESETTING) { + self->SetVar(u"IAmBuilt", false); + self->SetVar(u"AmActive", false); + + const auto objects = Game::entityManager->GetEntitiesInGroup("Facility"); + + if (!objects.empty()) { + objects[0]->NotifyObject(self, "ConsoleRightDown"); + } + } +} + +void FvConsoleRightQuickbuild::OnUse(Entity* self, Entity* user) { + if (self->GetVar<bool>(u"AmActive")) { + return; + } + + if (self->GetVar<bool>(u"IAmBuilt")) { + self->SetVar(u"AmActive", true); + + const auto objects = Game::entityManager->GetEntitiesInGroup("Facility"); + + if (!objects.empty()) { + objects[0]->NotifyObject(self, "ConsoleRightActive"); + } + } + + GameMessages::SendTerminateInteraction(user->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} diff --git a/dScripts/FvConsoleRightQuickbuild.h b/dScripts/ai/FV/FvConsoleRightQuickbuild.h similarity index 100% rename from dScripts/FvConsoleRightQuickbuild.h rename to dScripts/ai/FV/FvConsoleRightQuickbuild.h diff --git a/dScripts/ai/FV/FvDragonSmashingGolemQb.cpp b/dScripts/ai/FV/FvDragonSmashingGolemQb.cpp new file mode 100644 index 00000000..3f3121ef --- /dev/null +++ b/dScripts/ai/FV/FvDragonSmashingGolemQb.cpp @@ -0,0 +1,33 @@ +#include "FvDragonSmashingGolemQb.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "RenderComponent.h" +#include "Entity.h" +#include "eRebuildState.h" + +void FvDragonSmashingGolemQb::OnStartup(Entity* self) { + self->AddTimer("GolemBreakTimer", 10.5f); +} + +void FvDragonSmashingGolemQb::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "GolemBreakTimer") { + self->Smash(); + } +} + +void FvDragonSmashingGolemQb::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::COMPLETED) { + RenderComponent::PlayAnimation(self, u"dragonsmash"); + + const auto dragonId = self->GetVar<LWOOBJID>(u"Dragon"); + + auto* dragon = Game::entityManager->GetEntity(dragonId); + + if (dragon != nullptr) { + dragon->OnFireEventServerSide(self, "rebuildDone"); + } + + self->CancelTimer("GolemBreakTimer"); + self->AddTimer("GolemBreakTimer", 10.5f); + } +} diff --git a/dScripts/FvDragonSmashingGolemQb.h b/dScripts/ai/FV/FvDragonSmashingGolemQb.h similarity index 100% rename from dScripts/FvDragonSmashingGolemQb.h rename to dScripts/ai/FV/FvDragonSmashingGolemQb.h diff --git a/dScripts/ai/FV/FvFacilityBrick.cpp b/dScripts/ai/FV/FvFacilityBrick.cpp new file mode 100644 index 00000000..26c07647 --- /dev/null +++ b/dScripts/ai/FV/FvFacilityBrick.cpp @@ -0,0 +1,97 @@ +#include "FvFacilityBrick.h" +#include "GameMessages.h" +#include "dZoneManager.h" +#include "EntityManager.h" + +void FvFacilityBrick::OnStartup(Entity* self) { + self->SetVar(u"ConsoleLEFTActive", false); + self->SetVar(u"ConsoleRIGHTtActive", false); +} + +void FvFacilityBrick::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { + auto* brickSpawner = Game::zoneManager->GetSpawnersByName("ImaginationBrick")[0]; + auto* bugSpawner = Game::zoneManager->GetSpawnersByName("MaelstromBug")[0]; + auto* canisterSpawner = Game::zoneManager->GetSpawnersByName("BrickCanister")[0]; + + if (name == "ConsoleLeftUp") { + GameMessages::SendStopFXEffect(self, true, "LeftPipeOff"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2775, u"create", "LeftPipeEnergy"); + } else if (name == "ConsoleLeftDown") { + self->SetVar(u"ConsoleLEFTActive", false); + + GameMessages::SendStopFXEffect(self, true, "LeftPipeEnergy"); + GameMessages::SendStopFXEffect(self, true, "LeftPipeOn"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2774, u"create", "LeftPipeOff"); + } else if (name == "ConsoleLeftActive") { + self->SetVar(u"ConsoleLEFTActive", true); + + GameMessages::SendStopFXEffect(self, true, "LeftPipeEnergy"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2776, u"create", "LeftPipeOn"); + } + + else if (name == "ConsoleRightUp") { + GameMessages::SendStopFXEffect(self, true, "RightPipeOff"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2778, u"create", "RightPipeEnergy"); + } else if (name == "ConsoleRightDown") { + self->SetVar(u"ConsoleRIGHTActive", false); + + GameMessages::SendStopFXEffect(self, true, "RightPipeEnergy"); + GameMessages::SendStopFXEffect(self, true, "RightPipeOn"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2777, u"create", "RightPipeOff"); + } else if (name == "ConsoleRightActive") { + self->SetVar(u"ConsoleRIGHTActive", true); + + GameMessages::SendStopFXEffect(self, true, "RightPipeOff"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2779, u"create", "RightPipeEnergy"); + } + + if (self->GetVar<bool>(u"ConsoleLEFTActive") && self->GetVar<bool>(u"ConsoleRIGHTActive")) { + auto* object = Game::entityManager->GetEntitiesInGroup("Brick")[0]; + + if (object != nullptr) { + GameMessages::SendPlayFXEffect(object->GetObjectID(), 122, u"create", "bluebrick"); + GameMessages::SendPlayFXEffect(object->GetObjectID(), 1034, u"cast", "imaginationexplosion"); + } + + object = Game::entityManager->GetEntitiesInGroup("Canister")[0]; + + if (object != nullptr) { + object->Smash(self->GetObjectID(), eKillType::SILENT); + } + + canisterSpawner->Reset(); + canisterSpawner->Deactivate(); + } else if (self->GetVar<bool>(u"ConsoleLEFTActive") || self->GetVar<bool>(u"ConsoleRIGHTActive")) { + brickSpawner->Activate(); + + auto* object = Game::entityManager->GetEntitiesInGroup("Brick")[0]; + + if (object != nullptr) { + GameMessages::SendStopFXEffect(object, true, "bluebrick"); + } + + bugSpawner->Reset(); + bugSpawner->Deactivate(); + + canisterSpawner->Reset(); + canisterSpawner->Activate(); + } else { + brickSpawner->Reset(); + brickSpawner->Deactivate(); + + bugSpawner->Reset(); + bugSpawner->Activate(); + } +} + +void FvFacilityBrick::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args != "PlayFX") { + return; + } + + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2774, u"create", "LeftPipeOff"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2777, u"create", "RightPipeOff"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2750, u"create", "imagination_canister"); + GameMessages::SendPlayFXEffect(self->GetObjectID(), 2751, u"create", "canister_light_filler"); +} diff --git a/dScripts/FvFacilityBrick.h b/dScripts/ai/FV/FvFacilityBrick.h similarity index 100% rename from dScripts/FvFacilityBrick.h rename to dScripts/ai/FV/FvFacilityBrick.h diff --git a/dScripts/FvFacilityPipes.cpp b/dScripts/ai/FV/FvFacilityPipes.cpp similarity index 100% rename from dScripts/FvFacilityPipes.cpp rename to dScripts/ai/FV/FvFacilityPipes.cpp diff --git a/dScripts/FvFacilityPipes.h b/dScripts/ai/FV/FvFacilityPipes.h similarity index 100% rename from dScripts/FvFacilityPipes.h rename to dScripts/ai/FV/FvFacilityPipes.h diff --git a/dScripts/ai/FV/FvFlyingCreviceDragon.cpp b/dScripts/ai/FV/FvFlyingCreviceDragon.cpp new file mode 100644 index 00000000..c8ef9ab5 --- /dev/null +++ b/dScripts/ai/FV/FvFlyingCreviceDragon.cpp @@ -0,0 +1,108 @@ +#include "FvFlyingCreviceDragon.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "SkillComponent.h" +#include "GeneralUtils.h" +#include "RenderComponent.h" + +void FvFlyingCreviceDragon::OnStartup(Entity* self) { + self->AddTimer("waypoint", 5); +} + +void FvFlyingCreviceDragon::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "waypoint") { + auto point = self->GetVar<int32_t>(u"waypoint"); + + if (point >= 20) { + point = 0; + } + + self->SetVar<int32_t>(u"waypoint", point + 1); + + self->AddTimer("waypoint", 5); + + OnArrived(self); + + return; + } + + std::string groupName = ""; + + if (timerName == "platform1attack") { + groupName = "dragonFireballs1"; + } else if (timerName == "platform3attack") { + groupName = "dragonFireballs3"; + } + + const auto& group = Game::entityManager->GetEntitiesInGroup(groupName); + + if (group.empty()) { + return; + } + + auto* skillComponent = group[0]->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY, true); + } + + auto minionCount = 1; + for (size_t i = 1; i < group.size(); i++) { + if (minionCount == 4) { + return; + } + + if (/*GeneralUtils::GenerateRandomNumber<int32_t>(1, 5) > 3*/ true) { + skillComponent = group[i]->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY); + + ++minionCount; + } + } + } +} + +void FvFlyingCreviceDragon::OnArrived(Entity* self) { + auto point = self->GetVar<int32_t>(u"waypoint"); + + if (point == 4) { + RenderComponent::PlayAnimation(self, u"attack1", 2.0f); + self->AddTimer("platform1attack", 1.75f); + } else if (point == 12) { + RenderComponent::PlayAnimation(self, u"attack2", 2.0f); + + const auto& group2 = Game::entityManager->GetEntitiesInGroup("dragonFireballs2"); + + if (group2.empty()) { + return; + } + + auto* skillComponent = group2[0]->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY); + } + + auto minionCount = 1; + for (size_t i = 1; i < group2.size(); i++) { + if (minionCount == 4) { + return; + } + + if (GeneralUtils::GenerateRandomNumber<int32_t>(1, 5) > 3) { + skillComponent = group2[i]->GetComponent<SkillComponent>(); + + if (skillComponent != nullptr) { + skillComponent->CalculateBehavior(762, 12506, LWOOBJID_EMPTY, true); + + ++minionCount; + } + } + } + } else if (point == 16) { + RenderComponent::PlayAnimation(self, u"attack3", 2.0f); + self->AddTimer("platform3attack", 0.5f); + } +} diff --git a/dScripts/FvFlyingCreviceDragon.h b/dScripts/ai/FV/FvFlyingCreviceDragon.h similarity index 100% rename from dScripts/FvFlyingCreviceDragon.h rename to dScripts/ai/FV/FvFlyingCreviceDragon.h diff --git a/dScripts/ai/FV/FvFreeGfNinjas.cpp b/dScripts/ai/FV/FvFreeGfNinjas.cpp new file mode 100644 index 00000000..d690a6f7 --- /dev/null +++ b/dScripts/ai/FV/FvFreeGfNinjas.cpp @@ -0,0 +1,43 @@ +#include "FvFreeGfNinjas.h" +#include "Character.h" +#include "MissionComponent.h" +#include "eMissionState.h" + +void FvFreeGfNinjas::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID == 705 && missionState == eMissionState::AVAILABLE) { + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) + return; + + missionComponent->AcceptMission(701); + missionComponent->AcceptMission(702); + missionComponent->AcceptMission(703); + missionComponent->AcceptMission(704); + + auto* character = target->GetCharacter(); + if (character != nullptr) + character->SetPlayerFlag(68, true); + } else if (missionID == 786) { + auto* character = target->GetCharacter(); + if (character != nullptr) + character->SetPlayerFlag(81, true); + } +} + +void FvFreeGfNinjas::OnUse(Entity* self, Entity* user) { + // To allow player who already have the mission to progress. + auto* missionComponent = user->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) + return; + + if (missionComponent->GetMissionState(705) == eMissionState::ACTIVE) { + auto* character = user->GetCharacter(); + if (character != nullptr) + character->SetPlayerFlag(68, true); + + missionComponent->AcceptMission(701, true); + missionComponent->AcceptMission(702, true); + missionComponent->AcceptMission(703, true); + missionComponent->AcceptMission(704, true); + } +} diff --git a/dScripts/ai/FV/FvFreeGfNinjas.h b/dScripts/ai/FV/FvFreeGfNinjas.h new file mode 100644 index 00000000..01b0b6bd --- /dev/null +++ b/dScripts/ai/FV/FvFreeGfNinjas.h @@ -0,0 +1,8 @@ +#pragma once +#include "CppScripts.h" + +class FvFreeGfNinjas : public CppScripts::Script { +public: + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + void OnUse(Entity* self, Entity* user) override; +}; diff --git a/dScripts/FvMaelstromGeyser.cpp b/dScripts/ai/FV/FvMaelstromGeyser.cpp similarity index 100% rename from dScripts/FvMaelstromGeyser.cpp rename to dScripts/ai/FV/FvMaelstromGeyser.cpp diff --git a/dScripts/FvMaelstromGeyser.h b/dScripts/ai/FV/FvMaelstromGeyser.h similarity index 100% rename from dScripts/FvMaelstromGeyser.h rename to dScripts/ai/FV/FvMaelstromGeyser.h diff --git a/dScripts/ai/FV/FvNinjaGuard.cpp b/dScripts/ai/FV/FvNinjaGuard.cpp new file mode 100644 index 00000000..c087c6df --- /dev/null +++ b/dScripts/ai/FV/FvNinjaGuard.cpp @@ -0,0 +1,37 @@ +#include "FvNinjaGuard.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "RenderComponent.h" +#include "EntityManager.h" + +void FvNinjaGuard::OnStartup(Entity* self) { + if (self->GetLOT() == 7412) { + m_LeftGuard = self->GetObjectID(); + } else if (self->GetLOT() == 11128) { + m_RightGuard = self->GetObjectID(); + } +} + +void FvNinjaGuard::OnEmoteReceived(Entity* self, const int32_t emote, Entity* target) { + if (emote != 392) { + RenderComponent::PlayAnimation(self, u"no"); + + return; + } + + RenderComponent::PlayAnimation(self, u"scared"); + + if (self->GetLOT() == 7412) { + auto* rightGuard = Game::entityManager->GetEntity(m_RightGuard); + + if (rightGuard != nullptr) { + RenderComponent::PlayAnimation(rightGuard, u"laugh_rt"); + } + } else if (self->GetLOT() == 11128) { + auto* leftGuard = Game::entityManager->GetEntity(m_LeftGuard); + + if (leftGuard != nullptr) { + RenderComponent::PlayAnimation(leftGuard, u"laugh_lt"); + } + } +} diff --git a/dScripts/FvNinjaGuard.h b/dScripts/ai/FV/FvNinjaGuard.h similarity index 100% rename from dScripts/FvNinjaGuard.h rename to dScripts/ai/FV/FvNinjaGuard.h diff --git a/dScripts/ai/FV/FvPandaServer.cpp b/dScripts/ai/FV/FvPandaServer.cpp new file mode 100644 index 00000000..f29f7f2e --- /dev/null +++ b/dScripts/ai/FV/FvPandaServer.cpp @@ -0,0 +1,36 @@ +#include "FvPandaServer.h" +#include "PetComponent.h" +#include "Character.h" +#include "ePetTamingNotifyType.h" + +void FvPandaServer::OnStartup(Entity* self) { + const auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent != nullptr && petComponent->GetOwner() == nullptr) { + self->SetNetworkVar<std::string>(u"pandatamer", std::to_string(self->GetVar<LWOOBJID>(u"tamer"))); + self->AddTimer("killSelf", 45); + } +} + +void FvPandaServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) { + if (type == ePetTamingNotifyType::BEGIN) { + self->CancelAllTimers(); + } else if (type == ePetTamingNotifyType::QUIT || type == ePetTamingNotifyType::FAILED) { + self->Smash(); + } else if (type == ePetTamingNotifyType::SUCCESS) { + // TODO: Remove from groups + + auto* character = tamer->GetCharacter(); + if (character != nullptr) { + character->SetPlayerFlag(82, true); + } + } +} + +void FvPandaServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "killSelf") { + const auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent != nullptr && petComponent->GetOwner() == nullptr) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } + } +} diff --git a/dScripts/ai/FV/FvPandaServer.h b/dScripts/ai/FV/FvPandaServer.h new file mode 100644 index 00000000..5db060a0 --- /dev/null +++ b/dScripts/ai/FV/FvPandaServer.h @@ -0,0 +1,8 @@ +#pragma once +#include "CppScripts.h" + +class FvPandaServer : public CppScripts::Script { + void OnStartup(Entity* self) override; + void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) override; + void OnTimerDone(Entity* self, std::string timerName) override; +}; diff --git a/dScripts/ai/FV/FvPandaSpawnerServer.cpp b/dScripts/ai/FV/FvPandaSpawnerServer.cpp new file mode 100644 index 00000000..bc9f1c8a --- /dev/null +++ b/dScripts/ai/FV/FvPandaSpawnerServer.cpp @@ -0,0 +1,49 @@ +#include "FvPandaSpawnerServer.h" +#include "Character.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "EntityInfo.h" +#include "ScriptedActivityComponent.h" + +void FvPandaSpawnerServer::OnCollisionPhantom(Entity* self, Entity* target) { + auto* character = target->GetCharacter(); + if (character != nullptr && character->GetPlayerFlag(81)) { + + auto raceObjects = Game::entityManager->GetEntitiesInGroup("PandaRaceObject"); + if (raceObjects.empty()) + return; + + // Check if the player is currently in a footrace + auto* scriptedActivityComponent = raceObjects.at(0)->GetComponent<ScriptedActivityComponent>(); + if (scriptedActivityComponent == nullptr || !scriptedActivityComponent->IsPlayedBy(target)) + return; + + // If the player already spawned a panda + auto playerPandas = Game::entityManager->GetEntitiesInGroup("panda" + std::to_string(target->GetObjectID())); + if (!playerPandas.empty()) { + GameMessages::SendFireEventClientSide(self->GetObjectID(), target->GetSystemAddress(), u"playerPanda", + target->GetObjectID(), 0, 0, target->GetObjectID()); + return; + } + + // If there's already too many spawned pandas + auto pandas = Game::entityManager->GetEntitiesInGroup("pandas"); + if (pandas.size() > 4) { + GameMessages::SendFireEventClientSide(self->GetObjectID(), target->GetSystemAddress(), u"tooManyPandas", + target->GetObjectID(), 0, 0, target->GetObjectID()); + return; + } + + EntityInfo info{}; + info.spawnerID = target->GetObjectID(); + info.pos = self->GetPosition(); + info.lot = 5643; + info.settings = { + new LDFData<LWOOBJID>(u"tamer", target->GetObjectID()), + new LDFData<std::u16string>(u"groupID", u"panda" + (GeneralUtils::to_u16string(target->GetObjectID())) + u";pandas") + }; + + auto* panda = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(panda); + } +} diff --git a/dScripts/FvPandaSpawnerServer.h b/dScripts/ai/FV/FvPandaSpawnerServer.h similarity index 100% rename from dScripts/FvPandaSpawnerServer.h rename to dScripts/ai/FV/FvPandaSpawnerServer.h diff --git a/dScripts/FvPassThroughWall.cpp b/dScripts/ai/FV/FvPassThroughWall.cpp similarity index 100% rename from dScripts/FvPassThroughWall.cpp rename to dScripts/ai/FV/FvPassThroughWall.cpp diff --git a/dScripts/FvPassThroughWall.h b/dScripts/ai/FV/FvPassThroughWall.h similarity index 100% rename from dScripts/FvPassThroughWall.h rename to dScripts/ai/FV/FvPassThroughWall.h diff --git a/dScripts/ai/FV/TriggerGas.cpp b/dScripts/ai/FV/TriggerGas.cpp new file mode 100644 index 00000000..7e9762e3 --- /dev/null +++ b/dScripts/ai/FV/TriggerGas.cpp @@ -0,0 +1,49 @@ +#include "TriggerGas.h" +#include "InventoryComponent.h" +#include "SkillComponent.h" +#include "Entity.h" +#include "dLogger.h" + + +void TriggerGas::OnStartup(Entity* self) { + self->AddTimer(this->m_TimerName, this->m_Time); +} + +void TriggerGas::OnCollisionPhantom(Entity* self, Entity* target) { + if (!target->IsPlayer()) return; + auto players = self->GetVar<std::vector<Entity*>>(u"players"); + players.push_back(target); + self->SetVar(u"players", players); +} + +void TriggerGas::OnOffCollisionPhantom(Entity* self, Entity* target) { + auto players = self->GetVar<std::vector<Entity*>>(u"players"); + if (!target->IsPlayer() || players.empty()) return; + auto position = std::find(players.begin(), players.end(), target); + if (position != players.end()) players.erase(position); + self->SetVar(u"players", players); +} + +void TriggerGas::OnTimerDone(Entity* self, std::string timerName) { + if (timerName != this->m_TimerName) return; + auto players = self->GetVar<std::vector<Entity*>>(u"players"); + for (auto player : players) { + if (player->GetIsDead() || !player){ + auto position = std::find(players.begin(), players.end(), player); + if (position != players.end()) players.erase(position); + continue; + } + auto inventoryComponent = player->GetComponent<InventoryComponent>(); + if (inventoryComponent) { + if (!inventoryComponent->IsEquipped(this->m_MaelstromHelmet)) { + auto* skillComponent = self->GetComponent<SkillComponent>(); + if (skillComponent) { + skillComponent->CastSkill(this->m_FogDamageSkill, player->GetObjectID()); + } + } + } + } + self->SetVar(u"players", players); + self->AddTimer(this->m_TimerName, this->m_Time); +} + diff --git a/dScripts/ai/FV/TriggerGas.h b/dScripts/ai/FV/TriggerGas.h new file mode 100644 index 00000000..284f2485 --- /dev/null +++ b/dScripts/ai/FV/TriggerGas.h @@ -0,0 +1,14 @@ +#pragma once +#include "CppScripts.h" + +class TriggerGas : public CppScripts::Script { + void OnStartup(Entity* self) override; + void OnCollisionPhantom(Entity* self, Entity* target) override; + void OnOffCollisionPhantom(Entity* self, Entity* target) override; + void OnTimerDone(Entity* self, std::string timerName) override; +private: + std::string m_TimerName = "gasTriggerDamage"; + float m_Time = 3.0f; + uint32_t m_MaelstromHelmet = 3068; + uint32_t m_FogDamageSkill = 103; +}; diff --git a/dScripts/ai/GENERAL/CMakeLists.txt b/dScripts/ai/GENERAL/CMakeLists.txt new file mode 100644 index 00000000..da973658 --- /dev/null +++ b/dScripts/ai/GENERAL/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_AI_GENERAL + "InstanceExitTransferPlayerToLastNonInstance.cpp" + "LegoDieRoll.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/GENERAL/InstanceExitTransferPlayerToLastNonInstance.cpp b/dScripts/ai/GENERAL/InstanceExitTransferPlayerToLastNonInstance.cpp new file mode 100644 index 00000000..de1c62e0 --- /dev/null +++ b/dScripts/ai/GENERAL/InstanceExitTransferPlayerToLastNonInstance.cpp @@ -0,0 +1,57 @@ +#include "InstanceExitTransferPlayerToLastNonInstance.h" +#include "GameMessages.h" +#include "Player.h" +#include "Character.h" +#include "dServer.h" +#include "eTerminateType.h" + +void InstanceExitTransferPlayerToLastNonInstance::OnUse(Entity* self, Entity* user) { + auto transferText = self->GetVar<std::u16string>(u"transferText"); + if (transferText.empty()) + transferText = u"DRAGON_EXIT_QUESTION"; + + GameMessages::SendDisplayMessageBox( + user->GetObjectID(), + true, + self->GetObjectID(), + u"Instance_Exit", + 1, + transferText, + u"", + user->GetSystemAddress() + ); +} + +void InstanceExitTransferPlayerToLastNonInstance::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + auto* player = dynamic_cast<Player*>(sender); + if (player == nullptr) + return; + + auto* character = sender->GetCharacter(); + if (character != nullptr) { + if (identifier == u"Instance_Exit" && button == 1) { + auto lastInstance = character->GetLastNonInstanceZoneID(); + + // Sanity check + if (lastInstance == 0) { + switch (Game::server->GetZoneID()) { + case 2001: + lastInstance = 2000; + break; + case 1402: + lastInstance = 1400; + break; + default: + lastInstance = 1100; + break; + } + } + + player->SendToZone(lastInstance); + } + } + + GameMessages::SendTerminateInteraction(sender->GetObjectID(), eTerminateType::FROM_INTERACTION, self->GetObjectID()); +} + + diff --git a/dScripts/InstanceExitTransferPlayerToLastNonInstance.h b/dScripts/ai/GENERAL/InstanceExitTransferPlayerToLastNonInstance.h similarity index 100% rename from dScripts/InstanceExitTransferPlayerToLastNonInstance.h rename to dScripts/ai/GENERAL/InstanceExitTransferPlayerToLastNonInstance.h diff --git a/dScripts/ai/GENERAL/LegoDieRoll.cpp b/dScripts/ai/GENERAL/LegoDieRoll.cpp new file mode 100644 index 00000000..763a4704 --- /dev/null +++ b/dScripts/ai/GENERAL/LegoDieRoll.cpp @@ -0,0 +1,54 @@ +#include "LegoDieRoll.h" +#include "Entity.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "RenderComponent.h" +#include "eMissionState.h" + +void LegoDieRoll::OnStartup(Entity* self) { + self->AddTimer("DoneRolling", 10.0f); + self->AddTimer("ThrowDice", LegoDieRoll::animTime); +} + +void LegoDieRoll::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "DoneRolling") { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } else if (timerName == "ThrowDice") { + int dieRoll = GeneralUtils::GenerateRandomNumber<int>(1, 6); + + switch (dieRoll) { + case 1: + RenderComponent::PlayAnimation(self, u"roll-die-1"); + break; + case 2: + RenderComponent::PlayAnimation(self, u"roll-die-2"); + break; + case 3: + RenderComponent::PlayAnimation(self, u"roll-die-3"); + break; + case 4: + RenderComponent::PlayAnimation(self, u"roll-die-4"); + break; + case 5: + RenderComponent::PlayAnimation(self, u"roll-die-5"); + break; + case 6: + { + RenderComponent::PlayAnimation(self, u"roll-die-6"); + // tracking the It's Truly Random Achievement + auto* owner = self->GetOwner(); + auto* missionComponent = owner->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + const auto rollMissionState = missionComponent->GetMissionState(756); + if (rollMissionState == eMissionState::ACTIVE) { + missionComponent->ForceProgress(756, 1103, 1); + } + } + break; + } + default: + break; + } + } +} diff --git a/dScripts/LegoDieRoll.h b/dScripts/ai/GENERAL/LegoDieRoll.h similarity index 100% rename from dScripts/LegoDieRoll.h rename to dScripts/ai/GENERAL/LegoDieRoll.h diff --git a/dScripts/ai/GF/CMakeLists.txt b/dScripts/ai/GF/CMakeLists.txt new file mode 100644 index 00000000..9937618c --- /dev/null +++ b/dScripts/ai/GF/CMakeLists.txt @@ -0,0 +1,14 @@ +set(DSCRIPTS_SOURCES_AI_GF + "GfCampfire.cpp" + "GfOrgan.cpp" + "GfBanana.cpp" + "GfBananaCluster.cpp" + "GfJailkeepMission.cpp" + "TriggerAmbush.cpp" + "GfJailWalls.cpp" + "PetDigBuild.cpp" + "GfArchway.cpp" + "GfMaelstromGeyser.cpp" + "PirateRep.cpp" + "GfParrotCrash.cpp" + PARENT_SCOPE) diff --git a/dScripts/GfArchway.cpp b/dScripts/ai/GF/GfArchway.cpp similarity index 100% rename from dScripts/GfArchway.cpp rename to dScripts/ai/GF/GfArchway.cpp diff --git a/dScripts/GfArchway.h b/dScripts/ai/GF/GfArchway.h similarity index 100% rename from dScripts/GfArchway.h rename to dScripts/ai/GF/GfArchway.h diff --git a/dScripts/ai/GF/GfBanana.cpp b/dScripts/ai/GF/GfBanana.cpp new file mode 100644 index 00000000..6bc5c179 --- /dev/null +++ b/dScripts/ai/GF/GfBanana.cpp @@ -0,0 +1,94 @@ +#include "GfBanana.h" + +#include "Entity.h" +#include "DestroyableComponent.h" +#include "EntityInfo.h" +#include "EntityManager.h" + +void GfBanana::SpawnBanana(Entity* self) { + auto position = self->GetPosition(); + const auto rotation = self->GetRotation(); + + position.y += 12; + position.x -= rotation.GetRightVector().x * 5; + position.z -= rotation.GetRightVector().z * 5; + + EntityInfo info{}; + + info.pos = position; + info.rot = rotation; + info.lot = 6909; + info.spawnerID = self->GetObjectID(); + + auto* entity = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(entity); + + self->SetVar(u"banana", entity->GetObjectID()); + + entity->AddDieCallback([self]() { + self->SetVar(u"banana", LWOOBJID_EMPTY); + + self->AddTimer("bananaTimer", 30); + }); +} + +void GfBanana::OnStartup(Entity* self) { + SpawnBanana(self); +} + +void GfBanana::OnHit(Entity* self, Entity* attacker) { + auto* destroyable = self->GetComponent<DestroyableComponent>(); + + destroyable->SetHealth(9999); + + const auto bananaId = self->GetVar<LWOOBJID>(u"banana"); + + if (bananaId == LWOOBJID_EMPTY) return; + + auto* bananaEntity = Game::entityManager->GetEntity(bananaId); + + if (bananaEntity == nullptr) { + self->SetVar(u"banana", LWOOBJID_EMPTY); + + self->AddTimer("bananaTimer", 30); + + return; + } + + bananaEntity->SetPosition(bananaEntity->GetPosition() - NiPoint3::UNIT_Y * 8); + + auto* bananaDestroyable = bananaEntity->GetComponent<DestroyableComponent>(); + + bananaDestroyable->SetHealth(0); + + bananaDestroyable->Smash(attacker->GetObjectID()); + + /* + auto position = self->GetPosition(); + const auto rotation = self->GetRotation(); + + position.y += 12; + position.x -= rotation.GetRightVector().x * 5; + position.z -= rotation.GetRightVector().z * 5; + + EntityInfo info {}; + + info.pos = position; + info.rot = rotation; + info.lot = 6718; + info.spawnerID = self->GetObjectID(); + + auto* entity = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(entity, UNASSIGNED_SYSTEM_ADDRESS); + */ + + Game::entityManager->SerializeEntity(self); +} + +void GfBanana::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "bananaTimer") { + SpawnBanana(self); + } +} diff --git a/dScripts/GfBanana.h b/dScripts/ai/GF/GfBanana.h similarity index 100% rename from dScripts/GfBanana.h rename to dScripts/ai/GF/GfBanana.h diff --git a/dScripts/GfBananaCluster.cpp b/dScripts/ai/GF/GfBananaCluster.cpp similarity index 100% rename from dScripts/GfBananaCluster.cpp rename to dScripts/ai/GF/GfBananaCluster.cpp diff --git a/dScripts/GfBananaCluster.h b/dScripts/ai/GF/GfBananaCluster.h similarity index 100% rename from dScripts/GfBananaCluster.h rename to dScripts/ai/GF/GfBananaCluster.h diff --git a/dScripts/ai/GF/GfCampfire.cpp b/dScripts/ai/GF/GfCampfire.cpp new file mode 100644 index 00000000..aa03da89 --- /dev/null +++ b/dScripts/ai/GF/GfCampfire.cpp @@ -0,0 +1,101 @@ +#include "GfCampfire.h" +#include "RenderComponent.h" +#include "SkillComponent.h" +#include "MissionComponent.h" +#include "RenderComponent.h" +#include "EntityManager.h" +#include "eReplicaComponentType.h" + +void GfCampfire::OnStartup(Entity* self) { + self->SetI32(u"counter", static_cast<int32_t>(0)); + self->SetProximityRadius(2.0f, "placeholder"); + self->SetBoolean(u"isBurning", true); + + auto* render = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + if (render == nullptr) + return; + + render->PlayEffect(295, u"running", "Burn"); +} + +void GfCampfire::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args == "physicsReady") { + auto* render = static_cast<RenderComponent*>(self->GetComponent(eReplicaComponentType::RENDER)); + + render->PlayEffect(295, u"running", "Burn"); + } +} + +void GfCampfire::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + auto* skill = self->GetComponent<SkillComponent>(); + + if (self->GetBoolean(u"isBurning")) { + if (status == "ENTER") { + if (entering->GetCharacter()) { + int32_t counter = self->GetI32(u"counter"); + counter = counter + 1; + self->SetI32(u"counter", counter); + + if (counter == 1) { + skill->CalculateBehavior(m_skillCastId, 115, entering->GetObjectID()); + self->AddTimer("TimeBetweenCast", FIRE_COOLDOWN); + + //self->SetVar<LWOOBJID>("target", entering->GetObjectID()); + + auto* missionComponet = entering->GetComponent<MissionComponent>(); + + if (missionComponet != nullptr) { + missionComponet->ForceProgress(440, 658, 1); + } + } + } + } else { + int32_t counter = self->GetI32(u"counter"); + if (counter > 0) { + counter = counter - 1; + self->SetI32(u"counter", counter); + if (counter == 0) { + self->CancelAllTimers(); + } + } + } + } +} + +void GfCampfire::OnSkillEventFired(Entity* self, Entity* caster, const std::string& message) { + if (message == "waterspray" && self->GetVar<bool>(u"isBurning")) { + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + renderComponent->StopEffect("Burn"); + renderComponent->PlayEffect(295, u"idle", "Off"); + + self->SetVar<bool>(u"isBurning", false); + self->AddTimer("FireRestart", 37); + } + } +} + +void GfCampfire::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "TimeBetweenCast") { + /* + self->AddTimer("TimeBetweenCast", FIRE_COOLDOWN); + + const auto targetId = self->GetVar<LWOOBJID>("target"); + + auto* entering = Game::entityManager->GetEntity(targetId); + + if (entering == nullptr) + { + + } + */ + } else if (timerName == "FireRestart" && !self->GetVar<bool>(u"isBurning")) { + auto* renderComponent = self->GetComponent<RenderComponent>(); + if (renderComponent != nullptr) { + renderComponent->StopEffect("Off"); + renderComponent->PlayEffect(295, u"running", "Burn"); + self->SetVar<bool>(u"isBurning", true); + } + } +} diff --git a/dScripts/GfCampfire.h b/dScripts/ai/GF/GfCampfire.h similarity index 100% rename from dScripts/GfCampfire.h rename to dScripts/ai/GF/GfCampfire.h diff --git a/dScripts/ai/GF/GfJailWalls.cpp b/dScripts/ai/GF/GfJailWalls.cpp new file mode 100644 index 00000000..1efca1cf --- /dev/null +++ b/dScripts/ai/GF/GfJailWalls.cpp @@ -0,0 +1,30 @@ +#include "GfJailWalls.h" +#include "dZoneManager.h" +#include "GeneralUtils.h" +#include "eRebuildState.h" + +void GfJailWalls::OnRebuildComplete(Entity* self, Entity* target) { + const auto wall = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"Wall")); + + for (auto* spawner : Game::zoneManager->GetSpawnersByName("Jail0" + wall)) { + spawner->Deactivate(); + } + + for (auto* spawner : Game::zoneManager->GetSpawnersByName("JailCaptain0" + wall)) { + spawner->Deactivate(); + } +} + +void GfJailWalls::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state != eRebuildState::RESETTING) return; + + const auto wall = GeneralUtils::UTF16ToWTF8(self->GetVar<std::u16string>(u"Wall")); + + for (auto* spawner : Game::zoneManager->GetSpawnersByName("Jail0" + wall)) { + spawner->Activate(); + } + + for (auto* spawner : Game::zoneManager->GetSpawnersByName("JailCaptain0" + wall)) { + spawner->Activate(); + } +} diff --git a/dScripts/GfJailWalls.h b/dScripts/ai/GF/GfJailWalls.h similarity index 100% rename from dScripts/GfJailWalls.h rename to dScripts/ai/GF/GfJailWalls.h diff --git a/dScripts/ai/GF/GfJailkeepMission.cpp b/dScripts/ai/GF/GfJailkeepMission.cpp new file mode 100644 index 00000000..b8d4cd30 --- /dev/null +++ b/dScripts/ai/GF/GfJailkeepMission.cpp @@ -0,0 +1,39 @@ +#include "GfJailkeepMission.h" +#include "MissionComponent.h" +#include "Character.h" +#include "eMissionState.h" + +void GfJailkeepMission::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) + return; + + if (missionID == 385 && missionState == eMissionState::AVAILABLE) { + missionComponent->AcceptMission(386, true); + missionComponent->AcceptMission(387, true); + missionComponent->AcceptMission(388, true); + missionComponent->AcceptMission(390, true); + } else if (missionID == 385 && missionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { + auto* character = target->GetCharacter(); + if (character != nullptr && character->GetPlayerFlag(68)) { + missionComponent->AcceptMission(701); + missionComponent->AcceptMission(702); + missionComponent->AcceptMission(703); + missionComponent->AcceptMission(704); + } + } +} + +void GfJailkeepMission::OnUse(Entity* self, Entity* user) { + auto* missionComponent = user->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) + return; + + if (missionComponent->GetMissionState(385) == eMissionState::ACTIVE) { + missionComponent->AcceptMission(386, true); + missionComponent->AcceptMission(387, true); + missionComponent->AcceptMission(388, true); + missionComponent->AcceptMission(390, true); + } +} + diff --git a/dScripts/ai/GF/GfJailkeepMission.h b/dScripts/ai/GF/GfJailkeepMission.h new file mode 100644 index 00000000..651a6e1b --- /dev/null +++ b/dScripts/ai/GF/GfJailkeepMission.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class GfJailkeepMission final : public CppScripts::Script +{ +public: + void OnUse(Entity* self, Entity* user) override; + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +}; diff --git a/dScripts/GfMaelstromGeyser.cpp b/dScripts/ai/GF/GfMaelstromGeyser.cpp similarity index 100% rename from dScripts/GfMaelstromGeyser.cpp rename to dScripts/ai/GF/GfMaelstromGeyser.cpp diff --git a/dScripts/GfMaelstromGeyser.h b/dScripts/ai/GF/GfMaelstromGeyser.h similarity index 100% rename from dScripts/GfMaelstromGeyser.h rename to dScripts/ai/GF/GfMaelstromGeyser.h diff --git a/dScripts/ai/GF/GfOrgan.cpp b/dScripts/ai/GF/GfOrgan.cpp new file mode 100644 index 00000000..3184aa82 --- /dev/null +++ b/dScripts/ai/GF/GfOrgan.cpp @@ -0,0 +1,23 @@ +#include "GfOrgan.h" +#include "GameMessages.h" +#include "Entity.h" +#include "RenderComponent.h" + +void GfOrgan::OnUse(Entity* self, Entity* user) { + if (self->GetBoolean(u"bIsInUse")) { + m_canUse = false; + return; + } + + GameMessages::SendPlayNDAudioEmitter(self, UNASSIGNED_SYSTEM_ADDRESS, "{15d5f8bd-139a-4c31-8904-970c480cd70f}"); + self->SetBoolean(u"bIsInUse", true); + self->AddTimer("reset", 5.0f); + + RenderComponent::PlayAnimation(user, u"jig"); +} + +void GfOrgan::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "reset") { + self->SetBoolean(u"bIsInUse", false); + } +} diff --git a/dScripts/GfOrgan.h b/dScripts/ai/GF/GfOrgan.h similarity index 100% rename from dScripts/GfOrgan.h rename to dScripts/ai/GF/GfOrgan.h diff --git a/dScripts/GfParrotCrash.cpp b/dScripts/ai/GF/GfParrotCrash.cpp similarity index 100% rename from dScripts/GfParrotCrash.cpp rename to dScripts/ai/GF/GfParrotCrash.cpp diff --git a/dScripts/GfParrotCrash.h b/dScripts/ai/GF/GfParrotCrash.h similarity index 100% rename from dScripts/GfParrotCrash.h rename to dScripts/ai/GF/GfParrotCrash.h diff --git a/dScripts/ai/GF/PetDigBuild.cpp b/dScripts/ai/GF/PetDigBuild.cpp new file mode 100644 index 00000000..9732e56b --- /dev/null +++ b/dScripts/ai/GF/PetDigBuild.cpp @@ -0,0 +1,51 @@ +#include "PetDigBuild.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "MissionComponent.h" +#include "eMissionState.h" + +void PetDigBuild::OnRebuildComplete(Entity* self, Entity* target) { + auto flagNumber = self->GetVar<std::u16string>(u"flagNum"); + + EntityInfo info{}; + auto pos = self->GetPosition(); + pos.SetY(pos.GetY() + 0.5f); + info.pos = pos; + info.rot = self->GetRotation(); + info.spawnerID = self->GetSpawnerID(); + info.settings = { + new LDFData<LWOOBJID>(u"builder", target->GetObjectID()), + new LDFData<LWOOBJID>(u"X", self->GetObjectID()) + }; + + if (!flagNumber.empty()) { + info.lot = 7410; // Normal GF treasure + info.settings.push_back(new LDFData<std::u16string>(u"groupID", u"Flag" + flagNumber)); + } else { + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (missionComponent != nullptr && missionComponent->GetMissionState(746) == eMissionState::ACTIVE) { + info.lot = 9307; // Special Captain Jack treasure that drops a mission item + } else { + info.lot = 3495; // Normal AG treasure + } + } + + auto* treasure = Game::entityManager->CreateEntity(info); + Game::entityManager->ConstructEntity(treasure); + self->SetVar<LWOOBJID>(u"chestObj", treasure->GetObjectID()); +} + +void PetDigBuild::OnDie(Entity* self, Entity* killer) { + auto treasureID = self->GetVar<LWOOBJID>(u"chestObj"); + if (treasureID == LWOOBJID_EMPTY) + return; + + auto treasure = Game::entityManager->GetEntity(treasureID); + if (treasure == nullptr) + return; + + // If the quick build expired and the treasure was not collected, hide the treasure + if (!treasure->GetIsDead()) { + treasure->Smash(self->GetObjectID(), eKillType::SILENT); + } +} diff --git a/dScripts/PetDigBuild.h b/dScripts/ai/GF/PetDigBuild.h similarity index 100% rename from dScripts/PetDigBuild.h rename to dScripts/ai/GF/PetDigBuild.h diff --git a/dScripts/ai/GF/PirateRep.cpp b/dScripts/ai/GF/PirateRep.cpp new file mode 100644 index 00000000..33d6ab63 --- /dev/null +++ b/dScripts/ai/GF/PirateRep.cpp @@ -0,0 +1,14 @@ +#include "PirateRep.h" +#include "Character.h" +#include "eMissionState.h" +#include "Entity.h" +#include "ePlayerFlag.h" + +void PirateRep::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID == m_PirateRepMissionID && missionState >= eMissionState::READY_TO_COMPLETE) { + auto* character = target->GetCharacter(); + if (character) { + character->SetPlayerFlag(ePlayerFlag::GF_PIRATE_REP, true); + } + } +} diff --git a/dScripts/ai/GF/PirateRep.h b/dScripts/ai/GF/PirateRep.h new file mode 100644 index 00000000..754971be --- /dev/null +++ b/dScripts/ai/GF/PirateRep.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class PirateRep : public CppScripts::Script { +public: + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +private: + const int m_PirateRepMissionID = 301; +}; diff --git a/dScripts/ai/GF/TriggerAmbush.cpp b/dScripts/ai/GF/TriggerAmbush.cpp new file mode 100644 index 00000000..7ac329ea --- /dev/null +++ b/dScripts/ai/GF/TriggerAmbush.cpp @@ -0,0 +1,37 @@ +#include "TriggerAmbush.h" + +#include "dZoneManager.h" + +void TriggerAmbush::OnStartup(Entity* self) { + self->SetProximityRadius(20, "ambush"); +} + +void TriggerAmbush::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (name != "ambush" || status != "ENTER" || !entering->IsPlayer()) return; + + if (self->GetVar<bool>(u"triggered")) return; + + self->SetVar(u"triggered", true); + + const auto spawners = Game::zoneManager->GetSpawnersByName("Ambush"); + + for (auto* spawner : spawners) { + spawner->Activate(); + } + + self->AddTimer("TriggeredTimer", 45); +} + +void TriggerAmbush::OnTimerDone(Entity* self, std::string timerName) { + if (timerName != "TriggeredTimer") return; + + self->SetVar(u"triggered", false); + + const auto spawners = Game::zoneManager->GetSpawnersByName("Ambush"); + + for (auto* spawner : spawners) { + spawner->Reset(); + + spawner->Deactivate(); + } +} diff --git a/dScripts/TriggerAmbush.h b/dScripts/ai/GF/TriggerAmbush.h similarity index 100% rename from dScripts/TriggerAmbush.h rename to dScripts/ai/GF/TriggerAmbush.h diff --git a/dScripts/ai/MINIGAME/CMakeLists.txt b/dScripts/ai/MINIGAME/CMakeLists.txt new file mode 100644 index 00000000..d4517519 --- /dev/null +++ b/dScripts/ai/MINIGAME/CMakeLists.txt @@ -0,0 +1,9 @@ +set(DSCRIPTS_SOURCES_AI_MINIGAME) + +add_subdirectory(SG_GF) + +foreach(file ${DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF}) + set(DSCRIPTS_SOURCES_AI_MINIGAME ${DSCRIPTS_SOURCES_AI_MINIGAME} "SG_GF/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI_MINIGAME ${DSCRIPTS_SOURCES_AI_MINIGAME} PARENT_SCOPE) diff --git a/dScripts/ai/MINIGAME/SG_GF/CMakeLists.txt b/dScripts/ai/MINIGAME/SG_GF/CMakeLists.txt new file mode 100644 index 00000000..87bd879a --- /dev/null +++ b/dScripts/ai/MINIGAME/SG_GF/CMakeLists.txt @@ -0,0 +1,10 @@ +set(DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF + "ZoneSGServer.cpp") + +add_subdirectory(SERVER) + +foreach(file ${DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF_SERVER}) + set(DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF ${DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF} "SERVER/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF ${DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF} PARENT_SCOPE) diff --git a/dScripts/ai/MINIGAME/SG_GF/SERVER/CMakeLists.txt b/dScripts/ai/MINIGAME/SG_GF/SERVER/CMakeLists.txt new file mode 100644 index 00000000..fb79c0e9 --- /dev/null +++ b/dScripts/ai/MINIGAME/SG_GF/SERVER/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_AI_MINIGAME_SG_GF_SERVER + "SGCannon.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/MINIGAME/SG_GF/SERVER/SGCannon.cpp b/dScripts/ai/MINIGAME/SG_GF/SERVER/SGCannon.cpp new file mode 100644 index 00000000..66171b95 --- /dev/null +++ b/dScripts/ai/MINIGAME/SG_GF/SERVER/SGCannon.cpp @@ -0,0 +1,1066 @@ +#include "SGCannon.h" +#include "EntityManager.h" +#include "GameMessages.h" +#include "dZoneManager.h" +#include "Player.h" +#include "Character.h" +#include "ShootingGalleryComponent.h" +#include "PossessorComponent.h" +#include "CharacterComponent.h" +#include "SimplePhysicsComponent.h" +#include "MovementAIComponent.h" +#include "../dWorldServer/ObjectIDManager.h" +#include "MissionComponent.h" +#include "Loot.h" +#include "InventoryComponent.h" +#include "eMissionTaskType.h" +#include "eReplicaComponentType.h" +#include "RenderComponent.h" +#include "eGameActivity.h" + +void SGCannon::OnStartup(Entity* self) { + Game::logger->Log("SGCannon", "OnStartup"); + + m_Waves = GetWaves(); + constants = GetConstants(); + + ResetVars(self); + + self->SetVar<bool>(GameStartedVariable, false); + self->SetVar<Vector3>(InitialVelocityVariable, {}); + self->SetVar<uint32_t>(ImpactSkillVariale, constants.impactSkillID); + + auto* shootingGalleryComponent = self->GetComponent<ShootingGalleryComponent>(); + if (shootingGalleryComponent != nullptr) { + shootingGalleryComponent->SetStaticParams({ + Vector3 { -327.8609924316406, 256.8999938964844, 1.6482199430465698 }, + Vector3 { -181.4320068359375, 212.39999389648438, 2.5182199478149414 } + }); + + shootingGalleryComponent->SetDynamicParams({ + Vector3 { 0.0, 4.3, 9.0 }, + Vector3 { }, + 129.0, + 800.0, + 30.0, + 0.0, + -1.0, + 58.6 + }); + } + + self->SetVar<uint32_t>(TimeLimitVariable, 30); + self->SetVar<std::vector<LOT>>(ValidActorsVariable, { 3109, 3110, 3111, 3112, 3125, 3126 }); + self->SetVar<std::vector<LOT>>(ValidEffectsVariable, { 3122 }); + self->SetVar<std::vector<uint32_t>>(StreakBonusVariable, { 1, 2, 5, 10 }); + self->SetVar<bool>(SuperChargeActiveVariable, false); + self->SetVar<uint32_t>(MatrixVariable, 1); + self->SetVar<bool>(InitVariable, true); + + auto* simplePhysicsComponent = self->GetComponent<SimplePhysicsComponent>(); + + if (simplePhysicsComponent != nullptr) { + simplePhysicsComponent->SetPhysicsMotionState(5); + } +} + +void SGCannon::OnPlayerLoaded(Entity* self, Entity* player) { + Game::logger->Log("SGCannon", "Player loaded"); + self->SetVar<LWOOBJID>(PlayerIDVariable, player->GetObjectID()); +} + +void SGCannon::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + Script::OnFireEventServerSide(self, sender, args, param1, param2, param3); +} + +void SGCannon::OnActivityStateChangeRequest(Entity* self, LWOOBJID senderID, int32_t value1, int32_t value2, + const std::u16string& stringValue) { + Game::logger->Log("SGCannon", "Got activity state change request: %s", GeneralUtils::UTF16ToWTF8(stringValue).c_str()); + if (stringValue == u"clientready") { + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player != nullptr) { + Game::logger->Log("SGCannon", "Player is ready"); + /*GameMessages::SendSetStunned(player->GetObjectID(), eStateChangeType::PUSH, player->GetSystemAddress(), LWOOBJID_EMPTY, + true, true, true, true, true, true, true);*/ + + Game::logger->Log("SGCannon", "Sending ActivityEnter"); + + GameMessages::SendActivityEnter(self->GetObjectID(), player->GetSystemAddress()); + + auto* shootingGalleryComponent = self->GetComponent<ShootingGalleryComponent>(); + + if (shootingGalleryComponent != nullptr) { + shootingGalleryComponent->SetCurrentPlayerID(player->GetObjectID()); + + Game::logger->Log("SGCannon", "Setting player ID"); + + Game::entityManager->SerializeEntity(self); + } else { + Game::logger->Log("SGCannon", "Shooting gallery component is null"); + } + + auto* characterComponent = player->GetComponent<CharacterComponent>(); + + if (characterComponent != nullptr) { + characterComponent->SetIsRacing(true); + characterComponent->SetCurrentActivity(eGameActivity::SHOOTING_GALLERY); + auto possessor = player->GetComponent<PossessorComponent>(); + if (possessor) { + possessor->SetPossessable(self->GetObjectID()); + possessor->SetPossessableType(ePossessionType::NO_POSSESSION); + } + + Game::entityManager->SerializeEntity(player); + } + + self->SetNetworkVar<bool>(HideScoreBoardVariable, true); + self->SetNetworkVar<bool>(ReSetSuperChargeVariable, true); + self->SetNetworkVar<bool>(ShowLoadingUI, true); + + /* + GameMessages::SendTeleport( + player->GetObjectID(), + {-292.6415710449219, 230.20237731933594, -3.9090466499328613}, + {0.7067984342575073, -6.527870573336259e-05, 0.707414984703064, 0.00021762956748716533}, + player->GetSystemAddress(), true + ); + */ + + //GameMessages::SendRequestActivityEnter(self->GetObjectID(), player->GetSystemAddress(), false, player->GetObjectID()); + } else { + Game::logger->Log("SGCannon", "Player not found"); + } + } else if (value1 == 1200) { + StartGame(self); + } +} + +void SGCannon::OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) { + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (!player) return; + + if (identifier == u"Scoreboardinfo") { + GameMessages::SendDisplayMessageBox(player->GetObjectID(), true, + Game::zoneManager->GetZoneControlObject()->GetObjectID(), + u"Shooting_Gallery_Retry", 2, u"Retry?", + u"", player->GetSystemAddress()); + } else { + if ((button == 1 && (identifier == u"Shooting_Gallery_Retry" || identifier == u"RePlay")) || identifier == u"SG1" || button == 0) { + if (IsPlayerInActivity(self, player->GetObjectID())) return; + self->SetNetworkVar<bool>(ClearVariable, true); + StartGame(self); + } else if (button == 0 && ((identifier == u"Shooting_Gallery_Retry" || identifier == u"RePlay"))) { + RemovePlayer(player->GetObjectID()); + UpdatePlayer(self, player->GetObjectID(), true); + } else if (button == 1 && identifier == u"Shooting_Gallery_Exit") { + UpdatePlayer(self, player->GetObjectID(), true); + RemovePlayer(player->GetObjectID()); + } + } +} + +void SGCannon::OnActivityTimerDone(Entity* self, const std::string& name) { + if (name == SuperChargeTimer && !self->GetVar<bool>(SuperChargePausedVariable)) { + if (self->GetVar<bool>(WaveStatusVariable) || self->GetVar<uint32_t>(CurrentSuperChargedTimeVariable) < 1) { + self->SetNetworkVar<uint32_t>(ChargeCountingVariable, 99); + self->SetNetworkVar<uint32_t>(SuperChargeBarVariable, 0); + ToggleSuperCharge(self, false); + } + } else if (name == SpawnWaveTimer) { + if (self->GetVar<bool>(GameStartedVariable)) { + self->SetVar<bool>(WaveStatusVariable, true); + const auto wave = (int32_t)self->GetVar<uint32_t>(ThisWaveVariable); + + if (wave != 0 && self->GetVar<bool>(SuperChargePausedVariable)) { + StartChargedCannon(self, self->GetVar<uint32_t>(CurrentSuperChargedTimeVariable)); + self->SetVar<uint32_t>(CurrentSuperChargedTimeVariable, 0); + } + + TimerToggle(self, true); + + for (const auto& enemyToSpawn : m_Waves.at(self->GetVar<uint32_t>(ThisWaveVariable))) { + SpawnObject(self, enemyToSpawn, true); + } + + Game::logger->Log("SGCannon", "Current wave spawn: %i/%i", wave, m_Waves.size()); + + // All waves completed + const auto timeLimit = (float_t)self->GetVar<uint32_t>(TimeLimitVariable); + if (wave >= m_Waves.size()) { + ActivityTimerStart(self, GameOverTimer, timeLimit, timeLimit); + } else { + ActivityTimerStart(self, EndWaveTimer, timeLimit, timeLimit); + } + + const auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player != nullptr) { + GameMessages::SendPlayFXEffect(player->GetObjectID(), -1, u"SG-start", ""); + + GameMessages::SendStartActivityTime(self->GetObjectID(), timeLimit, player->GetSystemAddress()); + Game::logger->Log("SGCannon", "Sending ActivityPause false"); + + GameMessages::SendActivityPause(self->GetObjectID(), false, player->GetSystemAddress()); + } + } + } else if (name == EndWaveTimer) { + self->SetVar<bool>(WaveStatusVariable, false); + TimerToggle(self); + RecordPlayerScore(self); + + if (self->GetVar<uint32_t>(ThisWaveVariable) >= 2) { + GameMessages::SendActivityPause(self->GetObjectID(), true); + ActivityTimerStart(self, GameOverTimer, 0.1, 0.1); + return; + } + + self->SetVar<uint32_t>(ThisWaveVariable, self->GetVar<uint32_t>(ThisWaveVariable) + 1); + PlaySceneAnimation(self, u"wave" + GeneralUtils::to_u16string(self->GetVar<uint32_t>(ThisWaveVariable)), true, true, 1.7f); + self->SetNetworkVar<uint32_t>(WaveNumVariable, self->GetVar<uint32_t>(ThisWaveVariable) + 1); + self->SetNetworkVar<uint32_t>(WaveStrVariable, self->GetVar<uint32_t>(TimeLimitVariable)); + + Game::logger->Log("SGCannon", "Current wave: %i/%i", self->GetVar<uint32_t>(ThisWaveVariable), m_Waves.size()); + + if (self->GetVar<uint32_t>(ThisWaveVariable) >= m_Waves.size()) { + ActivityTimerStart(self, GameOverTimer, 0.1, 0.1); + } else { + ActivityTimerStart(self, SpawnWaveTimer, constants.inBetweenWavePause, constants.inBetweenWavePause); + } + + Game::logger->Log("SGCannon", "Sending ActivityPause true"); + + GameMessages::SendActivityPause(self->GetObjectID(), true); + if (self->GetVar<bool>(SuperChargeActiveVariable) && !self->GetVar<bool>(SuperChargePausedVariable)) { + PauseChargeCannon(self); + } + } else if (name == GameOverTimer) { + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player != nullptr) { + Game::logger->Log("SGCannon", "Sending ActivityPause true"); + + GameMessages::SendActivityPause(self->GetObjectID(), true, player->GetSystemAddress()); + + /*const auto leftoverCannonballs = Game::entityManager->GetEntitiesInGroup("cannonball"); + if (leftoverCannonballs.empty()) { + RecordPlayerScore(self); + + } else { + ActivityTimerStart(self, EndGameBufferTimer, 1, leftoverCannonballs.size()); + }*/ + + ActivityTimerStart(self, EndGameBufferTimer, 1, 1); + + TimerToggle(self); + } + } else if (name.rfind(DoSpawnTimer, 0) == 0) { + if (self->GetVar<bool>(GameStartedVariable)) { + const auto spawnNumber = (uint32_t)std::stoi(name.substr(7)); + const auto& activeSpawns = self->GetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable); + if (activeSpawns.size() < spawnNumber) { + Game::logger->Log("SGCannon", "Trying to spawn %i when spawns size is only %i", spawnNumber, activeSpawns.size()); + return; + } + const auto& toSpawn = activeSpawns.at(spawnNumber); + const auto pathIndex = GeneralUtils::GenerateRandomNumber<float_t>(0, toSpawn.spawnPaths.size() - 1); + const auto* path = Game::zoneManager->GetZone()->GetPath(toSpawn.spawnPaths.at(pathIndex)); + if (!path) { + Game::logger->Log("SGCannon", "Path %s at index %i is null", toSpawn.spawnPaths.at(pathIndex).c_str(), pathIndex); + return; + } + + auto info = EntityInfo{}; + info.lot = toSpawn.lot; + info.spawnerID = self->GetObjectID(); + info.pos = path->pathWaypoints.at(0).position; + + info.settings = { + new LDFData<SGEnemy>(u"SpawnData", toSpawn), + new LDFData<std::string>(u"custom_script_server", "scripts/ai/ACT/SG_TARGET.lua"), + new LDFData<std::string>(u"custom_script_client", "scripts/client/ai/SG_TARGET_CLIENT.lua"), + new LDFData<std::string>(u"attached_path", path->pathName), + new LDFData<uint32_t>(u"attached_path_start", 0), + new LDFData<std::u16string>(u"groupID", u"SGEnemy") + }; + + Game::logger->Log("SGCannon", "Spawning enemy %i on path %s", toSpawn.lot, path->pathName.c_str()); + + auto* enemy = Game::entityManager->CreateEntity(info, nullptr, self); + Game::entityManager->ConstructEntity(enemy); + + auto* movementAI = new MovementAIComponent(enemy, {}); + + enemy->AddComponent(eReplicaComponentType::MOVEMENT_AI, movementAI); + + movementAI->SetMaxSpeed(toSpawn.initialSpeed); + movementAI->SetCurrentSpeed(toSpawn.initialSpeed); + movementAI->SetHaltDistance(0.0f); + + std::vector<NiPoint3> pathWaypoints; + + for (const auto& waypoint : path->pathWaypoints) { + pathWaypoints.push_back(waypoint.position); + } + + if (GeneralUtils::GenerateRandomNumber<float_t>(0, 1) < 0.5f) { + std::reverse(pathWaypoints.begin(), pathWaypoints.end()); + } + + movementAI->SetPath(pathWaypoints); + + enemy->AddDieCallback([this, self, enemy, name]() { + RegisterHit(self, enemy, name); + }); + + // Save the enemy and tell it to start pathing + if (enemy != nullptr) { + const_cast<std::vector<LWOOBJID>&>(self->GetVar<std::vector<LWOOBJID>>(SpawnedObjects)).push_back(enemy->GetObjectID()); + GameMessages::SendPlatformResync(enemy, UNASSIGNED_SYSTEM_ADDRESS); + } + } + } else if (name == EndGameBufferTimer) { + RecordPlayerScore(self); + StopGame(self, false); + } +} + +void +SGCannon::OnActivityTimerUpdate(Entity* self, const std::string& name, float_t timeRemaining, float_t elapsedTime) { + ActivityManager::OnActivityTimerUpdate(self, name, timeRemaining, elapsedTime); +} + +void SGCannon::StartGame(Entity* self) { + if (self->GetVar<bool>(GameStartedVariable)) return; + self->SetNetworkVar<uint32_t>(TimeLimitVariable, self->GetVar<uint32_t>(TimeLimitVariable)); + self->SetNetworkVar<bool>(AudioStartIntroVariable, true); + self->SetVar<LOT>(CurrentRewardVariable, LOT_NULL); + + auto rewardObjects = Game::entityManager->GetEntitiesInGroup(constants.rewardModelGroup); + for (auto* reward : rewardObjects) { + reward->OnFireEventServerSide(self, ModelToBuildEvent); + } + + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player != nullptr) { + GetLeaderboardData(self, player->GetObjectID(), GetActivityID(self), 1); + Game::logger->Log("SGCannon", "Sending ActivityStart"); + GameMessages::SendActivityStart(self->GetObjectID(), player->GetSystemAddress()); + + GameMessages::SendPlayFXEffect(self->GetObjectID(), -1, u"start", ""); + + self->SetNetworkVar<bool>(ClearVariable, true); + DoGameStartup(self); + + if (!self->GetVar<bool>(FirstTimeDoneVariable)) { + TakeActivityCost(self, player->GetObjectID()); + } + + self->SetVar<bool>(FirstTimeDoneVariable, true); + } + + SpawnNewModel(self); +} + +void SGCannon::DoGameStartup(Entity* self) { + ResetVars(self); + self->SetVar<bool>(GameStartedVariable, true); + self->SetNetworkVar<bool>(ClearVariable, true); + self->SetVar<uint32_t>(ThisWaveVariable, 0); + + if (constants.firstWaveStartTime < 1) { + constants.firstWaveStartTime = 1; + } + + ActivityTimerStart(self, SpawnWaveTimer, constants.firstWaveStartTime, + constants.firstWaveStartTime); +} + +void SGCannon::SpawnNewModel(Entity* self) { + + // Add a new reward to the existing rewards + const auto currentReward = self->GetVar<LOT>(CurrentRewardVariable); + if (currentReward != -1) { + auto rewards = self->GetVar<std::vector<LOT>>(RewardsVariable); + rewards.push_back(currentReward); + self->SetNetworkVar<int32_t>(RewardAddedVariable, currentReward); + } + + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player != nullptr) { + for (auto* rewardModel : Game::entityManager->GetEntitiesInGroup(constants.rewardModelGroup)) { + uint32_t lootMatrix; + switch (self->GetVar<uint32_t>(MatrixVariable)) { + case 1: + lootMatrix = constants.scoreLootMatrix1; + break; + case 2: + lootMatrix = constants.scoreLootMatrix2; + break; + case 3: + lootMatrix = constants.scoreLootMatrix3; + break; + case 4: + lootMatrix = constants.scoreLootMatrix4; + break; + case 5: + lootMatrix = constants.scoreLootMatrix5; + break; + default: + lootMatrix = 0; + } + + if (lootMatrix != 0) { + std::unordered_map<LOT, int32_t> toDrop = {}; + toDrop = LootGenerator::Instance().RollLootMatrix(player, lootMatrix); + + for (auto drop : toDrop) { + rewardModel->OnFireEventServerSide(self, ModelToBuildEvent, drop.first); + self->SetVar<LOT>(CurrentRewardVariable, drop.first); + } + } + } + } +} + +void SGCannon::RemovePlayer(LWOOBJID playerID) { + auto* player = Game::entityManager->GetEntity(playerID); + if (player == nullptr) + return; + + auto* playerObject = dynamic_cast<Player*>(player); + if (playerObject == nullptr) + return; + + auto* character = playerObject->GetCharacter(); + if (character != nullptr) { + playerObject->SendToZone(character->GetLastNonInstanceZoneID()); + } +} + +void SGCannon::OnRequestActivityExit(Entity* self, LWOOBJID player, bool canceled) { + if (canceled) { + StopGame(self, canceled); + RemovePlayer(player); + } +} + + +void SGCannon::StartChargedCannon(Entity* self, uint32_t optionalTime) { + optionalTime = optionalTime == 0 ? constants.chargedTime : optionalTime; + self->SetVar<bool>(SuperChargePausedVariable, false); + ToggleSuperCharge(self, true); + ActivityTimerStart(self, SuperChargeTimer, 1, optionalTime); + + if (!self->GetVar<bool>(WaveStatusVariable)) { + PauseChargeCannon(self); + } +} + +void SGCannon::TimerToggle(Entity* self, bool start) { + if (start) { + self->SetNetworkVar<uint32_t>(CountVariable, self->GetVar<uint32_t>(TimeLimitVariable)); + self->SetVar<bool>(GameStartedVariable, true); + } else { + self->SetNetworkVar<bool>(StopVariable, true); + } +} + +void SGCannon::SpawnObject(Entity* self, const SGEnemy& toSpawn, bool spawnNow) { + auto activeSpawns = self->GetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable); + activeSpawns.push_back(toSpawn); + self->SetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable, activeSpawns); + + self->SetVar(SpawnNumberVariable, activeSpawns.size() - 1); + const auto timerName = DoSpawnTimer + std::to_string(activeSpawns.size() - 1); + + if (spawnNow) { + if (toSpawn.minSpawnTime > 0 && toSpawn.maxSpawnTime > 0) { + const auto spawnTime = GeneralUtils::GenerateRandomNumber<float_t>(toSpawn.minSpawnTime, toSpawn.maxSpawnTime); + + ActivityTimerStart(self, timerName, spawnTime, spawnTime); + } else { + ActivityTimerStart(self, timerName, 1, 1); + } + } else if (toSpawn.respawns) { + const auto spawnTime = GeneralUtils::GenerateRandomNumber<float_t>(toSpawn.minRespawnTime, toSpawn.maxRespawnTime); + + ActivityTimerStart(self, timerName, spawnTime, spawnTime); + } +} + +void SGCannon::RecordPlayerScore(Entity* self) { + const auto totalScore = self->GetVar<uint32_t>(TotalScoreVariable); + const auto currentWave = self->GetVar<uint32_t>(ThisWaveVariable); + + if (currentWave > 0) { + auto totalWaveScore = 0; + auto playerScores = self->GetVar<std::vector<int32_t>>(PlayerScoresVariable); + + for (const auto& waveScore : playerScores) { + totalWaveScore += waveScore; + } + + if (currentWave >= playerScores.size()) { + playerScores.push_back(totalWaveScore); + } else { + playerScores[currentWave] = totalWaveScore; + } + } +} + +void SGCannon::PlaySceneAnimation(Entity* self, const std::u16string& animationName, bool onCannon, bool onPlayer, float_t priority) { + for (auto* cannon : Game::entityManager->GetEntitiesInGroup("cannongroup")) { + RenderComponent::PlayAnimation(cannon, animationName, priority); + } + + if (onCannon) { + RenderComponent::PlayAnimation(self, animationName, priority); + } + + if (onPlayer) { + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player != nullptr) { + RenderComponent::PlayAnimation(player, animationName, priority); + } + } +} + +void SGCannon::PauseChargeCannon(Entity* self) { + const auto time = std::max((uint32_t)std::ceil(ActivityTimerGetCurrentTime(self, SuperChargeTimer)), (uint32_t)1); + + self->SetVar<bool>(SuperChargePausedVariable, true); + self->SetVar<uint32_t>(CurrentSuperChargedTimeVariable, time); + self->SetNetworkVar<uint32_t>(ChargeCountingVariable, time); + + ActivityTimerStop(self, SuperChargeTimer); +} + +void SGCannon::StopGame(Entity* self, bool cancel) { + self->SetNetworkVar<bool>(ReSetSuperChargeVariable, true); + self->SetNetworkVar<bool>(HideSuperChargeVariable, true); + + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player == nullptr) + return; + + ToggleSuperCharge(self, false); + + // The player won, store all the score and send rewards + if (!cancel) { + int32_t percentage = 0.0f; + auto misses = self->GetVar<uint32_t>(MissesVariable); + auto fired = self->GetVar<uint32_t>(ShotsFiredVariable); + + if (fired > 0) { + percentage = misses / fired; + } + + auto* missionComponent = player->GetComponent<MissionComponent>(); + + if (missionComponent != nullptr) { + missionComponent->Progress(eMissionTaskType::PERFORM_ACTIVITY, self->GetVar<uint32_t>(TotalScoreVariable), self->GetObjectID(), "performact_score"); + missionComponent->Progress(eMissionTaskType::PERFORM_ACTIVITY, self->GetVar<uint32_t>(MaxStreakVariable), self->GetObjectID(), "performact_streak"); + missionComponent->Progress(eMissionTaskType::ACTIVITY, m_CannonLot, 0, "", self->GetVar<uint32_t>(TotalScoreVariable)); + } + + LootGenerator::Instance().GiveActivityLoot(player, self, GetGameID(self), self->GetVar<uint32_t>(TotalScoreVariable)); + + SaveScore(self, player->GetObjectID(), + static_cast<float>(self->GetVar<uint32_t>(TotalScoreVariable)), static_cast<float>(self->GetVar<uint32_t>(MaxStreakVariable)), percentage); + + StopActivity(self, player->GetObjectID(), self->GetVar<uint32_t>(TotalScoreVariable), self->GetVar<uint32_t>(MaxStreakVariable), percentage); + self->SetNetworkVar<bool>(AudioFinalWaveDoneVariable, true); + + // Give the player the model rewards they earned + auto* inventory = player->GetComponent<InventoryComponent>(); + if (inventory != nullptr) { + for (const auto rewardLot : self->GetVar<std::vector<LOT>>(RewardsVariable)) { + inventory->AddItem(rewardLot, 1, eLootSourceType::ACTIVITY, eInventoryType::MODELS); + } + } + + self->SetNetworkVar<std::u16string>(u"UI_Rewards", + GeneralUtils::to_u16string(self->GetVar<uint32_t>(TotalScoreVariable)) + u"_0_0_0_0_0_0" + ); + } + + GameMessages::SendActivityStop(self->GetObjectID(), false, cancel, player->GetSystemAddress()); + self->SetVar<bool>(GameStartedVariable, false); + ActivityTimerStopAllTimers(self); + + // Destroy all spawners + for (auto* entity : Game::entityManager->GetEntitiesInGroup("SGEnemy")) { + entity->Kill(); + } + + ResetVars(self); +} + +void SGCannon::RegisterHit(Entity* self, Entity* target, const std::string& timerName) { + const auto& spawnInfo = target->GetVar<SGEnemy>(u"SpawnData"); + + if (spawnInfo.respawns) { + const auto respawnTime = GeneralUtils::GenerateRandomNumber<float_t>(spawnInfo.minRespawnTime, spawnInfo.maxRespawnTime); + + ActivityTimerStart(self, timerName, respawnTime, respawnTime); + } + + int score = spawnInfo.score; + + if (score > 0) { + score += score * GetCurrentBonus(self); + + if (!self->GetVar<bool>(SuperChargeActiveVariable)) { + self->SetVar<uint32_t>(u"m_curStreak", self->GetVar<uint32_t>(u"m_curStreak") + 1); + } + } else { + if (!self->GetVar<bool>(SuperChargeActiveVariable)) { + self->SetVar<uint32_t>(u"m_curStreak", 0); + } + + self->SetNetworkVar<bool>(u"hitFriend", true); + } + + auto lastSuperTotal = self->GetVar<uint32_t>(u"LastSuperTotal"); + + auto scScore = self->GetVar<uint32_t>(TotalScoreVariable) - lastSuperTotal; + + Game::logger->Log("SGCannon", "LastSuperTotal: %i, scScore: %i, constants.chargedPoints: %i", + lastSuperTotal, scScore, constants.chargedPoints + ); + + if (!self->GetVar<bool>(SuperChargeActiveVariable) && scScore >= constants.chargedPoints && score >= 0) { + StartChargedCannon(self); + self->SetNetworkVar<float>(u"SuperChargeBar", 100.0f); + self->SetVar<uint32_t>(u"LastSuperTotal", self->GetVar<uint32_t>(TotalScoreVariable)); + } + + UpdateStreak(self); + + GameMessages::SendNotifyClientShootingGalleryScore(self->GetObjectID(), UNASSIGNED_SYSTEM_ADDRESS, + 0.0f, + score, + target->GetObjectID(), + target->GetPosition() + ); + + auto newScore = (int)self->GetVar<uint32_t>(TotalScoreVariable) + score; + + if (newScore < 0) { + newScore = 0; + } + + self->SetVar<uint32_t>(TotalScoreVariable, newScore); + + self->SetNetworkVar<uint32_t>(u"updateScore", newScore); + + self->SetNetworkVar<std::u16string>(u"beatHighScore", GeneralUtils::to_u16string(newScore)); + + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + if (player == nullptr) return; + + auto missionComponent = player->GetComponent<MissionComponent>(); + if (missionComponent == nullptr) return; + + missionComponent->Progress(eMissionTaskType::SMASH, spawnInfo.lot, self->GetObjectID()); +} + +void SGCannon::UpdateStreak(Entity* self) { + const auto streakBonus = GetCurrentBonus(self); + + const auto curStreak = self->GetVar<uint32_t>(u"m_curStreak"); + + const auto marks = curStreak % 3; + + self->SetNetworkVar<uint32_t>(u"cStreak", curStreak); + + if (curStreak >= 0 && curStreak < 13) { + if (marks == 1) { + self->SetNetworkVar<bool>(u"Mark1", true); + } else if (marks == 2) { + self->SetNetworkVar<bool>(u"Mark2", true); + } else if (marks == 0 && curStreak > 0) { + self->SetVar<float_t>(u"StreakBonus", streakBonus); + self->SetNetworkVar<bool>(u"ShowStreak", streakBonus + 1); + self->SetNetworkVar<bool>(u"Mark3", true); + } else { + self->SetVar<float_t>(u"StreakBonus", streakBonus); + self->SetNetworkVar<bool>(u"UnMarkAll", true); + } + } + auto maxStreak = self->GetVar<uint32_t>(MaxStreakVariable); + if (maxStreak < curStreak) self->SetVar<uint32_t>(MaxStreakVariable, curStreak); +} + +float_t SGCannon::GetCurrentBonus(Entity* self) { + auto streak = self->GetVar<uint32_t>(u"m_curStreak"); + + if (streak > 12) { + streak = 12; + } + + return streak / 3; +} + +void SGCannon::ToggleSuperCharge(Entity* self, bool enable) { + if (enable && self->GetVar<bool>(SuperChargeActiveVariable)) + return; + + auto* player = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(PlayerIDVariable)); + + if (player == nullptr) { + Game::logger->Log("SGCannon", "Player not found in toggle super charge"); + return; + } + + auto* inventoryComponent = player->GetComponent<InventoryComponent>(); + + auto equippedItems = inventoryComponent->GetEquippedItems(); + + Game::logger->Log("SGCannon", "Player has %d equipped items", equippedItems.size()); + + auto skillID = constants.cannonSkill; + auto cooldown = constants.cannonRefireRate; + + auto* selfInventoryComponent = self->GetComponent<InventoryComponent>(); + + if (inventoryComponent == nullptr) { + Game::logger->Log("SGCannon", "Inventory component not found"); + return; + } + + if (enable) { + Game::logger->Log("SGCannon", "Player is activating super charge"); + selfInventoryComponent->UpdateSlot("greeble_r", { ObjectIDManager::GenerateRandomObjectID(), 6505, 1, 0 }); + selfInventoryComponent->UpdateSlot("greeble_l", { ObjectIDManager::GenerateRandomObjectID(), 6506, 1, 0 }); + + // TODO: Equip items + skillID = constants.cannonSuperChargeSkill; + cooldown = 400; + } else { + selfInventoryComponent->UpdateSlot("greeble_r", { ObjectIDManager::GenerateRandomObjectID(), 0, 0, 0 }); + selfInventoryComponent->UpdateSlot("greeble_l", { ObjectIDManager::GenerateRandomObjectID(), 0, 0, 0 }); + + self->SetNetworkVar<float>(u"SuperChargeBar", 0); + + Game::logger->Log("SGCannon", "Player disables super charge"); + + // TODO: Unequip items + for (const auto& equipped : equippedItems) { + if (equipped.first == "special_r" || equipped.first == "special_l") { + Game::logger->Log("SGCannon", "Trying to unequip a weapon, %i", equipped.second.lot); + + auto* item = inventoryComponent->FindItemById(equipped.second.id); + + if (item != nullptr) { + inventoryComponent->UnEquipItem(item); + } else { + Game::logger->Log("SGCannon", "Item not found, %i", equipped.second.lot); + } + } + } + cooldown = 800; + self->SetVar<uint32_t>(NumberOfChargesVariable, 0); + } + + const auto& constants = GetConstants(); + + auto* shootingGalleryComponent = self->GetComponent<ShootingGalleryComponent>(); + + if (shootingGalleryComponent == nullptr) { + return; + } + + DynamicShootingGalleryParams properties = shootingGalleryComponent->GetDynamicParams(); + + properties.cannonFOV = 58.6f; + properties.cannonVelocity = 129.0; + properties.cannonRefireRate = cooldown; + properties.cannonMinDistance = 30; + properties.cannonTimeout = -1; + + shootingGalleryComponent->SetDynamicParams(properties); + + Game::entityManager->SerializeEntity(self); + Game::entityManager->SerializeEntity(player); + + self->SetNetworkVar<uint64_t>(CannonBallSkillIDVariable, skillID); + self->SetVar<bool>(SuperChargeActiveVariable, enable); +} + +std::vector<std::vector<SGEnemy>> SGCannon::GetWaves() { + return { + // Wave 1 + { + // Ship 1 + { + std::vector<std::string> { "Wave_1_Ship_1", "Wave_1_Ship_3" }, + 6015, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 2 + { + std::vector<std::string> { "Wave_1_Ship_2", "Wave_1_Ship_4" }, + 6300, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Sub 1 + { + std::vector<std::string> { "Wave_1_Sub_1", "Wave_1_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 10.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Sub 2 + { + std::vector<std::string> { "Wave_2_Sub_1", "Wave_2_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Friendly + { + std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, + 2168,0.0,5.0,true, 2.0, 5.0, + 1.0, -1000, false, 0.0, 1.0, + 1.0, false,true + } + }, + + // Wave 2 + { + // Ship 1 + { + std::vector<std::string> { "Wave_1_Ship_1", "Wave_1_Ship_3" }, + 6015, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 2 + { + std::vector<std::string> { "Wave_1_Ship_2", "Wave_1_Ship_4" }, + 6300, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 3 + { + std::vector<std::string> { "Wave_2_Ship_1" }, + 6300, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 4 + { + std::vector<std::string> { "Wave_2_Ship_2" }, + 6015, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Sub 1 + { + std::vector<std::string> { "Wave_1_Sub_1", "Wave_1_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Sub 2 + { + std::vector<std::string> { "Wave_2_Sub_1", "Wave_2_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Duck + { + std::vector<std::string> { "Wave_1_Duck_1", "Wave_1_Duck_2" }, + 5946, 5.0, 10.0, true, 5.0, 10.0, + 4.0, 5000, false, 0.0, 1.0, + 1.0, false, true + }, + + // Friendly + { + std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, + 2168,0.0,5.0,true, 2.0, 5.0, + 1.0, -1000, false, 0.0, 1.0, + 1.0, false,true + } + }, + + // Wave 3 + { + // Ship 1 + { + std::vector<std::string> { "Wave_1_Ship_1", "Wave_1_Ship_3" }, + 6015, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 2 + { + std::vector<std::string> { "Wave_1_Ship_2", "Wave_1_Ship_4" }, + 6300, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 3 + { + std::vector<std::string> { "Wave_2_Ship_1", "Wave_2_Ship_2" }, + 6015, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ship 4 + { + std::vector<std::string> { "Wave_3_Ship_1", "Wave_3_Ship_2" }, + 6300, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1500, false, 0.0, 1.0, + 1.0, false, true + }, + + // Sub 1 + { + std::vector<std::string> { "Wave_1_Sub_1", "Wave_1_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Sub 2 + { + std::vector<std::string> { "Wave_2_Sub_1", "Wave_2_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Sub 3 + { + std::vector<std::string> { "Wave_3_Sub_1", "Wave_3_Sub_2" }, + 6016, 0.0, 2.0, true, 0.0, 2.0, + 2.0, 1000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Duck + { + std::vector<std::string> { "Wave_1_Duck_1", "Wave_1_Duck_2" }, + 5946, 5.0, 10.0, true, 5.0, 10.0, + 4.0, 5000, false, 0.0, 1.0, + 1.0, false, true + }, + + // Ness + { + std::vector<std::string> { "Wave_1_Ness_1", "Wave_1_Ness_2", "Wave_2_Ness_1" }, + 2565, 10.0, 15.0, true, 10.0, 15.0, + 2.0, 10000, false, 0.0, 1.0, + 1.0, true, true + }, + + // Friendly 1 + { + std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, + 2168,0.0,5.0,true, 2.0, 5.0, + 1.0, -1000, false, 0.0, 1.0, + 1.0, false,true + }, + + // Friendly 2 + { + std::vector<std::string> { "Wave_3_FShip_1", "Wave_3_FShip_2" }, + 2168,0.0,5.0,true, 2.0, 5.0, + 1.0, -1000, false, 0.0, 1.0, + 1.0, false,true + } + } + }; +} + +void SGCannon::ResetVars(Entity* self) { + self->SetVar<uint32_t>(SpawnNumberVariable, 0); + self->SetVar<uint32_t>(CurrentSpawnNumberVariable, 0); + self->SetVar<uint32_t>(ThisWaveVariable, 0); + self->SetVar<uint32_t>(GameScoreVariable, 0); + self->SetVar<uint32_t>(GameTimeVariable, 0); + self->SetVar<bool>(GameStartedVariable, false); + self->SetVar<uint32_t>(ShotsFiredVariable, 0); + self->SetVar<uint32_t>(MaxStreakVariable, 0); + self->SetVar<uint32_t>(MissesVariable, 0); + self->SetVar<uint32_t>(CurrentStreakVariable, 0); + self->SetVar<uint32_t>(CurrentSuperChargedTimeVariable, 0); + self->SetVar<std::vector<uint32_t>>(StreakBonusVariable, {}); + self->SetVar<uint32_t>(LastSuperTotalVariable, 0); + self->SetVar<LOT>(CurrentRewardVariable, LOT_NULL); + self->SetVar<std::vector<LOT>>(RewardsVariable, {}); + self->SetVar<uint32_t>(TotalScoreVariable, 0); + + const_cast<std::vector<SGEnemy>&>(self->GetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable)).clear(); + self->SetVar<std::vector<SGEnemy>>(ActiveSpawnsVariable, {}); + + const_cast<std::vector<LWOOBJID>&>(self->GetVar<std::vector<LWOOBJID>>(SpawnedObjects)).clear(); + self->SetVar<std::vector<LWOOBJID>>(SpawnedObjects, {}); + + if (self->GetVar<bool>(InitVariable)) { + ToggleSuperCharge(self, false); + } + + self->SetVar<uint32_t>(ImpactSkillVariale, constants.impactSkillID); + self->SetVar<std::vector<int32_t>>(PlayerScoresVariable, {}); + ActivityTimerStopAllTimers(self); +} + +SGConstants SGCannon::GetConstants() { + return { + Vector3 { -908.542480, 229.773178, -908.542480 }, + Quaternion { 0.91913521289825, 0, 0.39394217729568, 0 }, + 1864, + 34, + 1822, + Vector3 { 6.652, -2, 1.5 }, + 157, + 129.0, + 30.0, + 800.0, + Vector3 { 0, 4.3, 9 }, + 6297, + 1822, + 249, + 228, + -1, + 58.6, + true, + 2, + 10, + 25000, + "QBRewardGroup", + 1864, + 50000, + 157, + 100000, + 187, + 200000, + 188, + 400000, + 189, + 800000, + 190, + 4.0, + 7.0 + }; +} diff --git a/dScripts/ai/MINIGAME/SG_GF/SERVER/SGCannon.h b/dScripts/ai/MINIGAME/SG_GF/SERVER/SGCannon.h new file mode 100644 index 00000000..b20fae6d --- /dev/null +++ b/dScripts/ai/MINIGAME/SG_GF/SERVER/SGCannon.h @@ -0,0 +1,152 @@ +#pragma once +#include <regex> +#include "ActivityManager.h" + +struct SGEnemy { + std::vector<std::string> spawnPaths{}; + LOT lot; + float_t minSpawnTime; + float_t maxSpawnTime; + bool respawns; + float_t minRespawnTime; + float_t maxRespawnTime; + float_t initialSpeed; + int32_t score; + bool changeSpeedAtWaypoint; + float_t speedChangeChance; + float_t minSpeed; + float_t maxSpeed; + bool isMovingPlatform; + bool despawnOnLastWaypoint; +}; + +struct SGConstants { + Vector3 playerStartPosition; + Quaternion playerStartRotation; + LOT cannonLot; + uint32_t impactSkillID; + LOT projectileLot; + Vector3 playerOffset; + uint32_t rewardModelMatrix; + float_t cannonVelocity; + float_t cannonMinDistance; + float_t cannonRefireRate; + Vector3 cannonBarrelOffset; + LOT cannonSuperchargedProjectileLot; + LOT cannonProjectileLot; + uint32_t cannonSuperChargeSkill; + uint32_t cannonSkill; + int32_t cannonTimeout; + float_t cannonFOV; + bool useLeaderboards; + uint32_t streakModifier; + uint32_t chargedTime; + uint32_t chargedPoints; + std::string rewardModelGroup; + uint32_t activityID; + uint32_t scoreReward1; + uint32_t scoreLootMatrix1; + uint32_t scoreReward2; + uint32_t scoreLootMatrix2; + uint32_t scoreReward3; + uint32_t scoreLootMatrix3; + uint32_t scoreReward4; + uint32_t scoreLootMatrix4; + uint32_t scoreReward5; + uint32_t scoreLootMatrix5; + float_t firstWaveStartTime; + float_t inBetweenWavePause; +}; + +class SGCannon : public ActivityManager { +public: + void OnStartup(Entity* self) override; + void OnPlayerLoaded(Entity* self, Entity* player) override; + void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; + void OnActivityStateChangeRequest(Entity* self, LWOOBJID senderID, int32_t value1, int32_t value2, const std::u16string& stringValue) override; + void OnMessageBoxResponse(Entity* self, Entity* sender, int32_t button, const std::u16string& identifier, const std::u16string& userData) override; + void OnActivityTimerDone(Entity* self, const std::string& name) override; + void OnActivityTimerUpdate(Entity* self, const std::string& name, float_t timeRemaining, float_t elapsedTime) override; + void OnRequestActivityExit(Entity* self, LWOOBJID player, bool canceled) override; +private: + static std::vector<std::vector<SGEnemy>> GetWaves(); + static SGConstants GetConstants(); + void ResetVars(Entity* self); + void StartGame(Entity* self); + void DoGameStartup(Entity* self); + void SpawnNewModel(Entity* self); + void TimerToggle(Entity* self, bool start = false); + void SpawnObject(Entity* self, const SGEnemy& toSpawn, bool spawnNow = false); + void StartChargedCannon(Entity* self, uint32_t optionalTime = 0); + void ToggleSuperCharge(Entity* self, bool enable); + void RecordPlayerScore(Entity* self); + void PlaySceneAnimation(Entity* self, const std::u16string& animationName, bool onCannon, bool onPlayer, float_t priority); + static void RemovePlayer(LWOOBJID playerID); + void PauseChargeCannon(Entity* self); + void StopGame(Entity* self, bool cancel = false); + void RegisterHit(Entity* self, Entity* target, const std::string& timerName); + void UpdateStreak(Entity* self); + float_t GetCurrentBonus(Entity* self); + + LOT m_CannonLot = 1864; + std::u16string PlayerIDVariable = u"PlayerID"; + std::u16string HideScoreBoardVariable = u"HideScoreBoard"; + std::u16string ReSetSuperChargeVariable = u"ReSetSuperCharge"; + std::u16string ShowLoadingUI = u"showLoadingUI"; + std::u16string SpawnNumberVariable = u"SpawnNum"; + std::u16string CurrentSpawnNumberVariable = u"CurSpawnNum"; + std::u16string ThisWaveVariable = u"ThisWave"; + std::u16string GameScoreVariable = u"GameScore"; + std::u16string GameTimeVariable = u"GameTime"; + std::u16string GameStartedVariable = u"GameStarted"; + std::u16string ShotsFiredVariable = u"ShotsFired"; + std::u16string MaxStreakVariable = u"MaxStreak"; + std::u16string MissesVariable = u"Misses"; + std::u16string CurrentStreakVariable = u"CurrentStreak"; + std::u16string CurrentSuperChargedTimeVariable = u"CurrentSuperChargedTime"; + std::u16string StreakBonusVariable = u"StreakBonus"; + std::u16string LastSuperTotalVariable = u"LastSuperTotal"; + std::u16string CurrentRewardVariable = u"CurrentReward"; + std::u16string RewardsVariable = u"Rewards"; + std::u16string TotalScoreVariable = u"TotalScore"; + std::u16string InitVariable = u"Init"; + std::u16string ImpactSkillVariale = u"ImpactSkill"; + std::u16string PlayerScoresVariable = u"PlayerScores"; + std::u16string InitialVelocityVariable = u"InitialVelocity"; + std::u16string ValidActorsVariable = u"ValidActors"; + std::u16string ValidEffectsVariable = u"ValidEffects"; + std::u16string SuperChargeActiveVariable = u"SuperChargeActive"; + std::u16string MatrixVariable = u"Matrix"; + std::u16string TimeLimitVariable = u"game_timelimit"; + std::u16string AudioStartIntroVariable = u"Audio_Start_Intro"; + std::u16string ClearVariable = u"Clear"; + std::u16string FirstTimeDoneVariable = u"FirstTimeDone"; + std::u16string RewardAddedVariable = u"rewardAdded"; + std::u16string SuperChargePausedVariable = u"Super_Charge_Paused"; + std::u16string WaveStatusVariable = u"WaveStatus"; + std::u16string CountVariable = u"count"; + std::u16string StopVariable = u"Stop"; + std::u16string ActiveSpawnsVariable = u"ActiveSpawns"; + std::u16string SpawnedObjects = u"SpawnedObjects"; + std::u16string WaveNumVariable = u"wave.waveNum"; + std::u16string WaveStrVariable = u"wave.waveStr"; + std::u16string ChargeCountingVariable = u"charge_counting"; + std::u16string SuperChargeBarVariable = u"SuperChargeBar"; + std::u16string NumberOfChargesVariable = u"NumberOfCharges"; + std::u16string CannonBallSkillIDVariable = u"cbskill"; + std::u16string HideSuperChargeVariable = u"HideSuper"; + std::u16string AudioFinalWaveDoneVariable = u"Audio_Final_Wave_Done"; + + std::string SpawnWaveTimer = "SpawnWave"; + std::string EndWaveTimer = "EndWave"; + std::string GameOverTimer = "GameOver"; + std::string DoSpawnTimer = "DoSpawn"; + std::string EndGameBufferTimer = "endGameBuffer"; + std::string SuperChargeTimer = "SuperChargeTimer"; + + std::string ModelToBuildEvent = "modelToBuild"; + + std::regex DoSpawnRegex = std::regex("\\d*"); + SGConstants constants{}; + std::vector<std::vector<SGEnemy>> m_Waves{}; +}; diff --git a/dScripts/ai/MINIGAME/SG_GF/ZoneSGServer.cpp b/dScripts/ai/MINIGAME/SG_GF/ZoneSGServer.cpp new file mode 100644 index 00000000..b1de87c9 --- /dev/null +++ b/dScripts/ai/MINIGAME/SG_GF/ZoneSGServer.cpp @@ -0,0 +1,26 @@ +#include "ZoneSGServer.h" +#include "EntityManager.h" + +void ZoneSGServer::OnStartup(Entity* self) { + const auto cannons = Game::entityManager->GetEntitiesByLOT(1864); + for (const auto& cannon : cannons) + self->SetVar<LWOOBJID>(CannonIDVariable, cannon->GetObjectID()); +} + +void ZoneSGServer::OnActivityStateChangeRequest(Entity* self, const LWOOBJID senderID, const int32_t value1, + const int32_t value2, const std::u16string& stringValue) { + + auto* cannon = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(CannonIDVariable)); + if (cannon != nullptr) { + cannon->OnActivityStateChangeRequest(senderID, value1, value2, stringValue); + } +} + +void ZoneSGServer::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + + auto* cannon = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(CannonIDVariable)); + if (cannon != nullptr) { + cannon->OnFireEventServerSide(sender, args, param1, param2, param3); + } +} diff --git a/dScripts/ZoneSGServer.h b/dScripts/ai/MINIGAME/SG_GF/ZoneSGServer.h similarity index 100% rename from dScripts/ZoneSGServer.h rename to dScripts/ai/MINIGAME/SG_GF/ZoneSGServer.h diff --git a/dScripts/ai/NP/CMakeLists.txt b/dScripts/ai/NP/CMakeLists.txt new file mode 100644 index 00000000..39a7301a --- /dev/null +++ b/dScripts/ai/NP/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_AI_NP + "NpcNpSpacemanBob.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/NP/NpcNpSpacemanBob.cpp b/dScripts/ai/NP/NpcNpSpacemanBob.cpp new file mode 100644 index 00000000..2195f4b4 --- /dev/null +++ b/dScripts/ai/NP/NpcNpSpacemanBob.cpp @@ -0,0 +1,15 @@ +#include "NpcNpSpacemanBob.h" +#include "DestroyableComponent.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" + +void NpcNpSpacemanBob::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionState == eMissionState::READY_TO_COMPLETE && missionID == 173) { + DestroyableComponent* destroyable = static_cast<DestroyableComponent*>(target->GetComponent(eReplicaComponentType::DESTROYABLE)); + destroyable->SetImagination(6); + MissionComponent* mission = static_cast<MissionComponent*>(target->GetComponent(eReplicaComponentType::MISSION)); + + mission->CompleteMission(664); + } +} diff --git a/dScripts/ai/NP/NpcNpSpacemanBob.h b/dScripts/ai/NP/NpcNpSpacemanBob.h new file mode 100644 index 00000000..84c59deb --- /dev/null +++ b/dScripts/ai/NP/NpcNpSpacemanBob.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class NpcNpSpacemanBob : public CppScripts::Script +{ +public: + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState); +}; + diff --git a/dScripts/ai/NS/CMakeLists.txt b/dScripts/ai/NS/CMakeLists.txt new file mode 100644 index 00000000..e8ec84b1 --- /dev/null +++ b/dScripts/ai/NS/CMakeLists.txt @@ -0,0 +1,24 @@ +set(DSCRIPTS_SOURCES_AI_NS + "ClRing.cpp" + "NsConcertChoiceBuild.cpp" + "NsConcertInstrument.cpp" + "NsConcertQuickBuild.cpp" + "NsGetFactionMissionServer.cpp" + "NsJohnnyMissionServer.cpp" + "NsModularBuild.cpp" + "NsQbImaginationStatue.cpp" + "WhFans.cpp") + +add_subdirectory(NS_PP_01) + +foreach(file ${DSCRIPTS_SOURCES_AI_NS_NS_PP_01}) + set(DSCRIPTS_SOURCES_AI_NS ${DSCRIPTS_SOURCES_AI_NS} "NS_PP_01/${file}") +endforeach() + +add_subdirectory(WH) + +foreach(file ${DSCRIPTS_SOURCES_AI_NS_WH}) + set(DSCRIPTS_SOURCES_AI_NS ${DSCRIPTS_SOURCES_AI_NS} "WH/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI_NS ${DSCRIPTS_SOURCES_AI_NS} PARENT_SCOPE) diff --git a/dScripts/ClRing.cpp b/dScripts/ai/NS/ClRing.cpp similarity index 100% rename from dScripts/ClRing.cpp rename to dScripts/ai/NS/ClRing.cpp diff --git a/dScripts/ClRing.h b/dScripts/ai/NS/ClRing.h similarity index 100% rename from dScripts/ClRing.h rename to dScripts/ai/NS/ClRing.h diff --git a/dScripts/ai/NS/NS_PP_01/CMakeLists.txt b/dScripts/ai/NS/NS_PP_01/CMakeLists.txt new file mode 100644 index 00000000..3f26c135 --- /dev/null +++ b/dScripts/ai/NS/NS_PP_01/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_AI_NS_NS_PP_01 + "PropertyDeathPlane.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/NS/NS_PP_01/PropertyDeathPlane.cpp b/dScripts/ai/NS/NS_PP_01/PropertyDeathPlane.cpp new file mode 100644 index 00000000..8f8906d3 --- /dev/null +++ b/dScripts/ai/NS/NS_PP_01/PropertyDeathPlane.cpp @@ -0,0 +1,16 @@ +#include "PropertyDeathPlane.h" +#include "Entity.h" +#include "GameMessages.h" +#include "EntityManager.h" + +void PropertyDeathPlane::OnCollisionPhantom(Entity* self, Entity* target) { + const auto teleportGroup = Game::entityManager->GetEntitiesInGroup("Teleport"); + + if (teleportGroup.size() == 0) { + return; + } + + auto* teleport = teleportGroup[0]; + + GameMessages::SendTeleport(target->GetObjectID(), teleport->GetPosition(), teleport->GetRotation(), target->GetSystemAddress()); +} diff --git a/dScripts/PropertyDeathPlane.h b/dScripts/ai/NS/NS_PP_01/PropertyDeathPlane.h similarity index 100% rename from dScripts/PropertyDeathPlane.h rename to dScripts/ai/NS/NS_PP_01/PropertyDeathPlane.h diff --git a/dScripts/NsConcertChoiceBuild.cpp b/dScripts/ai/NS/NsConcertChoiceBuild.cpp similarity index 100% rename from dScripts/NsConcertChoiceBuild.cpp rename to dScripts/ai/NS/NsConcertChoiceBuild.cpp diff --git a/dScripts/NsConcertChoiceBuild.h b/dScripts/ai/NS/NsConcertChoiceBuild.h similarity index 100% rename from dScripts/NsConcertChoiceBuild.h rename to dScripts/ai/NS/NsConcertChoiceBuild.h diff --git a/dScripts/ai/NS/NsConcertInstrument.cpp b/dScripts/ai/NS/NsConcertInstrument.cpp new file mode 100644 index 00000000..bd397fbb --- /dev/null +++ b/dScripts/ai/NS/NsConcertInstrument.cpp @@ -0,0 +1,366 @@ +#include "NsConcertInstrument.h" +#include "GameMessages.h" +#include "Item.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "RebuildComponent.h" +#include "SoundTriggerComponent.h" +#include "InventoryComponent.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eMissionTaskType.h" +#include "RenderComponent.h" + +// Constants are at the bottom + +void NsConcertInstrument::OnStartup(Entity* self) { + self->SetVar<bool>(u"beingPlayed", false); + self->SetVar<LWOOBJID>(u"activePlayer", LWOOBJID_EMPTY); + self->SetVar<LWOOBJID>(u"oldItemLeft", LWOOBJID_EMPTY); + self->SetVar<LWOOBJID>(u"oldItemRight", LWOOBJID_EMPTY); +} + +void NsConcertInstrument::OnRebuildNotifyState(Entity* self, eRebuildState state) { + if (state == eRebuildState::RESETTING || state == eRebuildState::OPEN) { + self->SetVar<LWOOBJID>(u"activePlayer", LWOOBJID_EMPTY); + } +} + +void NsConcertInstrument::OnRebuildComplete(Entity* self, Entity* target) { + if (!target->GetIsDead()) { + self->SetVar<LWOOBJID>(u"activePlayer", target->GetObjectID()); + + self->AddCallbackTimer(0.2f, [self, target]() { + RepositionPlayer(self, target); + if (hideInstrumentOnPlay.at(GetInstrumentLot(self))) + self->SetNetworkVar<bool>(u"Hide", true); + }); + + self->AddCallbackTimer(0.1f, [self, target]() { + StartPlayingInstrument(self, target); + }); + } +} + +void NsConcertInstrument::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + if (args == "stopPlaying") { + const auto activePlayerID = self->GetVar<LWOOBJID>(u"activePlayer"); + if (activePlayerID == LWOOBJID_EMPTY) + return; + + const auto activePlayer = Game::entityManager->GetEntity(activePlayerID); + if (activePlayer == nullptr) + return; + + StopPlayingInstrument(self, activePlayer); + } +} + +void NsConcertInstrument::OnTimerDone(Entity* self, std::string name) { + const auto activePlayerID = self->GetVar<LWOOBJID>(u"activePlayer"); + if (activePlayerID == LWOOBJID_EMPTY) + return; + + // If for some reason the player becomes null (for example an unexpected leave), we need to clean up + const auto activePlayer = Game::entityManager->GetEntity(activePlayerID); + if (activePlayer == nullptr && name != "cleanupAfterStop") { + StopPlayingInstrument(self, nullptr); + return; + } + + if (activePlayer != nullptr && name == "checkPlayer" && self->GetVar<bool>(u"beingPlayed")) { + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"checkMovement", 0, 0, + activePlayer->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); + auto* stats = activePlayer->GetComponent<DestroyableComponent>(); + if (stats) { + if (stats->GetImagination() > 0) { + self->AddTimer("checkPlayer", updateFrequency); + } else { + StopPlayingInstrument(self, activePlayer); + } + } + } else if (activePlayer != nullptr && name == "deductImagination" && self->GetVar<bool>(u"beingPlayed")) { + auto* stats = activePlayer->GetComponent<DestroyableComponent>(); + if (stats) + stats->SetImagination(stats->GetImagination() - instrumentImaginationCost); + + self->AddTimer("deductImagination", instrumentCostFrequency); + } else if (name == "cleanupAfterStop") { + if (activePlayer != nullptr) { + UnEquipInstruments(self, activePlayer); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stopPlaying", 0, 0, + activePlayer->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); + } + + auto* rebuildComponent = self->GetComponent<RebuildComponent>(); + if (rebuildComponent != nullptr) + rebuildComponent->ResetRebuild(false); + + self->Smash(self->GetObjectID(), eKillType::VIOLENT); + self->SetVar<LWOOBJID>(u"activePlayer", LWOOBJID_EMPTY); + } else if (activePlayer != nullptr && name == "achievement") { + auto* missionComponent = activePlayer->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + missionComponent->ForceProgress(302, 462, self->GetLOT()); + } + self->AddTimer("achievement2", 10.0f); + } else if (activePlayer != nullptr && name == "achievement2") { + auto* missionComponent = activePlayer->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + missionComponent->ForceProgress(602, achievementTaskID.at(GetInstrumentLot(self)), self->GetLOT()); + } + } +} + +void NsConcertInstrument::StartPlayingInstrument(Entity* self, Entity* player) { + const auto instrumentLot = GetInstrumentLot(self); + self->SetVar<bool>(u"beingPlayed", true); + + // Stuff to notify the player + EquipInstruments(self, player); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"startPlaying", 0, 0, + player->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); + GameMessages::SendPlayCinematic(player->GetObjectID(), cinematics.at(instrumentLot), UNASSIGNED_SYSTEM_ADDRESS); + self->AddCallbackTimer(1.0f, [player, instrumentLot]() { + RenderComponent::PlayAnimation(player, animations.at(instrumentLot), 2.0f); + }); + + for (auto* soundBox : Game::entityManager->GetEntitiesInGroup("Audio-Concert")) { + auto* soundTrigger = soundBox->GetComponent<SoundTriggerComponent>(); + if (soundTrigger != nullptr) { + soundTrigger->ActivateMusicCue(music.at(instrumentLot)); + } + } + + // Add timers for deducting imagination and checking if the instruments can still be played + self->AddTimer("checkPlayer", updateFrequency); + self->AddTimer("deductImagination", instrumentCostFrequency); + self->AddTimer("achievement", 20.0f); +} + +void NsConcertInstrument::StopPlayingInstrument(Entity* self, Entity* player) { + // No use in stopping twice + if (!self->GetVar<bool>(u"beingPlayed")) + return; + + const auto instrumentLot = GetInstrumentLot(self); + + // Player might be null if they left + if (player != nullptr) { + auto* missions = player->GetComponent<MissionComponent>(); + if (missions != nullptr && missions->GetMissionState(176) == eMissionState::ACTIVE) { + missions->Progress(eMissionTaskType::SCRIPT, self->GetLOT()); + } + + GameMessages::SendEndCinematic(player->GetObjectID(), cinematics.at(instrumentLot), UNASSIGNED_SYSTEM_ADDRESS, 1.0f); + RenderComponent::PlayAnimation(player, smashAnimations.at(instrumentLot), 2.0f); + GameMessages::SendNotifyClientObject(self->GetObjectID(), u"stopCheckingMovement", 0, 0, + player->GetObjectID(), "", UNASSIGNED_SYSTEM_ADDRESS); + } + + self->SetVar<bool>(u"beingPlayed", false); + + for (auto* soundBox : Game::entityManager->GetEntitiesInGroup("Audio-Concert")) { + auto* soundTrigger = soundBox->GetComponent<SoundTriggerComponent>(); + if (soundTrigger != nullptr) { + soundTrigger->DeactivateMusicCue(music.at(instrumentLot)); + } + } + + self->CancelAllTimers(); + self->AddTimer("cleanupAfterStop", instrumentSmashAnimationTime.at(instrumentLot)); +} + +void NsConcertInstrument::EquipInstruments(Entity* self, Entity* player) { + auto* inventory = player->GetComponent<InventoryComponent>(); + if (inventory != nullptr) { + auto equippedItems = inventory->GetEquippedItems(); + + // Un equip the current left item + const auto equippedLeftItem = equippedItems.find("special_l"); + if (equippedLeftItem != equippedItems.end()) { + auto* leftItem = inventory->FindItemById(equippedLeftItem->second.id); + if (leftItem != nullptr) { + leftItem->UnEquip(); + self->SetVar<LWOOBJID>(u"oldItemLeft", leftItem->GetId()); + } + } + + // Un equip the current right item + const auto equippedRightItem = equippedItems.find("special_r"); + if (equippedRightItem != equippedItems.end()) { + auto* rightItem = inventory->FindItemById(equippedRightItem->second.id); + if (rightItem != nullptr) { + rightItem->UnEquip(); + self->SetVar<LWOOBJID>(u"oldItemRight", rightItem->GetId()); + } + } + + // Equip the left hand instrument + const auto leftInstrumentLot = instrumentLotLeft.find(GetInstrumentLot(self))->second; + if (leftInstrumentLot != LOT_NULL) { + inventory->AddItem(leftInstrumentLot, 1, eLootSourceType::NONE, TEMP_ITEMS, {}, LWOOBJID_EMPTY, false); + auto* leftInstrument = inventory->FindItemByLot(leftInstrumentLot, TEMP_ITEMS); + leftInstrument->Equip(); + } + + // Equip the right hand instrument + const auto rightInstrumentLot = instrumentLotRight.find(GetInstrumentLot(self))->second; + if (rightInstrumentLot != LOT_NULL) { + inventory->AddItem(rightInstrumentLot, 1, eLootSourceType::NONE, TEMP_ITEMS, {}, LWOOBJID_EMPTY, false); + auto* rightInstrument = inventory->FindItemByLot(rightInstrumentLot, TEMP_ITEMS); + rightInstrument->Equip(); + } + } +} + +void NsConcertInstrument::UnEquipInstruments(Entity* self, Entity* player) { + auto* inventory = player->GetComponent<InventoryComponent>(); + if (inventory != nullptr) { + auto equippedItems = inventory->GetEquippedItems(); + + // Un equip the current left instrument + const auto equippedInstrumentLeft = equippedItems.find("special_l"); + if (equippedInstrumentLeft != equippedItems.end()) { + auto* leftItem = inventory->FindItemById(equippedInstrumentLeft->second.id); + if (leftItem != nullptr) { + leftItem->UnEquip(); + inventory->RemoveItem(leftItem->GetLot(), 1, TEMP_ITEMS); + } + } + + // Un equip the current right instrument + const auto equippedInstrumentRight = equippedItems.find("special_r"); + if (equippedInstrumentRight != equippedItems.end()) { + auto* rightItem = inventory->FindItemById(equippedInstrumentRight->second.id); + if (rightItem != nullptr) { + rightItem->UnEquip(); + inventory->RemoveItem(rightItem->GetLot(), 1, TEMP_ITEMS); + } + } + + // Equip the old left hand item + const auto leftItemID = self->GetVar<LWOOBJID>(u"oldItemLeft"); + if (leftItemID != LWOOBJID_EMPTY) { + auto* item = inventory->FindItemById(leftItemID); + if (item != nullptr) + item->Equip(); + self->SetVar<LWOOBJID>(u"oldItemLeft", LWOOBJID_EMPTY); + } + + // Equip the old right hand item + const auto rightItemID = self->GetVar<LWOOBJID>(u"oldItemRight"); + if (rightItemID != LWOOBJID_EMPTY) { + auto* item = inventory->FindItemById(rightItemID); + if (item != nullptr) + item->Equip(); + self->SetVar<LWOOBJID>(u"oldItemRight", LWOOBJID_EMPTY); + } + } +} + +void NsConcertInstrument::RepositionPlayer(Entity* self, Entity* player) { + auto position = self->GetPosition(); + auto rotation = self->GetRotation(); + position.SetY(0.0f); + + switch (GetInstrumentLot(self)) { + case Bass: + case Guitar: + position.SetX(position.GetX() + 5.0f); + break; + case Keyboard: + position.SetX(position.GetX() - 0.45f); + position.SetZ(position.GetZ() + 0.75f); + rotation = NiQuaternion::CreateFromAxisAngle(position, -0.8f); // Slight rotation to make the animation sensible + break; + case Drum: + position.SetZ(position.GetZ() - 0.5f); + break; + } + + GameMessages::SendTeleport(player->GetObjectID(), position, rotation, player->GetSystemAddress()); +} + +InstrumentLot NsConcertInstrument::GetInstrumentLot(Entity* self) { + return static_cast<const InstrumentLot>(self->GetLOT()); +} + +// Static stuff needed for script execution + +const std::map<InstrumentLot, std::u16string> NsConcertInstrument::animations{ + { Guitar, u"guitar"}, + { Bass, u"bass"}, + { Keyboard, u"keyboard"}, + { Drum, u"drums"} +}; + +const std::map<InstrumentLot, std::u16string> NsConcertInstrument::smashAnimations{ + {Guitar, u"guitar-smash"}, + {Bass, u"bass-smash"}, + {Keyboard, u"keyboard-smash"}, + {Drum, u"keyboard-smash"} +}; + +const std::map<InstrumentLot, float> NsConcertInstrument::instrumentSmashAnimationTime{ + {Guitar, 2.167f}, + {Bass, 1.167f}, + {Keyboard, 1.0f}, + {Drum, 1.0f} +}; + +const std::map<InstrumentLot, std::string> NsConcertInstrument::music{ + {Guitar, "Concert_Guitar"}, + {Bass, "Concert_Bass"}, + {Keyboard, "Concert_Keys"}, + {Drum, "Concert_Drums"}, +}; + +const std::map<InstrumentLot, std::u16string> NsConcertInstrument::cinematics{ + {Guitar, u"Concert_Cam_G"}, + {Bass, u"Concert_Cam_B"}, + {Keyboard, u"Concert_Cam_K"}, + {Drum, u"Concert_Cam_D"}, +}; + +const std::map<InstrumentLot, LOT> NsConcertInstrument::instrumentLotLeft{ + {Guitar, 4991}, + {Bass, 4992}, + {Keyboard, LOT_NULL}, + {Drum, 4995}, +}; + +const std::map<InstrumentLot, LOT> NsConcertInstrument::instrumentLotRight{ + {Guitar, LOT_NULL}, + {Bass, LOT_NULL}, + {Keyboard, LOT_NULL}, + {Drum, 4996}, +}; + +const std::map<InstrumentLot, bool> NsConcertInstrument::hideInstrumentOnPlay{ + {Guitar, true}, + {Bass, true}, + {Keyboard, false}, + {Drum, false}, +}; + +const std::map<InstrumentLot, float> NsConcertInstrument::instrumentEquipTime{ + {Guitar, 1.033}, + {Bass, 0.75}, + {Keyboard, -1}, + {Drum, 0}, +}; + +const std::map<InstrumentLot, uint32_t> NsConcertInstrument::achievementTaskID{ + {Guitar, 911}, + {Bass, 912}, + {Keyboard, 913}, + {Drum, 914}, +}; + +const uint32_t NsConcertInstrument::instrumentImaginationCost = 2; + +const float NsConcertInstrument::instrumentCostFrequency = 4.0f; + +const float NsConcertInstrument::updateFrequency = 1.0f; diff --git a/dScripts/NsConcertInstrument.h b/dScripts/ai/NS/NsConcertInstrument.h similarity index 100% rename from dScripts/NsConcertInstrument.h rename to dScripts/ai/NS/NsConcertInstrument.h diff --git a/dScripts/ai/NS/NsConcertQuickBuild.cpp b/dScripts/ai/NS/NsConcertQuickBuild.cpp new file mode 100644 index 00000000..960b90d5 --- /dev/null +++ b/dScripts/ai/NS/NsConcertQuickBuild.cpp @@ -0,0 +1,224 @@ +#include "NsConcertQuickBuild.h" +#include "EntityManager.h" +#include "NsConcertChoiceBuildManager.h" +#include "DestroyableComponent.h" +#include "GameMessages.h" +#include "MovingPlatformComponent.h" +#include "MissionComponent.h" + +const float NsConcertQuickBuild::resetTime = 40.0f; +const float NsConcertQuickBuild::resetBlinkTime = 6.0f; +const float NsConcertQuickBuild::resetStageTime = 66.5f; +const float NsConcertQuickBuild::resetActivatorTime = 30.0f; +const std::map<LOT, QuickBuildSet> NsConcertQuickBuild::quickBuildSets{ + {5846, QuickBuildSet {"laser", {"discoball", "discofloor", "stagelights", "spotlight"}}}, + {5847, QuickBuildSet {"spotlight", {"spotlight", "stagelights"}}}, + {5848, QuickBuildSet {"rocket", {"flamethrower"}}}, + {5845, QuickBuildSet {"speaker", {"speaker", "speakerHill", "stagelights", "spotlight"}}} +}; + +const std::map<std::string, std::string> NsConcertQuickBuild::quickBuildFX{ + {"discoball", "effectsDiscoball"}, + {"speaker", "effectsShell"}, + {"speakerHill", "effectsHill"}, + {"spotlight", "effectsHill"}, + {"discofloor", "effectsShell"}, + {"flamethrower", "effectsShell"}, + {"stagelights", "effectsShell"} +}; + +std::vector<LWOOBJID> NsConcertQuickBuild::finishedQuickBuilds = {}; + +void NsConcertQuickBuild::OnStartup(Entity* self) { + const auto groups = self->GetGroups(); + if (groups.empty()) + return; + + // Groups are of the form Concert_Laser_QB_1, Concert_Laser_QB_2, etc. + auto group = groups.at(0); + const auto splitGroup = GeneralUtils::SplitString(group, '_'); + if (splitGroup.size() < 4) + return; + + // Get the manager of the crate of this quick build + const auto groupNumber = std::stoi(splitGroup.at(3)); + const auto managerObjects = Game::entityManager->GetEntitiesInGroup("CB_" + std::to_string(groupNumber)); + if (managerObjects.empty()) + return; + + auto* managerObject = managerObjects.at(0); + self->SetVar<LWOOBJID>(u"managerObject", managerObject->GetObjectID()); + self->SetVar<int32_t>(u"groupNumber", groupNumber); + + // Makes the quick build blink after a certain amount of time + self->AddCallbackTimer(GetBlinkTime(resetActivatorTime), [self]() { + self->SetNetworkVar<float>(u"startEffect", NsConcertQuickBuild::GetBlinkTime(resetActivatorTime)); + }); + + // Destroys the quick build after a while if it wasn't built + self->AddCallbackTimer(resetActivatorTime, [self]() { + self->SetNetworkVar<float>(u"startEffect", -1.0f); + self->Smash(self->GetObjectID(), eKillType::SILENT); + }); +} + +float NsConcertQuickBuild::GetBlinkTime(float time) { + return time <= NsConcertQuickBuild::resetBlinkTime ? 1.0f : time - NsConcertQuickBuild::resetBlinkTime; +} + +void NsConcertQuickBuild::OnDie(Entity* self, Entity* killer) { + auto* managerObject = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"managerObject")); + if (managerObject) { + managerObject->CancelAllTimers(); + managerObject->AddCallbackTimer(1.0f, [managerObject]() { + NsConcertChoiceBuildManager::SpawnCrate(managerObject); + }); + } + + auto position = std::find(finishedQuickBuilds.begin(), finishedQuickBuilds.end(), self->GetObjectID()); + if (position != finishedQuickBuilds.end()) + finishedQuickBuilds.erase(position); +} + +void NsConcertQuickBuild::OnRebuildComplete(Entity* self, Entity* target) { + const auto groupNumber = self->GetVar<int32_t>(u"groupNumber"); + finishedQuickBuilds.push_back(self->GetObjectID()); + self->SetNetworkVar<float>(u"startEffect", -1.0f); + + ProgressStageCraft(self, target); + + // Find all the quick build objects of the same lot + auto finishedQuickBuildObjects = std::vector<Entity*>(); + for (auto quickBuildID : finishedQuickBuilds) { + const auto quickBuildObject = Game::entityManager->GetEntity(quickBuildID); + if (quickBuildObject && quickBuildObject->GetLOT() == self->GetLOT()) { + quickBuildObject->SetVar<LWOOBJID>(u"Player_" + (GeneralUtils::to_u16string(groupNumber)), target->GetObjectID()); + finishedQuickBuildObjects.push_back(quickBuildObject); + } + } + + // If all 4 sets were built, do cool stuff + if (finishedQuickBuildObjects.size() >= 4) { + + // Move all the platforms so the user can collect the imagination brick + const auto movingPlatforms = Game::entityManager->GetEntitiesInGroup("ConcertPlatforms"); + for (auto* movingPlatform : movingPlatforms) { + auto* component = movingPlatform->GetComponent<MovingPlatformComponent>(); + if (component) { + component->WarpToWaypoint(component->GetLastWaypointIndex()); + + movingPlatform->AddCallbackTimer(resetStageTime, [movingPlatform, component]() { + component->WarpToWaypoint(0); + }); + } + } + + ProgressLicensedTechnician(self); + + // Reset all timers for the quickbuilds and make them indestructible + for (auto quickBuild : finishedQuickBuildObjects) { + quickBuild->SetNetworkVar<float>(u"startEffect", -1.0f); + quickBuild->CancelAllTimers(); + + // Indicate that the stage will reset + quickBuild->AddCallbackTimer(GetBlinkTime(resetStageTime), [quickBuild]() { + quickBuild->SetNetworkVar<float>(u"startEffect", GetBlinkTime(resetTime)); + }); + + // Reset the stage + quickBuild->AddCallbackTimer(resetStageTime, [quickBuild]() { + CancelEffects(quickBuild); + quickBuild->SetNetworkVar<float>(u"startEffect", -1); + quickBuild->Smash(); + }); + + auto* destroyableComponent = quickBuild->GetComponent<DestroyableComponent>(); + if (destroyableComponent) + destroyableComponent->SetFaction(-1); + } + + UpdateEffects(self); + return; + } + + // If not all 4 sets were built, reset the timers that were set on spawn + self->CancelAllTimers(); + + // Makes the quick build blink after a certain amount of time + self->AddCallbackTimer(GetBlinkTime(resetTime), [self]() { + self->SetNetworkVar<float>(u"startEffect", NsConcertQuickBuild::GetBlinkTime(resetActivatorTime)); + }); + + // Destroys the quick build after a while if it wasn't built + self->AddCallbackTimer(resetTime, [self]() { + self->SetNetworkVar<float>(u"startEffect", -1.0f); + self->Smash(self->GetObjectID()); + }); +} + +void NsConcertQuickBuild::ProgressStageCraft(Entity* self, Entity* player) { + auto* missionComponent = player->GetComponent<MissionComponent>(); + if (missionComponent) { + + // Has to be forced as to not accidentally trigger the licensed technician achievement + switch (self->GetLOT()) { + case 5845: + missionComponent->ForceProgress(283, 432, 5845); + break; + case 5846: + missionComponent->ForceProgress(283, 433, 5846); + break; + case 5847: + missionComponent->ForceProgress(283, 434, 5847); + break; + case 5848: + missionComponent->ForceProgress(283, 435, 5848); + break; + default: + break; + } + } +} + +void NsConcertQuickBuild::ProgressLicensedTechnician(Entity* self) { + for (auto i = 1; i < 5; i++) { + const auto playerID = self->GetVar<LWOOBJID>(u"Player_" + (GeneralUtils::to_u16string(i))); + if (playerID != LWOOBJID_EMPTY) { + const auto player = Game::entityManager->GetEntity(playerID); + if (player) { + auto playerMissionComponent = player->GetComponent<MissionComponent>(); + if (playerMissionComponent) + playerMissionComponent->ForceProgress(598, 903, self->GetLOT()); + } + } + } +} + +void NsConcertQuickBuild::UpdateEffects(Entity* self) { + CancelEffects(self); + + auto setIterator = quickBuildSets.find(self->GetLOT()); + if (setIterator == quickBuildSets.end()) + return; + + for (const auto& effectName : setIterator->second.effects) { + const auto effectObjects = Game::entityManager->GetEntitiesInGroup(quickBuildFX.at(effectName)); + for (auto* effectObject : effectObjects) { + GameMessages::SendPlayFXEffect(effectObject, 0, GeneralUtils::ASCIIToUTF16(effectName), + effectName + "Effect", LWOOBJID_EMPTY, 1, 1, true); + } + } +} + +void NsConcertQuickBuild::CancelEffects(Entity* self) { + auto setIterator = quickBuildSets.find(self->GetLOT()); + if (setIterator == quickBuildSets.end()) + return; + + for (const auto& effectName : setIterator->second.effects) { + const auto effectObjects = Game::entityManager->GetEntitiesInGroup(quickBuildFX.at(effectName)); + for (auto* effectObject : effectObjects) { + GameMessages::SendStopFXEffect(effectObject, true, effectName + "Effect"); + } + } +} diff --git a/dScripts/NsConcertQuickBuild.h b/dScripts/ai/NS/NsConcertQuickBuild.h similarity index 100% rename from dScripts/NsConcertQuickBuild.h rename to dScripts/ai/NS/NsConcertQuickBuild.h diff --git a/dScripts/ai/NS/NsGetFactionMissionServer.cpp b/dScripts/ai/NS/NsGetFactionMissionServer.cpp new file mode 100644 index 00000000..185bd344 --- /dev/null +++ b/dScripts/ai/NS/NsGetFactionMissionServer.cpp @@ -0,0 +1,56 @@ +#include "NsGetFactionMissionServer.h" +#include "GameMessages.h" +#include "MissionComponent.h" +#include "Character.h" +#include "eReplicaComponentType.h" +#include "ePlayerFlag.h" + +void NsGetFactionMissionServer::OnRespondToMission(Entity* self, int missionID, Entity* player, int reward) { + if (missionID != 474) return; + + if (reward != LOT_NULL) { + std::vector<int> factionMissions; + int celebrationID = -1; + int32_t flagID = -1; + + if (reward == 6980) { + // Venture League + factionMissions = { 555, 556 }; + celebrationID = 14; + flagID = 46; + } else if (reward == 6979) { + // Assembly + factionMissions = { 544, 545 }; + celebrationID = 15; + flagID = 47; + } else if (reward == 6981) { + // Paradox + factionMissions = { 577, 578 }; + celebrationID = 16; + flagID = 48; + } else if (reward == 6978) { + // Sentinel + factionMissions = { 566, 567 }; + celebrationID = 17; + flagID = 49; + } + + factionMissions.push_back(778); + + if (celebrationID != -1) { + GameMessages::SendStartCelebrationEffect(player, player->GetSystemAddress(), celebrationID); + } + + if (flagID != -1) { + player->GetCharacter()->SetPlayerFlag(ePlayerFlag::JOINED_A_FACTION, true); + player->GetCharacter()->SetPlayerFlag(flagID, true); + } + + MissionComponent* mis = static_cast<MissionComponent*>(player->GetComponent(eReplicaComponentType::MISSION)); + + for (int mission : factionMissions) { + mis->AcceptMission(mission); + mis->CompleteMission(mission); + } + } +} diff --git a/dScripts/NsGetFactionMissionServer.h b/dScripts/ai/NS/NsGetFactionMissionServer.h similarity index 100% rename from dScripts/NsGetFactionMissionServer.h rename to dScripts/ai/NS/NsGetFactionMissionServer.h diff --git a/dScripts/ai/NS/NsJohnnyMissionServer.cpp b/dScripts/ai/NS/NsJohnnyMissionServer.cpp new file mode 100644 index 00000000..107d3c44 --- /dev/null +++ b/dScripts/ai/NS/NsJohnnyMissionServer.cpp @@ -0,0 +1,15 @@ +#include "NsJohnnyMissionServer.h" +#include "MissionComponent.h" +#include "eMissionState.h" + +void NsJohnnyMissionServer::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + if (missionID == 773 && missionState <= eMissionState::ACTIVE) { + auto* missionComponent = target->GetComponent<MissionComponent>(); + if (missionComponent != nullptr) { + missionComponent->AcceptMission(774); + missionComponent->AcceptMission(775); + missionComponent->AcceptMission(776); + missionComponent->AcceptMission(777); + } + } +} diff --git a/dScripts/ai/NS/NsJohnnyMissionServer.h b/dScripts/ai/NS/NsJohnnyMissionServer.h new file mode 100644 index 00000000..c37ea06c --- /dev/null +++ b/dScripts/ai/NS/NsJohnnyMissionServer.h @@ -0,0 +1,6 @@ +#pragma once +#include "CppScripts.h" + +class NsJohnnyMissionServer : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +}; diff --git a/dScripts/ai/NS/NsModularBuild.cpp b/dScripts/ai/NS/NsModularBuild.cpp new file mode 100644 index 00000000..1922eb17 --- /dev/null +++ b/dScripts/ai/NS/NsModularBuild.cpp @@ -0,0 +1,18 @@ +#include "NsModularBuild.h" +#include "MissionComponent.h" +#include "eMissionState.h" +#include "eReplicaComponentType.h" + +void NsModularBuild::OnModularBuildExit(Entity* self, Entity* player, bool bCompleted, std::vector<LOT> modules) { + if (bCompleted) { + MissionComponent* mission = static_cast<MissionComponent*>(player->GetComponent(eReplicaComponentType::MISSION)); + + if (mission->GetMissionState(m_MissionNum) == eMissionState::ACTIVE) { + for (LOT mod : modules) { + if (mod == 9516 || mod == 9517 || mod == 9518) { + mission->ForceProgress(m_MissionNum, 1178, 1); + } + } + } + } +} diff --git a/dScripts/NsModularBuild.h b/dScripts/ai/NS/NsModularBuild.h similarity index 100% rename from dScripts/NsModularBuild.h rename to dScripts/ai/NS/NsModularBuild.h diff --git a/dScripts/ai/NS/NsQbImaginationStatue.cpp b/dScripts/ai/NS/NsQbImaginationStatue.cpp new file mode 100644 index 00000000..8f9a0b09 --- /dev/null +++ b/dScripts/ai/NS/NsQbImaginationStatue.cpp @@ -0,0 +1,40 @@ +#include "NsQbImaginationStatue.h" +#include "EntityManager.h" +#include "GameMessages.h" + +void NsQbImaginationStatue::OnStartup(Entity* self) { + +} + +void NsQbImaginationStatue::OnRebuildComplete(Entity* self, Entity* target) { + if (target == nullptr) return; + + self->SetVar(u"Player", target->GetObjectID()); + + SpawnLoot(self); + + self->AddTimer("SpawnDelay", 1.5f); + + self->AddTimer("StopSpawner", 10.0f); +} + +void NsQbImaginationStatue::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "SpawnDelay") { + SpawnLoot(self); + + self->AddTimer("SpawnDelay", 1.5f); + } else if (timerName == "StopSpawner") { + self->CancelAllTimers(); + } +} + +void NsQbImaginationStatue::SpawnLoot(Entity* self) { + const auto playerId = self->GetVar<LWOOBJID>(u"Player"); + + auto* player = Game::entityManager->GetEntity(playerId); + + if (player == nullptr) return; + + GameMessages::SendDropClientLoot(player, self->GetObjectID(), 935, 0); + GameMessages::SendDropClientLoot(player, self->GetObjectID(), 935, 0); +} diff --git a/dScripts/NsQbImaginationStatue.h b/dScripts/ai/NS/NsQbImaginationStatue.h similarity index 100% rename from dScripts/NsQbImaginationStatue.h rename to dScripts/ai/NS/NsQbImaginationStatue.h diff --git a/dScripts/ai/NS/WH/CMakeLists.txt b/dScripts/ai/NS/WH/CMakeLists.txt new file mode 100644 index 00000000..179a417f --- /dev/null +++ b/dScripts/ai/NS/WH/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_AI_NS_WH + "RockHydrantSmashable.cpp" + "RockHydrantBroken.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/NS/WH/RockHydrantBroken.cpp b/dScripts/ai/NS/WH/RockHydrantBroken.cpp new file mode 100644 index 00000000..243761d3 --- /dev/null +++ b/dScripts/ai/NS/WH/RockHydrantBroken.cpp @@ -0,0 +1,38 @@ +#include "RockHydrantBroken.h" +#include "EntityManager.h" +#include "GameMessages.h" + +void RockHydrantBroken::OnStartup(Entity* self) { + self->AddTimer("playEffect", 1); + + const auto hydrant = "hydrant" + self->GetVar<std::string>(u"hydrant"); + + const auto bouncers = Game::entityManager->GetEntitiesInGroup(hydrant); + + for (auto* bouncer : bouncers) { + self->SetVar<LWOOBJID>(u"bouncer", bouncer->GetObjectID()); + + + GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), true, UNASSIGNED_SYSTEM_ADDRESS); + + GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"enableCollision", UNASSIGNED_SYSTEM_ADDRESS); + } + + self->AddTimer("KillBroken", 10); +} + +void RockHydrantBroken::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "KillBroken") { + auto* bouncer = Game::entityManager->GetEntity(self->GetVar<LWOOBJID>(u"bouncer")); + + if (bouncer != nullptr) { + GameMessages::SendBouncerActiveStatus(bouncer->GetObjectID(), false, UNASSIGNED_SYSTEM_ADDRESS); + + GameMessages::SendNotifyObject(bouncer->GetObjectID(), self->GetObjectID(), u"disableCollision", UNASSIGNED_SYSTEM_ADDRESS); + } + + self->Kill(); + } else if (timerName == "playEffect") { + GameMessages::SendPlayFXEffect(self->GetObjectID(), 4737, u"water", "water", LWOOBJID_EMPTY, 1, 1, true); + } +} diff --git a/dScripts/RockHydrantBroken.h b/dScripts/ai/NS/WH/RockHydrantBroken.h similarity index 100% rename from dScripts/RockHydrantBroken.h rename to dScripts/ai/NS/WH/RockHydrantBroken.h diff --git a/dScripts/ai/NS/WH/RockHydrantSmashable.cpp b/dScripts/ai/NS/WH/RockHydrantSmashable.cpp new file mode 100644 index 00000000..d388baac --- /dev/null +++ b/dScripts/ai/NS/WH/RockHydrantSmashable.cpp @@ -0,0 +1,21 @@ +#include "RockHydrantSmashable.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "GeneralUtils.h" + +void RockHydrantSmashable::OnDie(Entity* self, Entity* killer) { + const auto hydrantName = self->GetVar<std::u16string>(u"hydrant"); + + LDFBaseData* data = new LDFData<std::string>(u"hydrant", GeneralUtils::UTF16ToWTF8(hydrantName)); + + EntityInfo info{}; + info.lot = ROCK_HYDRANT_BROKEN; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.settings = { data }; + info.spawnerID = self->GetSpawnerID(); + + auto* hydrant = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(hydrant); +} diff --git a/dScripts/RockHydrantSmashable.h b/dScripts/ai/NS/WH/RockHydrantSmashable.h similarity index 100% rename from dScripts/RockHydrantSmashable.h rename to dScripts/ai/NS/WH/RockHydrantSmashable.h diff --git a/dScripts/ai/NS/WhFans.cpp b/dScripts/ai/NS/WhFans.cpp new file mode 100644 index 00000000..a41a2c23 --- /dev/null +++ b/dScripts/ai/NS/WhFans.cpp @@ -0,0 +1,73 @@ +#include "WhFans.h" + +#include "RenderComponent.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "PhantomPhysicsComponent.h" +#include "RenderComponent.h" +#include "Entity.h" + +void WhFans::OnStartup(Entity* self) { + self->SetVar<bool>(u"alive", true); + self->SetVar<bool>(u"on", false); + + ToggleFX(self, false); +} + +void WhFans::ToggleFX(Entity* self, bool hit) { + std::string fanGroup; + const auto& groups = self->GetGroups(); + if (!groups.empty()) { + fanGroup = groups[0]; + } else { + fanGroup = ""; + } + + std::vector<Entity*> fanVolumes = Game::entityManager->GetEntitiesInGroup(fanGroup); + + auto* renderComponent = self->GetComponent<RenderComponent>(); + + if (renderComponent == nullptr) return; + + if (fanVolumes.size() == 0 || !self->GetVar<bool>(u"alive")) return; + + if (self->GetVar<bool>(u"on")) { + RenderComponent::PlayAnimation(self, u"fan-off"); + + renderComponent->StopEffect("fanOn"); + self->SetVar<bool>(u"on", false); + + for (Entity* volume : fanVolumes) { + auto volumePhys = volume->GetComponent<PhantomPhysicsComponent>(); + if (!volumePhys) continue; + volumePhys->SetPhysicsEffectActive(false); + Game::entityManager->SerializeEntity(volume); + } + } else if (!self->GetVar<bool>(u"on") && self->GetVar<bool>(u"alive")) { + RenderComponent::PlayAnimation(self, u"fan-on"); + + self->SetVar<bool>(u"on", true); + + for (Entity* volume : fanVolumes) { + auto volumePhys = volume->GetComponent<PhantomPhysicsComponent>(); + if (!volumePhys) continue; + volumePhys->SetPhysicsEffectActive(true); + Game::entityManager->SerializeEntity(volume); + } + } +} + +void WhFans::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, + int32_t param3) { + if (args.length() == 0 || !self->GetVar<bool>(u"alive")) return; + + if ((args == "turnOn" && self->GetVar<bool>(u"on")) || (args == "turnOff" && !self->GetVar<bool>(u"on"))) return; + ToggleFX(self, false); +} + +void WhFans::OnDie(Entity* self, Entity* killer) { + if (self->GetVar<bool>(u"on")) { + ToggleFX(self, true); + } + self->SetVar<bool>(u"alive", false); +} diff --git a/dScripts/WhFans.h b/dScripts/ai/NS/WhFans.h similarity index 100% rename from dScripts/WhFans.h rename to dScripts/ai/NS/WhFans.h diff --git a/dScripts/ai/PETS/CMakeLists.txt b/dScripts/ai/PETS/CMakeLists.txt new file mode 100644 index 00000000..93a9012d --- /dev/null +++ b/dScripts/ai/PETS/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_AI_PETS + "HydrantSmashable.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/PETS/HydrantSmashable.cpp b/dScripts/ai/PETS/HydrantSmashable.cpp new file mode 100644 index 00000000..fc83a5d3 --- /dev/null +++ b/dScripts/ai/PETS/HydrantSmashable.cpp @@ -0,0 +1,21 @@ +#include "HydrantSmashable.h" +#include "EntityManager.h" +#include "EntityInfo.h" +#include "GeneralUtils.h" + +void HydrantSmashable::OnDie(Entity* self, Entity* killer) { + const auto hydrantName = self->GetVar<std::u16string>(u"hydrant"); + + LDFBaseData* data = new LDFData<std::string>(u"hydrant", GeneralUtils::UTF16ToWTF8(hydrantName)); + + EntityInfo info{}; + info.lot = HYDRANT_BROKEN; + info.pos = self->GetPosition(); + info.rot = self->GetRotation(); + info.settings = { data }; + info.spawnerID = self->GetSpawnerID(); + + auto* hydrant = Game::entityManager->CreateEntity(info); + + Game::entityManager->ConstructEntity(hydrant); +} diff --git a/dScripts/HydrantSmashable.h b/dScripts/ai/PETS/HydrantSmashable.h similarity index 100% rename from dScripts/HydrantSmashable.h rename to dScripts/ai/PETS/HydrantSmashable.h diff --git a/dScripts/ai/PROPERTY/AG/AgPropGuard.cpp b/dScripts/ai/PROPERTY/AG/AgPropGuard.cpp new file mode 100644 index 00000000..e8e94b53 --- /dev/null +++ b/dScripts/ai/PROPERTY/AG/AgPropGuard.cpp @@ -0,0 +1,40 @@ +#include "AgPropGuard.h" +#include "Entity.h" +#include "Character.h" +#include "EntityManager.h" +#include "InventoryComponent.h" +#include "MissionComponent.h" +#include "Item.h" +#include "eMissionState.h" + +void AgPropGuard::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + auto* character = target->GetCharacter(); + auto* missionComponent = target->GetComponent<MissionComponent>(); + auto* inventoryComponent = target->GetComponent<InventoryComponent>(); + + const auto state = missionComponent->GetMissionState(320); + if (missionID == 768 && missionState == eMissionState::AVAILABLE) { + if (!character->GetPlayerFlag(71)) { + // TODO: Cinematic "MissionCam" + } + } else if (missionID == 768 && missionState >= eMissionState::READY_TO_COMPLETE) { + //remove the inventory items + for (int item : gearSets) { + auto* id = inventoryComponent->FindItemByLot(item); + + if (id) { + inventoryComponent->UnEquipItem(id); + inventoryComponent->RemoveItem(id->GetLot(), id->GetCount()); + } + } + } else if ( + (missionID == 320 && state == eMissionState::AVAILABLE) /*|| + (state == eMissionState::COMPLETE && missionID == 891 && missionState == eMissionState::READY_TO_COMPLETE)*/ + ) { + //GameMessages::SendNotifyClientObject(Game::entityManager->GetZoneControlEntity()->GetObjectID(), u"GuardChat", target->GetObjectID(), 0, target->GetObjectID(), "", target->GetSystemAddress()); + + target->GetCharacter()->SetPlayerFlag(113, true); + + Game::entityManager->GetZoneControlEntity()->AddTimer("GuardFlyAway", 1.0f); + } +} diff --git a/dScripts/ai/PROPERTY/AG/AgPropGuard.h b/dScripts/ai/PROPERTY/AG/AgPropGuard.h new file mode 100644 index 00000000..2b41f006 --- /dev/null +++ b/dScripts/ai/PROPERTY/AG/AgPropGuard.h @@ -0,0 +1,11 @@ +#pragma once +#include "CppScripts.h" + +class AgPropGuard final : public CppScripts::Script +{ +public: + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; + +private: + std::vector<int> gearSets = { 14359,14321,14353,14315 }; +}; diff --git a/dScripts/ai/PROPERTY/AG/CMakeLists.txt b/dScripts/ai/PROPERTY/AG/CMakeLists.txt new file mode 100644 index 00000000..f2139463 --- /dev/null +++ b/dScripts/ai/PROPERTY/AG/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_AI_PROPERTY_AG + "AgPropGuard.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/PROPERTY/AgPropguards.cpp b/dScripts/ai/PROPERTY/AgPropguards.cpp new file mode 100644 index 00000000..c514031a --- /dev/null +++ b/dScripts/ai/PROPERTY/AgPropguards.cpp @@ -0,0 +1,57 @@ +#include "AgPropguards.h" +#include "Character.h" +#include "GameMessages.h" +#include "EntityManager.h" +#include "dZoneManager.h" +#include "eMissionState.h" + +void AgPropguards::OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) { + auto* character = target->GetCharacter(); + if (character == nullptr) + return; + + const auto flag = GetFlagForMission(missionID); + if (flag == 0) + return; + + if ((missionState == eMissionState::AVAILABLE || missionState == eMissionState::ACTIVE) + && !character->GetPlayerFlag(flag)) { + // If the player just started the mission, play a cinematic highlighting the target + GameMessages::SendPlayCinematic(target->GetObjectID(), u"MissionCam", target->GetSystemAddress()); + } else if (missionState == eMissionState::COMPLETE_READY_TO_COMPLETE) { + // Makes the guard disappear once the mission has been completed + const auto zoneControlID = Game::entityManager->GetZoneControlEntity()->GetObjectID(); + GameMessages::SendNotifyClientObject(zoneControlID, u"GuardChat", 0, 0, self->GetObjectID(), + "", UNASSIGNED_SYSTEM_ADDRESS); + + self->AddCallbackTimer(5.0f, [self]() { + auto spawnerName = self->GetVar<std::string>(u"spawner_name"); + if (spawnerName.empty()) + spawnerName = "Guard"; + + auto spawners = Game::zoneManager->GetSpawnersByName(spawnerName); + for (auto* spawner : spawners) { + spawner->Deactivate(); + } + + self->Smash(); + }); + } +} + +int32_t AgPropguards::GetFlagForMission(uint32_t missionID) { + switch (missionID) { + case 872: + return 97; + case 873: + return 98; + case 874: + return 99; + case 1293: + return 118; + case 1322: + return 122; + default: + return 0; + } +} diff --git a/dScripts/ai/PROPERTY/AgPropguards.h b/dScripts/ai/PROPERTY/AgPropguards.h new file mode 100644 index 00000000..ed2e3cb0 --- /dev/null +++ b/dScripts/ai/PROPERTY/AgPropguards.h @@ -0,0 +1,8 @@ +#pragma once +#include "CppScripts.h" + +class AgPropguards : public CppScripts::Script { + void OnMissionDialogueOK(Entity* self, Entity* target, int missionID, eMissionState missionState) override; +private: + static int32_t GetFlagForMission(uint32_t missionID); +}; diff --git a/dScripts/ai/PROPERTY/CMakeLists.txt b/dScripts/ai/PROPERTY/CMakeLists.txt new file mode 100644 index 00000000..295137b4 --- /dev/null +++ b/dScripts/ai/PROPERTY/CMakeLists.txt @@ -0,0 +1,11 @@ +set(DSCRIPTS_SOURCES_AI_PROPERTY + "AgPropguards.cpp" + "PropertyFXDamage.cpp") + +add_subdirectory(AG) + +foreach(file ${DSCRIPTS_SOURCES_AI_PROPERTY_AG}) + set(DSCRIPTS_SOURCES_AI_PROPERTY ${DSCRIPTS_SOURCES_AI_PROPERTY} "AG/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI_PROPERTY ${DSCRIPTS_SOURCES_AI_PROPERTY} PARENT_SCOPE) diff --git a/dScripts/ai/PROPERTY/PropertyFXDamage.cpp b/dScripts/ai/PROPERTY/PropertyFXDamage.cpp new file mode 100644 index 00000000..56079384 --- /dev/null +++ b/dScripts/ai/PROPERTY/PropertyFXDamage.cpp @@ -0,0 +1,18 @@ +#include "PropertyFXDamage.h" +#include "DestroyableComponent.h" +#include "SkillComponent.h" + +void PropertyFXDamage::OnCollisionPhantom(Entity* self, Entity* target) { + if (target == nullptr) + return; + + auto* skills = self->GetComponent<SkillComponent>(); + auto* targetStats = target->GetComponent<DestroyableComponent>(); + + if (skills != nullptr && targetStats != nullptr) { + auto targetFactions = targetStats->GetFactionIDs(); + if (std::find(targetFactions.begin(), targetFactions.end(), 1) != targetFactions.end()) { + skills->CalculateBehavior(692, 11386, target->GetObjectID()); + } + } +} diff --git a/dScripts/PropertyFXDamage.h b/dScripts/ai/PROPERTY/PropertyFXDamage.h similarity index 100% rename from dScripts/PropertyFXDamage.h rename to dScripts/ai/PROPERTY/PropertyFXDamage.h diff --git a/dScripts/ai/RACING/CMakeLists.txt b/dScripts/ai/RACING/CMakeLists.txt new file mode 100644 index 00000000..0c1918de --- /dev/null +++ b/dScripts/ai/RACING/CMakeLists.txt @@ -0,0 +1,9 @@ +set(DSCRIPTS_SOURCES_AI_RACING) + +add_subdirectory(OBJECTS) + +foreach(file ${DSCRIPTS_SOURCES_AI_RACING_OBJECTS}) + set(DSCRIPTS_SOURCES_AI_RACING ${DSCRIPTS_SOURCES_AI_RACING} "OBJECTS/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_AI_RACING ${DSCRIPTS_SOURCES_AI_RACING} PARENT_SCOPE) diff --git a/dScripts/ai/RACING/OBJECTS/CMakeLists.txt b/dScripts/ai/RACING/OBJECTS/CMakeLists.txt new file mode 100644 index 00000000..4ef427d5 --- /dev/null +++ b/dScripts/ai/RACING/OBJECTS/CMakeLists.txt @@ -0,0 +1,6 @@ +set(DSCRIPTS_SOURCES_AI_RACING_OBJECTS + "RaceImagineCrateServer.cpp" + "RaceImaginePowerup.cpp" + "FvRaceSmashEggImagineServer.cpp" + "RaceSmashServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/RACING/OBJECTS/FvRaceSmashEggImagineServer.cpp b/dScripts/ai/RACING/OBJECTS/FvRaceSmashEggImagineServer.cpp new file mode 100644 index 00000000..2bdb0364 --- /dev/null +++ b/dScripts/ai/RACING/OBJECTS/FvRaceSmashEggImagineServer.cpp @@ -0,0 +1,39 @@ +#include "FvRaceSmashEggImagineServer.h" +#include "CharacterComponent.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "PossessableComponent.h" +#include "eRacingTaskParam.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" + +void FvRaceSmashEggImagineServer::OnDie(Entity* self, Entity* killer) { + if (killer != nullptr) { + auto* destroyableComponent = killer->GetComponent<DestroyableComponent>(); + if (destroyableComponent != nullptr) { + destroyableComponent->SetImagination(destroyableComponent->GetImagination() + 10); + Game::entityManager->SerializeEntity(killer); + } + + // get possessor to progress statistics and tasks. + auto* possessableComponent = killer->GetComponent<PossessableComponent>(); + if (possessableComponent != nullptr) { + + auto* possessor = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); + if (possessor != nullptr) { + + auto* missionComponent = possessor->GetComponent<MissionComponent>(); + auto* characterComponent = possessor->GetComponent<CharacterComponent>(); + if (characterComponent != nullptr) { + characterComponent->UpdatePlayerStatistic(ImaginationPowerUpsCollected); + characterComponent->UpdatePlayerStatistic(RacingSmashablesSmashed); + } + if (missionComponent == nullptr) return; + // Dragon eggs have their own smash server so we handle mission progression for them here. + missionComponent->Progress(eMissionTaskType::RACING, 0, (LWOOBJID)eRacingTaskParam::SMASHABLES); + missionComponent->Progress(eMissionTaskType::RACING, self->GetLOT(), (LWOOBJID)eRacingTaskParam::SMASH_SPECIFIC_SMASHABLE); + } + } + + } +} diff --git a/dScripts/FvRaceSmashEggImagineServer.h b/dScripts/ai/RACING/OBJECTS/FvRaceSmashEggImagineServer.h similarity index 100% rename from dScripts/FvRaceSmashEggImagineServer.h rename to dScripts/ai/RACING/OBJECTS/FvRaceSmashEggImagineServer.h diff --git a/dScripts/ai/RACING/OBJECTS/RaceImagineCrateServer.cpp b/dScripts/ai/RACING/OBJECTS/RaceImagineCrateServer.cpp new file mode 100644 index 00000000..20a3c0cc --- /dev/null +++ b/dScripts/ai/RACING/OBJECTS/RaceImagineCrateServer.cpp @@ -0,0 +1,56 @@ +#include "CharacterComponent.h" +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "PossessableComponent.h" +#include "RaceImagineCrateServer.h" +#include "eRacingTaskParam.h" +#include "MissionComponent.h" +#include "SkillComponent.h" +#include "eMissionTaskType.h" + +void RaceImagineCrateServer::OnDie(Entity* self, Entity* killer) { + if (self->GetVar<bool>(u"bIsDead")) { + return; + } + + self->SetVar<bool>(u"bIsDead", true); + + if (killer == nullptr) { + return; + } + + auto* skillComponent = killer->GetComponent<SkillComponent>(); + + if (skillComponent == nullptr) { + return; + } + + auto* destroyableComponent = killer->GetComponent<DestroyableComponent>(); + + if (destroyableComponent != nullptr) { + destroyableComponent->SetImagination(60); + + Game::entityManager->SerializeEntity(killer); + } + + // Find possessor of race car to progress missions and update stats. + auto* possessableComponent = killer->GetComponent<PossessableComponent>(); + if (possessableComponent != nullptr) { + + auto* possessor = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); + if (possessor != nullptr) { + + auto* missionComponent = possessor->GetComponent<MissionComponent>(); + auto* characterComponent = possessor->GetComponent<CharacterComponent>(); + + if (characterComponent != nullptr) { + characterComponent->UpdatePlayerStatistic(RacingImaginationCratesSmashed); + characterComponent->UpdatePlayerStatistic(RacingSmashablesSmashed); + } + + // Progress racing smashable missions + if (missionComponent == nullptr) return; + missionComponent->Progress(eMissionTaskType::RACING, 0, (LWOOBJID)eRacingTaskParam::SMASHABLES); + } + } +} diff --git a/dScripts/RaceImagineCrateServer.h b/dScripts/ai/RACING/OBJECTS/RaceImagineCrateServer.h similarity index 100% rename from dScripts/RaceImagineCrateServer.h rename to dScripts/ai/RACING/OBJECTS/RaceImagineCrateServer.h diff --git a/dScripts/ai/RACING/OBJECTS/RaceImaginePowerup.cpp b/dScripts/ai/RACING/OBJECTS/RaceImaginePowerup.cpp new file mode 100644 index 00000000..d55eeffd --- /dev/null +++ b/dScripts/ai/RACING/OBJECTS/RaceImaginePowerup.cpp @@ -0,0 +1,37 @@ +#include "DestroyableComponent.h" +#include "EntityManager.h" +#include "PossessorComponent.h" +#include "RaceImaginePowerup.h" +#include "eRacingTaskParam.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" + +void RaceImaginePowerup::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, + int32_t param2, int32_t param3) { + if (sender->IsPlayer() && args == "powerup") { + auto* possessorComponent = sender->GetComponent<PossessorComponent>(); + + if (possessorComponent == nullptr) { + return; + } + + auto* vehicle = Game::entityManager->GetEntity(possessorComponent->GetPossessable()); + + if (vehicle == nullptr) { + return; + } + + auto* destroyableComponent = vehicle->GetComponent<DestroyableComponent>(); + + if (destroyableComponent == nullptr) { + return; + } + + destroyableComponent->Imagine(10); + + auto* missionComponent = sender->GetComponent<MissionComponent>(); + + if (missionComponent == nullptr) return; + missionComponent->Progress(eMissionTaskType::RACING, self->GetLOT(), (LWOOBJID)eRacingTaskParam::COLLECT_IMAGINATION); + } +} diff --git a/dScripts/RaceImaginePowerup.h b/dScripts/ai/RACING/OBJECTS/RaceImaginePowerup.h similarity index 100% rename from dScripts/RaceImaginePowerup.h rename to dScripts/ai/RACING/OBJECTS/RaceImaginePowerup.h diff --git a/dScripts/ai/RACING/OBJECTS/RaceSmashServer.cpp b/dScripts/ai/RACING/OBJECTS/RaceSmashServer.cpp new file mode 100644 index 00000000..5fcb2ff4 --- /dev/null +++ b/dScripts/ai/RACING/OBJECTS/RaceSmashServer.cpp @@ -0,0 +1,31 @@ +#include "CharacterComponent.h" +#include "EntityManager.h" +#include "PossessableComponent.h" +#include "RaceSmashServer.h" +#include "eRacingTaskParam.h" +#include "MissionComponent.h" +#include "eMissionTaskType.h" + +void RaceSmashServer::OnDie(Entity* self, Entity* killer) { + // Crate is smashed by the car + auto* possessableComponent = killer->GetComponent<PossessableComponent>(); + if (possessableComponent != nullptr) { + + auto* possessor = Game::entityManager->GetEntity(possessableComponent->GetPossessor()); + if (possessor != nullptr) { + + auto* missionComponent = possessor->GetComponent<MissionComponent>(); + auto* characterComponent = possessor->GetComponent<CharacterComponent>(); + + if (characterComponent != nullptr) { + characterComponent->UpdatePlayerStatistic(RacingSmashablesSmashed); + } + + // Progress racing smashable missions + if (missionComponent == nullptr) return; + missionComponent->Progress(eMissionTaskType::RACING, 0, (LWOOBJID)eRacingTaskParam::SMASHABLES); + // Progress missions that ask us to smash a specific smashable. + missionComponent->Progress(eMissionTaskType::RACING, self->GetLOT(), (LWOOBJID)eRacingTaskParam::SMASH_SPECIFIC_SMASHABLE); + } + } +} diff --git a/dScripts/RaceSmashServer.h b/dScripts/ai/RACING/OBJECTS/RaceSmashServer.h similarity index 100% rename from dScripts/RaceSmashServer.h rename to dScripts/ai/RACING/OBJECTS/RaceSmashServer.h diff --git a/dScripts/ai/SPEC/CMakeLists.txt b/dScripts/ai/SPEC/CMakeLists.txt new file mode 100644 index 00000000..42dbf8f8 --- /dev/null +++ b/dScripts/ai/SPEC/CMakeLists.txt @@ -0,0 +1,5 @@ +set(DSCRIPTS_SOURCES_AI_SPEC + "SpecialCoinSpawner.cpp" + "SpecialPowerupSpawner.cpp" + "SpecialSpeedBuffSpawner.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/SPEC/SpecialCoinSpawner.cpp b/dScripts/ai/SPEC/SpecialCoinSpawner.cpp new file mode 100644 index 00000000..ff494845 --- /dev/null +++ b/dScripts/ai/SPEC/SpecialCoinSpawner.cpp @@ -0,0 +1,16 @@ +#include "SpecialCoinSpawner.h" +#include "CharacterComponent.h" + +void SpecialCoinSpawner::OnStartup(Entity* self) { + self->SetProximityRadius(1.5f, "powerupEnter"); +} + +void SpecialCoinSpawner::OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) { + if (name != "powerupEnter" && status != "ENTER") return; + if (!entering->IsPlayer()) return; + auto character = entering->GetCharacter(); + if (!character) return; + GameMessages::SendPlayFXEffect(self, -1, u"pickup", "", LWOOBJID_EMPTY, 1, 1, true); + character->SetCoins(character->GetCoins() + this->m_CurrencyDenomination, eLootSourceType::CURRENCY); + self->Smash(entering->GetObjectID(), eKillType::SILENT); +} diff --git a/dScripts/ai/SPEC/SpecialCoinSpawner.h b/dScripts/ai/SPEC/SpecialCoinSpawner.h new file mode 100644 index 00000000..5af6f24a --- /dev/null +++ b/dScripts/ai/SPEC/SpecialCoinSpawner.h @@ -0,0 +1,13 @@ +#pragma once +#include "CppScripts.h" + +class SpecialCoinSpawner : public CppScripts::Script { +public: + SpecialCoinSpawner(uint32_t CurrencyDenomination) { + m_CurrencyDenomination = CurrencyDenomination; + }; + void OnStartup(Entity* self) override; + void OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) override; +private: + int32_t m_CurrencyDenomination = 0; +}; diff --git a/dScripts/ai/SPEC/SpecialPowerupSpawner.cpp b/dScripts/ai/SPEC/SpecialPowerupSpawner.cpp new file mode 100644 index 00000000..72565923 --- /dev/null +++ b/dScripts/ai/SPEC/SpecialPowerupSpawner.cpp @@ -0,0 +1,26 @@ +#include "SpecialPowerupSpawner.h" + +#include "GameMessages.h" +#include "SkillComponent.h" +#include "EntityManager.h" +#include "eReplicaComponentType.h" + +void SpecialPowerupSpawner::OnStartup(Entity* self) { + self->SetProximityRadius(1.5f, "powerupEnter"); + self->SetVar(u"bIsDead", false); +} + +void SpecialPowerupSpawner::OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) { + if (name != "powerupEnter" && status != "ENTER") return; + if (!entering->IsPlayer()) return; + if (self->GetVar<bool>(u"bIsDead")) return; + + GameMessages::SendPlayFXEffect(self, -1, u"pickup", "", LWOOBJID_EMPTY, 1, 1, true); + + auto skillComponent = self->GetComponent<SkillComponent>(); + if (!skillComponent) return; + skillComponent->CastSkill(this->m_SkillId, entering->GetObjectID()); + + self->SetVar(u"bIsDead", true); + self->Smash(entering->GetObjectID(), eKillType::SILENT); +} diff --git a/dScripts/ai/SPEC/SpecialPowerupSpawner.h b/dScripts/ai/SPEC/SpecialPowerupSpawner.h new file mode 100644 index 00000000..b27e9789 --- /dev/null +++ b/dScripts/ai/SPEC/SpecialPowerupSpawner.h @@ -0,0 +1,13 @@ +#pragma once +#include "CppScripts.h" + +class SpecialPowerupSpawner : public CppScripts::Script { +public: + SpecialPowerupSpawner(uint32_t skillId) { + m_SkillId = skillId; + }; + void OnStartup(Entity* self) override; + void OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) override; +private: + uint32_t m_SkillId = 0; +}; diff --git a/dScripts/ai/SPEC/SpecialSpeedBuffSpawner.cpp b/dScripts/ai/SPEC/SpecialSpeedBuffSpawner.cpp new file mode 100644 index 00000000..d3109806 --- /dev/null +++ b/dScripts/ai/SPEC/SpecialSpeedBuffSpawner.cpp @@ -0,0 +1,26 @@ +#include "SpecialSpeedBuffSpawner.h" + +#include "GameMessages.h" +#include "SkillComponent.h" +#include "EntityManager.h" +#include "eReplicaComponentType.h" + +void SpecialSpeedBuffSpawner::OnStartup(Entity* self) { + self->SetProximityRadius(1.5f, "powerupEnter"); + self->SetVar(u"bIsDead", false); +} + +void SpecialSpeedBuffSpawner::OnProximityUpdate(Entity* self, Entity* entering, const std::string name, const std::string status) { + if (name != "powerupEnter" && status != "ENTER") return; + if (!entering->IsPlayer()) return; + if (self->GetVar<bool>(u"bIsDead")) return; + + GameMessages::SendPlayFXEffect(self, -1, u"pickup", "", LWOOBJID_EMPTY, 1, 1, true); + + auto skillComponent = entering->GetComponent<SkillComponent>(); + if (!skillComponent) return; + skillComponent->CastSkill(this->m_SkillId, entering->GetObjectID()); + + self->SetVar(u"bIsDead", true); + self->Smash(entering->GetObjectID(), eKillType::SILENT); +} diff --git a/dScripts/ai/SPEC/SpecialSpeedBuffSpawner.h b/dScripts/ai/SPEC/SpecialSpeedBuffSpawner.h new file mode 100644 index 00000000..e1741691 --- /dev/null +++ b/dScripts/ai/SPEC/SpecialSpeedBuffSpawner.h @@ -0,0 +1,10 @@ +#pragma once +#include "CppScripts.h" + +class SpecialSpeedBuffSpawner : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) override; +private: + uint32_t m_SkillId = 500; +}; diff --git a/dScripts/AllCrateChicken.cpp b/dScripts/ai/WILD/AllCrateChicken.cpp similarity index 100% rename from dScripts/AllCrateChicken.cpp rename to dScripts/ai/WILD/AllCrateChicken.cpp diff --git a/dScripts/AllCrateChicken.h b/dScripts/ai/WILD/AllCrateChicken.h similarity index 100% rename from dScripts/AllCrateChicken.h rename to dScripts/ai/WILD/AllCrateChicken.h diff --git a/dScripts/ai/WILD/CMakeLists.txt b/dScripts/ai/WILD/CMakeLists.txt new file mode 100644 index 00000000..446ce0d4 --- /dev/null +++ b/dScripts/ai/WILD/CMakeLists.txt @@ -0,0 +1,11 @@ +set(DSCRIPTS_SOURCES_AI_WILD + "AllCrateChicken.cpp" + "WildAmbients.cpp" + "WildAmbientCrab.cpp" + "WildAndScared.cpp" + "WildGfGlowbug.cpp" + "WildNinjaBricks.cpp" + "WildNinjaStudent.cpp" + "WildNinjaSensei.cpp" + "WildPants.cpp" + PARENT_SCOPE) diff --git a/dScripts/ai/WILD/WildAmbientCrab.cpp b/dScripts/ai/WILD/WildAmbientCrab.cpp new file mode 100644 index 00000000..7b5b3d45 --- /dev/null +++ b/dScripts/ai/WILD/WildAmbientCrab.cpp @@ -0,0 +1,27 @@ +#include "WildAmbientCrab.h" +#include "GameMessages.h" + +void WildAmbientCrab::OnStartup(Entity* self){ + self->SetVar(u"flipped", true); + GameMessages::SendPlayAnimation(self, u"idle"); +} + +void WildAmbientCrab::OnUse(Entity* self, Entity* user) { + auto flipped = self->GetVar<bool>(u"flipped"); + if (flipped) { + self->AddTimer("Flipping", 0.6f); + GameMessages::SendPlayAnimation(self, u"flip-over"); + self->SetVar(u"flipped", false); + } else if (!flipped) { + self->AddTimer("Flipback", 0.8f); + GameMessages::SendPlayAnimation(self, u"flip-back"); + self->SetVar(u"flipped", true); + } +} + +void WildAmbientCrab::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "Flipping") GameMessages::SendPlayAnimation(self, u"over-idle"); + else if (timerName == "Flipback") GameMessages::SendPlayAnimation(self, u"idle"); +} + + diff --git a/dScripts/ai/WILD/WildAmbientCrab.h b/dScripts/ai/WILD/WildAmbientCrab.h new file mode 100644 index 00000000..0e1b3877 --- /dev/null +++ b/dScripts/ai/WILD/WildAmbientCrab.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class WildAmbientCrab final : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnTimerDone(Entity* self, std::string timerName) override; + void OnUse(Entity* self, Entity* user) override; +}; diff --git a/dScripts/ai/WILD/WildAmbients.cpp b/dScripts/ai/WILD/WildAmbients.cpp new file mode 100644 index 00000000..c21b6d76 --- /dev/null +++ b/dScripts/ai/WILD/WildAmbients.cpp @@ -0,0 +1,7 @@ +#include "WildAmbients.h" +#include "GameMessages.h" +#include "RenderComponent.h" + +void WildAmbients::OnUse(Entity* self, Entity* user) { + RenderComponent::PlayAnimation(self, u"interact"); +} diff --git a/dScripts/WildAmbients.h b/dScripts/ai/WILD/WildAmbients.h similarity index 100% rename from dScripts/WildAmbients.h rename to dScripts/ai/WILD/WildAmbients.h diff --git a/dScripts/ai/WILD/WildAndScared.cpp b/dScripts/ai/WILD/WildAndScared.cpp new file mode 100644 index 00000000..d2e89c40 --- /dev/null +++ b/dScripts/ai/WILD/WildAndScared.cpp @@ -0,0 +1,6 @@ +#include "WildAndScared.h" +#include "GameMessages.h" + +void WildAndScared::OnUse(Entity* self, Entity* user) { + GameMessages::SendPlayAnimation(self, u"scared"); +} diff --git a/dScripts/ai/WILD/WildAndScared.h b/dScripts/ai/WILD/WildAndScared.h new file mode 100644 index 00000000..c94fb06d --- /dev/null +++ b/dScripts/ai/WILD/WildAndScared.h @@ -0,0 +1,7 @@ +#pragma once +#include "CppScripts.h" + +class WildAndScared : public CppScripts::Script { +public: + void OnUse(Entity* self, Entity* user) override; +}; diff --git a/dScripts/ai/WILD/WildGfGlowbug.cpp b/dScripts/ai/WILD/WildGfGlowbug.cpp new file mode 100644 index 00000000..d834f5b5 --- /dev/null +++ b/dScripts/ai/WILD/WildGfGlowbug.cpp @@ -0,0 +1,28 @@ +#include "WildGfGlowbug.h" +#include "GameMessages.h" + +void WildGfGlowbug::OnStartup(Entity* self){ + self->SetVar(u"switch", false); +} + +void WildGfGlowbug::OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) { + if (args == "physicsReady") { + auto switchState = self->GetVar<bool>(u"switch"); + if (!switchState) { + GameMessages::SendStopFXEffect(self, true, "glowlight"); + } else if (switchState) { + GameMessages::SendPlayFXEffect(self, -1, u"light", "glowlight", LWOOBJID_EMPTY); + } + } +} + +void WildGfGlowbug::OnUse(Entity* self, Entity* user) { + auto switchState = self->GetVar<bool>(u"switch"); + if (switchState) { + GameMessages::SendStopFXEffect(self, true, "glowlight"); + self->SetVar(u"switch", false); + } else if (!switchState) { + GameMessages::SendPlayFXEffect(self, -1, u"light", "glowlight", LWOOBJID_EMPTY); + self->SetVar(u"switch", true); + } +} diff --git a/dScripts/ai/WILD/WildGfGlowbug.h b/dScripts/ai/WILD/WildGfGlowbug.h new file mode 100644 index 00000000..03242372 --- /dev/null +++ b/dScripts/ai/WILD/WildGfGlowbug.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class WildGfGlowbug : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnFireEventServerSide(Entity* self, Entity* sender, std::string args, int32_t param1, int32_t param2, int32_t param3) override; + void OnUse(Entity* self, Entity* user) override; +}; diff --git a/dScripts/ai/WILD/WildNinjaBricks.cpp b/dScripts/ai/WILD/WildNinjaBricks.cpp new file mode 100644 index 00000000..4fa65b01 --- /dev/null +++ b/dScripts/ai/WILD/WildNinjaBricks.cpp @@ -0,0 +1,13 @@ +#include "WildNinjaBricks.h" +#include "Entity.h" + +void WildNinjaBricks::OnStartup(Entity* self) { + self->AddToGroup("Ninjastuff"); +} + +void WildNinjaBricks::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { + if (name == "Crane") GameMessages::SendPlayAnimation(self, u"crane"); + else if (name == "Tiger") GameMessages::SendPlayAnimation(self, u"tiger"); + else if (name == "Mantis") GameMessages::SendPlayAnimation(self, u"mantis"); +} + diff --git a/dScripts/ai/WILD/WildNinjaBricks.h b/dScripts/ai/WILD/WildNinjaBricks.h new file mode 100644 index 00000000..9578e37a --- /dev/null +++ b/dScripts/ai/WILD/WildNinjaBricks.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class WildNinjaBricks : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1 = 0, int32_t param2 = 0) override; +}; + diff --git a/dScripts/ai/WILD/WildNinjaSensei.cpp b/dScripts/ai/WILD/WildNinjaSensei.cpp new file mode 100644 index 00000000..0941e41c --- /dev/null +++ b/dScripts/ai/WILD/WildNinjaSensei.cpp @@ -0,0 +1,36 @@ +#include "WildNinjaSensei.h" +#include "Entity.h" + +void WildNinjaSensei::OnStartup(Entity* self) { + GameMessages::SendPlayAnimation(self, u"bow"); + self->AddTimer("CraneStart", 5); +} + +void WildNinjaSensei::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "CraneStart") { + auto ninjas = Game::entityManager->GetEntitiesInGroup("Ninjastuff"); + for (auto ninja : ninjas) ninja->NotifyObject(self, "Crane"); + self->AddTimer("Bow", 15.5f); + self->AddTimer("TigerStart", 25); + GameMessages::SendPlayAnimation(self, u"crane"); + } else if (timerName == "TigerStart") { + auto ninjas = Game::entityManager->GetEntitiesInGroup("Ninjastuff"); + GameMessages::SendPlayAnimation(self, u"bow"); + for (auto ninja : ninjas) ninja->NotifyObject(self, "Tiger"); + self->AddTimer("Bow", 15.5f); + self->AddTimer("MantisStart", 25); + GameMessages::SendPlayAnimation(self, u"tiger"); + } else if (timerName == "MantisStart") { + auto ninjas = Game::entityManager->GetEntitiesInGroup("Ninjastuff"); + GameMessages::SendPlayAnimation(self, u"tiger"); + for (auto ninja : ninjas) ninja->NotifyObject(self, "Mantis"); + self->AddTimer("Bow", 15.5f); + self->AddTimer("CraneStart", 25); + GameMessages::SendPlayAnimation(self, u"mantis"); + } else if (timerName == "Bow") { + auto ninjas = Game::entityManager->GetEntitiesInGroup("Ninjastuff"); + for (auto ninja : ninjas) ninja->NotifyObject(self, "Bow"); + GameMessages::SendPlayAnimation(self, u"bow"); + } +} + diff --git a/dScripts/ai/WILD/WildNinjaSensei.h b/dScripts/ai/WILD/WildNinjaSensei.h new file mode 100644 index 00000000..c14b6f08 --- /dev/null +++ b/dScripts/ai/WILD/WildNinjaSensei.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class WildNinjaSensei : public CppScripts::Script { +public: + void OnStartup(Entity* self); + void OnTimerDone(Entity* self, std::string timerName); +}; + diff --git a/dScripts/ai/WILD/WildNinjaStudent.cpp b/dScripts/ai/WILD/WildNinjaStudent.cpp new file mode 100644 index 00000000..b7e2f585 --- /dev/null +++ b/dScripts/ai/WILD/WildNinjaStudent.cpp @@ -0,0 +1,14 @@ +#include "WildNinjaStudent.h" +#include "GameMessages.h" + +void WildNinjaStudent::OnStartup(Entity* self) { + self->AddToGroup("Ninjastuff"); + GameMessages::SendPlayAnimation(self, u"bow"); +} + +void WildNinjaStudent::OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1, int32_t param2) { + if (name == "Crane") GameMessages::SendPlayAnimation(self, u"crane"); + else if (name == "Tiger") GameMessages::SendPlayAnimation(self, u"tiger"); + else if (name == "Mantis") GameMessages::SendPlayAnimation(self, u"mantis"); + else if (name == "Bow") GameMessages::SendPlayAnimation(self, u"bow"); +} diff --git a/dScripts/ai/WILD/WildNinjaStudent.h b/dScripts/ai/WILD/WildNinjaStudent.h new file mode 100644 index 00000000..b76e5fa5 --- /dev/null +++ b/dScripts/ai/WILD/WildNinjaStudent.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class WildNinjaStudent : public CppScripts::Script { +public: + void OnStartup(Entity* self) override; + void OnNotifyObject(Entity* self, Entity* sender, const std::string& name, int32_t param1 = 0, int32_t param2 = 0) override; +}; + diff --git a/dScripts/ai/WILD/WildPants.cpp b/dScripts/ai/WILD/WildPants.cpp new file mode 100644 index 00000000..7c5e1cd2 --- /dev/null +++ b/dScripts/ai/WILD/WildPants.cpp @@ -0,0 +1,10 @@ +#include "WildPants.h" +#include "GameMessages.h" + +void WildPants::OnStartup(Entity* self) { + self->SetProximityRadius(5, "scardyPants"); +} + +void WildPants::OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) { + if (status == "ENTER") GameMessages::SendPlayAnimation(self, u"scared"); +} diff --git a/dScripts/ai/WILD/WildPants.h b/dScripts/ai/WILD/WildPants.h new file mode 100644 index 00000000..c6968045 --- /dev/null +++ b/dScripts/ai/WILD/WildPants.h @@ -0,0 +1,9 @@ +#pragma once +#include "CppScripts.h" + +class WildPants : public CppScripts::Script +{ +public: + void OnStartup(Entity* self) override; + void OnProximityUpdate(Entity* self, Entity* entering, std::string name, std::string status) override; +}; diff --git a/dScripts/client/CMakeLists.txt b/dScripts/client/CMakeLists.txt new file mode 100644 index 00000000..c2777508 --- /dev/null +++ b/dScripts/client/CMakeLists.txt @@ -0,0 +1,9 @@ +set(DSCRIPTS_SOURCES_CLIENT) + +add_subdirectory(ai) + +foreach(file ${DSCRIPTS_SOURCES_CLIENT_AI}) + set(DSCRIPTS_SOURCES_CLIENT ${DSCRIPTS_SOURCES_CLIENT} "ai/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_CLIENT ${DSCRIPTS_SOURCES_CLIENT} PARENT_SCOPE) diff --git a/dScripts/client/ai/CMakeLists.txt b/dScripts/client/ai/CMakeLists.txt new file mode 100644 index 00000000..c1358c57 --- /dev/null +++ b/dScripts/client/ai/CMakeLists.txt @@ -0,0 +1,9 @@ +set(DSCRIPTS_SOURCES_CLIENT_AI) + +add_subdirectory(PR) + +foreach(file ${DSCRIPTS_SOURCES_CLIENT_AI_PR}) + set(DSCRIPTS_SOURCES_CLIENT_AI ${DSCRIPTS_SOURCES_CLIENT_AI} "PR/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_CLIENT_AI ${DSCRIPTS_SOURCES_CLIENT_AI} PARENT_SCOPE) diff --git a/dScripts/client/ai/PR/CMakeLists.txt b/dScripts/client/ai/PR/CMakeLists.txt new file mode 100644 index 00000000..ef7d5d6a --- /dev/null +++ b/dScripts/client/ai/PR/CMakeLists.txt @@ -0,0 +1,4 @@ +set(DSCRIPTS_SOURCES_CLIENT_AI_PR + "PrWhistle.cpp" + "CrabServer.cpp" + PARENT_SCOPE) diff --git a/dScripts/client/ai/PR/CrabServer.cpp b/dScripts/client/ai/PR/CrabServer.cpp new file mode 100644 index 00000000..f30142ba --- /dev/null +++ b/dScripts/client/ai/PR/CrabServer.cpp @@ -0,0 +1,44 @@ +#include "CrabServer.h" +#include "PetComponent.h" +#include "ePetTamingNotifyType.h" + +void CrabServer::OnStartup(Entity* self) { + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr || petComponent->GetOwner() != nullptr) + return; + + // Triggers the local crab script for taming etc. + auto tamer = self->GetVar<LWOOBJID>(u"tamer"); + // Client compares this with player:GetID() which is a string, so we'll have to give it a string + self->SetNetworkVar(u"crabtamer", std::to_string(tamer)); + + // Kill if the player decides that the crab is not worthy + self->AddTimer("killself", 45.0f); +} + +void CrabServer::OnTimerDone(Entity* self, std::string timerName) { + if (timerName == "killself") { + + // Don't accidentally kill a pet that is already owned + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr || petComponent->GetOwner() != nullptr) + return; + + self->Smash(self->GetObjectID(), eKillType::SILENT); + } +} + +void CrabServer::OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) { + if (type == ePetTamingNotifyType::BEGIN) { + self->CancelTimer("killself"); + } else if (type == ePetTamingNotifyType::QUIT || type == ePetTamingNotifyType::FAILED) { + self->Smash(self->GetObjectID(), eKillType::SILENT); + } else if (type == ePetTamingNotifyType::SUCCESS) { + auto* petComponent = self->GetComponent<PetComponent>(); + if (petComponent == nullptr) + return; + // TODO: Remove custom group? + // Command the pet to the player as it may otherwise go to its spawn point which is non existant + // petComponent->Command(NiPoint3::ZERO, LWOOBJID_EMPTY, 6, 202, true); + } +} diff --git a/dScripts/client/ai/PR/CrabServer.h b/dScripts/client/ai/PR/CrabServer.h new file mode 100644 index 00000000..8c689dbd --- /dev/null +++ b/dScripts/client/ai/PR/CrabServer.h @@ -0,0 +1,10 @@ +#pragma once +#include "CppScripts.h" + +class CrabServer : public CppScripts::Script +{ +public: + void OnStartup(Entity* self) override; + void OnTimerDone(Entity* self, std::string timerName) override; + void OnNotifyPetTamingMinigame(Entity* self, Entity* tamer, ePetTamingNotifyType type) override; +}; diff --git a/dScripts/PrWhistle.cpp b/dScripts/client/ai/PR/PrWhistle.cpp similarity index 100% rename from dScripts/PrWhistle.cpp rename to dScripts/client/ai/PR/PrWhistle.cpp diff --git a/dScripts/PrWhistle.h b/dScripts/client/ai/PR/PrWhistle.h similarity index 100% rename from dScripts/PrWhistle.h rename to dScripts/client/ai/PR/PrWhistle.h diff --git a/dScripts/zone/AG/CMakeLists.txt b/dScripts/zone/AG/CMakeLists.txt new file mode 100644 index 00000000..14426a46 --- /dev/null +++ b/dScripts/zone/AG/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_ZONE_AG + "ZoneAgSurvival.cpp" + PARENT_SCOPE) diff --git a/dScripts/ZoneAgSurvival.cpp b/dScripts/zone/AG/ZoneAgSurvival.cpp similarity index 100% rename from dScripts/ZoneAgSurvival.cpp rename to dScripts/zone/AG/ZoneAgSurvival.cpp diff --git a/dScripts/ZoneAgSurvival.h b/dScripts/zone/AG/ZoneAgSurvival.h similarity index 100% rename from dScripts/ZoneAgSurvival.h rename to dScripts/zone/AG/ZoneAgSurvival.h diff --git a/dScripts/zone/CMakeLists.txt b/dScripts/zone/CMakeLists.txt new file mode 100644 index 00000000..5d800031 --- /dev/null +++ b/dScripts/zone/CMakeLists.txt @@ -0,0 +1,21 @@ +set(DSCRIPTS_SOURCES_ZONE) + +add_subdirectory(AG) + +foreach(file ${DSCRIPTS_SOURCES_ZONE_AG}) + set(DSCRIPTS_SOURCES_ZONE ${DSCRIPTS_SOURCES_ZONE} "AG/${file}") +endforeach() + +add_subdirectory(LUPs) + +foreach(file ${DSCRIPTS_SOURCES_ZONE_LUPS}) + set(DSCRIPTS_SOURCES_ZONE ${DSCRIPTS_SOURCES_ZONE} "LUPs/${file}") +endforeach() + +add_subdirectory(PROPERTY) + +foreach(file ${DSCRIPTS_SOURCES_ZONE_PROPERTY}) + set(DSCRIPTS_SOURCES_ZONE ${DSCRIPTS_SOURCES_ZONE} "PROPERTY/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_ZONE ${DSCRIPTS_SOURCES_ZONE} PARENT_SCOPE) diff --git a/dScripts/zone/LUPs/CMakeLists.txt b/dScripts/zone/LUPs/CMakeLists.txt new file mode 100644 index 00000000..b3b55ad6 --- /dev/null +++ b/dScripts/zone/LUPs/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_ZONE_LUPS + "WblGenericZone.cpp" + PARENT_SCOPE) diff --git a/dScripts/WblGenericZone.cpp b/dScripts/zone/LUPs/WblGenericZone.cpp similarity index 100% rename from dScripts/WblGenericZone.cpp rename to dScripts/zone/LUPs/WblGenericZone.cpp diff --git a/dScripts/WblGenericZone.h b/dScripts/zone/LUPs/WblGenericZone.h similarity index 100% rename from dScripts/WblGenericZone.h rename to dScripts/zone/LUPs/WblGenericZone.h diff --git a/dScripts/zone/PROPERTY/CMakeLists.txt b/dScripts/zone/PROPERTY/CMakeLists.txt new file mode 100644 index 00000000..b0588181 --- /dev/null +++ b/dScripts/zone/PROPERTY/CMakeLists.txt @@ -0,0 +1,21 @@ +set(DSCRIPTS_SOURCES_ZONE_PROPERTY) + +add_subdirectory(FV) + +foreach(file ${DSCRIPTS_SOURCES_ZONE_PROPERTY_FV}) + set(DSCRIPTS_SOURCES_ZONE_PROPERTY ${DSCRIPTS_SOURCES_ZONE_PROPERTY} "FV/${file}") +endforeach() + +add_subdirectory(GF) + +foreach(file ${DSCRIPTS_SOURCES_ZONE_PROPERTY_GF}) + set(DSCRIPTS_SOURCES_ZONE_PROPERTY ${DSCRIPTS_SOURCES_ZONE_PROPERTY} "GF/${file}") +endforeach() + +add_subdirectory(NS) + +foreach(file ${DSCRIPTS_SOURCES_ZONE_PROPERTY_NS}) + set(DSCRIPTS_SOURCES_ZONE_PROPERTY ${DSCRIPTS_SOURCES_ZONE_PROPERTY} "NS/${file}") +endforeach() + +set(DSCRIPTS_SOURCES_ZONE_PROPERTY ${DSCRIPTS_SOURCES_ZONE_PROPERTY} PARENT_SCOPE) diff --git a/dScripts/zone/PROPERTY/FV/CMakeLists.txt b/dScripts/zone/PROPERTY/FV/CMakeLists.txt new file mode 100644 index 00000000..60da048d --- /dev/null +++ b/dScripts/zone/PROPERTY/FV/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_ZONE_PROPERTY_FV + "ZoneFvProperty.cpp" + PARENT_SCOPE) diff --git a/dScripts/zone/PROPERTY/FV/ZoneFvProperty.cpp b/dScripts/zone/PROPERTY/FV/ZoneFvProperty.cpp new file mode 100644 index 00000000..8b8072ad --- /dev/null +++ b/dScripts/zone/PROPERTY/FV/ZoneFvProperty.cpp @@ -0,0 +1,40 @@ +#include "ZoneFvProperty.h" +#include "Entity.h" + +void ZoneFvProperty::SetGameVariables(Entity* self) { + self->SetVar<std::string>(ClaimMarkerGroup, "Platform"); + self->SetVar<std::string>(GeneratorGroup, "Generator"); + self->SetVar<std::string>(GuardGroup, "Guard"); + self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); + self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); + self->SetVar<std::string>(SpotsGroup, "Spots"); + self->SetVar<std::string>(MSCloudsGroup, "Clouds"); + self->SetVar<std::string>(EnemiesGroup, "Enemies"); + self->SetVar<std::string>(FXManagerGroup, "FXManager"); + self->SetVar<std::string>(ImagOrbGroup, "Orb"); + self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); + + self->SetVar<std::vector<std::string>>(EnemiesSpawner, + { "RoninWander", "RoninGen", "HorsemenGen" }); + self->SetVar<std::string>(ClaimMarkerSpawner, "Platform"); + self->SetVar<std::string>(GeneratorSpawner, "Generator"); + self->SetVar<std::string>(DamageFXSpawner, "Clouds"); + self->SetVar<std::string>(FXSpotsSpawner, "Spots"); + self->SetVar<std::string>(PropertyMGSpawner, "Guard"); + self->SetVar<std::string>(ImageOrbSpawner, "Orb"); + self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); + self->SetVar<std::string>(SmashablesSpawner, "Smashables"); + self->SetVar<std::string>(FXManagerSpawner, "FXManager"); + self->SetVar<std::string>(PropObjsSpawner, "BankObj"); + self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Ash", "FX", "Fog" }); + self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, {}); + + self->SetVar<int32_t>(defeatedProperyFlag, 99); + self->SetVar<int32_t>(placedModelFlag, 107); + self->SetVar<uint32_t>(guardMissionFlag, 874); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 950); + self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); + self->SetVar<LOT>(generatorIdFlag, 11023); + self->SetVar<LOT>(orbIDFlag, 10226); + self->SetVar<LOT>(behaviorQBID, 11011); +} diff --git a/dScripts/ZoneFvProperty.h b/dScripts/zone/PROPERTY/FV/ZoneFvProperty.h similarity index 100% rename from dScripts/ZoneFvProperty.h rename to dScripts/zone/PROPERTY/FV/ZoneFvProperty.h diff --git a/dScripts/zone/PROPERTY/GF/CMakeLists.txt b/dScripts/zone/PROPERTY/GF/CMakeLists.txt new file mode 100644 index 00000000..fed6033b --- /dev/null +++ b/dScripts/zone/PROPERTY/GF/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_ZONE_PROPERTY_GF + "ZoneGfProperty.cpp" + PARENT_SCOPE) diff --git a/dScripts/zone/PROPERTY/GF/ZoneGfProperty.cpp b/dScripts/zone/PROPERTY/GF/ZoneGfProperty.cpp new file mode 100644 index 00000000..87b2345d --- /dev/null +++ b/dScripts/zone/PROPERTY/GF/ZoneGfProperty.cpp @@ -0,0 +1,40 @@ +#include "ZoneGfProperty.h" +#include "Entity.h" + +void ZoneGfProperty::SetGameVariables(Entity* self) { + self->SetVar<std::string>(ClaimMarkerGroup, "BehavQB"); + self->SetVar<std::string>(GeneratorGroup, "Generator"); + self->SetVar<std::string>(GuardGroup, "Guard"); + self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); + self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); + self->SetVar<std::string>(SpotsGroup, "Spots"); + self->SetVar<std::string>(MSCloudsGroup, "Clouds"); + self->SetVar<std::string>(EnemiesGroup, "Enemies"); + self->SetVar<std::string>(FXManagerGroup, "FXManager"); + self->SetVar<std::string>(ImagOrbGroup, "Orb"); + self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); + + self->SetVar<std::vector<std::string>>(EnemiesSpawner, + { "PiratesWander", "PiratesGen", "AdmiralsWander", "AdmiralsGen" }); + self->SetVar<std::string>(ClaimMarkerSpawner, "BehavPlat"); + self->SetVar<std::string>(GeneratorSpawner, "Generator"); + self->SetVar<std::string>(DamageFXSpawner, "Clouds"); + self->SetVar<std::string>(FXSpotsSpawner, "Spots"); + self->SetVar<std::string>(PropertyMGSpawner, "Guard"); + self->SetVar<std::string>(ImageOrbSpawner, "Orb"); + self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); + self->SetVar<std::string>(SmashablesSpawner, "Smashables"); + self->SetVar<std::string>(FXManagerSpawner, "FXManager"); + self->SetVar<std::string>(PropObjsSpawner, "BankObj"); + self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Birds", "Falls", "Sunbeam" }); + self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, { "TrappedPlatform", "IceBarrier", "FireBeast" }); + + self->SetVar<int32_t>(defeatedProperyFlag, 98); + self->SetVar<int32_t>(placedModelFlag, 106); + self->SetVar<uint32_t>(guardMissionFlag, 873); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 949); + self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); + self->SetVar<LOT>(generatorIdFlag, 11109); + self->SetVar<LOT>(orbIDFlag, 10226); + self->SetVar<LOT>(behaviorQBID, 11001); +} diff --git a/dScripts/ZoneGfProperty.h b/dScripts/zone/PROPERTY/GF/ZoneGfProperty.h similarity index 100% rename from dScripts/ZoneGfProperty.h rename to dScripts/zone/PROPERTY/GF/ZoneGfProperty.h diff --git a/dScripts/zone/PROPERTY/NS/CMakeLists.txt b/dScripts/zone/PROPERTY/NS/CMakeLists.txt new file mode 100644 index 00000000..0e1f392c --- /dev/null +++ b/dScripts/zone/PROPERTY/NS/CMakeLists.txt @@ -0,0 +1,3 @@ +set(DSCRIPTS_SOURCES_ZONE_PROPERTY_NS + "ZoneNsProperty.cpp" + PARENT_SCOPE) diff --git a/dScripts/zone/PROPERTY/NS/ZoneNsProperty.cpp b/dScripts/zone/PROPERTY/NS/ZoneNsProperty.cpp new file mode 100644 index 00000000..52bb02e2 --- /dev/null +++ b/dScripts/zone/PROPERTY/NS/ZoneNsProperty.cpp @@ -0,0 +1,41 @@ +#include "ZoneNsProperty.h" +#include "Entity.h" + +void ZoneNsProperty::SetGameVariables(Entity* self) { + self->SetVar<std::string>(ClaimMarkerGroup, "Rhino"); + self->SetVar<std::string>(GeneratorGroup, "Generator"); + self->SetVar<std::string>(GuardGroup, "Guard"); + self->SetVar<std::string>(PropertyPlaqueGroup, "PropertyPlaque"); + self->SetVar<std::string>(PropertyVendorGroup, "PropertyVendor"); + self->SetVar<std::string>(SpotsGroup, "Spots"); + self->SetVar<std::string>(MSCloudsGroup, "Clouds"); + self->SetVar<std::string>(EnemiesGroup, "Enemies"); + self->SetVar<std::string>(FXManagerGroup, "FXManager"); + self->SetVar<std::string>(ImagOrbGroup, "Orb"); + self->SetVar<std::string>(GeneratorFXGroup, "GeneratorFX"); + + self->SetVar<std::vector<std::string>>(EnemiesSpawner, { + "StrombieWander", "StrombieGen", "PirateWander", "PirateGen", "RoninGen" + }); + self->SetVar<std::string>(ClaimMarkerSpawner, "ClaimMarker"); + self->SetVar<std::string>(GeneratorSpawner, "Generator"); + self->SetVar<std::string>(DamageFXSpawner, "MSClouds"); + self->SetVar<std::string>(FXSpotsSpawner, "Spots"); + self->SetVar<std::string>(PropertyMGSpawner, "Guard"); + self->SetVar<std::string>(ImageOrbSpawner, "Orb"); + self->SetVar<std::string>(GeneratorFXSpawner, "GeneratorFX"); + self->SetVar<std::string>(SmashablesSpawner, "Smashables"); + self->SetVar<std::string>(FXManagerSpawner, "FXManager"); + self->SetVar<std::string>(PropObjsSpawner, "BankObj"); + self->SetVar<std::vector<std::string>>(AmbientFXSpawner, { "Rockets" }); + self->SetVar<std::vector<std::string>>(BehaviorObjsSpawner, { "Cage", "Platform", "Door" }); + + self->SetVar<int32_t>(defeatedProperyFlag, 97); + self->SetVar<int32_t>(placedModelFlag, 105); + self->SetVar<uint32_t>(guardMissionFlag, 872); + self->SetVar<uint32_t>(brickLinkMissionIDFlag, 948); + self->SetVar<std::string>(passwordFlag, "s3kratK1ttN"); + self->SetVar<LOT>(generatorIdFlag, 11031); + self->SetVar<LOT>(orbIDFlag, 10226); + self->SetVar<LOT>(behaviorQBID, 11009); +} diff --git a/dScripts/ZoneNsProperty.h b/dScripts/zone/PROPERTY/NS/ZoneNsProperty.h similarity index 100% rename from dScripts/ZoneNsProperty.h rename to dScripts/zone/PROPERTY/NS/ZoneNsProperty.h diff --git a/dWorldServer/CMakeLists.txt b/dWorldServer/CMakeLists.txt index fcf29838..c616da87 100644 --- a/dWorldServer/CMakeLists.txt +++ b/dWorldServer/CMakeLists.txt @@ -1,6 +1,11 @@ -set(DWORLDSERVER_SOURCES "ObjectIDManager.cpp" - "PerformanceManager.cpp" - "WorldServer.cpp") +set(DWORLDSERVER_SOURCES + "ObjectIDManager.cpp" + "PerformanceManager.cpp" +) + +add_library(dWorldServer ${DWORLDSERVER_SOURCES}) +add_executable(WorldServer "WorldServer.cpp") + +target_link_libraries(dWorldServer ${COMMON_LIBRARIES}) +target_link_libraries(WorldServer ${COMMON_LIBRARIES} dChatFilter dGame dZoneManager dPhysics Detour Recast tinyxml2 dWorldServer dNavigation) -add_executable(WorldServer ${DWORLDSERVER_SOURCES}) -target_link_libraries(WorldServer ${COMMON_LIBRARIES} dChatFilter dGame dZoneManager Detour Recast dPhysics tinyxml2 dNavigation) diff --git a/dWorldServer/ObjectIDManager.cpp b/dWorldServer/ObjectIDManager.cpp index ef5fb9a5..9490c0a7 100644 --- a/dWorldServer/ObjectIDManager.cpp +++ b/dWorldServer/ObjectIDManager.cpp @@ -1,8 +1,5 @@ #include "ObjectIDManager.h" -// Std -#include <random> - // Custom Classes #include "MasterPackets.h" #include "Database.h" @@ -48,11 +45,7 @@ void ObjectIDManager::HandleRequestPersistentIDResponse(uint64_t requestID, uint //! Handles cases where we have to get a unique object ID synchronously uint32_t ObjectIDManager::GenerateRandomObjectID() { - std::random_device rd; - - std::mt19937 rng(rd()); - - return uni(rng); + return uni(Game::randomEngine); } diff --git a/dWorldServer/PerformanceManager.cpp b/dWorldServer/PerformanceManager.cpp index 85546f28..19f38d00 100644 --- a/dWorldServer/PerformanceManager.cpp +++ b/dWorldServer/PerformanceManager.cpp @@ -2,23 +2,18 @@ #include "UserManager.h" -//Times are 1 / fps, in ms -#define HIGH 16 //60 fps -#define MEDIUM 33 //30 fps -#define LOW 66 //15 fps - -#define SOCIAL { LOW } -#define SOCIAL_HUB { MEDIUM } //Added to compensate for the large playercounts in NS and NT -#define BATTLE { HIGH } -#define BATTLE_INSTANCE { MEDIUM } -#define RACE { HIGH } -#define PROPERTY { LOW } +#define SOCIAL { lowFrameDelta } +#define SOCIAL_HUB { mediumFrameDelta } //Added to compensate for the large playercounts in NS and NT +#define BATTLE { highFrameDelta } +#define BATTLE_INSTANCE { mediumFrameDelta } +#define RACE { highFrameDelta } +#define PROPERTY { lowFrameDelta } PerformanceProfile PerformanceManager::m_CurrentProfile = SOCIAL; PerformanceProfile PerformanceManager::m_DefaultProfile = SOCIAL; -PerformanceProfile PerformanceManager::m_InactiveProfile = { LOW }; +PerformanceProfile PerformanceManager::m_InactiveProfile = { lowFrameDelta }; std::map<LWOMAPID, PerformanceProfile> PerformanceManager::m_Profiles = { // VE @@ -72,13 +67,6 @@ std::map<LWOMAPID, PerformanceProfile> PerformanceManager::m_Profiles = { { 2001, BATTLE_INSTANCE }, }; - -PerformanceManager::PerformanceManager() { -} - -PerformanceManager::~PerformanceManager() { -} - void PerformanceManager::SelectProfile(LWOMAPID mapID) { const auto pair = m_Profiles.find(mapID); @@ -91,10 +79,10 @@ void PerformanceManager::SelectProfile(LWOMAPID mapID) { m_CurrentProfile = pair->second; } -uint32_t PerformanceManager::GetServerFramerate() { +uint32_t PerformanceManager::GetServerFrameDelta() { if (UserManager::Instance()->GetUserCount() == 0) { - return m_InactiveProfile.serverFramerate; + return m_InactiveProfile.serverFrameDelta; } - return m_CurrentProfile.serverFramerate; + return m_CurrentProfile.serverFrameDelta; } diff --git a/dWorldServer/PerformanceManager.h b/dWorldServer/PerformanceManager.h index b8a090e0..c584d4ac 100644 --- a/dWorldServer/PerformanceManager.h +++ b/dWorldServer/PerformanceManager.h @@ -5,21 +5,16 @@ #include "dCommonVars.h" struct PerformanceProfile { - uint32_t serverFramerate; + uint32_t serverFrameDelta; }; - class PerformanceManager { public: - ~PerformanceManager(); - static void SelectProfile(LWOMAPID mapID); - static uint32_t GetServerFramerate(); + static uint32_t GetServerFrameDelta(); private: - PerformanceManager(); - static PerformanceProfile m_CurrentProfile; static PerformanceProfile m_DefaultProfile; static PerformanceProfile m_InactiveProfile; diff --git a/dWorldServer/WorldServer.cpp b/dWorldServer/WorldServer.cpp index 495053fa..f07c17e3 100644 --- a/dWorldServer/WorldServer.cpp +++ b/dWorldServer/WorldServer.cpp @@ -17,6 +17,7 @@ #include "Metrics.hpp" #include "PerformanceManager.h" #include "Diagnostics.h" +#include "BinaryPathFinder.h" //RakNet includes: #include "RakNetDefines.h" @@ -30,7 +31,6 @@ #include "PacketUtils.h" #include "WorldPackets.h" #include "UserManager.h" -#include "dMessageIdentifiers.h" #include "CDClientManager.h" #include "CDClientDatabase.h" #include "GeneralUtils.h" @@ -41,13 +41,15 @@ #include "CharacterComponent.h" #include "EntityManager.h" +#include "EntityInfo.h" +#include "User.h" +#include "Loot.h" #include "Entity.h" #include "Character.h" #include "ChatPackets.h" #include "GameMessageHandler.h" #include "GameMessages.h" #include "Mail.h" -#include "dLocale.h" #include "TeamManager.h" #include "SkillComponent.h" #include "DestroyableComponent.h" @@ -55,33 +57,46 @@ #include "MasterPackets.h" #include "Player.h" #include "PropertyManagementComponent.h" - +#include "AssetManager.h" +#include "LevelProgressionComponent.h" +#include "eBlueprintSaveResponseType.h" +#include "Amf3.h" +#include "NiPoint3.h" +#include "eServerDisconnectIdentifiers.h" +#include "eObjectBits.h" +#include "eConnectionType.h" +#include "eServerMessageType.h" +#include "eChatInternalMessageType.h" +#include "eWorldMessageType.h" +#include "eMasterMessageType.h" +#include "eGameMessageType.h" #include "ZCompression.h" +#include "EntityManager.h" namespace Game { - dLogger* logger; - dServer* server; - dZoneManager* zoneManager; - dpWorld* physicsWorld; - dChatFilter* chatFilter; - dConfig* config; - dLocale* locale; + dLogger* logger = nullptr; + dServer* server = nullptr; + dpWorld* physicsWorld = nullptr; + dChatFilter* chatFilter = nullptr; + dConfig* config = nullptr; + AssetManager* assetManager = nullptr; + RakPeerInterface* chatServer = nullptr; std::mt19937 randomEngine; - - RakPeerInterface* chatServer; SystemAddress chatSysAddr; -} + bool shouldShutdown = false; + EntityManager* entityManager = nullptr; + dZoneManager* zoneManager = nullptr; +} // namespace Game bool chatDisabled = false; bool chatConnected = false; -bool worldShutdownSequenceStarted = false; bool worldShutdownSequenceComplete = false; void WorldShutdownSequence(); void WorldShutdownProcess(uint32_t zoneId); void FinalizeShutdown(); void SendShutdownMessageToMaster(); -dLogger* SetupLogger(int zoneID, int instanceID); +dLogger* SetupLogger(uint32_t zoneID, uint32_t instanceID); void HandlePacketChat(Packet* packet); void HandlePacket(Packet* packet); @@ -91,8 +106,8 @@ struct tempSessionInfo { }; std::map<std::string, tempSessionInfo> m_PendingUsers; -int instanceID = 0; -int g_CloneID = 0; +uint32_t instanceID = 0; +uint32_t g_CloneID = 0; std::string databaseChecksum = ""; int main(int argc, char** argv) { @@ -106,13 +121,13 @@ int main(int argc, char** argv) { signal(SIGINT, [](int) { WorldShutdownSequence(); }); signal(SIGTERM, [](int) { WorldShutdownSequence(); }); - int zoneID = 1000; - int cloneID = 0; - int maxClients = 8; - int ourPort = 2007; + uint32_t zoneID = 1000; + uint32_t cloneID = 0; + uint32_t maxClients = 8; + uint32_t ourPort = 2007; //Check our arguments: - for (int i = 0; i < argc; ++i) { + for (int32_t i = 0; i < argc; ++i) { std::string argument(argv[i]); if (argument == "-zone") zoneID = atoi(argv[i + 1]); @@ -124,58 +139,67 @@ int main(int argc, char** argv) { //Create all the objects we need to run our service: Game::logger = SetupLogger(zoneID, instanceID); - if (!Game::logger) return 0; + if (!Game::logger) return EXIT_FAILURE; + + //Read our config: + Game::config = new dConfig((BinaryPathFinder::GetBinaryDir() / "worldconfig.ini").string()); + Game::logger->SetLogToConsole(Game::config->GetValue("log_to_console") != "0"); + Game::logger->SetLogDebugStatements(Game::config->GetValue("log_debug_statements") == "1"); - Game::logger->SetLogToConsole(true); //We want this info to always be logged. Game::logger->Log("WorldServer", "Starting World server..."); Game::logger->Log("WorldServer", "Version: %i.%i", PROJECT_VERSION_MAJOR, PROJECT_VERSION_MINOR); Game::logger->Log("WorldServer", "Compiled on: %s", __TIMESTAMP__); -#ifndef _DEBUG - Game::logger->SetLogToConsole(false); //By default, turn it back off if not in debug. -#endif + if (Game::config->GetValue("disable_chat") == "1") chatDisabled = true; - //Read our config: - dConfig config("worldconfig.ini"); - Game::config = &config; - Game::logger->SetLogToConsole(bool(std::stoi(config.GetValue("log_to_console")))); - Game::logger->SetLogDebugStatements(config.GetValue("log_debug_statements") == "1"); - if (config.GetValue("disable_chat") == "1") chatDisabled = true; + try { + std::string clientPathStr = Game::config->GetValue("client_location"); + if (clientPathStr.empty()) clientPathStr = "./res"; + std::filesystem::path clientPath = std::filesystem::path(clientPathStr); + if (clientPath.is_relative()) { + clientPath = BinaryPathFinder::GetBinaryDir() / clientPath; + } + Game::assetManager = new AssetManager(clientPath); + } catch (std::runtime_error& ex) { + Game::logger->Log("WorldServer", "Got an error while setting up assets: %s", ex.what()); + + return EXIT_FAILURE; + } // Connect to CDClient try { - CDClientDatabase::Connect("./res/CDServer.sqlite"); + CDClientDatabase::Connect((BinaryPathFinder::GetBinaryDir() / "resServer" / "CDServer.sqlite").string()); } catch (CppSQLite3Exception& e) { Game::logger->Log("WorldServer", "Unable to connect to CDServer SQLite Database"); Game::logger->Log("WorldServer", "Error: %s", e.errorMessage()); Game::logger->Log("WorldServer", "Error Code: %i", e.errorCode()); - return -1; + return EXIT_FAILURE; } - CDClientManager::Instance()->Initialize(); + CDClientManager::Instance(); //Connect to the MySQL Database - std::string mysql_host = config.GetValue("mysql_host"); - std::string mysql_database = config.GetValue("mysql_database"); - std::string mysql_username = config.GetValue("mysql_username"); - std::string mysql_password = config.GetValue("mysql_password"); + std::string mysql_host = Game::config->GetValue("mysql_host"); + std::string mysql_database = Game::config->GetValue("mysql_database"); + std::string mysql_username = Game::config->GetValue("mysql_username"); + std::string mysql_password = Game::config->GetValue("mysql_password"); - Diagnostics::SetProduceMemoryDump(config.GetValue("generate_dump") == "1"); + Diagnostics::SetProduceMemoryDump(Game::config->GetValue("generate_dump") == "1"); - if (!config.GetValue("dump_folder").empty()) { - Diagnostics::SetOutDirectory(config.GetValue("dump_folder")); + if (!Game::config->GetValue("dump_folder").empty()) { + Diagnostics::SetOutDirectory(Game::config->GetValue("dump_folder")); } try { Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); } catch (sql::SQLException& ex) { Game::logger->Log("WorldServer", "Got an error while connecting to the database: %s", ex.what()); - return 0; + return EXIT_FAILURE; } //Find out the master's IP: std::string masterIP = "localhost"; - int masterPort = 1000; + uint32_t masterPort = 1000; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -189,13 +213,13 @@ int main(int argc, char** argv) { ObjectIDManager::Instance()->Initialize(); UserManager::Instance()->Initialize(); LootGenerator::Instance(); - Game::chatFilter = new dChatFilter("./res/chatplus_en_us", bool(std::stoi(config.GetValue("dont_generate_dcf")))); + Game::chatFilter = new dChatFilter(Game::assetManager->GetResPath().string() + "/chatplus_en_us", bool(std::stoi(Game::config->GetValue("dont_generate_dcf")))); - Game::server = new dServer(masterIP, ourPort, instanceID, maxClients, false, true, Game::logger, masterIP, masterPort, ServerType::World, zoneID); + Game::server = new dServer(masterIP, ourPort, instanceID, maxClients, false, true, Game::logger, masterIP, masterPort, ServerType::World, Game::config, &Game::shouldShutdown, zoneID); //Connect to the chat server: - int chatPort = 1501; - if (config.GetValue("chat_server_port") != "") chatPort = std::atoi(config.GetValue("chat_server_port").c_str()); + uint32_t chatPort = 1501; + if (Game::config->GetValue("chat_server_port") != "") chatPort = std::atoi(Game::config->GetValue("chat_server_port").c_str()); auto chatSock = SocketDescriptor(uint16_t(ourPort + 2), 0); Game::chatServer = RakNetworkFactory::GetRakPeerInterface(); @@ -204,38 +228,38 @@ int main(int argc, char** argv) { //Set up other things: Game::randomEngine = std::mt19937(time(0)); - Game::locale = new dLocale(); //Run it until server gets a kill message from Master: auto lastTime = std::chrono::high_resolution_clock::now(); auto t = std::chrono::high_resolution_clock::now(); Packet* packet = nullptr; - int framesSinceLastFlush = 0; - int framesSinceMasterDisconnect = 0; - int framesSinceChatDisconnect = 0; - int framesSinceLastUsersSave = 0; - int framesSinceLastSQLPing = 0; - int framesSinceLastUser = 0; + uint32_t framesSinceLastFlush = 0; + uint32_t framesSinceMasterDisconnect = 0; + uint32_t framesSinceChatDisconnect = 0; + uint32_t framesSinceLastUsersSave = 0; + uint32_t framesSinceLastSQLPing = 0; + uint32_t framesSinceLastUser = 0; const float maxPacketProcessingTime = 1.5f; //0.015f; - const int maxPacketsToProcess = 1024; + const uint32_t maxPacketsToProcess = 1024; bool ready = false; - int framesSinceMasterStatus = 0; - int framesSinceShutdownSequence = 0; - int currentFramerate = highFrameRate; + uint32_t framesSinceMasterStatus = 0; + uint32_t framesSinceShutdownSequence = 0; + uint32_t currentFramerate = highFramerate; - int ghostingStepCount = 0; + uint32_t ghostingStepCount = 0; auto ghostingLastTime = std::chrono::high_resolution_clock::now(); PerformanceManager::SelectProfile(zoneID); + Game::entityManager = new EntityManager(); + Game::zoneManager = new dZoneManager(); //Load our level: if (zoneID != 0) { dpWorld::Instance().Initialize(zoneID); - Game::physicsWorld = &dpWorld::Instance(); //just in case some old code references it - dZoneManager::Instance()->Initialize(LWOZONEID(zoneID, instanceID, cloneID)); + Game::zoneManager->Initialize(LWOZONEID(zoneID, instanceID, cloneID)); g_CloneID = cloneID; // pre calculate the FDB checksum @@ -243,20 +267,20 @@ int main(int argc, char** argv) { std::ifstream fileStream; static const std::vector<std::string> aliases = { - "res/CDServers.fdb", - "res/cdserver.fdb", - "res/CDClient.fdb", - "res/cdclient.fdb", + "CDServers.fdb", + "cdserver.fdb", + "CDClient.fdb", + "cdclient.fdb", }; for (const auto& file : aliases) { - fileStream.open(file, std::ios::binary | std::ios::in); + fileStream.open(Game::assetManager->GetResPath() / file, std::ios::binary | std::ios::in); if (fileStream.is_open()) { break; } } - const int bufferSize = 1024; + const int32_t bufferSize = 1024; MD5* md5 = new MD5(); char fileStreamBuffer[1024] = {}; @@ -278,8 +302,19 @@ int main(int argc, char** argv) { Game::logger->Log("WorldServer", "FDB Checksum calculated as: %s", databaseChecksum.c_str()); } + } else { + Game::entityManager->Initialize(); } + uint32_t currentFrameDelta = highFrameDelta; + // These values are adjust them selves to the current framerate should it update. + uint32_t logFlushTime = 15 * currentFramerate; // 15 seconds in frames + uint32_t shutdownTimeout = 10 * 60 * currentFramerate; // 10 minutes in frames + uint32_t noMasterConnectionTimeout = 5 * currentFramerate; // 5 seconds in frames + uint32_t chatReconnectionTime = 30 * currentFramerate; // 30 seconds in frames + uint32_t saveTime = 10 * 60 * currentFramerate; // 10 minutes in frames + uint32_t sqlPingTime = 10 * 60 * currentFramerate; // 10 minutes in frames + uint32_t emptyShutdownTime = (cloneID == 0 ? 30 : 5) * 60 * currentFramerate; // 30 minutes for main worlds, 5 for all others. while (true) { Metrics::StartMeasurement(MetricVariable::Frame); Metrics::StartMeasurement(MetricVariable::GameLoop); @@ -292,25 +327,47 @@ int main(int argc, char** argv) { const auto occupied = UserManager::Instance()->GetUserCount() != 0; + uint32_t newFrameDelta = currentFrameDelta; if (!ready) { - currentFramerate = highFrameRate; + newFrameDelta = highFrameDelta; } else { - currentFramerate = PerformanceManager::GetServerFramerate(); + newFrameDelta = PerformanceManager::GetServerFrameDelta(); + } + + // Update to the new framerate and scale all timings to said new framerate + if (newFrameDelta != currentFrameDelta) { + float_t ratioBeforeToAfter = (float)currentFrameDelta / (float)newFrameDelta; + currentFrameDelta = newFrameDelta; + currentFramerate = MS_TO_FRAMES(newFrameDelta); + Game::logger->LogDebug("WorldServer", "Framerate for zone/instance/clone %i/%i/%i is now %i", zoneID, instanceID, cloneID, currentFramerate); + logFlushTime = 15 * currentFramerate; // 15 seconds in frames + framesSinceLastFlush *= ratioBeforeToAfter; + shutdownTimeout = 10 * 60 * currentFramerate; // 10 minutes in frames + framesSinceLastUser *= ratioBeforeToAfter; + noMasterConnectionTimeout = 5 * currentFramerate; // 5 seconds in frames + framesSinceMasterDisconnect *= ratioBeforeToAfter; + chatReconnectionTime = 30 * currentFramerate; // 30 seconds in frames + framesSinceChatDisconnect *= ratioBeforeToAfter; + saveTime = 10 * 60 * currentFramerate; // 10 minutes in frames + framesSinceLastUsersSave *= ratioBeforeToAfter; + sqlPingTime = 10 * 60 * currentFramerate; // 10 minutes in frames + framesSinceLastSQLPing *= ratioBeforeToAfter; + emptyShutdownTime = (cloneID == 0 ? 30 : 5) * 60 * currentFramerate; // 30 minutes for main worlds, 5 for all others. + framesSinceLastUser *= ratioBeforeToAfter; } //Warning if we ran slow - if (deltaTime > currentFramerate) { - Game::logger->Log("WorldServer", "We're running behind, dT: %f > %f (framerate)", deltaTime, currentFramerate); + if (deltaTime > currentFrameDelta) { + Game::logger->Log("WorldServer", "We're running behind, dT: %f > %f (framerate %i)", deltaTime, currentFrameDelta, currentFramerate); } //Check if we're still connected to master: if (!Game::server->GetIsConnectedToMaster()) { framesSinceMasterDisconnect++; - int framesToWaitForMaster = ready ? 10 : 200; - if (framesSinceMasterDisconnect >= framesToWaitForMaster && !worldShutdownSequenceStarted) { - Game::logger->Log("WorldServer", "Game loop running but no connection to master for %d frames, shutting down", framesToWaitForMaster); - worldShutdownSequenceStarted = true; + if (framesSinceMasterDisconnect >= noMasterConnectionTimeout && !Game::shouldShutdown) { + Game::logger->Log("WorldServer", "Game loop running but no connection to master for %d frames, shutting down", noMasterConnectionTimeout); + Game::shouldShutdown = true; } } else framesSinceMasterDisconnect = 0; @@ -318,8 +375,7 @@ int main(int argc, char** argv) { if (!chatConnected) { framesSinceChatDisconnect++; - // Attempt to reconnect every 30 seconds. - if (framesSinceChatDisconnect >= 2000) { + if (framesSinceChatDisconnect >= chatReconnectionTime) { framesSinceChatDisconnect = 0; Game::chatServer->Connect(masterIP.c_str(), chatPort, "3.25 ND1", 8); @@ -334,18 +390,18 @@ int main(int argc, char** argv) { Metrics::EndMeasurement(MetricVariable::Physics); Metrics::StartMeasurement(MetricVariable::UpdateEntities); - EntityManager::Instance()->UpdateEntities(deltaTime); + Game::entityManager->UpdateEntities(deltaTime); Metrics::EndMeasurement(MetricVariable::UpdateEntities); Metrics::StartMeasurement(MetricVariable::Ghosting); if (std::chrono::duration<float>(currentTime - ghostingLastTime).count() >= 1.0f) { - EntityManager::Instance()->UpdateGhosting(); + Game::entityManager->UpdateGhosting(); ghostingLastTime = currentTime; } Metrics::EndMeasurement(MetricVariable::Ghosting); Metrics::StartMeasurement(MetricVariable::UpdateSpawners); - dZoneManager::Instance()->Update(deltaTime); + Game::zoneManager->Update(deltaTime); Metrics::EndMeasurement(MetricVariable::UpdateSpawners); } @@ -371,14 +427,14 @@ int main(int argc, char** argv) { UserManager::Instance()->DeletePendingRemovals(); auto t1 = std::chrono::high_resolution_clock::now(); - for (int curPacket = 0; curPacket < maxPacketsToProcess && timeSpent < maxPacketProcessingTime; curPacket++) { + for (uint32_t curPacket = 0; curPacket < maxPacketsToProcess && timeSpent < maxPacketProcessingTime; curPacket++) { packet = Game::server->Receive(); if (packet) { auto t1 = std::chrono::high_resolution_clock::now(); HandlePacket(packet); auto t2 = std::chrono::high_resolution_clock::now(); - timeSpent += std::chrono::duration_cast<std::chrono::milliseconds>(t2 - t1).count(); + timeSpent += std::chrono::duration_cast<std::chrono::duration<float>>(t2 - t1).count(); Game::server->DeallocatePacket(packet); packet = nullptr; } else { @@ -396,7 +452,7 @@ int main(int argc, char** argv) { Metrics::EndMeasurement(MetricVariable::UpdateReplica); //Push our log every 15s: - if (framesSinceLastFlush >= 1000) { + if (framesSinceLastFlush >= logFlushTime) { Game::logger->Flush(); framesSinceLastFlush = 0; } else framesSinceLastFlush++; @@ -405,15 +461,15 @@ int main(int argc, char** argv) { framesSinceLastUser++; //If we haven't had any players for a while, time out and shut down: - if (framesSinceLastUser == (cloneID != 0 ? 4000 : 40000)) { - worldShutdownSequenceStarted = true; + if (framesSinceLastUser >= emptyShutdownTime) { + Game::shouldShutdown = true; } } else { framesSinceLastUser = 0; } //Save all connected users every 10 minutes: - if (framesSinceLastUsersSave >= 40000 && zoneID != 0) { + if (framesSinceLastUsersSave >= saveTime && zoneID != 0) { UserManager::Instance()->SaveAllActiveCharacters(); framesSinceLastUsersSave = 0; @@ -423,10 +479,10 @@ int main(int argc, char** argv) { } else framesSinceLastUsersSave++; //Every 10 min we ping our sql server to keep it alive hopefully: - if (framesSinceLastSQLPing >= 40000) { + if (framesSinceLastSQLPing >= sqlPingTime) { //Find out the master's IP for absolutely no reason: std::string masterIP; - int masterPort; + uint32_t masterPort; sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); auto res = stmt->executeQuery(); while (res->next()) { @@ -444,7 +500,7 @@ int main(int argc, char** argv) { Metrics::StartMeasurement(MetricVariable::Sleep); - t += std::chrono::milliseconds(currentFramerate); + t += std::chrono::milliseconds(currentFrameDelta); std::this_thread::sleep_until(t); Metrics::EndMeasurement(MetricVariable::Sleep); @@ -463,7 +519,7 @@ int main(int argc, char** argv) { } } - if (worldShutdownSequenceStarted && !worldShutdownSequenceComplete) { + if (Game::shouldShutdown && !worldShutdownSequenceComplete) { WorldShutdownProcess(zoneID); break; } @@ -475,8 +531,8 @@ int main(int argc, char** argv) { return EXIT_SUCCESS; } -dLogger* SetupLogger(int zoneID, int instanceID) { - std::string logPath = "./logs/WorldServer_" + std::to_string(zoneID) + "_" + std::to_string(instanceID) + "_" + std::to_string(time(nullptr)) + ".log"; +dLogger* SetupLogger(uint32_t zoneID, uint32_t instanceID) { + std::string logPath = (BinaryPathFinder::GetBinaryDir() / ("logs/WorldServer_" + std::to_string(zoneID) + "_" + std::to_string(instanceID) + "_" + std::to_string(time(nullptr)) + ".log")).string(); bool logToConsole = false; bool logDebugStatements = false; #ifdef _DEBUG @@ -502,22 +558,21 @@ void HandlePacketChat(Packet* packet) { } if (packet->data[0] == ID_USER_PACKET_ENUM) { - if (packet->data[1] == CHAT_INTERNAL) { - switch (packet->data[3]) { - case MSG_CHAT_INTERNAL_ROUTE_TO_PLAYER: { - CINSTREAM; + if (static_cast<eConnectionType>(packet->data[1]) == eConnectionType::CHAT_INTERNAL) { + switch (static_cast<eChatInternalMessageType>(packet->data[3])) { + case eChatInternalMessageType::ROUTE_TO_PLAYER: { + CINSTREAM_SKIP_HEADER; LWOOBJID playerID; inStream.Read(playerID); - inStream.Read(playerID); - auto player = EntityManager::Instance()->GetEntity(playerID); + auto player = Game::entityManager->GetEntity(playerID); if (!player) return; auto sysAddr = player->GetSystemAddress(); //Write our stream outwards: CBITSTREAM; - for (int i = 0; i < inStream.GetNumberOfBytesUsed(); i++) { + for (BitSize_t i = 0; i < inStream.GetNumberOfBytesUsed(); i++) { bitStream.Write(packet->data[i + 16]); //16 bytes == header + playerID to skip } @@ -526,17 +581,15 @@ void HandlePacketChat(Packet* packet) { break; } - case MSG_CHAT_INTERNAL_ANNOUNCEMENT: { - CINSTREAM; - LWOOBJID header; - inStream.Read(header); + case eChatInternalMessageType::ANNOUNCEMENT: { + CINSTREAM_SKIP_HEADER; std::string title; std::string msg; uint32_t len; inStream.Read<uint32_t>(len); - for (int i = 0; len > i; i++) { + for (uint32_t i = 0; len > i; i++) { char character; inStream.Read<char>(character); title += character; @@ -544,7 +597,7 @@ void HandlePacketChat(Packet* packet) { len = 0; inStream.Read<uint32_t>(len); - for (int i = 0; len > i; i++) { + for (uint32_t i = 0; len > i; i++) { char character; inStream.Read<char>(character); msg += character; @@ -552,28 +605,23 @@ void HandlePacketChat(Packet* packet) { //Send to our clients: AMFArrayValue args; - auto* titleValue = new AMFStringValue(); - titleValue->SetStringValue(title.c_str()); - auto* messageValue = new AMFStringValue(); - messageValue->SetStringValue(msg.c_str()); - args.InsertValue("title", titleValue); - args.InsertValue("message", messageValue); + args.Insert("title", title); + args.Insert("message", msg); - GameMessages::SendUIMessageServerToAllClients("ToggleAnnounce", &args); + GameMessages::SendUIMessageServerToAllClients("ToggleAnnounce", args); break; } - case MSG_CHAT_INTERNAL_MUTE_UPDATE: { - CINSTREAM; + case eChatInternalMessageType::MUTE_UPDATE: { + CINSTREAM_SKIP_HEADER; LWOOBJID playerId; time_t expire = 0; inStream.Read(playerId); - inStream.Read(playerId); inStream.Read(expire); - auto* entity = EntityManager::Instance()->GetEntity(playerId); + auto* entity = Game::entityManager->GetEntity(playerId); if (entity != nullptr) { entity->GetParentUser()->SetMuteExpire(expire); @@ -584,10 +632,8 @@ void HandlePacketChat(Packet* packet) { break; } - case MSG_CHAT_INTERNAL_TEAM_UPDATE: { - CINSTREAM; - LWOOBJID header; - inStream.Read(header); + case eChatInternalMessageType::TEAM_UPDATE: { + CINSTREAM_SKIP_HEADER; LWOOBJID teamID = 0; char lootOption = 0; @@ -639,7 +685,7 @@ void HandlePacket(Packet* packet) { return; } - auto* entity = EntityManager::Instance()->GetEntity(c->GetObjectID()); + auto* entity = Game::entityManager->GetEntity(c->GetObjectID()); if (!entity) { entity = Player::GetPlayer(packet->systemAddress); @@ -656,12 +702,12 @@ void HandlePacket(Packet* packet) { Game::logger->Log("WorldServer", "Deleting player %llu", entity->GetObjectID()); - EntityManager::Instance()->DestroyEntity(entity); + Game::entityManager->DestroyEntity(entity); } { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_PLAYER_REMOVED_NOTIFICATION); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::PLAYER_REMOVED_NOTIFICATION); bitStream.Write(user->GetLoggedInChar()); Game::chatServer->Send(&bitStream, SYSTEM_PRIORITY, RELIABLE, 0, Game::chatSysAddr, false); } @@ -673,35 +719,35 @@ void HandlePacket(Packet* packet) { } CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_PLAYER_REMOVED); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::PLAYER_REMOVED); bitStream.Write((LWOMAPID)Game::server->GetZoneID()); bitStream.Write((LWOINSTANCEID)instanceID); Game::server->SendToMaster(&bitStream); } - if (packet->data[0] != ID_USER_PACKET_ENUM) return; - if (packet->data[1] == SERVER) { - if (packet->data[3] == MSG_SERVER_VERSION_CONFIRM) { + if (packet->data[0] != ID_USER_PACKET_ENUM || packet->length < 4) return; + if (static_cast<eConnectionType>(packet->data[1]) == eConnectionType::SERVER) { + if (static_cast<eServerMessageType>(packet->data[3]) == eServerMessageType::VERSION_CONFIRM) { AuthPackets::HandleHandshake(Game::server, packet); } } - if (packet->data[1] == MASTER) { - switch (packet->data[3]) { - case MSG_MASTER_REQUEST_PERSISTENT_ID_RESPONSE: { + if (static_cast<eConnectionType>(packet->data[1]) == eConnectionType::MASTER) { + switch (static_cast<eMasterMessageType>(packet->data[3])) { + case eMasterMessageType::REQUEST_PERSISTENT_ID_RESPONSE: { uint64_t requestID = PacketUtils::ReadPacketU64(8, packet); uint32_t objectID = PacketUtils::ReadPacketU32(16, packet); ObjectIDManager::Instance()->HandleRequestPersistentIDResponse(requestID, objectID); break; } - case MSG_MASTER_REQUEST_ZONE_TRANSFER_RESPONSE: { + case eMasterMessageType::REQUEST_ZONE_TRANSFER_RESPONSE: { uint64_t requestID = PacketUtils::ReadPacketU64(8, packet); ZoneInstanceManager::Instance()->HandleRequestZoneTransferResponse(requestID, packet); break; } - case MSG_MASTER_SESSION_KEY_RESPONSE: { + case eMasterMessageType::SESSION_KEY_RESPONSE: { //Read our session key and to which user it belongs: RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); @@ -722,7 +768,7 @@ void HandlePacket(Packet* packet) { //Verify it: if (userHash != it->second.hash) { Game::logger->Log("WorldServer", "SOMEONE IS TRYING TO HACK? SESSION KEY MISMATCH: ours: %s != master: %s", userHash.c_str(), it->second.hash.c_str()); - Game::server->Disconnect(it->second.sysAddr, SERVER_DISCON_INVALID_SESSION_KEY); + Game::server->Disconnect(it->second.sysAddr, eServerDisconnectIdentifiers::INVALID_SESSION_KEY); return; } else { Game::logger->Log("WorldServer", "User %s authenticated with correct key.", username.c_str()); @@ -732,7 +778,7 @@ void HandlePacket(Packet* packet) { //Create our user and send them in: UserManager::Instance()->CreateUser(it->second.sysAddr, username, userHash); - auto zone = dZoneManager::Instance()->GetZone(); + auto zone = Game::zoneManager->GetZone(); if (zone) { float x = 0.0f; float y = 0.0f; @@ -758,7 +804,7 @@ void HandlePacket(Packet* packet) { //Notify master: { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_PLAYER_ADDED); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::PLAYER_ADDED); bitStream.Write((LWOMAPID)Game::server->GetZoneID()); bitStream.Write((LWOINSTANCEID)instanceID); Game::server->SendToMaster(&bitStream); @@ -767,27 +813,27 @@ void HandlePacket(Packet* packet) { break; } - case MSG_MASTER_AFFIRM_TRANSFER_REQUEST: { + case eMasterMessageType::AFFIRM_TRANSFER_REQUEST: { const uint64_t requestID = PacketUtils::ReadPacketU64(8, packet); Game::logger->Log("MasterServer", "Got affirmation request of transfer %llu", requestID); CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_AFFIRM_TRANSFER_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::AFFIRM_TRANSFER_RESPONSE); bitStream.Write(requestID); Game::server->SendToMaster(&bitStream); break; } - case MSG_MASTER_SHUTDOWN: { - worldShutdownSequenceStarted = true; + case eMasterMessageType::SHUTDOWN: { + Game::shouldShutdown = true; Game::logger->Log("WorldServer", "Got shutdown request from master, zone (%i), instance (%i)", Game::server->GetZoneID(), Game::server->GetInstanceID()); break; } - case MSG_MASTER_NEW_SESSION_ALERT: { + case eMasterMessageType::NEW_SESSION_ALERT: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); uint32_t sessionKey = inStream.Read(sessionKey); @@ -797,7 +843,7 @@ void HandlePacket(Packet* packet) { uint32_t len; inStream.Read(len); - for (int i = 0; i < len; i++) { + for (uint32_t i = 0; i < len; i++) { char character; inStream.Read<char>(character); username += character; } @@ -812,7 +858,7 @@ void HandlePacket(Packet* packet) { //Check the key: if (sessionKey != std::atoi(user->GetSessionKey().c_str())) { Game::logger->Log("WorldServer", "Got new session alert for user %s, but the session key is invalid.", username.c_str()); - Game::server->Disconnect(user->GetSystemAddress(), SERVER_DISCON_INVALID_SESSION_KEY); + Game::server->Disconnect(user->GetSystemAddress(), eServerDisconnectIdentifiers::INVALID_SESSION_KEY); return; } break; @@ -825,10 +871,10 @@ void HandlePacket(Packet* packet) { return; } - if (packet->data[1] != WORLD) return; + if (static_cast<eConnectionType>(packet->data[1]) != eConnectionType::WORLD) return; - switch (packet->data[3]) { - case MSG_WORLD_CLIENT_VALIDATION: { + switch (static_cast<eWorldMessageType>(packet->data[3])) { + case eWorldMessageType::VALIDATION: { std::string username = PacketUtils::ReadString(0x08, packet, true); std::string sessionKey = PacketUtils::ReadString(74, packet, true); std::string clientDatabaseChecksum = PacketUtils::ReadString(packet->length - 33, packet, false); @@ -853,14 +899,14 @@ void HandlePacket(Packet* packet) { // Developers may skip this check if (gmLevel < 8 && clientDatabaseChecksum != databaseChecksum) { Game::logger->Log("WorldServer", "Client's database checksum does not match the server's, aborting connection."); - Game::server->Disconnect(packet->systemAddress, SERVER_DISCON_KICK); + Game::server->Disconnect(packet->systemAddress, eServerDisconnectIdentifiers::WRONG_GAME_VERSION); return; } } //Request the session info from Master: CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_REQUEST_SESSION_KEY); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::REQUEST_SESSION_KEY); PacketUtils::WriteString(bitStream, username, 64); Game::server->SendToMaster(&bitStream); @@ -873,12 +919,12 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_CHARACTER_LIST_REQUEST: { + case eWorldMessageType::CHARACTER_LIST_REQUEST: { //We need to delete the entity first, otherwise the char list could delete it while it exists in the world! if (Game::server->GetZoneID() != 0) { auto user = UserManager::Instance()->GetUser(packet->systemAddress); if (!user) return; - EntityManager::Instance()->DestroyEntity(user->GetLastUsedChar()->GetEntity()); + Game::entityManager->DestroyEntity(user->GetLastUsedChar()->GetEntity()); } //This loops prevents users who aren't authenticated to double-request the char list, which @@ -895,12 +941,12 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_GAME_MSG: { + case eWorldMessageType::GAME_MSG: { RakNet::BitStream bitStream(packet->data, packet->length, false); uint64_t header; LWOOBJID objectID; - uint16_t messageID; + eGameMessageType messageID; bitStream.Read(header); bitStream.Read(objectID); @@ -909,23 +955,23 @@ void HandlePacket(Packet* packet) { RakNet::BitStream dataStream; bitStream.Read(dataStream, bitStream.GetNumberOfUnreadBits()); - GameMessageHandler::HandleMessage(&dataStream, packet->systemAddress, objectID, GAME_MSG(messageID)); + GameMessageHandler::HandleMessage(&dataStream, packet->systemAddress, objectID, messageID); break; } - case MSG_WORLD_CLIENT_CHARACTER_CREATE_REQUEST: { + case eWorldMessageType::CHARACTER_CREATE_REQUEST: { UserManager::Instance()->CreateCharacter(packet->systemAddress, packet); break; } - case MSG_WORLD_CLIENT_LOGIN_REQUEST: { + case eWorldMessageType::LOGIN_REQUEST: { RakNet::BitStream inStream(packet->data, packet->length, false); uint64_t header = inStream.Read(header); LWOOBJID playerID = 0; inStream.Read(playerID); - playerID = GeneralUtils::ClearBit(playerID, OBJECT_BIT_CHARACTER); - playerID = GeneralUtils::ClearBit(playerID, OBJECT_BIT_PERSISTENT); + GeneralUtils::ClearBit(playerID, eObjectBits::CHARACTER); + GeneralUtils::ClearBit(playerID, eObjectBits::PERSISTENT); auto user = UserManager::Instance()->GetUser(packet->systemAddress); @@ -934,7 +980,7 @@ void HandlePacket(Packet* packet) { // This means we swapped characters and we need to remove the previous player from the container. if (static_cast<uint32_t>(lastCharacter) != playerID) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_PLAYER_REMOVED_NOTIFICATION); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::PLAYER_REMOVED_NOTIFICATION); bitStream.Write(lastCharacter); Game::chatServer->Send(&bitStream, SYSTEM_PRIORITY, RELIABLE, 0, Game::chatSysAddr, false); } @@ -944,18 +990,18 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_CHARACTER_DELETE_REQUEST: { + case eWorldMessageType::CHARACTER_DELETE_REQUEST: { UserManager::Instance()->DeleteCharacter(packet->systemAddress, packet); UserManager::Instance()->RequestCharacterList(packet->systemAddress); break; } - case MSG_WORLD_CLIENT_CHARACTER_RENAME_REQUEST: { + case eWorldMessageType::CHARACTER_RENAME_REQUEST: { UserManager::Instance()->RenameCharacter(packet->systemAddress, packet); break; } - case MSG_WORLD_CLIENT_LEVEL_LOAD_COMPLETE: { + case eWorldMessageType::LEVEL_LOAD_COMPLETE: { Game::logger->Log("WorldServer", "Received level load complete from user."); User* user = UserManager::Instance()->GetUser(packet->systemAddress); if (user) { @@ -966,29 +1012,49 @@ void HandlePacket(Packet* packet) { EntityInfo info{}; info.lot = 1; - Entity* player = EntityManager::Instance()->CreateEntity(info, UserManager::Instance()->GetUser(packet->systemAddress)); + Entity* player = Game::entityManager->CreateEntity(info, UserManager::Instance()->GetUser(packet->systemAddress)); WorldPackets::SendCreateCharacter(packet->systemAddress, player, c->GetXMLData(), username, c->GetGMLevel()); WorldPackets::SendServerState(packet->systemAddress); - const auto respawnPoint = player->GetCharacter()->GetRespawnPoint(dZoneManager::Instance()->GetZone()->GetWorldID()); + const auto respawnPoint = player->GetCharacter()->GetRespawnPoint(Game::zoneManager->GetZone()->GetWorldID()); - EntityManager::Instance()->ConstructEntity(player, UNASSIGNED_SYSTEM_ADDRESS, true); + Game::entityManager->ConstructEntity(player, UNASSIGNED_SYSTEM_ADDRESS, true); if (respawnPoint != NiPoint3::ZERO) { GameMessages::SendPlayerReachedRespawnCheckpoint(player, respawnPoint, NiQuaternion::IDENTITY); } - EntityManager::Instance()->ConstructAllEntities(packet->systemAddress); + Game::entityManager->ConstructAllEntities(packet->systemAddress); auto* characterComponent = player->GetComponent<CharacterComponent>(); if (characterComponent) { player->GetComponent<CharacterComponent>()->RocketUnEquip(player); } - c->SetRetroactiveFlags(); + // Do charxml fixes here + auto* levelComponent = player->GetComponent<LevelProgressionComponent>(); + if (!levelComponent) return; - player->RetroactiveVaultSize(); + auto version = levelComponent->GetCharacterVersion(); + switch(version) { + case eCharacterVersion::RELEASE: + // TODO: Implement, super low priority + case eCharacterVersion::LIVE: + Game::logger->Log("WorldServer", "Updating Character Flags"); + c->SetRetroactiveFlags(); + levelComponent->SetCharacterVersion(eCharacterVersion::PLAYER_FACTION_FLAGS); + case eCharacterVersion::PLAYER_FACTION_FLAGS: + Game::logger->Log("WorldServer", "Updating Vault Size"); + player->RetroactiveVaultSize(); + levelComponent->SetCharacterVersion(eCharacterVersion::VAULT_SIZE); + case eCharacterVersion::VAULT_SIZE: + Game::logger->Log("WorldServer", "Updaing Speedbase"); + levelComponent->SetRetroactiveBaseSpeed(); + levelComponent->SetCharacterVersion(eCharacterVersion::UP_TO_DATE); + case eCharacterVersion::UP_TO_DATE: + break; + } player->GetCharacter()->SetTargetScene(""); @@ -1001,7 +1067,7 @@ void HandlePacket(Packet* packet) { //Tell the player to generate BBB models, if any: if (g_CloneID != 0) { - const auto& worldId = dZoneManager::Instance()->GetZone()->GetZoneID(); + const auto& worldId = Game::zoneManager->GetZone()->GetZoneID(); const auto zoneId = Game::server->GetZoneID(); const auto cloneId = g_CloneID; @@ -1020,7 +1086,7 @@ void HandlePacket(Packet* packet) { //Check for BBB models: auto stmt = Database::CreatePreppedStmt("SELECT ugc_id FROM properties_contents WHERE lot=14 AND property_id=?"); - int templateId = result.getIntField(0); + int32_t templateId = result.getIntField(0); result.finalize(); @@ -1059,25 +1125,16 @@ void HandlePacket(Packet* packet) { //Send message: { LWOOBJID blueprintID = res->getUInt(1); - blueprintID = GeneralUtils::SetBit(blueprintID, OBJECT_BIT_CHARACTER); - blueprintID = GeneralUtils::SetBit(blueprintID, OBJECT_BIT_PERSISTENT); + GeneralUtils::SetBit(blueprintID, eObjectBits::CHARACTER); + GeneralUtils::SetBit(blueprintID, eObjectBits::PERSISTENT); CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CLIENT, MSG_CLIENT_BLUEPRINT_SAVE_RESPONSE); - bitStream.Write<LWOOBJID>(0); //always zero so that a check on the client passes - bitStream.Write<unsigned int>(0); - bitStream.Write<unsigned int>(1); + PacketUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::BLUEPRINT_SAVE_RESPONSE); + bitStream.Write<LWOOBJID>(LWOOBJID_EMPTY); //always zero so that a check on the client passes + bitStream.Write(eBlueprintSaveResponseType::EverythingWorked); + bitStream.Write<uint32_t>(1); bitStream.Write(blueprintID); - bitStream.Write<uint32_t>(lxfmlSize + 9); - - //Write a fake sd0 header: - bitStream.Write<unsigned char>(0x73); //s - bitStream.Write<unsigned char>(0x64); //d - bitStream.Write<unsigned char>(0x30); //0 - bitStream.Write<unsigned char>(0x01); //1 - bitStream.Write<unsigned char>(0xFF); //end magic - bitStream.Write<uint32_t>(lxfmlSize); for (size_t i = 0; i < lxfmlSize; ++i) @@ -1113,14 +1170,14 @@ void HandlePacket(Packet* packet) { //RakNet::RakString playerName(player->GetCharacter()->GetName().c_str()); CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT_INTERNAL, MSG_CHAT_INTERNAL_PLAYER_ADDED_NOTIFICATION); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT_INTERNAL, eChatInternalMessageType::PLAYER_ADDED_NOTIFICATION); bitStream.Write(player->GetObjectID()); bitStream.Write<uint32_t>(playerName.size()); for (size_t i = 0; i < playerName.size(); i++) { bitStream.Write(playerName[i]); } - auto zone = dZoneManager::Instance()->GetZone()->GetZoneID(); + auto zone = Game::zoneManager->GetZone()->GetZoneID(); bitStream.Write(zone.GetMapID()); bitStream.Write(zone.GetInstanceID()); bitStream.Write(zone.GetCloneID()); @@ -1130,7 +1187,7 @@ void HandlePacket(Packet* packet) { } } else { Game::logger->Log("WorldServer", "Couldn't find character to log in with for user %s (%i)!", user->GetUsername().c_str(), user->GetAccountID()); - Game::server->Disconnect(packet->systemAddress, SERVER_DISCON_CHARACTER_NOT_FOUND); + Game::server->Disconnect(packet->systemAddress, eServerDisconnectIdentifiers::CHARACTER_NOT_FOUND); } } else { Game::logger->Log("WorldServer", "Couldn't get user for level load complete!"); @@ -1138,12 +1195,12 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_POSITION_UPDATE: { + case eWorldMessageType::POSITION_UPDATE: { ClientPackets::HandleClientPositionUpdate(packet->systemAddress, packet); break; } - case MSG_WORLD_CLIENT_MAIL: { + case eWorldMessageType::MAIL: { RakNet::BitStream bitStream(packet->data, packet->length, false); LWOOBJID space; bitStream.Read(space); @@ -1151,12 +1208,10 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_ROUTE_PACKET: { + case eWorldMessageType::ROUTE_PACKET: { //Yeet to chat - CINSTREAM; - uint64_t header = 0; + CINSTREAM_SKIP_HEADER; uint32_t size = 0; - inStream.Read(header); inStream.Read(size); if (size > 20000) { @@ -1166,7 +1221,7 @@ void HandlePacket(Packet* packet) { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, CHAT, packet->data[14]); + PacketUtils::WriteHeader(bitStream, eConnectionType::CHAT, packet->data[14]); //We need to insert the player's objectID so the chat server can find who originated this request: LWOOBJID objectID = 0; @@ -1187,12 +1242,12 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_STRING_CHECK: { + case eWorldMessageType::STRING_CHECK: { ClientPackets::HandleChatModerationRequest(packet->systemAddress, packet); break; } - case MSG_WORLD_CLIENT_GENERAL_CHAT_MESSAGE: { + case eWorldMessageType::GENERAL_CHAT_MESSAGE: { if (chatDisabled) { ChatPackets::SendMessageFail(packet->systemAddress); } else { @@ -1202,7 +1257,7 @@ void HandlePacket(Packet* packet) { break; } - case MSG_WORLD_CLIENT_HANDLE_FUNNESS: { + case eWorldMessageType::HANDLE_FUNNESS: { //This means the client is running slower or faster than it should. //Could be insane lag, but I'mma just YEET them as it's usually speedhacking. //This is updated to now count the amount of times we've been caught "speedhacking" to kick with a delay @@ -1215,7 +1270,7 @@ void HandlePacket(Packet* packet) { if (user) { user->UserOutOfSync(); } else { - Game::server->Disconnect(packet->systemAddress, SERVER_DISCON_KICK); + Game::server->Disconnect(packet->systemAddress, eServerDisconnectIdentifiers::KICK); } break; } @@ -1255,17 +1310,17 @@ void WorldShutdownProcess(uint32_t zoneId) { while (Game::server->GetReplicaManager()->GetParticipantCount() > 0) { const auto& player = Game::server->GetReplicaManager()->GetParticipantAtIndex(0); - Game::server->Disconnect(player, SERVER_DISCON_KICK); + Game::server->Disconnect(player, eServerDisconnectIdentifiers::SERVER_SHUTDOWN); } SendShutdownMessageToMaster(); } void WorldShutdownSequence() { - if (worldShutdownSequenceStarted || worldShutdownSequenceComplete) { + if (Game::shouldShutdown || worldShutdownSequenceComplete) { return; } - worldShutdownSequenceStarted = true; + Game::shouldShutdown = true; Game::logger->Log("WorldServer", "Zone (%i) instance (%i) shutting down outside of main loop!", Game::server->GetZoneID(), instanceID); WorldShutdownProcess(Game::server->GetZoneID()); @@ -1273,17 +1328,17 @@ void WorldShutdownSequence() { } void FinalizeShutdown() { - //Delete our objects here: - if (Game::physicsWorld) Game::physicsWorld = nullptr; - if (Game::zoneManager) delete Game::zoneManager; - Game::logger->Log("WorldServer", "Shutdown complete, zone (%i), instance (%i)", Game::server->GetZoneID(), instanceID); + //Delete our objects here: Metrics::Clear(); Database::Destroy("WorldServer"); - delete Game::chatFilter; - delete Game::server; - delete Game::logger; + if (Game::chatFilter) delete Game::chatFilter; + if (Game::zoneManager) delete Game::zoneManager; + if (Game::server) delete Game::server; + if (Game::config) delete Game::config; + if (Game::entityManager) delete Game::entityManager; + if (Game::logger) delete Game::logger; worldShutdownSequenceComplete = true; @@ -1292,6 +1347,6 @@ void FinalizeShutdown() { void SendShutdownMessageToMaster() { CBITSTREAM; - PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SHUTDOWN_RESPONSE); + PacketUtils::WriteHeader(bitStream, eConnectionType::MASTER, eMasterMessageType::SHUTDOWN_RESPONSE); Game::server->SendToMaster(&bitStream); } diff --git a/dZoneManager/LUTriggers.h b/dZoneManager/LUTriggers.h new file mode 100644 index 00000000..75662778 --- /dev/null +++ b/dZoneManager/LUTriggers.h @@ -0,0 +1,34 @@ +#ifndef __LUTRIGGERS__H__ +#define __LUTRIGGERS__H__ + +#include <string> +#include <vector> +#include <cstdint> + +class Command; +class Event; +enum class eTriggerCommandType; +enum class eTriggerEventType; + + +namespace LUTriggers { + struct Command { + eTriggerCommandType id; + std::string target; + std::string targetName; + std::string args; + }; + + struct Event { + eTriggerEventType id; + std::vector<Command*> commands; + }; + + struct Trigger { + uint32_t id; + bool enabled; + std::vector<Event*> events; + }; +}; + +#endif //!__LUTRIGGERS__H__ diff --git a/dZoneManager/Level.cpp b/dZoneManager/Level.cpp index ebfca4cc..7d248588 100644 --- a/dZoneManager/Level.cpp +++ b/dZoneManager/Level.cpp @@ -13,17 +13,22 @@ #include "EntityManager.h" #include "CDFeatureGatingTable.h" #include "CDClientManager.h" +#include "AssetManager.h" Level::Level(Zone* parentZone, const std::string& filepath) { m_ParentZone = parentZone; - std::ifstream file(filepath, std::ios_base::in | std::ios_base::binary); - if (file) { - ReadChunks(file); - } else { + + auto buffer = Game::assetManager->GetFileAsBuffer(filepath.c_str()); + + if (!buffer.m_Success) { Game::logger->Log("Level", "Failed to load %s", filepath.c_str()); + return; } - file.close(); + std::istream file(&buffer); + ReadChunks(file); + + buffer.close(); } Level::~Level() { @@ -41,7 +46,7 @@ const void Level::PrintAllObjects() { } } -void Level::ReadChunks(std::ifstream& file) { +void Level::ReadChunks(std::istream& file) { const uint32_t CHNK_HEADER = ('C' + ('H' << 8) + ('N' << 16) + ('K' << 24)); while (!file.eof()) { @@ -139,7 +144,7 @@ void Level::ReadChunks(std::ifstream& file) { } } -void Level::ReadFileInfoChunk(std::ifstream& file, Header& header) { +void Level::ReadFileInfoChunk(std::istream& file, Header& header) { FileInfoChunk* fi = new FileInfoChunk; BinaryIO::BinaryRead(file, fi->version); BinaryIO::BinaryRead(file, fi->revision); @@ -152,12 +157,12 @@ void Level::ReadFileInfoChunk(std::ifstream& file, Header& header) { if (header.fileInfo->revision == 3452816845 && m_ParentZone->GetZoneID().GetMapID() == 1100) header.fileInfo->revision = 26; } -void Level::ReadSceneObjectDataChunk(std::ifstream& file, Header& header) { +void Level::ReadSceneObjectDataChunk(std::istream& file, Header& header) { SceneObjectDataChunk* chunk = new SceneObjectDataChunk; uint32_t objectsCount = 0; BinaryIO::BinaryRead(file, objectsCount); - CDFeatureGatingTable* featureGatingTable = CDClientManager::Instance()->GetTable<CDFeatureGatingTable>("FeatureGating"); + CDFeatureGatingTable* featureGatingTable = CDClientManager::Instance().GetTable<CDFeatureGatingTable>(); for (uint32_t i = 0; i < objectsCount; ++i) { SceneObject obj; @@ -174,8 +179,8 @@ void Level::ReadSceneObjectDataChunk(std::ifstream& file, Header& header) { //This is a little bit of a bodge, but because the alpha client (HF) doesn't store the //spawn position / rotation like the later versions do, we need to check the LOT for the spawn pos & set it. if (obj.lot == LOT_MARKER_PLAYER_START) { - dZoneManager::Instance()->GetZone()->SetSpawnPos(obj.position); - dZoneManager::Instance()->GetZone()->SetSpawnRot(obj.rotation); + Game::zoneManager->GetZone()->SetSpawnPos(obj.position); + Game::zoneManager->GetZone()->SetSpawnRot(obj.rotation); } std::u16string ldfString = u""; @@ -266,7 +271,7 @@ void Level::ReadSceneObjectDataChunk(std::ifstream& file, Header& header) { spawnInfo.respawnTime = std::stof(data->GetValueAsString()); } else if (data->GetValueType() == eLDFType::LDF_TYPE_U32) // Ints are in ms? { - spawnInfo.respawnTime = std::stoi(data->GetValueAsString()) / 1000; + spawnInfo.respawnTime = std::stoul(data->GetValueAsString()) / 1000; } } if (data->GetKey() == u"spawnsGroupOnSmash") { @@ -292,7 +297,7 @@ void Level::ReadSceneObjectDataChunk(std::ifstream& file, Header& header) { } } Spawner* spawner = new Spawner(spawnInfo); - dZoneManager::Instance()->AddSpawner(obj.id, spawner); + Game::zoneManager->AddSpawner(obj.id, spawner); } else { //Regular object EntityInfo info; info.spawnerID = 0; @@ -323,11 +328,11 @@ void Level::ReadSceneObjectDataChunk(std::ifstream& file, Header& header) { if (!clientOnly) { // We should never have more than 1 zone control object - const auto zoneControlObject = dZoneManager::Instance()->GetZoneControlObject(); + const auto zoneControlObject = Game::zoneManager->GetZoneControlObject(); if (zoneControlObject != nullptr && info.lot == zoneControlObject->GetLOT()) goto deleteSettings; - EntityManager::Instance()->CreateEntity(info, nullptr); + Game::entityManager->CreateEntity(info, nullptr); } else { deleteSettings: diff --git a/dZoneManager/Level.h b/dZoneManager/Level.h index e724363f..0f8fa72d 100644 --- a/dZoneManager/Level.h +++ b/dZoneManager/Level.h @@ -30,12 +30,6 @@ public: struct SceneObjectDataChunk { std::map<LWOOBJID, SceneObject> objects; - SceneObject& GetObject(LWOOBJID id) { - for (std::map<LWOOBJID, SceneObject>::iterator it = objects.begin(); it != objects.end(); ++it) { - if (it->first == id) return it->second; - } - } - const void PrintAllObjects() { for (std::map<LWOOBJID, SceneObject>::iterator it = objects.begin(); it != objects.end(); ++it) { std::cout << "\t ID: " << it->first << " LOT: " << it->second.lot << std::endl; @@ -67,7 +61,7 @@ private: Zone* m_ParentZone; //private functions: - void ReadChunks(std::ifstream& file); - void ReadFileInfoChunk(std::ifstream& file, Header& header); - void ReadSceneObjectDataChunk(std::ifstream& file, Header& header); + void ReadChunks(std::istream& file); + void ReadFileInfoChunk(std::istream& file, Header& header); + void ReadSceneObjectDataChunk(std::istream& file, Header& header); }; diff --git a/dZoneManager/Spawner.cpp b/dZoneManager/Spawner.cpp index 57b4e988..bd1970c6 100644 --- a/dZoneManager/Spawner.cpp +++ b/dZoneManager/Spawner.cpp @@ -46,9 +46,9 @@ Spawner::Spawner(const SpawnerInfo info) { } if (m_Info.spawnOnSmashGroupName != "") { - std::vector<Entity*> spawnSmashEntities = EntityManager::Instance()->GetEntitiesInGroup(m_Info.spawnOnSmashGroupName); - std::vector<Spawner*> spawnSmashSpawners = dZoneManager::Instance()->GetSpawnersInGroup(m_Info.spawnOnSmashGroupName); - std::vector<Spawner*> spawnSmashSpawnersN = dZoneManager::Instance()->GetSpawnersByName(m_Info.spawnOnSmashGroupName); + std::vector<Entity*> spawnSmashEntities = Game::entityManager->GetEntitiesInGroup(m_Info.spawnOnSmashGroupName); + std::vector<Spawner*> spawnSmashSpawners = Game::zoneManager->GetSpawnersInGroup(m_Info.spawnOnSmashGroupName); + std::vector<Spawner*> spawnSmashSpawnersN = Game::zoneManager->GetSpawnersByName(m_Info.spawnOnSmashGroupName); for (Entity* ssEntity : spawnSmashEntities) { m_SpawnSmashFoundGroup = true; ssEntity->AddDieCallback([=]() { @@ -102,11 +102,11 @@ Entity* Spawner::Spawn(std::vector<SpawnerNode*> freeNodes, const bool force) { m_EntityInfo.spawnerID = m_Info.spawnerID; } - Entity* rezdE = EntityManager::Instance()->CreateEntity(m_EntityInfo, nullptr); + Entity* rezdE = Game::entityManager->CreateEntity(m_EntityInfo, nullptr); rezdE->GetGroups() = m_Info.groups; - EntityManager::Instance()->ConstructEntity(rezdE); + Game::entityManager->ConstructEntity(rezdE); m_Entities.insert({ rezdE->GetObjectID(), spawnNode }); spawnNode->entities.push_back(rezdE->GetObjectID()); @@ -134,24 +134,23 @@ void Spawner::AddEntitySpawnedCallback(std::function<void(Entity*)> callback) { void Spawner::Reset() { m_Start = true; - - for (auto* node : m_Info.nodes) { - for (const auto& spawned : node->entities) { - auto* entity = EntityManager::Instance()->GetEntity(spawned); - - if (entity == nullptr) continue; - - entity->Kill(); - } - - node->entities.clear(); - } - + DestroyAllEntities(); m_Entities.clear(); m_AmountSpawned = 0; m_NeedsUpdate = true; } +void Spawner::DestroyAllEntities(){ + for (auto* node : m_Info.nodes) { + for (const auto& element : node->entities) { + auto* entity = Game::entityManager->GetEntity(element); + if (entity == nullptr) continue; + entity->Kill(); + } + node->entities.clear(); + } +} + void Spawner::SoftReset() { m_Start = true; m_AmountSpawned = 0; diff --git a/dZoneManager/Spawner.h b/dZoneManager/Spawner.h index 908bb3a3..1f610b71 100644 --- a/dZoneManager/Spawner.h +++ b/dZoneManager/Spawner.h @@ -9,6 +9,7 @@ #include <string> #include <functional> #include "LDFFormat.h" +#include "EntityInfo.h" struct SpawnerNode { NiPoint3 position = NiPoint3::ZERO; @@ -60,6 +61,7 @@ public: void AddEntitySpawnedCallback(std::function<void(Entity*)> callback); void SetSpawnLot(LOT lot); void Reset(); + void DestroyAllEntities(); void SoftReset(); void SetRespawnTime(float time); void SetNumToMaintain(int32_t value); diff --git a/dZoneManager/WorldConfig.h b/dZoneManager/WorldConfig.h new file mode 100644 index 00000000..a98433a1 --- /dev/null +++ b/dZoneManager/WorldConfig.h @@ -0,0 +1,67 @@ +#ifndef __WORLDCONFIG__H__ +#define __WORLDCONFIG__H__ + +#include <cstdint> +#include <string> + +struct WorldConfig { + int32_t worldConfigID{}; //! Primary key for WorlcConfig table + float peGravityValue{}; //! Unknown + float peBroadphaseWorldSize{}; //! Unknown + float peGameObjScaleFactor{}; //! Unknown + float characterRotationSpeed{}; //! The players' rotation speed + float characterWalkForwardSpeed{}; //! The players' walk forward speed + float characterWalkBackwardSpeed{}; //! The players' walk backwards speed + float characterWalkStrafeSpeed{}; //! The players' strafe speed + float characterWalkStrafeForwardSpeed{}; //! The players' walk strafe forward speed + float characterWalkStrafeBackwardSpeed{}; //! The players' walk strage backwards speed + float characterRunBackwardSpeed{}; //! The players' run backwards speed + float characterRunStrafeSpeed{}; //! The players' run strafe speed + float characterRunStrafeForwardSpeed{}; //! The players' run strafe forward speed + float characterRunStrafeBackwardSpeed{}; //! The players' run strage backwards speed + float globalCooldown{}; //! The global ability cooldown + float characterGroundedTime{}; //! Unknown + float characterGroundedSpeed{}; //! Unknown + float globalImmunityTime{}; //! Unknown + float characterMaxSlope{}; //! Unknown + float defaultRespawnTime{}; //! Unknown + float missionTooltipTimeout{}; + float vendorBuyMultiplier{}; //! The buy scalar for buying from vendors + float petFollowRadius{}; //! The players' pet follow radius + float characterEyeHeight{}; //! The players' eye height + float flightVerticalVelocity{}; //! Unknown + float flightAirspeed{}; //! Unknown + float flightFuelRatio{}; //! Unknown + float flightMaxAirspeed{}; //! Unknown + float fReputationPerVote{}; //! Unknown + int32_t propertyCloneLimit{}; //! Unknown + int32_t defaultHomespaceTemplate{}; //! Unknown + float coinsLostOnDeathPercent{}; //! The percentage of coins to lose on a player death + int32_t coinsLostOnDeathMin{}; //! The minimum number of coins to lose on a player death + int32_t coinsLostOnDeathMax{}; //! The maximum number of coins to lose on a player death + int32_t characterVotesPerDay{}; //! Unknown + int32_t propertyModerationRequestApprovalCost{};//! Unknown + int32_t propertyModerationRequestReviewCost{}; //! Unknown + int32_t propertyModRequestsAllowedSpike{}; //! Unknown + int32_t propertyModRequestsAllowedInterval{}; //! Unknown + int32_t propertyModRequestsAllowedTotal{}; //! Unknown + int32_t propertyModRequestsSpikeDuration{}; //! Unknown + int32_t propertyModRequestsIntervalDuration{}; //! Unknown + bool modelModerateOnCreate{}; //! Unknown + float defaultPropertyMaxHeight{}; //! Unknown + float reputationPerVoteCast{}; //! Unknown + float reputationPerVoteReceived{}; //! Unknown + int32_t showcaseTopModelConsiderationBattles{}; //! Unknown + float reputationPerBattlePromotion{}; //! Unknown + float coinsLostOnDeathMinTimeout{}; //! Unknown + float coinsLostOnDeathMaxTimeout{}; //! Unknown + int32_t mailBaseFee{}; //! The base fee to take when a player sends mail + float mailPercentAttachmentFee{}; //! The scalar multiplied by an items base cost to determine how much that item costs to be mailed + int32_t propertyReputationDelay{}; //! Unknown + int32_t levelCap{}; //! The maximum player level + std::string levelUpBehaviorEffect{}; //! Unknown + int32_t characterVersion{}; //! Unknown + int32_t levelCapCurrencyConversion{}; //! The ratio of UScore (LEGO Score) to coins +}; + +#endif //! __WORLDCONFIG__H__ diff --git a/dZoneManager/Zone.cpp b/dZoneManager/Zone.cpp index 54246c28..6b153706 100644 --- a/dZoneManager/Zone.cpp +++ b/dZoneManager/Zone.cpp @@ -6,12 +6,17 @@ #include "dLogger.h" #include "GeneralUtils.h" #include "BinaryIO.h" +#include "LUTriggers.h" +#include "AssetManager.h" #include "CDClientManager.h" #include "CDZoneTableTable.h" #include "Spawner.h" #include "dZoneManager.h" +#include "eTriggerCommandType.h" +#include "eTriggerEventType.h" + Zone::Zone(const LWOMAPID& mapID, const LWOINSTANCEID& instanceID, const LWOCLONEID& cloneID) : m_ZoneID(mapID, instanceID, cloneID) { m_NumberOfScenesLoaded = 0; @@ -40,7 +45,14 @@ void Zone::LoadZoneIntoMemory() { m_ZonePath = m_ZoneFilePath.substr(0, m_ZoneFilePath.rfind('/') + 1); if (m_ZoneFilePath == "ERR") return; - std::ifstream file(m_ZoneFilePath, std::ios::binary); + AssetMemoryBuffer buffer = Game::assetManager->GetFileAsBuffer(m_ZoneFilePath.c_str()); + + if (!buffer.m_Success) { + Game::logger->Log("Zone", "Failed to load %s", m_ZoneFilePath.c_str()); + throw std::runtime_error("Aborting Zone loading due to no Zone File."); + } + + std::istream file(&buffer); if (file) { BinaryIO::BinaryRead(file, m_ZoneFileFormatVersion); @@ -90,15 +102,12 @@ void Zone::LoadZoneIntoMemory() { if (m_ZoneFileFormatVersion >= Zone::ZoneFileFormatVersion::EarlyAlpha) { BinaryIO::BinaryRead(file, m_PathDataLength); - uint32_t unknown; - uint32_t pathCount; + BinaryIO::BinaryRead(file, m_PathChunkVersion); // always should be 1 - BinaryIO::BinaryRead(file, unknown); + uint32_t pathCount; BinaryIO::BinaryRead(file, pathCount); - for (uint32_t i = 0; i < pathCount; ++i) { - LoadPath(file); - } + for (uint32_t i = 0; i < pathCount; ++i) LoadPath(file); for (Path path : m_Paths) { if (path.pathType == PathType::Spawner) { @@ -140,29 +149,25 @@ void Zone::LoadZoneIntoMemory() { info.activeOnLoad = path.spawner.spawnerNetActive; info.isNetwork = true; Spawner* spawner = new Spawner(info); - dZoneManager::Instance()->AddSpawner(info.spawnerID, spawner); + Game::zoneManager->AddSpawner(info.spawnerID, spawner); } } - - - //m_PathData.resize(m_PathDataLength); - //file.read((char*)&m_PathData[0], m_PathDataLength); } } else { Game::logger->Log("Zone", "Failed to open: %s", m_ZoneFilePath.c_str()); } m_ZonePath = m_ZoneFilePath.substr(0, m_ZoneFilePath.rfind('/') + 1); - file.close(); + buffer.close(); } std::string Zone::GetFilePathForZoneID() { //We're gonna go ahead and presume we've got the db loaded already: - CDZoneTableTable* zoneTable = CDClientManager::Instance()->GetTable<CDZoneTableTable>("ZoneTable"); + CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable<CDZoneTableTable>(); const CDZoneTable* zone = zoneTable->Query(this->GetZoneID().GetMapID()); if (zone != nullptr) { - std::string toReturn = "./res/maps/" + zone->zoneName; + std::string toReturn = "maps/" + zone->zoneName; std::transform(toReturn.begin(), toReturn.end(), toReturn.begin(), ::tolower); return toReturn; } @@ -222,7 +227,7 @@ const void Zone::PrintAllGameObjects() { } } -void Zone::LoadScene(std::ifstream& file) { +void Zone::LoadScene(std::istream& file) { SceneRef scene; scene.level = nullptr; LWOSCENEID lwoSceneID(LWOZONEID_INVALID, 0); @@ -232,7 +237,8 @@ void Zone::LoadScene(std::ifstream& file) { scene.filename = BinaryIO::ReadString(file, sceneFilenameLength); std::string luTriggersPath = scene.filename.substr(0, scene.filename.size() - 4) + ".lutriggers"; - std::vector<LUTriggers::Trigger*> triggers = LoadLUTriggers(luTriggersPath, scene.id); + std::vector<LUTriggers::Trigger*> triggers; + if(Game::assetManager->HasFile((m_ZonePath + luTriggersPath).c_str())) triggers = LoadLUTriggers(luTriggersPath, scene.id); for (LUTriggers::Trigger* trigger : triggers) { scene.triggers.insert({ trigger->id, trigger }); @@ -247,15 +253,6 @@ void Zone::LoadScene(std::ifstream& file) { scene.name = BinaryIO::ReadString(file, sceneNameLength); file.ignore(3); - /* - if (m_Scenes.find(scene.id) != m_Scenes.end()) { - //Extract the layer id from the filename (bad I know, but it's reliable at least): - std::string layer = scene.filename.substr(scene.filename.rfind('x') + 1); - layer = layer.substr(0, layer.find('_')); - lwoSceneID.SetLayerID(std::atoi(layer.c_str())); - } - */ - lwoSceneID.SetLayerID(scene.sceneType); m_Scenes.insert(std::make_pair(lwoSceneID, scene)); @@ -264,10 +261,20 @@ void Zone::LoadScene(std::ifstream& file) { std::vector<LUTriggers::Trigger*> Zone::LoadLUTriggers(std::string triggerFile, LWOSCENEID sceneID) { std::vector<LUTriggers::Trigger*> lvlTriggers; - std::ifstream file(m_ZonePath + triggerFile); + + auto buffer = Game::assetManager->GetFileAsBuffer((m_ZonePath + triggerFile).c_str()); + + if (!buffer.m_Success) { + Game::logger->Log("Zone", "Failed to load %s from disk. Skipping loading triggers", (m_ZonePath + triggerFile).c_str()); + return lvlTriggers; + } + + std::istream file(&buffer); std::stringstream data; data << file.rdbuf(); + buffer.close(); + if (data.str().size() == 0) return lvlTriggers; tinyxml2::XMLDocument* doc = new tinyxml2::XMLDocument(); @@ -292,11 +299,11 @@ std::vector<LUTriggers::Trigger*> Zone::LoadLUTriggers(std::string triggerFile, auto currentEvent = currentTrigger->FirstChildElement("event"); while (currentEvent) { LUTriggers::Event* newEvent = new LUTriggers::Event(); - newEvent->eventID = currentEvent->Attribute("id"); + newEvent->id = TriggerEventType::StringToTriggerEventType(currentEvent->Attribute("id")); auto currentCommand = currentEvent->FirstChildElement("command"); while (currentCommand) { LUTriggers::Command* newCommand = new LUTriggers::Command(); - newCommand->id = currentCommand->Attribute("id"); + newCommand->id = TriggerCommandType::StringToTriggerCommandType(currentCommand->Attribute("id")); newCommand->target = currentCommand->Attribute("target"); if (currentCommand->Attribute("targetName") != NULL) { newCommand->targetName = currentCommand->Attribute("targetName"); @@ -336,12 +343,13 @@ const Path* Zone::GetPath(std::string name) const { return nullptr; } -void Zone::LoadSceneTransition(std::ifstream& file) { +void Zone::LoadSceneTransition(std::istream& file) { SceneTransition sceneTrans; - if (m_ZoneFileFormatVersion < Zone::ZoneFileFormatVersion::LateAlpha) { + if (m_ZoneFileFormatVersion < Zone::ZoneFileFormatVersion::Auramar) { uint8_t length; BinaryIO::BinaryRead(file, length); sceneTrans.name = BinaryIO::ReadString(file, length); + BinaryIO::BinaryRead(file, sceneTrans.width); } //BR�THER MAY I HAVE SOME L��PS? @@ -354,22 +362,18 @@ void Zone::LoadSceneTransition(std::ifstream& file) { m_SceneTransitions.push_back(sceneTrans); } -SceneTransitionInfo Zone::LoadSceneTransitionInfo(std::ifstream& file) { +SceneTransitionInfo Zone::LoadSceneTransitionInfo(std::istream& file) { SceneTransitionInfo info; BinaryIO::BinaryRead(file, info.sceneID); BinaryIO::BinaryRead(file, info.position); return info; } -void Zone::LoadPath(std::ifstream& file) { - // Currently only spawner (type 4) paths are supported +void Zone::LoadPath(std::istream& file) { Path path = Path(); - uint32_t unknown1; - uint32_t pathType; - uint32_t pathBehavior; - BinaryIO::BinaryRead(file, path.pathVersion); + uint8_t stringLength; BinaryIO::BinaryRead(file, stringLength); for (uint8_t i = 0; i < stringLength; ++i) { @@ -377,16 +381,14 @@ void Zone::LoadPath(std::ifstream& file) { BinaryIO::BinaryRead(file, character); path.pathName.push_back(character); } - BinaryIO::BinaryRead(file, pathType); - path.pathType = PathType(pathType); - BinaryIO::BinaryRead(file, unknown1); - BinaryIO::BinaryRead(file, pathBehavior); - path.pathType = PathType(pathType); + + BinaryIO::BinaryRead(file, path.pathType); + BinaryIO::BinaryRead(file, path.flags); + BinaryIO::BinaryRead(file, path.pathBehavior); if (path.pathType == PathType::MovingPlatform) { if (path.pathVersion >= 18) { - uint8_t unknown; - BinaryIO::BinaryRead(file, unknown); + BinaryIO::BinaryRead(file, path.movingPlatform.timeBasedMovement); } else if (path.pathVersion >= 13) { uint8_t count; BinaryIO::BinaryRead(file, count); @@ -397,35 +399,43 @@ void Zone::LoadPath(std::ifstream& file) { } } } else if (path.pathType == PathType::Property) { - int32_t unknown; - BinaryIO::BinaryRead(file, unknown); + BinaryIO::BinaryRead(file, path.property.pathType); BinaryIO::BinaryRead(file, path.property.price); - BinaryIO::BinaryRead(file, path.property.rentalTime); - BinaryIO::BinaryRead(file, path.property.associatedZone); - uint8_t count1; - BinaryIO::BinaryRead(file, count1); - for (uint8_t i = 0; i < count1; ++i) { - uint16_t character; - BinaryIO::BinaryRead(file, character); - path.property.displayName.push_back(character); - } - uint32_t count2; - BinaryIO::BinaryRead(file, count2); - for (uint8_t i = 0; i < count2; ++i) { - uint16_t character; - BinaryIO::BinaryRead(file, character); - path.property.displayDesc.push_back(character); - } - int32_t unknown1; - BinaryIO::BinaryRead(file, unknown1); - BinaryIO::BinaryRead(file, path.property.cloneLimit); - BinaryIO::BinaryRead(file, path.property.repMultiplier); BinaryIO::BinaryRead(file, path.property.rentalTimeUnit); - BinaryIO::BinaryRead(file, path.property.achievementRequired); - BinaryIO::BinaryRead(file, path.property.playerZoneCoords.x); - BinaryIO::BinaryRead(file, path.property.playerZoneCoords.y); - BinaryIO::BinaryRead(file, path.property.playerZoneCoords.z); - BinaryIO::BinaryRead(file, path.property.maxBuildHeight); + BinaryIO::BinaryRead(file, path.property.associatedZone); + + if (path.pathVersion >= 5) { + uint8_t count1; + BinaryIO::BinaryRead(file, count1); + for (uint8_t i = 0; i < count1; ++i) { + uint16_t character; + BinaryIO::BinaryRead(file, character); + path.property.displayName.push_back(character); + } + uint32_t count2; + BinaryIO::BinaryRead(file, count2); + for (uint8_t i = 0; i < count2; ++i) { + uint16_t character; + BinaryIO::BinaryRead(file, character); + path.property.displayDesc.push_back(character); + } + } + + if (path.pathVersion >= 6) BinaryIO::BinaryRead(file, path.property.type); + + if (path.pathVersion >= 7) { + BinaryIO::BinaryRead(file, path.property.cloneLimit); + BinaryIO::BinaryRead(file, path.property.repMultiplier); + BinaryIO::BinaryRead(file, path.property.rentalTimeUnit); + } + + if (path.pathVersion >= 8) { + BinaryIO::BinaryRead(file, path.property.achievementRequired); + BinaryIO::BinaryRead(file, path.property.playerZoneCoords.x); + BinaryIO::BinaryRead(file, path.property.playerZoneCoords.y); + BinaryIO::BinaryRead(file, path.property.playerZoneCoords.z); + BinaryIO::BinaryRead(file, path.property.maxBuildHeight); + } } else if (path.pathType == PathType::Camera) { uint8_t count; BinaryIO::BinaryRead(file, count); @@ -435,11 +445,10 @@ void Zone::LoadPath(std::ifstream& file) { path.camera.nextPath.push_back(character); } if (path.pathVersion >= 14) { - uint8_t unknown; - BinaryIO::BinaryRead(file, unknown); + BinaryIO::BinaryRead(file, path.camera.rotatePlayer); + } } else if (path.pathType == PathType::Spawner) { - //SpawnerPath* path = static_cast<SpawnerPath*>(path); // Convert to a spawner path BinaryIO::BinaryRead(file, path.spawner.spawnedLOT); BinaryIO::BinaryRead(file, path.spawner.respawnTime); BinaryIO::BinaryRead(file, path.spawner.maxToSpawn); @@ -460,7 +469,7 @@ void Zone::LoadPath(std::ifstream& file) { BinaryIO::BinaryRead(file, waypoint.position.z); - if (path.pathType == PathType::Spawner || path.pathType == PathType::MovingPlatform || path.pathType == PathType::Race) { + if (path.pathType == PathType::Spawner || path.pathType == PathType::MovingPlatform || path.pathType == PathType::Race || path.pathType == PathType::Camera || path.pathType == PathType::Rail) { BinaryIO::BinaryRead(file, waypoint.rotation.w); BinaryIO::BinaryRead(file, waypoint.rotation.x); BinaryIO::BinaryRead(file, waypoint.rotation.y); @@ -488,33 +497,19 @@ void Zone::LoadPath(std::ifstream& file) { } } } else if (path.pathType == PathType::Camera) { - float unknown; - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); BinaryIO::BinaryRead(file, waypoint.camera.time); - BinaryIO::BinaryRead(file, unknown); + BinaryIO::BinaryRead(file, waypoint.camera.fov); BinaryIO::BinaryRead(file, waypoint.camera.tension); BinaryIO::BinaryRead(file, waypoint.camera.continuity); BinaryIO::BinaryRead(file, waypoint.camera.bias); } else if (path.pathType == PathType::Race) { - uint8_t unknown; - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); - float unknown1; - BinaryIO::BinaryRead(file, unknown1); - BinaryIO::BinaryRead(file, unknown1); - BinaryIO::BinaryRead(file, unknown1); + BinaryIO::BinaryRead(file, waypoint.racing.isResetNode); + BinaryIO::BinaryRead(file, waypoint.racing.isNonHorizontalCamera); + BinaryIO::BinaryRead(file, waypoint.racing.planeWidth); + BinaryIO::BinaryRead(file, waypoint.racing.planeHeight); + BinaryIO::BinaryRead(file, waypoint.racing.shortestDistanceToEnd); } else if (path.pathType == PathType::Rail) { - float unknown; - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); - BinaryIO::BinaryRead(file, unknown); - if (path.pathVersion >= 17) { - BinaryIO::BinaryRead(file, unknown); - } + if (path.pathVersion > 16) BinaryIO::BinaryRead(file, waypoint.rail.speed); } // object LDF configs @@ -538,8 +533,14 @@ void Zone::LoadPath(std::ifstream& file) { BinaryIO::BinaryRead(file, character); value.push_back(character); } - LDFBaseData* ldfConfig = LDFBaseData::DataFromString(parameter + "=" + value); - waypoint.config.push_back(ldfConfig); + + LDFBaseData* ldfConfig = nullptr; + if (path.pathType == PathType::Movement || path.pathType == PathType::Rail) { + ldfConfig = LDFBaseData::DataFromString(parameter + "=0:" + value); + } else { + ldfConfig = LDFBaseData::DataFromString(parameter + "=" + value); + } + if (ldfConfig) waypoint.config.push_back(ldfConfig); } } diff --git a/dZoneManager/Zone.h b/dZoneManager/Zone.h index 50530273..b3e72036 100644 --- a/dZoneManager/Zone.h +++ b/dZoneManager/Zone.h @@ -1,35 +1,18 @@ #pragma once + #include "dZMCommon.h" #include "LDFFormat.h" -#include "../thirdparty/tinyxml2/tinyxml2.h" +#include "tinyxml2.h" #include <string> #include <vector> #include <map> -class Level; - -class LUTriggers { -public: - - struct Command { - std::string id; - std::string target; - std::string targetName; - std::string args; - }; - - struct Event { - std::string eventID; - std::vector<Command*> commands; - }; - - struct Trigger { - uint32_t id; - bool enabled; - std::vector<Event*> events; - }; +namespace LUTriggers { + struct Trigger; }; +class Level; + struct SceneRef { std::string filename; uint32_t id; @@ -47,6 +30,7 @@ struct SceneTransitionInfo { struct SceneTransition { std::string name; std::vector<SceneTransitionInfo> points; + float width; }; struct MovingPlatformPathWaypoint { @@ -59,16 +43,31 @@ struct MovingPlatformPathWaypoint { struct CameraPathWaypoint { float time; + float fov; float tension; float continuity; float bias; }; +struct RacingPathWaypoint { + uint8_t isResetNode; + uint8_t isNonHorizontalCamera; + float planeWidth; + float planeHeight; + float shortestDistanceToEnd; +}; + +struct RailPathWaypoint { + float speed; +}; + struct PathWaypoint { NiPoint3 position; NiQuaternion rotation; // not included in all, but it's more convenient here MovingPlatformPathWaypoint movingPlatform; CameraPathWaypoint camera; + RacingPathWaypoint racing; + RailPathWaypoint rail; std::vector<LDFBaseData*> config; }; @@ -89,6 +88,19 @@ enum class PathBehavior : uint32_t { Once = 2 }; +enum class PropertyPathType : int32_t { + Path = 0, + EntireZone = 1, + GenetatedRectangle = 2 +}; + +enum class PropertyType : int32_t { + Premiere = 0, + Prize = 1, + LUP = 2, + Headspace = 3 +}; + enum class PropertyRentalTimeUnit : int32_t { Forever = 0, Seconds = 1, @@ -116,17 +128,19 @@ enum class PropertyAchievmentRequired : int32_t { struct MovingPlatformPath { std::string platformTravelSound; + uint8_t timeBasedMovement; }; struct PropertyPath { + PropertyPathType pathType; int32_t price; - int32_t rentalTime; + PropertyRentalTimeUnit rentalTimeUnit; uint64_t associatedZone; std::string displayName; std::string displayDesc; + PropertyType type; int32_t cloneLimit; float repMultiplier; - PropertyRentalTimeUnit rentalTimeUnit; PropertyAchievmentRequired achievementRequired; NiPoint3 playerZoneCoords; float maxBuildHeight; @@ -134,6 +148,7 @@ struct PropertyPath { struct CameraPath { std::string nextPath; + uint8_t rotatePlayer; }; struct SpawnerPath { @@ -150,6 +165,7 @@ struct Path { uint32_t pathVersion; PathType pathType; std::string pathName; + uint32_t flags; PathBehavior pathBehavior; uint32_t waypointCount; std::vector<PathWaypoint> pathWaypoints; @@ -190,7 +206,7 @@ public: uint32_t GetWorldID() const { return m_WorldID; } [[nodiscard]] std::string GetZoneName() const { return m_ZoneName; } - std::string GetZoneRawPath() const { return m_ZoneRawPath;} + std::string GetZoneRawPath() const { return m_ZoneRawPath; } std::string GetZonePath() const { return m_ZonePath; } const NiPoint3& GetSpawnPos() const { return m_Spawnpoint; } @@ -219,15 +235,17 @@ private: std::map<LWOSCENEID, SceneRef, mapCompareLwoSceneIDs> m_Scenes; std::vector<SceneTransition> m_SceneTransitions; + uint32_t m_PathDataLength; - //std::vector<char> m_PathData; //Binary path data + uint32_t m_PathChunkVersion; std::vector<Path> m_Paths; + std::map<LWOSCENEID, uint32_t, mapCompareLwoSceneIDs> m_MapRevisions; //rhs is the revision! //private ("helper") functions: - void LoadScene(std::ifstream& file); + void LoadScene(std::istream& file); std::vector<LUTriggers::Trigger*> LoadLUTriggers(std::string triggerFile, LWOSCENEID sceneID); - void LoadSceneTransition(std::ifstream& file); - SceneTransitionInfo LoadSceneTransitionInfo(std::ifstream& file); - void LoadPath(std::ifstream& file); + void LoadSceneTransition(std::istream& file); + SceneTransitionInfo LoadSceneTransitionInfo(std::istream& file); + void LoadPath(std::istream& file); }; diff --git a/dZoneManager/dZoneManager.cpp b/dZoneManager/dZoneManager.cpp index 102fb3af..7b352a57 100644 --- a/dZoneManager/dZoneManager.cpp +++ b/dZoneManager/dZoneManager.cpp @@ -8,12 +8,15 @@ #include "DestroyableComponent.h" #include "GameMessages.h" #include "VanityUtilities.h" +#include "WorldConfig.h" +#include "CDZoneTableTable.h" #include <chrono> +#include "eObjectBits.h" +#include "CDZoneTableTable.h" +#include "AssetManager.h" #include "../dWorldServer/ObjectIDManager.h" -dZoneManager* dZoneManager::m_Address = nullptr; - void dZoneManager::Initialize(const LWOZONEID& zoneID) { Game::logger->Log("dZoneManager", "Preparing zone: %i/%i/%i", zoneID.GetMapID(), zoneID.GetInstanceID(), zoneID.GetCloneID()); @@ -26,7 +29,7 @@ void dZoneManager::Initialize(const LWOZONEID& zoneID) { LOT zoneControlTemplate = 2365; - CDZoneTableTable* zoneTable = CDClientManager::Instance()->GetTable<CDZoneTableTable>("ZoneTable"); + CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable<CDZoneTableTable>(); if (zoneTable != nullptr) { const CDZoneTable* zone = zoneTable->Query(zoneID.GetMapID()); @@ -34,8 +37,8 @@ void dZoneManager::Initialize(const LWOZONEID& zoneID) { zoneControlTemplate = zone->zoneControlTemplate != -1 ? zone->zoneControlTemplate : 2365; const auto min = zone->ghostdistance_min != -1.0f ? zone->ghostdistance_min : 100; const auto max = zone->ghostdistance != -1.0f ? zone->ghostdistance : 100; - EntityManager::Instance()->SetGhostDistanceMax(max + min); - EntityManager::Instance()->SetGhostDistanceMin(max); + Game::entityManager->SetGhostDistanceMax(max + min); + Game::entityManager->SetGhostDistanceMin(max); m_PlayerLoseCoinsOnDeath = zone->PlayerLoseCoinsOnDeath; } } @@ -43,16 +46,23 @@ void dZoneManager::Initialize(const LWOZONEID& zoneID) { Game::logger->Log("dZoneManager", "Creating zone control object %i", zoneControlTemplate); // Create ZoneControl object + if (!Game::entityManager) { + Game::logger->Log("dZoneManager", "ERROR: No entity manager loaded. Cannot proceed."); + throw std::invalid_argument("No entity manager loaded. Cannot proceed."); + } + Game::entityManager->Initialize(); EntityInfo info; info.lot = zoneControlTemplate; info.id = 70368744177662; - Entity* zoneControl = EntityManager::Instance()->CreateEntity(info, nullptr, nullptr, true); + Entity* zoneControl = Game::entityManager->CreateEntity(info, nullptr, nullptr, true); m_ZoneControlObject = zoneControl; m_pZone->Initalize(); endTime = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::steady_clock::now().time_since_epoch()).count(); + LoadWorldConfig(); + Game::logger->Log("dZoneManager", "Zone prepared in: %llu ms", (endTime - startTime)); VanityUtilities::SpawnVanity(); @@ -69,6 +79,7 @@ dZoneManager::~dZoneManager() { m_Spawners.erase(p.first); } + if (m_WorldConfig) delete m_WorldConfig; } Zone* dZoneManager::GetZone() { @@ -117,24 +128,6 @@ LWOZONEID dZoneManager::GetZoneID() const { return m_ZoneID; } -uint32_t dZoneManager::GetMaxLevel() { - if (m_MaxLevel == 0) { - auto tableData = CDClientDatabase::ExecuteQuery("SELECT LevelCap FROM WorldConfig WHERE WorldConfigID = 1 LIMIT 1;"); - m_MaxLevel = tableData.getIntField(0, -1); - tableData.finalize(); - } - return m_MaxLevel; -} - -int32_t dZoneManager::GetLevelCapCurrencyConversion() { - if (m_CurrencyConversionRate == 0) { - auto tableData = CDClientDatabase::ExecuteQuery("SELECT LevelCapCurrencyConversion FROM WorldConfig WHERE WorldConfigID = 1 LIMIT 1;"); - m_CurrencyConversionRate = tableData.getIntField(0, -1); - tableData.finalize(); - } - return m_CurrencyConversionRate; -} - void dZoneManager::Update(float deltaTime) { for (auto spawner : m_Spawners) { spawner.second->Update(deltaTime); @@ -146,8 +139,7 @@ LWOOBJID dZoneManager::MakeSpawner(SpawnerInfo info) { if (objectId == LWOOBJID_EMPTY) { objectId = ObjectIDManager::Instance()->GenerateObjectID(); - - objectId = GeneralUtils::SetBit(objectId, OBJECT_BIT_CLIENT); + GeneralUtils::SetBit(objectId, eObjectBits::CLIENT); info.spawnerID = objectId; } @@ -159,9 +151,9 @@ LWOOBJID dZoneManager::MakeSpawner(SpawnerInfo info) { entityInfo.id = objectId; entityInfo.lot = 176; - auto* entity = EntityManager::Instance()->CreateEntity(entityInfo, nullptr, nullptr, false, objectId); + auto* entity = Game::entityManager->CreateEntity(entityInfo, nullptr, nullptr, false, objectId); - EntityManager::Instance()->ConstructEntity(entity); + Game::entityManager->ConstructEntity(entity); AddSpawner(objectId, spawner); @@ -186,7 +178,7 @@ void dZoneManager::RemoveSpawner(const LWOOBJID id) { return; } - auto* entity = EntityManager::Instance()->GetEntity(id); + auto* entity = Game::entityManager->GetEntity(id); if (entity != nullptr) { entity->Kill(); @@ -195,17 +187,7 @@ void dZoneManager::RemoveSpawner(const LWOOBJID id) { Game::logger->Log("dZoneManager", "Failed to find spawner entity (%llu)", id); } - for (auto* node : spawner->m_Info.nodes) { - for (const auto& element : node->entities) { - auto* nodeEntity = EntityManager::Instance()->GetEntity(element); - - if (nodeEntity == nullptr) continue; - - nodeEntity->Kill(); - } - - node->entities.clear(); - } + spawner->DestroyAllEntities(); spawner->Deactivate(); @@ -249,3 +231,88 @@ uint32_t dZoneManager::GetUniqueMissionIdStartingValue() { } return m_UniqueMissionIdStart; } + +bool dZoneManager::CheckIfAccessibleZone(LWOMAPID zoneID) { + //We're gonna go ahead and presume we've got the db loaded already: + CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable<CDZoneTableTable>(); + const CDZoneTable* zone = zoneTable->Query(zoneID); + if (zone != nullptr) { + return Game::assetManager->HasFile(("maps/" + zone->zoneName).c_str()); + } else { + return false; + } +} + +void dZoneManager::LoadWorldConfig() { + Game::logger->Log("dZoneManager", "Loading WorldConfig into memory"); + + auto worldConfig = CDClientDatabase::ExecuteQuery("SELECT * FROM WorldConfig;"); + + if (!m_WorldConfig) m_WorldConfig = new WorldConfig(); + + if (worldConfig.eof()) { + Game::logger->Log("dZoneManager", "WorldConfig table is empty. Is this intended?"); + return; + } + + // Now read in the giant table + m_WorldConfig->worldConfigID = worldConfig.getIntField("WorldConfigID"); + m_WorldConfig->peGravityValue = worldConfig.getFloatField("pegravityvalue"); + m_WorldConfig->peBroadphaseWorldSize = worldConfig.getFloatField("pebroadphaseworldsize"); + m_WorldConfig->peGameObjScaleFactor = worldConfig.getFloatField("pegameobjscalefactor"); + m_WorldConfig->characterRotationSpeed = worldConfig.getFloatField("character_rotation_speed"); + m_WorldConfig->characterWalkForwardSpeed = worldConfig.getFloatField("character_walk_forward_speed"); + m_WorldConfig->characterWalkBackwardSpeed = worldConfig.getFloatField("character_walk_backward_speed"); + m_WorldConfig->characterWalkStrafeSpeed = worldConfig.getFloatField("character_walk_strafe_speed"); + m_WorldConfig->characterWalkStrafeForwardSpeed = worldConfig.getFloatField("character_walk_strafe_forward_speed"); + m_WorldConfig->characterWalkStrafeBackwardSpeed = worldConfig.getFloatField("character_walk_strafe_backward_speed"); + m_WorldConfig->characterRunBackwardSpeed = worldConfig.getFloatField("character_run_backward_speed"); + m_WorldConfig->characterRunStrafeSpeed = worldConfig.getFloatField("character_run_strafe_speed"); + m_WorldConfig->characterRunStrafeForwardSpeed = worldConfig.getFloatField("character_run_strafe_forward_speed"); + m_WorldConfig->characterRunStrafeBackwardSpeed = worldConfig.getFloatField("character_run_strafe_backward_speed"); + m_WorldConfig->globalCooldown = worldConfig.getFloatField("global_cooldown"); + m_WorldConfig->characterGroundedTime = worldConfig.getFloatField("characterGroundedTime"); + m_WorldConfig->characterGroundedSpeed = worldConfig.getFloatField("characterGroundedSpeed"); + m_WorldConfig->globalImmunityTime = worldConfig.getFloatField("globalImmunityTime"); + m_WorldConfig->characterMaxSlope = worldConfig.getFloatField("character_max_slope"); + m_WorldConfig->defaultRespawnTime = worldConfig.getFloatField("defaultrespawntime"); + m_WorldConfig->missionTooltipTimeout = worldConfig.getFloatField("mission_tooltip_timeout"); + m_WorldConfig->vendorBuyMultiplier = worldConfig.getFloatField("vendor_buy_multiplier", 0.1); + m_WorldConfig->petFollowRadius = worldConfig.getFloatField("pet_follow_radius"); + m_WorldConfig->characterEyeHeight = worldConfig.getFloatField("character_eye_height"); + m_WorldConfig->flightVerticalVelocity = worldConfig.getFloatField("flight_vertical_velocity"); + m_WorldConfig->flightAirspeed = worldConfig.getFloatField("flight_airspeed"); + m_WorldConfig->flightFuelRatio = worldConfig.getFloatField("flight_fuel_ratio"); + m_WorldConfig->flightMaxAirspeed = worldConfig.getFloatField("flight_max_airspeed"); + m_WorldConfig->fReputationPerVote = worldConfig.getFloatField("fReputationPerVote"); + m_WorldConfig->propertyCloneLimit = worldConfig.getIntField("nPropertyCloneLimit"); + m_WorldConfig->defaultHomespaceTemplate = worldConfig.getIntField("defaultHomespaceTemplate"); + m_WorldConfig->coinsLostOnDeathPercent = worldConfig.getFloatField("coins_lost_on_death_percent"); + m_WorldConfig->coinsLostOnDeathMin = worldConfig.getIntField("coins_lost_on_death_min"); + m_WorldConfig->coinsLostOnDeathMax = worldConfig.getIntField("coins_lost_on_death_max"); + m_WorldConfig->characterVotesPerDay = worldConfig.getIntField("character_votes_per_day"); + m_WorldConfig->propertyModerationRequestApprovalCost = worldConfig.getIntField("property_moderation_request_approval_cost"); + m_WorldConfig->propertyModerationRequestReviewCost = worldConfig.getIntField("property_moderation_request_review_cost"); + m_WorldConfig->propertyModRequestsAllowedSpike = worldConfig.getIntField("propertyModRequestsAllowedSpike"); + m_WorldConfig->propertyModRequestsAllowedInterval = worldConfig.getIntField("propertyModRequestsAllowedInterval"); + m_WorldConfig->propertyModRequestsAllowedTotal = worldConfig.getIntField("propertyModRequestsAllowedTotal"); + m_WorldConfig->propertyModRequestsSpikeDuration = worldConfig.getIntField("propertyModRequestsSpikeDuration"); + m_WorldConfig->propertyModRequestsIntervalDuration = worldConfig.getIntField("propertyModRequestsIntervalDuration"); + m_WorldConfig->modelModerateOnCreate = worldConfig.getIntField("modelModerateOnCreate") != 0; + m_WorldConfig->defaultPropertyMaxHeight = worldConfig.getFloatField("defaultPropertyMaxHeight"); + m_WorldConfig->reputationPerVoteCast = worldConfig.getFloatField("reputationPerVoteCast"); + m_WorldConfig->reputationPerVoteReceived = worldConfig.getFloatField("reputationPerVoteReceived"); + m_WorldConfig->showcaseTopModelConsiderationBattles = worldConfig.getIntField("showcaseTopModelConsiderationBattles"); + m_WorldConfig->reputationPerBattlePromotion = worldConfig.getFloatField("reputationPerBattlePromotion"); + m_WorldConfig->coinsLostOnDeathMinTimeout = worldConfig.getFloatField("coins_lost_on_death_min_timeout"); + m_WorldConfig->coinsLostOnDeathMaxTimeout = worldConfig.getFloatField("coins_lost_on_death_max_timeout"); + m_WorldConfig->mailBaseFee = worldConfig.getIntField("mail_base_fee"); + m_WorldConfig->mailPercentAttachmentFee = worldConfig.getFloatField("mail_percent_attachment_fee"); + m_WorldConfig->propertyReputationDelay = worldConfig.getIntField("propertyReputationDelay"); + m_WorldConfig->levelCap = worldConfig.getIntField("LevelCap"); + m_WorldConfig->levelUpBehaviorEffect = worldConfig.getStringField("LevelUpBehaviorEffect"); + m_WorldConfig->characterVersion = worldConfig.getIntField("CharacterVersion"); + m_WorldConfig->levelCapCurrencyConversion = worldConfig.getIntField("LevelCapCurrencyConversion"); + worldConfig.finalize(); + Game::logger->Log("dZoneManager", "Loaded WorldConfig into memory"); +} diff --git a/dZoneManager/dZoneManager.h b/dZoneManager/dZoneManager.h index b2fef1e3..1e08b008 100644 --- a/dZoneManager/dZoneManager.h +++ b/dZoneManager/dZoneManager.h @@ -4,6 +4,8 @@ #include "Spawner.h" #include <map> +class WorldConfig; + class dZoneManager { public: enum class dZoneNotifier { @@ -16,15 +18,13 @@ public: InvalidNotifier }; +private: + /** + * Reads the WorldConfig from the CDClientDatabase into memory + */ + void LoadWorldConfig(); + public: - static dZoneManager* Instance() { - if (!m_Address) { - m_Address = new dZoneManager(); - } - - return m_Address; - } - void Initialize(const LWOZONEID& zoneID); ~dZoneManager(); @@ -33,8 +33,6 @@ public: void NotifyZone(const dZoneNotifier& notifier, const LWOOBJID& objectID); //Notifies the zone of a certain event or command. void AddSpawner(LWOOBJID id, Spawner* spawner); LWOZONEID GetZoneID() const; - uint32_t GetMaxLevel(); - int32_t GetLevelCapCurrencyConversion(); LWOOBJID MakeSpawner(SpawnerInfo info); Spawner* GetSpawner(LWOOBJID id); void RemoveSpawner(LWOOBJID id); @@ -44,28 +42,25 @@ public: Entity* GetZoneControlObject() { return m_ZoneControlObject; } bool GetPlayerLoseCoinOnDeath() { return m_PlayerLoseCoinsOnDeath; } uint32_t GetUniqueMissionIdStartingValue(); + bool CheckIfAccessibleZone(LWOMAPID zoneID); + + // The world config should not be modified by a caller. + const WorldConfig* GetWorldConfig() { + if (!m_WorldConfig) LoadWorldConfig(); + return m_WorldConfig; + }; private: - /** - * The maximum level of the world. - */ - uint32_t m_MaxLevel = 0; - - /** - * The ratio of LEGO Score to currency when the character has hit the max level. - */ - int32_t m_CurrencyConversionRate = 0; - /** * The starting unique mission ID. */ uint32_t m_UniqueMissionIdStart = 0; - static dZoneManager* m_Address; //Singleton - Zone* m_pZone; + Zone* m_pZone = nullptr; LWOZONEID m_ZoneID; bool m_PlayerLoseCoinsOnDeath; //Do players drop coins in this zone when smashed std::map<LWOOBJID, Spawner*> m_Spawners; + WorldConfig* m_WorldConfig = nullptr; - Entity* m_ZoneControlObject; + Entity* m_ZoneControlObject = nullptr; }; diff --git a/docker-compose.yml b/docker-compose.yml index 4b997e53..d8c3c40a 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -12,6 +12,7 @@ services: - DATABASE_USER=${MARIADB_USER:-darkflame} - DATABASE_PASSWORD=${MARIADB_PASSWORD:-darkflame} - EXTERNAL_IP=${EXTERNAL_IP:-darkflame} + - BUILD_VERSION=${BUILD_VERSION:?171022} volumes: - ${CLIENT_PATH:?missing_client_path}:/client - shared_configs:/docker/ diff --git a/docker/Dockerfile b/docker/Dockerfile index 50f5b083..39153110 100644 --- a/docker/Dockerfile +++ b/docker/Dockerfile @@ -1,4 +1,4 @@ -FROM gcc:11 as build +FROM gcc:12 as build WORKDIR /build @@ -6,7 +6,7 @@ RUN --mount=type=cache,id=build-apt-cache,target=/var/cache/apt \ echo "Install build dependencies" && \ apt update && \ apt remove -y libmysqlcppconn7v5 libmysqlcppconn-dev && \ - apt install cmake zlib1g zlib1g-dev unzip -yqq --no-install-recommends && \ + apt install cmake zlib1g zlib1g-dev -yqq --no-install-recommends && \ rm -rf /var/lib/apt/lists/* COPY dAuthServer/ /build/dAuthServer @@ -32,16 +32,14 @@ COPY .clang-* CMake* LICENSE /build/ ARG BUILD_THREADS=1 ARG BUILD_VERSION=171022 -RUN echo "Build server" && \ - mkdir -p cmake_build && \ - cd cmake_build && \ - sed -i -e "s/171022/${BUILD_VERSION}/g" ../CMakeVariables.txt && \ - cmake .. -DCMAKE_BUILD_RPATH_USE_ORIGIN=TRUE && \ - make -j $BUILD_THREADS +RUN echo "Build server" +RUN sed -i -e "s/__maria_db_connector_compile_jobs__=.*/__maria_db_connector_compile_jobs__=${BUILD_THREADS}/g" CMakeVariables.txt +RUN mkdir -p cmake_build +RUN cd cmake_build && \ + cmake .. -DCMAKE_BUILD_RPATH_USE_ORIGIN=TRUE && \ + make -j$BUILD_THREADS -RUN unzip /build/resources/navmeshes.zip -d /build/cmake_build/res/maps - -FROM gcc:11 as runtime +FROM gcc:12 as runtime RUN --mount=type=cache,id=runtime-apt-cache,target=/var/cache/apt \ apt update && \ diff --git a/docker/setup.Dockerfile b/docker/setup.Dockerfile index 2664e2fa..18bb2d06 100644 --- a/docker/setup.Dockerfile +++ b/docker/setup.Dockerfile @@ -1,23 +1,12 @@ -FROM rust:alpine3.14 as LUnpack - -WORKDIR /build_LUnpack - -COPY ./thirdparty/LUnpack . - -RUN apk add musl-dev --no-cache && cargo build --release - FROM python:3.10-alpine3.14 as prep -RUN apk add sqlite bash --no-cache +RUN apk add bash --no-cache WORKDIR /setup # copy needed files from repo COPY resources/ resources/ -COPY migrations/cdserver/ migrations/cdserver -COPY --from=LUnpack /build_LUnpack/target/release/lunpack /usr/local/bin/lunpack -ADD thirdparty/docker-utils/utils/*.py utils/ COPY docker/setup.sh /setup.sh -CMD [ "/setup.sh" ] \ No newline at end of file +CMD [ "/setup.sh" ] diff --git a/docker/setup.sh b/docker/setup.sh index 1a95f4de..1944bbba 100755 --- a/docker/setup.sh +++ b/docker/setup.sh @@ -7,7 +7,7 @@ function update_ini() { FILE="/docker/configs/$1" KEY=$2 NEW_VALUE=$3 - sed -i "/^$KEY=/s/=.*/=$NEW_VALUE/" $FILE + sed -i "s~$2=.*~$2=$3~" $FILE } function update_database_ini_values_for() { @@ -17,9 +17,8 @@ function update_database_ini_values_for() { update_ini $INI_FILE mysql_database $DATABASE update_ini $INI_FILE mysql_username $DATABASE_USER update_ini $INI_FILE mysql_password $DATABASE_PASSWORD - if [[ "$INI_FILE" != "worldconfig.ini" ]]; then - update_ini $INI_FILE external_ip $EXTERNAL_IP - fi + update_ini $INI_FILE client_net_version $BUILD_VERSION + update_ini $INI_FILE external_ip $EXTERNAL_IP } function update_ini_values() { @@ -30,66 +29,13 @@ function update_ini_values() { cp resources/authconfig.ini /docker/configs/ cp resources/chatconfig.ini /docker/configs/ cp resources/worldconfig.ini /docker/configs/ - - update_ini worldconfig.ini chat_server_port $CHAT_SERVER_PORT - update_ini worldconfig.ini max_clients $MAX_CLIENTS + cp resources/sharedconfig.ini /docker/configs/ # always use the internal docker hostname update_ini masterconfig.ini master_ip "darkflame" + update_ini sharedconfig.ini client_location "/client" - update_database_ini_values_for masterconfig.ini - update_database_ini_values_for authconfig.ini - update_database_ini_values_for chatconfig.ini - update_database_ini_values_for worldconfig.ini -} - -function fdb_to_sqlite() { - echo "Run fdb_to_sqlite" - python3 utils/fdb_to_sqlite.py /client/client/res/cdclient.fdb --sqlite_path /client/client/res/CDServer.sqlite - - ( - cd migrations/cdserver - readarray -d '' entries < <(printf '%s\0' *.sql | sort -zV) - for entry in "${entries[@]}"; do - echo "Execute $entry" - sqlite3 /client/client/res/CDServer.sqlite < $entry - done - ) + update_database_ini_values_for sharedconfig.ini } update_ini_values - -if [[ ! -d "/client" ]]; then - echo "Client not found." - echo "Did you forget to mount the client into the \"/client\" directory?" - exit 1 -fi - -if [[ ! -f "/client/extracted" ]]; then - echo "Start client resource extraction" - - touch globs.txt - - echo "client/res/macros/**" >> globs.txt - echo "client/res/BrickModels/**" >> globs.txt - echo "client/res/maps/**" >> globs.txt - echo "*.fdb" >> globs.txt - - lunpack -g ./globs.txt /client/ - - touch /client/extracted -else - echo "Client already extracted. Skip this step..." - echo "If you want to force a re-extract, just delete the file called \"extracted\" in the client directory" -fi - -if [[ ! -f "/client/migrated" ]]; then - echo "Start client db migration" - - fdb_to_sqlite - - touch /client/migrated -else - echo "Client db already migrated. Skip this step..." - echo "If you want to force a re-migrate, just delete the file called \"migrated\" in the client directory" -fi diff --git a/docker/start_server.sh b/docker/start_server.sh index feb61361..2e2e8c28 100755 --- a/docker/start_server.sh +++ b/docker/start_server.sh @@ -1,23 +1,5 @@ #!/bin/bash -function symlink_client_files() { - echo "Creating symlinks for client files" - ln -s /client/client/res/macros/ /app/res/macros - ln -s /client/client/res/BrickModels/ /app/res/BrickModels - ln -s /client/client/res/chatplus_en_us.txt /app/res/chatplus_en_us.txt - ln -s /client/client/res/names/ /app/res/names - ln -s /client/client/res/CDServer.sqlite /app/res/CDServer.sqlite - ln -s /client/client/locale/locale.xml /app/locale/locale.xml - # need to iterate over entries in maps due to maps already being a directory with navmeshes/ in it - ( - cd /client/client/res/maps - readarray -d '' entries < <(printf '%s\0' * | sort -zV) - for entry in "${entries[@]}"; do - ln -s /client/client/res/maps/$entry /app/res/maps/ - done - ) -} - function symlink_config_files() { echo "Creating symlinks for config files" rm /app/*.ini @@ -25,17 +7,11 @@ function symlink_config_files() { ln -s /shared_configs/configs/chatconfig.ini /app/chatconfig.ini ln -s /shared_configs/configs/masterconfig.ini /app/masterconfig.ini ln -s /shared_configs/configs/worldconfig.ini /app/worldconfig.ini + ln -s /shared_configs/configs/sharedconfig.ini /app/sharedconfig.ini } -# check to make sure the setup has completed -while [ ! -f "/client/extracted" ] || [ ! -f "/client/migrated" ]; do - echo "Client setup not finished. Waiting for setup container to complete..." - sleep 5 -done - if [[ ! -f "/app/initialized" ]]; then # setup symlinks for volume files - symlink_client_files symlink_config_files # do not run symlinks more than once touch /app/initialized @@ -46,4 +22,4 @@ fi # start the server echo "Starting MasterServer" ./MasterServer -tail -f /dev/null \ No newline at end of file +tail -f /dev/null diff --git a/docs/Commands.md b/docs/Commands.md index 95ec28f9..c997c3c4 100644 --- a/docs/Commands.md +++ b/docs/Commands.md @@ -1,6 +1,5 @@ # In-game commands - -Here is a summary of the commands available in-game. All commands are prefixed by `/` and typed in the in-game chat window. Some commands requires admin privileges. Operands within `<>` are required, operands within `()` are not. For the full list of in-game commands, please checkout [the source file](../dGame/dUtilities/SlashCommandHandler.cpp). +* All commands are prefixed by `/` and typed in the in-game chat window. Some commands require elevated gmlevel privileges. Operands within `<>` are required, operands within `()` are not. ## General Commands @@ -14,8 +13,6 @@ Here is a summary of the commands available in-game. All commands are prefixed b |pvp|`/pvp`|Toggle your PVP flag.|| |resurrect|`/resurrect`|Resurrects the player.|| |requestmailcount|`/requestmailcount`|Sends notification with number of unread messages in the player's mailbox.|| -|skip-ags|`/skip-ags`|Skips the Avant Gardens Survival minigame mission, "Impress the Sentinel Faction".|| -|skip-sg|`/skip-sg`|Skips the Shooting Gallery minigame mission, "Monarch of the Sea".|| |who|`/who`|Displays in chat all players on the instance.|| ## Moderation Commands @@ -39,8 +36,6 @@ Here is a summary of the commands available in-game. All commands are prefixed b |Command|Usage|Description|Admin Level Requirement| |--- |--- |--- |--- | |announce|`/announce`|Sends a announcement. `/setanntitle` and `/setannmsg` must be called first to configure the announcement.|8| -|config-set|`/config-set <key> <value>`|Set configuration item.|8| -|config-get|`/config-get <key>`|Get current value of a configuration item.|8| |kill|`/kill <username>`|Smashes the character whom the given user is playing.|8| |metrics|`/metrics`|Prints some information about the server's performance.|8| |setannmsg|`/setannmsg <title>`|Sets the message of an announcement.|8| @@ -53,6 +48,7 @@ These commands are primarily for development and testing. The usage of many of t |Command|Usage|Description|Admin Level Requirement| |--- |--- |--- |--- | +|togglenameplate|`/togglenameplate`|Turns the nameplate above your head that is visible to other players off and on.|8 or if `allow_nameplate_off` is set to exactly `1` in the settings| |fix-stats|`/fix-stats`|Resets skills, buffs, and destroyables.|| |join|`/join <password>`|Joins a private zone with given password.|| |leave-zone|`/leave-zone`|If you are in an instanced zone, transfers you to the closest main world. For example, if you are in an instance of Avant Gardens Survival or the Spider Queen Battle, you are sent to Avant Gardens. If you are in the Battle of Nimbus Station, you are sent to Nimbus Station.|| @@ -73,6 +69,7 @@ These commands are primarily for development and testing. The usage of many of t |createprivate|`/createprivate <zone id> <clone id> <password>`|Creates a private zone with password.|8| |debugui|`/debugui`|Toggle Debug UI.|8| |dismount|`/dismount`|Dismounts you from the vehicle or mount.|8| +|reloadconfig|`/reloadconfig`|Reloads the server with the new config values.|8| |force-save|`/force-save`|While saving to database usually happens on regular intervals and when you disconnect from the server, this command saves your player's data to the database.|8| |freecam|`/freecam`|Toggles freecam mode.|8| |freemoney|`/freemoney <coins>`|Gives coins.|8| @@ -82,7 +79,7 @@ These commands are primarily for development and testing. The usage of many of t |inspect|`/inspect <component> (-m <waypoint> \| -a <animation> \| -s \| -p \| -f (faction) \| -t)`|Finds the closest entity with the given component or LDF variable (ignoring players and racing cars), printing its ID, distance from the player, and whether it is sleeping, as well as the the IDs of all components the entity has. See [Detailed `/inspect` Usage](#detailed-inspect-usage) below.|8| |list-spawns|`/list-spawns`|Lists all the character spawn points in the zone. Additionally, this command will display the current scene that plays when the character lands in the next zone, if there is one.|8| |locrow|`/locrow`|Prints the your current position and rotation information to the console.|8| -|lookup|`/lookup <query>`|Searches through the Objects table in the client SQLite database for items whose display name, name, or description contains the query.|8| +|lookup|`/lookup <query>`|Searches through the Objects table in the client SQLite database for items whose display name, name, or description contains the query. Query can be multiple words delimited by spaces.|8| |playanimation|`/playanimation <id>`|Plays animation with given ID. Alias: `/playanim`.|8| |playeffect|`/playeffect <effect id> <effect type> <effect name>`|Plays an effect.|8| |playlvlfx|`/playlvlfx`|Plays the level up animation on your character.|8| @@ -96,7 +93,7 @@ These commands are primarily for development and testing. The usage of many of t |setcontrolscheme|`/setcontrolscheme <scheme number>`|Sets the character control scheme to the specified number.|8| |setcurrency|`/setcurrency <coins>`|Sets your coins.|8| |setflag|`/setflag (value) <flag id>`|Sets the given inventory or health flag to the given value, where value can be one of "on" or "off". If no value is given, by default this adds the flag to your character (equivalent of calling `/setflag on <flag id>`).|8| -|setinventorysize|`/setinventorysize <size>`|Sets your inventory size to the given size. Alias: `/setinvsize`|8| +|setinventorysize|`/setinventorysize <size> (inventory)`|Sets your inventory size to the given size. If `inventory` is provided, the number or string will be used to set that inventory to the requested size. Alias: `/setinvsize`|8| |setuistate|`/setuistate <ui state>`|Changes UI state.|8| |spawn|`/spawn <id>`|Spawns an object at your location by id.|8| |speedboost|`/speedboost <amount>`|Sets the speed multiplier to the given amount. `/speedboost 1.5` will set the speed multiplier to 1.5x the normal speed.|8| @@ -131,13 +128,13 @@ There are 9 Game master levels |Level|Variable Name|Description| |--- |--- |--- | -|0|GAME_MASTER_LEVEL_CIVILIAN|Normal player| -|1|GAME_MASTER_LEVEL_FORUM_MODERATOR|Forum moderator. No permissions on live servers.| -|2|GAME_MASTER_LEVEL_JUNIOR_MODERATOR|Can kick/mute and pull chat logs| -|3|GAME_MASTER_LEVEL_MODERATOR|Can return lost items| -|4|GAME_MASTER_LEVEL_SENIOR_MODERATOR|Can ban| -|5|GAME_MASTER_LEVEL_LEAD_MODERATOR|Can approve properties| -|6|GAME_MASTER_LEVEL_JUNIOR_DEVELOPER|Junior developer & future content team. Civilan on live.| -|7|GAME_MASTER_LEVEL_INACTIVE_DEVELOPER|Inactive developer, limited permissions.| -|8|GAME_MASTER_LEVEL_DEVELOPER|Active developer, full permissions on live.| -|9|GAME_MASTER_LEVEL_OPERATOR|Can shutdown server for restarts & updates.| +|0|CIVILIAN|Normal player| +|1|FORUM_MODERATOR|Forum moderator. No permissions on live servers.| +|2|JUNIOR_MODERATOR|Can kick/mute and pull chat logs| +|3|MODERATOR|Can return lost items| +|4|SENIOR_MODERATOR|Can ban| +|5|LEAD_MODERATOR|Can approve properties| +|6|JUNIOR_DEVELOPER|Junior developer & future content team. Civilan on live.| +|7|INACTIVE_DEVELOPER|Inactive developer, limited permissions.| +|8|DEVELOPER|Active developer, full permissions on live.| +|9|OPERATOR|Can shutdown server for restarts & updates.| diff --git a/migrations/cdserver/0_nt_footrace.sql b/migrations/cdserver/0_nt_footrace.sql index fd37599e..0a40cfef 100644 --- a/migrations/cdserver/0_nt_footrace.sql +++ b/migrations/cdserver/0_nt_footrace.sql @@ -1,6 +1,2 @@ -BEGIN TRANSACTION; - UPDATE ComponentsRegistry SET component_id = 1901 WHERE id = 12916 AND component_type = 39; INSERT INTO ActivityRewards (objectTemplate, ActivityRewardIndex, activityRating, LootMatrixIndex, CurrencyIndex, ChallengeRating, description) VALUES (1901, 166, -1, 598, 1, 4, 'NT Foot Race'); - -COMMIT; diff --git a/migrations/cdserver/5_serratorizer_chargeup_fix.sql b/migrations/cdserver/5_serratorizer_chargeup_fix.sql new file mode 100644 index 00000000..a61da2c2 --- /dev/null +++ b/migrations/cdserver/5_serratorizer_chargeup_fix.sql @@ -0,0 +1 @@ +UPDATE behaviorParameter SET value = 20 WHERE behaviorID = 21001 AND parameterID = "value 2"; diff --git a/migrations/cdserver/6_ninja_sensei.sql b/migrations/cdserver/6_ninja_sensei.sql new file mode 100644 index 00000000..f3828b07 --- /dev/null +++ b/migrations/cdserver/6_ninja_sensei.sql @@ -0,0 +1,2 @@ +INSERT INTO ScriptComponent (id, script_name, client_script_name) VALUES (228, 'scripts\ai\FV\L_ACT_NINJA_SENSEI.lua', null); +UPDATE ComponentsRegistry SET component_id = 228 WHERE id = 2489 AND component_type = 5; diff --git a/migrations/dlu/5_brick_model_sd0.sql b/migrations/dlu/5_brick_model_sd0.sql new file mode 100644 index 00000000..895f8b34 --- /dev/null +++ b/migrations/dlu/5_brick_model_sd0.sql @@ -0,0 +1 @@ +# This file is here as a mock. The real migration is located in BrickByBrickFix.cpp diff --git a/migrations/dlu/6_property_behaviors.sql b/migrations/dlu/6_property_behaviors.sql new file mode 100644 index 00000000..b858db67 --- /dev/null +++ b/migrations/dlu/6_property_behaviors.sql @@ -0,0 +1,11 @@ +ALTER TABLE properties_contents + ADD COLUMN model_name TEXT NOT NULL DEFAULT "", + ADD COLUMN model_description TEXT NOT NULL DEFAULT "", + ADD COLUMN behavior_1 INT NOT NULL DEFAULT 0, + ADD COLUMN behavior_2 INT NOT NULL DEFAULT 0, + ADD COLUMN behavior_3 INT NOT NULL DEFAULT 0, + ADD COLUMN behavior_4 INT NOT NULL DEFAULT 0, + ADD COLUMN behavior_5 INT NOT NULL DEFAULT 0; + +UPDATE properties_contents SET model_name = CONCAT("Objects_", lot, "_name") WHERE model_name = ""; +CREATE TABLE IF NOT EXISTS behaviors (id INT NOT NULL, behavior_info TEXT NOT NULL); diff --git a/migrations/dlu/7_make_play_key_id_nullable.sql b/migrations/dlu/7_make_play_key_id_nullable.sql new file mode 100644 index 00000000..11239967 --- /dev/null +++ b/migrations/dlu/7_make_play_key_id_nullable.sql @@ -0,0 +1 @@ +ALTER TABLE accounts MODIFY play_key_id INT DEFAULT 0; diff --git a/migrations/dlu/8_foreign_play_key.sql b/migrations/dlu/8_foreign_play_key.sql new file mode 100644 index 00000000..6f171bb5 --- /dev/null +++ b/migrations/dlu/8_foreign_play_key.sql @@ -0,0 +1 @@ +ALTER TABLE accounts MODIFY play_key_id INT DEFAULT NULL; diff --git a/migrations/dlu/9_Update_Leaderboard_Storage.sql b/migrations/dlu/9_Update_Leaderboard_Storage.sql new file mode 100644 index 00000000..c87e3501 --- /dev/null +++ b/migrations/dlu/9_Update_Leaderboard_Storage.sql @@ -0,0 +1,18 @@ +ALTER TABLE leaderboard + ADD COLUMN tertiaryScore FLOAT NOT NULL DEFAULT 0, + ADD COLUMN numWins INT NOT NULL DEFAULT 0, + ADD COLUMN timesPlayed INT NOT NULL DEFAULT 1, + MODIFY time INT NOT NULL DEFAULT 0; + +/* Can only ALTER one column at a time... */ +ALTER TABLE leaderboard CHANGE score primaryScore FLOAT NOT NULL DEFAULT 0; +ALTER TABLE leaderboard CHANGE time secondaryScore FLOAT NOT NULL DEFAULT 0 AFTER primaryScore; + +/* A bit messy, but better than going through a bunch of code fixes all to be run once. */ +UPDATE leaderboard SET + primaryScore = secondaryScore, + secondaryScore = 0 WHERE game_id IN (1, 44, 46, 47, 48, 49, 53, 103, 104, 108, 1901); + +/* Do this last so we dont update entry times erroneously */ +ALTER TABLE leaderboard + CHANGE last_played last_played TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP() ON UPDATE CURRENT_TIMESTAMP(); diff --git a/resources/authconfig.ini b/resources/authconfig.ini index 40ca146e..ec414bc0 100644 --- a/resources/authconfig.ini +++ b/resources/authconfig.ini @@ -1,27 +1,6 @@ -# MySQL connection info: -mysql_host= -mysql_database= -mysql_username= -mysql_password= - -# The public facing IP address. Can be 'localhost' for locally hosted servers -external_ip=localhost - # Port number. The client has the authserver port hardcoded to 1001 port=1001 -# Where to put crashlogs -dump_folder= - -# How many clients can be connected to the server at once -max_clients=999 - -# 0 or 1, should log to console -log_to_console=1 - -# 0 or 1, should log debug (developer only) statements to console for debugging, not needed for normal operation -log_debug_statements=0 - # 0 or 1, should ignore playkeys # If 1 everyone with an account will be able to login, regardless of if they have a key or not dont_use_keys=0 diff --git a/resources/chatconfig.ini b/resources/chatconfig.ini index f30fb8f9..26b26cc7 100644 --- a/resources/chatconfig.ini +++ b/resources/chatconfig.ini @@ -1,26 +1,2 @@ -# MySQL connection info: -mysql_host= -mysql_database= -mysql_username= -mysql_password= - -# The public facing IP address. Can be 'localhost' for locally hosted servers -external_ip=localhost - # Port number port=2005 - -# Where to put crashlogs -dump_folder= - -# How many clients can be connected to the server at once -max_clients=999 - -# 0 or 1, should log to console -log_to_console=1 - -# 0 or 1, should log debug (developer only) statements to console for debugging, not needed for normal operation -log_debug_statements=0 - -# 0 or 1, should not compile chat hash map to file -dont_generate_dcf=0 diff --git a/resources/masterconfig.ini b/resources/masterconfig.ini index c2d884a5..4864d8cb 100644 --- a/resources/masterconfig.ini +++ b/resources/masterconfig.ini @@ -1,12 +1,3 @@ -# MySQL connection info: -mysql_host= -mysql_database= -mysql_username= -mysql_password= - -# The public facing IP address. Can be 'localhost' for locally hosted servers -external_ip=localhost - # The internal ip of the master server master_ip=localhost @@ -26,17 +17,5 @@ use_sudo_chat=0 # Use sudo when launching world servers use_sudo_world=0 -# Where to put crashlogs -dump_folder= - -# How many clients can be connected to the server at once -max_clients=999 - -# 0 or 1, should log to console -log_to_console=1 - -# 0 or 1, should log debug (developer only) statements to console for debugging, not needed for normal operation -log_debug_statements=0 - # 0 or 1, should autostart auth, chat, and char servers prestart_servers=1 diff --git a/resources/sharedconfig.ini b/resources/sharedconfig.ini new file mode 100644 index 00000000..ad587b18 --- /dev/null +++ b/resources/sharedconfig.ini @@ -0,0 +1,45 @@ +# MySQL connection info: +mysql_host= +mysql_database= +mysql_username= +mysql_password= + +# 0 or 1, should log to console +log_to_console=1 + +# 0 or 1, should log debug (developer only) statements to console for debugging, not needed for normal operation +log_debug_statements=0 + +# The public facing IP address. Can be 'localhost' for locally hosted servers +external_ip=localhost + +# 0 or 1, should not compile chat hash map to file +dont_generate_dcf=0 + +# How many clients can be connected to the server at once +max_clients=999 + +# Where to put crashlogs +dump_folder= + +# The location of the client +# Either the folder with /res or with /client and /versions +client_location= + +# The maximum outgoing bandwidth in bits. If your clients are having +# issues with enemies taking a while to catch up to them, increse this value. +maximum_outgoing_bandwidth=80000 + +# The Maximum Translation Unit (MTU) size for packets. If players are +# getting stuck at 55% on the loading screen, lower this number to +# reduce the chances of packet loss. This value only has an effect +# from 512 <= maximum_mtu_size <= 1492 so make sure to keep this +# value within that range. +maximum_mtu_size=1228 + +# The client network version to allow to connect to this server. +# Client's that do not match this value will be kicked from the server. +# If you are using a Darkflame Universe client, set this value to 171023. +# This cannot just be any arbitrary number. This has to match the same value that is in your client. +# If you do not know what this value is, default it to 171022. +client_net_version=171022 diff --git a/resources/worldconfig.ini b/resources/worldconfig.ini index 931da28c..b05614b4 100644 --- a/resources/worldconfig.ini +++ b/resources/worldconfig.ini @@ -1,9 +1,3 @@ -# MySQL connection info: -mysql_host= -mysql_database= -mysql_username= -mysql_password= - # URL to the code repository for the hosted server # If you fork this repository and/or make changes to the code, reflect that here to comply with AGPLv3 source=https://github.com/DarkflameUniverse/DarkflameServer @@ -11,21 +5,6 @@ source=https://github.com/DarkflameUniverse/DarkflameServer # Port to the chat server, same as in chatconfig.ini chat_server_port=2005 -# Where to put crashlogs -dump_folder= - -# How many clients can be connected to the server at once -max_clients=999 - -# 0 or 1, should log to console -log_to_console=1 - -# 0 or 1, should log debug (developer only) statements to console for debugging, not needed for normal operation -log_debug_statements=0 - -# 0 or 1, should not compile chat hash map to file -dont_generate_dcf=0 - # 0 or 1, should disable chat disable_chat=0 @@ -63,4 +42,22 @@ pets_take_imagination=1 # If you would like to increase the maximum number of best friends a player can have on the server # Change the value below to what you would like this to be (5 is live accurate) -max_number_of_best_friends=5 \ No newline at end of file +max_number_of_best_friends=5 + +# Disables loot drops +disable_drops=0 + +# Hardcore mode settings +hardcore_mode=0 + +# Drop your entire inventory on death + coins (drops on the ground, so can be retrieved) +hardcore_dropinventory_on_death=1 + +# Enemies drop their max hp * this value. 0 will effectively disable it. +hardcore_uscore_enemies_multiplier=2 + +# Percentage of u-score to lose on player death +hardcore_lose_uscore_on_death_percent=10 + +# Allow civilian players the ability to turn the nameplate above their head off. Must be exactly 1 to be enabled for civilians. +allow_nameplate_off=0 diff --git a/tests/AMFDeserializeTests.cpp b/tests/AMFDeserializeTests.cpp deleted file mode 100644 index 03941a85..00000000 --- a/tests/AMFDeserializeTests.cpp +++ /dev/null @@ -1,409 +0,0 @@ -#include <chrono> -#include <fstream> -#include <iostream> -#include <memory> - -#include "AMFDeserialize.h" -#include "AMFFormat.h" -#include "CommonCxxTests.h" - -std::unique_ptr<AMFValue> ReadFromBitStream(RakNet::BitStream* bitStream) { - AMFDeserialize deserializer; - std::unique_ptr<AMFValue> returnValue(deserializer.Read(bitStream)); - return returnValue; -} - -int ReadAMFUndefinedFromBitStream() { - CBITSTREAM; - bitStream.Write<uint8_t>(0x00); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFUndefined); - return 0; -} - -int ReadAMFNullFromBitStream() { - CBITSTREAM; - bitStream.Write<uint8_t>(0x01); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFNull); - return 0; -} - -int ReadAMFFalseFromBitStream() { - CBITSTREAM; - bitStream.Write<uint8_t>(0x02); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFFalse); - return 0; -} - -int ReadAMFTrueFromBitStream() { - CBITSTREAM; - bitStream.Write<uint8_t>(0x03); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFTrue); - return 0; -} - -int ReadAMFIntegerFromBitStream() { - CBITSTREAM; - { - bitStream.Write<uint8_t>(0x04); - // 127 == 01111111 - bitStream.Write<uint8_t>(127); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFInteger); - // Check that the max value of a byte can be read correctly - ASSERT_EQ(static_cast<AMFIntegerValue*>(res.get())->GetIntegerValue(), 127); - } - bitStream.Reset(); - { - bitStream.Write<uint8_t>(0x04); - bitStream.Write<uint32_t>(UINT32_MAX); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFInteger); - // Check that we can read the maximum value correctly - ASSERT_EQ(static_cast<AMFIntegerValue*>(res.get())->GetIntegerValue(), 536870911); - } - bitStream.Reset(); - { - bitStream.Write<uint8_t>(0x04); - // 131 == 10000011 - bitStream.Write<uint8_t>(131); - // 255 == 11111111 - bitStream.Write<uint8_t>(255); - // 127 == 01111111 - bitStream.Write<uint8_t>(127); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFInteger); - // Check that short max can be read correctly - ASSERT_EQ(static_cast<AMFIntegerValue*>(res.get())->GetIntegerValue(), UINT16_MAX); - } - bitStream.Reset(); - { - bitStream.Write<uint8_t>(0x04); - // 255 == 11111111 - bitStream.Write<uint8_t>(255); - // 127 == 01111111 - bitStream.Write<uint8_t>(127); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFInteger); - // Check that 2 byte max can be read correctly - ASSERT_EQ(static_cast<AMFIntegerValue*>(res.get())->GetIntegerValue(), 16383); - } - return 0; -} - -int ReadAMFDoubleFromBitStream() { - CBITSTREAM; - bitStream.Write<uint8_t>(0x05); - bitStream.Write<double>(25346.4f); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFDouble); - ASSERT_EQ(static_cast<AMFDoubleValue*>(res.get())->GetDoubleValue(), 25346.4f); - return 0; -} - -int ReadAMFStringFromBitStream() { - CBITSTREAM; - bitStream.Write<uint8_t>(0x06); - bitStream.Write<uint8_t>(0x0F); - std::string toWrite = "stateID"; - for (auto e : toWrite) bitStream.Write<char>(e); - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFString); - ASSERT_EQ(static_cast<AMFStringValue*>(res.get())->GetStringValue(), "stateID"); - return 0; -} - -int ReadAMFArrayFromBitStream() { - CBITSTREAM; - // Test empty AMFArray - bitStream.Write<uint8_t>(0x09); - bitStream.Write<uint8_t>(0x01); - bitStream.Write<uint8_t>(0x01); - { - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFArray); - ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetAssociativeMap().size(), 0); - ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetDenseArray().size(), 0); - } - bitStream.Reset(); - // Test a key'd value and dense value - bitStream.Write<uint8_t>(0x09); - bitStream.Write<uint8_t>(0x03); - bitStream.Write<uint8_t>(0x15); - for (auto e : "BehaviorID") if (e != '\0') bitStream.Write<char>(e); - bitStream.Write<uint8_t>(0x06); - bitStream.Write<uint8_t>(0x0B); - for (auto e : "10447") if (e != '\0') bitStream.Write<char>(e); - bitStream.Write<uint8_t>(0x01); - bitStream.Write<uint8_t>(0x06); - bitStream.Write<uint8_t>(0x0B); - for (auto e : "10447") if (e != '\0') bitStream.Write<char>(e); - { - std::unique_ptr<AMFValue> res(ReadFromBitStream(&bitStream)); - ASSERT_EQ(res->GetValueType(), AMFValueType::AMFArray); - ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetAssociativeMap().size(), 1); - ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetDenseArray().size(), 1); - ASSERT_EQ(static_cast<AMFStringValue*>(static_cast<AMFArrayValue*>(res.get())->FindValue("BehaviorID"))->GetStringValue(), "10447"); - ASSERT_EQ(static_cast<AMFStringValue*>(static_cast<AMFArrayValue*>(res.get())->GetDenseArray()[0])->GetStringValue(), "10447"); - } - // Test a dense array - return 0; -} - -/** - * This test checks that if we recieve an unimplemented AMFValueType - * we correctly throw an error and can actch it. - */ -int TestUnimplementedAMFValues() { - std::vector<AMFValueType> unimplementedValues = { - AMFValueType::AMFXMLDoc, - AMFValueType::AMFDate, - AMFValueType::AMFObject, - AMFValueType::AMFXML, - AMFValueType::AMFByteArray, - AMFValueType::AMFVectorInt, - AMFValueType::AMFVectorUInt, - AMFValueType::AMFVectorDouble, - AMFValueType::AMFVectorObject, - AMFValueType::AMFDictionary - }; - // Run unimplemented tests to check that errors are thrown if - // unimplemented AMF values are attempted to be parsed. - std::ifstream fileStream; - fileStream.open("AMFBitStreamUnimplementedTest.bin", std::ios::binary); - - // Read a test BitStream from a file - std::vector<char> baseBitStream; - char byte = 0; - while (fileStream.get(byte)) { - baseBitStream.push_back(byte); - } - - fileStream.close(); - - for (auto amfValueType : unimplementedValues) { - RakNet::BitStream testBitStream; - for (auto element : baseBitStream) { - testBitStream.Write(element); - } - testBitStream.Write(amfValueType); - bool caughtException = false; - try { - ReadFromBitStream(&testBitStream); - } catch (AMFValueType unimplementedValueType) { - caughtException = true; - } - std::cout << "Testing unimplemented value " << amfValueType << " Did we catch an exception: " << (caughtException ? "YES" : "NO") << std::endl; - ASSERT_EQ(caughtException, true); - } - return 0; -} - -int TestLiveCapture() { - std::ifstream testFileStream; - testFileStream.open("AMFBitStreamTest.bin", std::ios::binary); - - // Read a test BitStream from a file - RakNet::BitStream testBitStream; - char byte = 0; - while (testFileStream.get(byte)) { - testBitStream.Write<char>(byte); - } - - testFileStream.close(); - - auto resultFromFn = ReadFromBitStream(&testBitStream); - auto result = static_cast<AMFArrayValue*>(resultFromFn.get()); - // Test the outermost array - - ASSERT_EQ(dynamic_cast<AMFStringValue*>(result->FindValue("BehaviorID"))->GetStringValue(), "10447"); - ASSERT_EQ(dynamic_cast<AMFStringValue*>(result->FindValue("objectID"))->GetStringValue(), "288300744895913279") - - // Test the execution state array - auto executionState = dynamic_cast<AMFArrayValue*>(result->FindValue("executionState")); - ASSERT_NE(executionState, nullptr); - - auto strips = dynamic_cast<AMFArrayValue*>(executionState->FindValue("strips"))->GetDenseArray(); - - ASSERT_EQ(strips.size(), 1); - - auto stripsPosition0 = dynamic_cast<AMFArrayValue*>(strips[0]); - - auto actionIndex = dynamic_cast<AMFDoubleValue*>(stripsPosition0->FindValue("actionIndex")); - - ASSERT_EQ(actionIndex->GetDoubleValue(), 0.0f); - - auto stripIDExecution = dynamic_cast<AMFDoubleValue*>(stripsPosition0->FindValue("id")); - - ASSERT_EQ(stripIDExecution->GetDoubleValue(), 0.0f); - - auto stateIDExecution = dynamic_cast<AMFDoubleValue*>(executionState->FindValue("stateID")); - - ASSERT_EQ(stateIDExecution->GetDoubleValue(), 0.0f); - - auto states = dynamic_cast<AMFArrayValue*>(result->FindValue("states"))->GetDenseArray(); - - ASSERT_EQ(states.size(), 1); - - auto firstState = dynamic_cast<AMFArrayValue*>(states[0]); - - auto stateID = dynamic_cast<AMFDoubleValue*>(firstState->FindValue("id")); - - ASSERT_EQ(stateID->GetDoubleValue(), 0.0f); - - auto stripsInState = dynamic_cast<AMFArrayValue*>(firstState->FindValue("strips"))->GetDenseArray(); - - ASSERT_EQ(stripsInState.size(), 1); - - auto firstStrip = dynamic_cast<AMFArrayValue*>(stripsInState[0]); - - auto actionsInFirstStrip = dynamic_cast<AMFArrayValue*>(firstStrip->FindValue("actions"))->GetDenseArray(); - - ASSERT_EQ(actionsInFirstStrip.size(), 3); - - auto actionID = dynamic_cast<AMFDoubleValue*>(firstStrip->FindValue("id")); - - ASSERT_EQ(actionID->GetDoubleValue(), 0.0f) - - auto uiArray = dynamic_cast<AMFArrayValue*>(firstStrip->FindValue("ui")); - - auto xPos = dynamic_cast<AMFDoubleValue*>(uiArray->FindValue("x")); - auto yPos = dynamic_cast<AMFDoubleValue*>(uiArray->FindValue("y")); - - ASSERT_EQ(xPos->GetDoubleValue(), 103.0f); - ASSERT_EQ(yPos->GetDoubleValue(), 82.0f); - - auto stripID = dynamic_cast<AMFDoubleValue*>(firstStrip->FindValue("id")); - - ASSERT_EQ(stripID->GetDoubleValue(), 0.0f) - - auto firstAction = dynamic_cast<AMFArrayValue*>(actionsInFirstStrip[0]); - - auto firstType = dynamic_cast<AMFStringValue*>(firstAction->FindValue("Type")); - - ASSERT_EQ(firstType->GetStringValue(), "OnInteract"); - - auto firstCallback = dynamic_cast<AMFStringValue*>(firstAction->FindValue("__callbackID__")); - - ASSERT_EQ(firstCallback->GetStringValue(), ""); - - auto secondAction = dynamic_cast<AMFArrayValue*>(actionsInFirstStrip[1]); - - auto secondType = dynamic_cast<AMFStringValue*>(secondAction->FindValue("Type")); - - ASSERT_EQ(secondType->GetStringValue(), "FlyUp"); - - auto secondCallback = dynamic_cast<AMFStringValue*>(secondAction->FindValue("__callbackID__")); - - ASSERT_EQ(secondCallback->GetStringValue(), ""); - - auto secondDistance = dynamic_cast<AMFDoubleValue*>(secondAction->FindValue("Distance")); - - ASSERT_EQ(secondDistance->GetDoubleValue(), 25.0f); - - auto thirdAction = dynamic_cast<AMFArrayValue*>(actionsInFirstStrip[2]); - - auto thirdType = dynamic_cast<AMFStringValue*>(thirdAction->FindValue("Type")); - - ASSERT_EQ(thirdType->GetStringValue(), "FlyDown"); - - auto thirdCallback = dynamic_cast<AMFStringValue*>(thirdAction->FindValue("__callbackID__")); - - ASSERT_EQ(thirdCallback->GetStringValue(), ""); - - auto thirdDistance = dynamic_cast<AMFDoubleValue*>(thirdAction->FindValue("Distance")); - - ASSERT_EQ(thirdDistance->GetDoubleValue(), 25.0f); - - return 0; -} - -int TestNullStream() { - auto result = ReadFromBitStream(nullptr); - ASSERT_EQ(result.get(), nullptr); - return 0; -} - -int AMFDeserializeTests(int argc, char** const argv) { - std::cout << "Checking that using a null bitstream doesnt cause exception" << std::endl; - if (TestNullStream()) return 1; - std::cout << "passed nullptr test, checking basic tests" << std::endl; - if (ReadAMFUndefinedFromBitStream() != 0) return 1; - if (ReadAMFNullFromBitStream() != 0) return 1; - if (ReadAMFFalseFromBitStream() != 0) return 1; - if (ReadAMFTrueFromBitStream() != 0) return 1; - if (ReadAMFIntegerFromBitStream() != 0) return 1; - if (ReadAMFDoubleFromBitStream() != 0) return 1; - if (ReadAMFStringFromBitStream() != 0) return 1; - if (ReadAMFArrayFromBitStream() != 0) return 1; - std::cout << "Passed basic test, checking live capture" << std::endl; - if (TestLiveCapture() != 0) return 1; - std::cout << "Passed live capture, checking unimplemented amf values" << std::endl; - if (TestUnimplementedAMFValues() != 0) return 1; - std::cout << "Passed all tests." << std::endl; - return 0; -} - -/** - * Below is the AMF that is in the AMFBitStreamTest.bin file that we are reading in - * from a bitstream to test. -args: amf3! -{ - "objectID": "288300744895913279", - "BehaviorID": "10447", - "executionState": amf3! - { - "strips": amf3! - [ - amf3! - { - "actionIndex": 0.0, - "id": 0.0, - }, - ], - "stateID": 0.0, - }, - "states": amf3! - [ - amf3! - { - "id": 0.0, - "strips": amf3! - [ - amf3! - { - "actions": amf3! - [ - amf3! - { - "Type": "OnInteract", - "__callbackID__": "", - }, - amf3! - { - "Distance": 25.0, - "Type": "FlyUp", - "__callbackID__": "", - }, - amf3! - { - "Distance": 25.0, - "Type": "FlyDown", - "__callbackID__": "", - }, - ], - "id": 0.0, - "ui": amf3! - { - "x": 103.0, - "y": 82.0, - }, - }, - ], - }, - ], -} - */ diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt index fb1ed5ac..9ba75a2f 100644 --- a/tests/CMakeLists.txt +++ b/tests/CMakeLists.txt @@ -1,34 +1,21 @@ -# create the testing file and list of tests -create_test_sourcelist (Tests - CommonCxxTests.cpp - AMFDeserializeTests.cpp - TestNiPoint3.cpp - TestLDFFormat.cpp - TestEncoding.cpp +message (STATUS "Testing is enabled. Fetching gtest...") +enable_testing() + +include(FetchContent) +FetchContent_Declare( + googletest + GIT_REPOSITORY https://github.com/google/googletest.git + GIT_TAG release-1.12.1 ) -# add the executable -add_executable (CommonCxxTests ${Tests}) -target_link_libraries(CommonCxxTests ${COMMON_LIBRARIES}) +# For Windows: Prevent overriding the parent project's compiler/linker settings +set(gtest_force_shared_crt ON CACHE BOOL "" FORCE) -# remove the test driver source file -set (TestsToRun ${Tests}) -remove (TestsToRun CommonCxxTests.cpp) +FetchContent_MakeAvailable(GoogleTest) +include(GoogleTest) -# Copy test files to testing directory -configure_file( - ${CMAKE_SOURCE_DIR}/tests/TestBitStreams/AMFBitStreamTest.bin ${PROJECT_BINARY_DIR}/tests/AMFBitStreamTest.bin - COPYONLY -) +message(STATUS "gtest fetched and is now ready.") -configure_file( - ${CMAKE_SOURCE_DIR}/tests/TestBitStreams/AMFBitStreamUnimplementedTest.bin ${PROJECT_BINARY_DIR}/tests/AMFBitStreamUnimplementedTest.bin - COPYONLY -) - -# Add all the ADD_TEST for each test -foreach (test ${TestsToRun}) - get_filename_component (TName ${test} NAME_WE) - add_test (NAME ${TName} COMMAND CommonCxxTests ${TName}) - set_property(TEST ${TName} PROPERTY ENVIRONMENT CTEST_OUTPUT_ON_FAILURE=1) -endforeach () +# Add the subdirectories +add_subdirectory(dCommonTests) +add_subdirectory(dGameTests) diff --git a/tests/CommonCxxTests.cpp b/tests/CommonCxxTests.cpp deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/CommonCxxTests.h b/tests/CommonCxxTests.h deleted file mode 100644 index f1894927..00000000 --- a/tests/CommonCxxTests.h +++ /dev/null @@ -1,4 +0,0 @@ -#include <cstdio> - -#define ASSERT_EQ(a,b) { if (!(a == b)) { printf("Failed assertion: " #a " == " #b " \n in %s:%d\n", __FILE__, __LINE__); return 1; }} -#define ASSERT_NE(a,b) { if (!(a != b)) { printf("Failed assertion: " #a " != " #b " \n in %s:%d\n", __FILE__, __LINE__); return 1; }} diff --git a/tests/TestEncoding.cpp b/tests/TestEncoding.cpp deleted file mode 100644 index c23e2db6..00000000 --- a/tests/TestEncoding.cpp +++ /dev/null @@ -1,52 +0,0 @@ -#include <stdexcept> -#include <string> - -#include "GeneralUtils.h" -#include "CommonCxxTests.h" - -int TestEncoding(int argc, char** const argv) { - std::string x = "Hello World!"; - std::string_view v(x); - - uint32_t out; - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 'H'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 'e'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 'l'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 'l'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 'o'); - ASSERT_EQ(GeneralUtils::_NextUTF8Char(v, out), true); - - x = u8"Frühling"; - v = x; - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'F'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'r'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'ü'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'h'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'l'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'i'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'n'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'g'); - ASSERT_EQ(GeneralUtils::_NextUTF8Char(v, out), false); - - x = "中文字"; - v = x; - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'中'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'文'); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, U'字'); - ASSERT_EQ(GeneralUtils::_NextUTF8Char(v, out), false); - - x = "👨‍⚖️"; - v = x; - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 0x1F468); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 0x200D); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 0x2696); - GeneralUtils::_NextUTF8Char(v, out); ASSERT_EQ(out, 0xFE0F); - ASSERT_EQ(GeneralUtils::_NextUTF8Char(v, out), false); - - ASSERT_EQ(GeneralUtils::UTF8ToUTF16("Hello World!"), u"Hello World!"); - ASSERT_EQ(GeneralUtils::UTF8ToUTF16("Frühling"), u"Frühling"); - ASSERT_EQ(GeneralUtils::UTF8ToUTF16("中文字"), u"中文字"); - ASSERT_EQ(GeneralUtils::UTF8ToUTF16("👨‍⚖️"), u"👨‍⚖️"); - - return 0; -} diff --git a/tests/TestLDFFormat.cpp b/tests/TestLDFFormat.cpp deleted file mode 100644 index a7433e96..00000000 --- a/tests/TestLDFFormat.cpp +++ /dev/null @@ -1,31 +0,0 @@ -#include "LDFFormat.h" -#include "CommonCxxTests.h" - -/** - * @brief Test parsing an LDF value - * - * @param argc Number of command line arguments for this test - * @param argv Command line arguments - * @return 0 on success, non-zero on failure - */ -int TestLDFFormat(int argc, char** const argv) { - // Create - auto* data = LDFBaseData::DataFromString("KEY=0:VALUE"); - - // Check that the data type is correct - ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_16); - - // Check that the key is correct - ASSERT_EQ(data->GetKey(), u"KEY"); - - // Check that the value is correct - ASSERT_EQ(((LDFData<std::u16string>*)data)->GetValue(), u"VALUE"); - - // Check that the serialization is correct - ASSERT_EQ(data->GetString(), "KEY=0:VALUE"); - - // Cleanup the object - delete data; - - return 0; -} diff --git a/tests/TestNiPoint3.cpp b/tests/TestNiPoint3.cpp deleted file mode 100644 index d1588b8b..00000000 --- a/tests/TestNiPoint3.cpp +++ /dev/null @@ -1,13 +0,0 @@ -#include <stdexcept> - -#include "NiPoint3.h" -#include "CommonCxxTests.h" - -int TestNiPoint3(int argc, char** const argv) { - // Check that Unitize works - ASSERT_EQ(NiPoint3(3, 0, 0).Unitize(), NiPoint3::UNIT_X); - // Check what unitize does to a vector of length 0 - ASSERT_EQ(NiPoint3::ZERO.Unitize(), NiPoint3::ZERO); - // If we get here, all was successful - return 0; -} diff --git a/tests/dCommonTests/AMFDeserializeTests.cpp b/tests/dCommonTests/AMFDeserializeTests.cpp new file mode 100644 index 00000000..5a9d91e6 --- /dev/null +++ b/tests/dCommonTests/AMFDeserializeTests.cpp @@ -0,0 +1,447 @@ +#include <fstream> +#include <memory> +#include <gtest/gtest.h> + +#include "AMFDeserialize.h" +#include "Amf3.h" + +#include "Game.h" +#include "dLogger.h" + +/** + * Helper method that all tests use to get their respective AMF. + */ +AMFBaseValue* ReadFromBitStream(RakNet::BitStream* bitStream) { + AMFDeserialize deserializer; + AMFBaseValue* returnValue(deserializer.Read(bitStream)); + return returnValue; +} + +/** + * @brief Test reading an AMFUndefined value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFUndefinedTest) { + CBITSTREAM; + bitStream.Write<uint8_t>(0x00); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Undefined); +} + +/** + * @brief Test reading an AMFNull value from a BitStream. + * + */ +TEST(dCommonTests, AMFDeserializeAMFNullTest) { + CBITSTREAM; + bitStream.Write<uint8_t>(0x01); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Null); +} + +/** + * @brief Test reading an AMFFalse value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFFalseTest) { + CBITSTREAM; + bitStream.Write<uint8_t>(0x02); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::False); +} + +/** + * @brief Test reading an AMFTrue value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFTrueTest) { + CBITSTREAM; + bitStream.Write<uint8_t>(0x03); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::True); +} + +/** + * @brief Test reading an AMFInteger value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFIntegerTest) { + CBITSTREAM; + { + bitStream.Write<uint8_t>(0x04); + // 127 == 01111111 + bitStream.Write<uint8_t>(127); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Integer); + // Check that the max value of a byte can be read correctly + ASSERT_EQ(static_cast<AMFIntValue*>(res.get())->GetValue(), 127); + } + bitStream.Reset(); + { + bitStream.Write<uint8_t>(0x04); + bitStream.Write<uint32_t>(UINT32_MAX); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Integer); + // Check that we can read the maximum value correctly + ASSERT_EQ(static_cast<AMFIntValue*>(res.get())->GetValue(), 536870911); + } + bitStream.Reset(); + { + bitStream.Write<uint8_t>(0x04); + // 131 == 10000011 + bitStream.Write<uint8_t>(131); + // 255 == 11111111 + bitStream.Write<uint8_t>(255); + // 127 == 01111111 + bitStream.Write<uint8_t>(127); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Integer); + // Check that short max can be read correctly + ASSERT_EQ(static_cast<AMFIntValue*>(res.get())->GetValue(), UINT16_MAX); + } + bitStream.Reset(); + { + bitStream.Write<uint8_t>(0x04); + // 255 == 11111111 + bitStream.Write<uint8_t>(255); + // 127 == 01111111 + bitStream.Write<uint8_t>(127); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Integer); + // Check that 2 byte max can be read correctly + ASSERT_EQ(static_cast<AMFIntValue*>(res.get())->GetValue(), 16383); + } +} + +/** + * @brief Test reading an AMFDouble value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFDoubleTest) { + CBITSTREAM; + bitStream.Write<uint8_t>(0x05); + bitStream.Write<double>(25346.4f); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Double); + ASSERT_EQ(static_cast<AMFDoubleValue*>(res.get())->GetValue(), 25346.4f); +} + +/** + * @brief Test reading an AMFString value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFStringTest) { + CBITSTREAM; + bitStream.Write<uint8_t>(0x06); + bitStream.Write<uint8_t>(0x0F); + std::string toWrite = "stateID"; + for (auto e : toWrite) bitStream.Write<char>(e); + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::String); + ASSERT_EQ(static_cast<AMFStringValue*>(res.get())->GetValue(), "stateID"); +} + +/** + * @brief Test reading an AMFArray value from a BitStream. + */ +TEST(dCommonTests, AMFDeserializeAMFArrayTest) { + CBITSTREAM; + // Test empty AMFArray + bitStream.Write<uint8_t>(0x09); + bitStream.Write<uint8_t>(0x01); + bitStream.Write<uint8_t>(0x01); + { + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Array); + ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetAssociative().size(), 0); + ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetDense().size(), 0); + } + bitStream.Reset(); + // Test a key'd value and dense value + bitStream.Write<uint8_t>(0x09); + bitStream.Write<uint8_t>(0x03); + bitStream.Write<uint8_t>(0x15); + for (auto e : "BehaviorID") if (e != '\0') bitStream.Write<char>(e); + bitStream.Write<uint8_t>(0x06); + bitStream.Write<uint8_t>(0x0B); + for (auto e : "10447") if (e != '\0') bitStream.Write<char>(e); + bitStream.Write<uint8_t>(0x01); + bitStream.Write<uint8_t>(0x06); + bitStream.Write<uint8_t>(0x0B); + for (auto e : "10447") if (e != '\0') bitStream.Write<char>(e); + { + std::unique_ptr<AMFBaseValue> res(ReadFromBitStream(&bitStream)); + ASSERT_EQ(res->GetValueType(), eAmf::Array); + ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetAssociative().size(), 1); + ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->GetDense().size(), 1); + ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->Get<std::string>("BehaviorID")->GetValue(), "10447"); + ASSERT_EQ(static_cast<AMFArrayValue*>(res.get())->Get<std::string>(0)->GetValue(), "10447"); + } +} + +/** + * @brief This test checks that if we recieve an unimplemented eAmf + * we correctly throw an error and can actch it. + * Yes this leaks memory. + */ +TEST(dCommonTests, AMFDeserializeUnimplementedValuesTest) { + std::vector<eAmf> unimplementedValues = { + eAmf::XMLDoc, + eAmf::Date, + eAmf::Object, + eAmf::XML, + eAmf::ByteArray, + eAmf::VectorInt, + eAmf::VectorUInt, + eAmf::VectorDouble, + eAmf::VectorObject, + eAmf::Dictionary + }; + // Run unimplemented tests to check that errors are thrown if + // unimplemented AMF values are attempted to be parsed. + std::ifstream fileStream; + fileStream.open("AMFBitStreamUnimplementedTest.bin", std::ios::binary); + + // Read a test BitStream from a file + std::vector<char> baseBitStream; + char byte = 0; + while (fileStream.get(byte)) { + baseBitStream.push_back(byte); + } + + fileStream.close(); + + for (auto value : unimplementedValues) { + RakNet::BitStream testBitStream; + for (auto element : baseBitStream) { + testBitStream.Write(element); + } + testBitStream.Write(value); + bool caughtException = false; + try { + ReadFromBitStream(&testBitStream); + } catch (eAmf unimplementedValueType) { + caughtException = true; + } + + ASSERT_EQ(caughtException, true); + } +} + +/** + * @brief Test reading a packet capture from live from a BitStream + */ +TEST(dCommonTests, AMFDeserializeLivePacketTest) { + std::ifstream testFileStream; + testFileStream.open("AMFBitStreamTest.bin", std::ios::binary); + + // Read a test BitStream from a file + RakNet::BitStream testBitStream; + char byte = 0; + while (testFileStream.get(byte)) { + testBitStream.Write<char>(byte); + } + + testFileStream.close(); + + std::unique_ptr<AMFBaseValue> resultFromFn(ReadFromBitStream(&testBitStream)); + auto result = static_cast<AMFArrayValue*>(resultFromFn.get()); + // Test the outermost array + + ASSERT_EQ(result->Get<std::string>("BehaviorID")->GetValue(), "10447"); + ASSERT_EQ(result->Get<std::string>("objectID")->GetValue(), "288300744895913279"); + + // Test the execution state array + auto executionState = result->GetArray("executionState"); + + ASSERT_NE(executionState, nullptr); + + auto strips = executionState->GetArray("strips")->GetDense(); + + ASSERT_EQ(strips.size(), 1); + + auto stripsPosition0 = dynamic_cast<AMFArrayValue*>(strips[0]); + + auto actionIndex = stripsPosition0->Get<double>("actionIndex"); + + ASSERT_EQ(actionIndex->GetValue(), 0.0f); + + auto stripIdExecution = stripsPosition0->Get<double>("id"); + + ASSERT_EQ(stripIdExecution->GetValue(), 0.0f); + + auto stateIdExecution = executionState->Get<double>("stateID"); + + ASSERT_EQ(stateIdExecution->GetValue(), 0.0f); + + auto states = result->GetArray("states")->GetDense(); + + ASSERT_EQ(states.size(), 1); + + auto firstState = dynamic_cast<AMFArrayValue*>(states[0]); + + auto stateID = firstState->Get<double>("id"); + + ASSERT_EQ(stateID->GetValue(), 0.0f); + + auto stripsInState = firstState->GetArray("strips")->GetDense(); + + ASSERT_EQ(stripsInState.size(), 1); + + auto firstStrip = dynamic_cast<AMFArrayValue*>(stripsInState[0]); + + auto actionsInFirstStrip = firstStrip->GetArray("actions")->GetDense(); + + ASSERT_EQ(actionsInFirstStrip.size(), 3); + + auto actionID = firstStrip->Get<double>("id"); + + ASSERT_EQ(actionID->GetValue(), 0.0f); + + auto uiArray = firstStrip->GetArray("ui"); + + auto xPos = uiArray->Get<double>("x"); + auto yPos = uiArray->Get<double>("y"); + + ASSERT_EQ(xPos->GetValue(), 103.0f); + ASSERT_EQ(yPos->GetValue(), 82.0f); + + auto stripId = firstStrip->Get<double>("id"); + + ASSERT_EQ(stripId->GetValue(), 0.0f); + + auto firstAction = dynamic_cast<AMFArrayValue*>(actionsInFirstStrip[0]); + + auto firstType = firstAction->Get<std::string>("Type"); + + ASSERT_EQ(firstType->GetValue(), "OnInteract"); + + auto firstCallback = firstAction->Get<std::string>("__callbackID__"); + + ASSERT_EQ(firstCallback->GetValue(), ""); + + auto secondAction = dynamic_cast<AMFArrayValue*>(actionsInFirstStrip[1]); + + auto secondType = secondAction->Get<std::string>("Type"); + + ASSERT_EQ(secondType->GetValue(), "FlyUp"); + + auto secondCallback = secondAction->Get<std::string>("__callbackID__"); + + ASSERT_EQ(secondCallback->GetValue(), ""); + + auto secondDistance = secondAction->Get<double>("Distance"); + + ASSERT_EQ(secondDistance->GetValue(), 25.0f); + + auto thirdAction = dynamic_cast<AMFArrayValue*>(actionsInFirstStrip[2]); + + auto thirdType = thirdAction->Get<std::string>("Type"); + + ASSERT_EQ(thirdType->GetValue(), "FlyDown"); + + auto thirdCallback = thirdAction->Get<std::string>("__callbackID__"); + + ASSERT_EQ(thirdCallback->GetValue(), ""); + + auto thirdDistance = thirdAction->Get<double>("Distance"); + + ASSERT_EQ(thirdDistance->GetValue(), 25.0f); +} + +/** + * @brief Tests that having no BitStream returns a nullptr. + */ +TEST(dCommonTests, AMFDeserializeNullTest) { + std::unique_ptr<AMFBaseValue> result(ReadFromBitStream(nullptr)); + ASSERT_EQ(result.get(), nullptr); +} + +TEST(dCommonTests, AMFBadConversionTest) { + std::ifstream testFileStream; + testFileStream.open("AMFBitStreamTest.bin", std::ios::binary); + + // Read a test BitStream from a file + RakNet::BitStream testBitStream; + char byte = 0; + while (testFileStream.get(byte)) { + testBitStream.Write<char>(byte); + } + + testFileStream.close(); + + std::unique_ptr<AMFBaseValue> resultFromFn(ReadFromBitStream(&testBitStream)); + auto result = static_cast<AMFArrayValue*>(resultFromFn.get()); + + // Actually a string value. + ASSERT_EQ(result->Get<double>("BehaviorID"), nullptr); + + // Does not exist in the associative portion + ASSERT_EQ(result->Get<nullptr_t>("DOES_NOT_EXIST"), nullptr); + + result->Push(true); + + // Exists and is correct type + ASSERT_NE(result->Get<bool>(0), nullptr); + + // Value exists but is wrong typing + ASSERT_EQ(result->Get<std::string>(0), nullptr); + + // Value is out of bounds + ASSERT_EQ(result->Get<bool>(1), nullptr); +} + +/** + * Below is the AMF that is in the AMFBitStreamTest.bin file that we are reading in + * from a bitstream to test. +args: amf3! +{ + "objectID": "288300744895913279", + "BehaviorID": "10447", + "executionState": amf3! + { + "strips": amf3! + [ + amf3! + { + "actionIndex": 0.0, + "id": 0.0, + }, + ], + "stateID": 0.0, + }, + "states": amf3! + [ + amf3! + { + "id": 0.0, + "strips": amf3! + [ + amf3! + { + "actions": amf3! + [ + amf3! + { + "Type": "OnInteract", + "__callbackID__": "", + }, + amf3! + { + "Distance": 25.0, + "Type": "FlyUp", + "__callbackID__": "", + }, + amf3! + { + "Distance": 25.0, + "Type": "FlyDown", + "__callbackID__": "", + }, + ], + "id": 0.0, + "ui": amf3! + { + "x": 103.0, + "y": 82.0, + }, + }, + ], + }, + ], +} + */ diff --git a/tests/dCommonTests/Amf3Tests.cpp b/tests/dCommonTests/Amf3Tests.cpp new file mode 100644 index 00000000..a51fe4ba --- /dev/null +++ b/tests/dCommonTests/Amf3Tests.cpp @@ -0,0 +1,116 @@ +#include <gtest/gtest.h> + +#include <vector> + +#include "Amf3.h" + +TEST(dCommonTests, AMF3AssociativeArrayTest) { + + AMFArrayValue array; + array.Insert("true", true); + array.Insert("false", false); + + // test associative can insert values + ASSERT_EQ(array.GetAssociative().size(), 2); + ASSERT_EQ(array.Get<bool>("true")->GetValueType(), eAmf::True); + ASSERT_EQ(array.Get<bool>("false")->GetValueType(), eAmf::False); + + // Test associative can remove values + array.Remove("true"); + ASSERT_EQ(array.GetAssociative().size(), 1); + ASSERT_EQ(array.Get<bool>("true"), nullptr); + ASSERT_EQ(array.Get<bool>("false")->GetValueType(), eAmf::False); + + array.Remove("false"); + ASSERT_EQ(array.GetAssociative().size(), 0); + ASSERT_EQ(array.Get<bool>("true"), nullptr); + ASSERT_EQ(array.Get<bool>("false"), nullptr); + + // Test that multiple of the same key respect only the first element of that key + array.Insert("true", true); + array.Insert("true", false); + ASSERT_EQ(array.GetAssociative().size(), 1); + ASSERT_EQ(array.Get<bool>("true")->GetValueType(), eAmf::True); + array.Remove("true"); + + // Now test the dense portion + // Get some out of bounds values and cast to incorrect template types + array.Push(true); + array.Push(false); + + ASSERT_EQ(array.GetDense().size(), 2); + ASSERT_EQ(array.Get<bool>(0)->GetValueType(), eAmf::True); + ASSERT_EQ(array.Get<std::string>(0), nullptr); + ASSERT_EQ(array.Get<bool>(1)->GetValueType(), eAmf::False); + ASSERT_EQ(array.Get<bool>(155), nullptr); + + array.Pop(); + + ASSERT_EQ(array.GetDense().size(), 1); + ASSERT_EQ(array.Get<bool>(0)->GetValueType(), eAmf::True); + ASSERT_EQ(array.Get<std::string>(0), nullptr); + ASSERT_EQ(array.Get<bool>(1), nullptr); + + array.Pop(); + + ASSERT_EQ(array.GetDense().size(), 0); + ASSERT_EQ(array.Get<bool>(0), nullptr); + ASSERT_EQ(array.Get<std::string>(0), nullptr); + ASSERT_EQ(array.Get<bool>(1), nullptr); +} + +TEST(dCommonTests, AMF3InsertionAssociativeTest) { + AMFArrayValue array; + array.Insert("CString", "string"); + array.Insert("String", std::string("string")); + array.Insert("False", false); + array.Insert("True", true); + array.Insert<int32_t>("Integer", 42U); + array.Insert("Double", 42.0); + array.InsertArray("Array"); + array.Insert<std::vector<uint32_t>>("Undefined", {}); + array.Insert("Null", nullptr); + + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<const char*>("CString")->GetValueType(), eAmf::String); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<std::string>("String")->GetValueType(), eAmf::String); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<bool>("False")->GetValueType(), eAmf::False); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<bool>("True")->GetValueType(), eAmf::True); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<int32_t>("Integer")->GetValueType(), eAmf::Integer); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<double>("Double")->GetValueType(), eAmf::Double); + std::cout << "test" << std::endl; + ASSERT_EQ(array.GetArray("Array")->GetValueType(), eAmf::Array); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<nullptr_t>("Null")->GetValueType(), eAmf::Null); + std::cout << "test" << std::endl; + ASSERT_EQ(array.Get<std::vector<uint32_t>>("Undefined")->GetValueType(), eAmf::Undefined); + std::cout << "test" << std::endl; +} + +TEST(dCommonTests, AMF3InsertionDenseTest) { + AMFArrayValue array; + array.Push<std::string>("string"); + array.Push("CString"); + array.Push(false); + array.Push(true); + array.Push<int32_t>(42U); + array.Push(42.0); + array.PushArray(); + array.Push(nullptr); + array.Push<std::vector<uint32_t>>({}); + + ASSERT_EQ(array.Get<std::string>(0)->GetValueType(), eAmf::String); + ASSERT_EQ(array.Get<const char*>(1)->GetValueType(), eAmf::String); + ASSERT_EQ(array.Get<bool>(2)->GetValueType(), eAmf::False); + ASSERT_EQ(array.Get<bool>(3)->GetValueType(), eAmf::True); + ASSERT_EQ(array.Get<int32_t>(4)->GetValueType(), eAmf::Integer); + ASSERT_EQ(array.Get<double>(5)->GetValueType(), eAmf::Double); + ASSERT_EQ(array.GetArray(6)->GetValueType(), eAmf::Array); + ASSERT_EQ(array.Get<nullptr_t>(7)->GetValueType(), eAmf::Null); + ASSERT_EQ(array.Get<std::vector<uint32_t>>(8)->GetValueType(), eAmf::Undefined); +} diff --git a/tests/dCommonTests/CMakeLists.txt b/tests/dCommonTests/CMakeLists.txt new file mode 100644 index 00000000..a345863d --- /dev/null +++ b/tests/dCommonTests/CMakeLists.txt @@ -0,0 +1,22 @@ +set(DCOMMONTEST_SOURCES + "AMFDeserializeTests.cpp" + "Amf3Tests.cpp" + "HeaderSkipTest.cpp" + "TestLDFFormat.cpp" + "TestNiPoint3.cpp" + "TestEncoding.cpp" + "dCommonDependencies.cpp" +) + +# Set our executable +add_executable(dCommonTests ${DCOMMONTEST_SOURCES}) + +# Link needed libraries +target_link_libraries(dCommonTests ${COMMON_LIBRARIES} GTest::gtest_main) + +# Copy test files to testing directory +add_subdirectory(TestBitStreams) +file(COPY ${TESTBITSTREAMS} DESTINATION ${CMAKE_CURRENT_BINARY_DIR}) + +# Discover the tests +gtest_discover_tests(dCommonTests) diff --git a/tests/dCommonTests/HeaderSkipTest.cpp b/tests/dCommonTests/HeaderSkipTest.cpp new file mode 100644 index 00000000..a8f78078 --- /dev/null +++ b/tests/dCommonTests/HeaderSkipTest.cpp @@ -0,0 +1,37 @@ +#include <gtest/gtest.h> + +#include "dCommonDependencies.h" +#include "dCommonVars.h" +#include "BitStream.h" + +#define PacketUniquePtr std::unique_ptr<Packet> + +TEST(dCommonTests, HeaderSkipExcessTest) { + PacketUniquePtr packet = std::make_unique<Packet>(); + unsigned char headerAndData[] = { 0x53, 0x02, 0x00, 0x07, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00 }; // positive + packet->data = headerAndData; + packet->length = sizeof(headerAndData); + CINSTREAM_SKIP_HEADER; + ASSERT_EQ(inStream.GetNumberOfUnreadBits(), 64); + ASSERT_EQ(inStream.GetNumberOfBitsAllocated(), 128); +} + +TEST(dCommonTests, HeaderSkipExactDataTest) { + PacketUniquePtr packet = std::make_unique<Packet>(); + unsigned char header[] = { 0x53, 0x02, 0x00, 0x07, 0x00, 0x00, 0x00, 0x00 }; // positive + packet->data = header; + packet->length = sizeof(header); + CINSTREAM_SKIP_HEADER; + ASSERT_EQ(inStream.GetNumberOfUnreadBits(), 0); + ASSERT_EQ(inStream.GetNumberOfBitsAllocated(), 64); +} + +TEST(dCommonTests, HeaderSkipNotEnoughDataTest) { + PacketUniquePtr packet = std::make_unique<Packet>(); + unsigned char notEnoughData[] = { 0x53, 0x02, 0x00, 0x07, 0x00, 0x00 }; // negative + packet->data = notEnoughData; + packet->length = sizeof(notEnoughData); + CINSTREAM_SKIP_HEADER; + ASSERT_EQ(inStream.GetNumberOfUnreadBits(), 0); + ASSERT_EQ(inStream.GetNumberOfBitsAllocated(), 48); +} diff --git a/tests/TestBitStreams/AMFBitStreamTest.bin b/tests/dCommonTests/TestBitStreams/AMFBitStreamTest.bin similarity index 100% rename from tests/TestBitStreams/AMFBitStreamTest.bin rename to tests/dCommonTests/TestBitStreams/AMFBitStreamTest.bin diff --git a/tests/TestBitStreams/AMFBitStreamUnimplementedTest.bin b/tests/dCommonTests/TestBitStreams/AMFBitStreamUnimplementedTest.bin similarity index 100% rename from tests/TestBitStreams/AMFBitStreamUnimplementedTest.bin rename to tests/dCommonTests/TestBitStreams/AMFBitStreamUnimplementedTest.bin diff --git a/tests/dCommonTests/TestBitStreams/CMakeLists.txt b/tests/dCommonTests/TestBitStreams/CMakeLists.txt new file mode 100644 index 00000000..93606df6 --- /dev/null +++ b/tests/dCommonTests/TestBitStreams/CMakeLists.txt @@ -0,0 +1,11 @@ +set(TESTBITSTREAMS + "AMFBitStreamTest.bin" + "AMFBitStreamUnimplementedTest.bin" +) + +# Get the folder name and prepend it to the files above +get_filename_component(thisFolderName ${CMAKE_CURRENT_SOURCE_DIR} NAME) +list(TRANSFORM TESTBITSTREAMS PREPEND "${thisFolderName}/") + +# Export our list of files +set(TESTBITSTREAMS ${TESTBITSTREAMS} PARENT_SCOPE) diff --git a/tests/dCommonTests/TestEncoding.cpp b/tests/dCommonTests/TestEncoding.cpp new file mode 100644 index 00000000..c103ccbf --- /dev/null +++ b/tests/dCommonTests/TestEncoding.cpp @@ -0,0 +1,68 @@ +#include <string> +#include <gtest/gtest.h> +#include <string_view> + +#include "GeneralUtils.h" + +class EncodingTest : public ::testing::Test { +protected: + std::string originalWord; + std::string_view originalWordSv; + uint32_t out; +}; + +TEST_F(EncodingTest, TestEncodingHello) { + originalWord = "Hello World!"; + originalWordSv = originalWord; + + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 'H'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 'e'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 'l'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 'l'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 'o'); + EXPECT_EQ(GeneralUtils::_NextUTF8Char(originalWordSv, out), true); + + EXPECT_EQ(GeneralUtils::UTF8ToUTF16("Hello World!"), u"Hello World!"); +}; + +TEST_F(EncodingTest, TestEncodingUmlaut) { + originalWord = u8"Frühling"; + originalWordSv = originalWord; + + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'F'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'r'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'ü'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'h'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'l'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'i'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'n'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'g'); + EXPECT_EQ(GeneralUtils::_NextUTF8Char(originalWordSv, out), false); + + EXPECT_EQ(GeneralUtils::UTF8ToUTF16("Frühling"), u"Frühling"); +}; + +TEST_F(EncodingTest, TestEncodingChinese) { + originalWord = "中文字"; + originalWordSv = originalWord; + + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'中'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'文'); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, U'字'); + EXPECT_EQ(GeneralUtils::_NextUTF8Char(originalWordSv, out), false); + + EXPECT_EQ(GeneralUtils::UTF8ToUTF16("中文字"), u"中文字"); +}; + +TEST_F(EncodingTest, TestEncodingEmoji) { + originalWord = "👨‍⚖️"; + originalWordSv = originalWord; + + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 0x1F468); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 0x200D); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 0x2696); + GeneralUtils::_NextUTF8Char(originalWordSv, out); EXPECT_EQ(out, 0xFE0F); + EXPECT_EQ(GeneralUtils::_NextUTF8Char(originalWordSv, out), false); + + EXPECT_EQ(GeneralUtils::UTF8ToUTF16("👨‍⚖️"), u"👨‍⚖️"); +}; diff --git a/tests/dCommonTests/TestLDFFormat.cpp b/tests/dCommonTests/TestLDFFormat.cpp new file mode 100644 index 00000000..36326e38 --- /dev/null +++ b/tests/dCommonTests/TestLDFFormat.cpp @@ -0,0 +1,252 @@ +#include "LDFFormat.h" + +#include <gtest/gtest.h> + +#include "Game.h" +#include "dCommonDependencies.h" +#include "dLogger.h" + +class LDFTests : public dCommonDependenciesTest { +protected: + void SetUp() override { + SetUpDependencies(); + } + + void TearDown() override { + TearDownDependencies(); + } +}; + +#define LdfUniquePtr std::unique_ptr<LDFBaseData> + +// Suite of tests for parsing LDF values + +TEST_F(LDFTests, LDFUTF16Test) { + std::string testWord = "KEY=0:IAmA weird string with :::: and spac,./;'][\\es that I expect to be parsed correctly...; "; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_16); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::u16string>*)data.get())->GetValue(), u"IAmA weird string with :::: and spac,./;'][\\es that I expect to be parsed correctly...; "); + ASSERT_EQ(data->GetString(), testWord); +} + +TEST_F(LDFTests, LDFUTF16EmptyTest) { + std::string testWord = "KEY=0:"; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_16); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::u16string>*)data.get())->GetValue(), u""); + ASSERT_EQ(data->GetString(), testWord); +} + +TEST_F(LDFTests, LDFUTF16ColonTest) { + std::string testWord = "KEY=0:::"; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_16); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::u16string>*)data.get())->GetValue(), u"::"); + ASSERT_EQ(data->GetString(), testWord); +} + +TEST_F(LDFTests, LDFUTF16EqualsTest) { + std::string testWord = "KEY=0:=="; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_16); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::u16string>*)data.get())->GetValue(), u"=="); + ASSERT_EQ(data->GetString(), testWord); +} + +TEST_F(LDFTests, LDFS32Test) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=1:-15")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_S32); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<int32_t>*)data.get())->GetValue(), -15); + ASSERT_EQ(data->GetString(), "KEY=1:-15"); +} +TEST_F(LDFTests, LDFU32Test) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=5:15")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_U32); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<uint32_t>*)data.get())->GetValue(), 15); + ASSERT_EQ(data->GetString(), "KEY=5:15"); +} + +TEST_F(LDFTests, LDFU32TrueTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=5:true")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_U32); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<uint32_t>*)data.get())->GetValue(), 1); + ASSERT_EQ(data->GetString(), "KEY=5:1"); +} + +TEST_F(LDFTests, LDFU32FalseTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=5:false")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_U32); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<uint32_t>*)data.get())->GetValue(), 0); + ASSERT_EQ(data->GetString(), "KEY=5:0"); +} + + +// Use find since floats and doubles generally have appended 0s +TEST_F(LDFTests, LDFFloatTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=3:15.5")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_FLOAT); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<float>*)data.get())->GetValue(), 15.5f); + ASSERT_EQ(data->GetString().find("KEY=3:15.5"), 0); +} + +TEST_F(LDFTests, LDFDoubleTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=4:15.5")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_DOUBLE); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<double>*)data.get())->GetValue(), 15.5); + ASSERT_EQ(data->GetString().find("KEY=4:15.5"), 0); +} + + +TEST_F(LDFTests, LDFBoolTrueTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=7:true")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_BOOLEAN); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<bool>*)data.get())->GetValue(), true); + ASSERT_EQ(data->GetString(), "KEY=7:1"); +} + +TEST_F(LDFTests, LDFBoolFalseTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=7:false")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_BOOLEAN); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<bool>*)data.get())->GetValue(), false); + ASSERT_EQ(data->GetString(), "KEY=7:0"); +} + +TEST_F(LDFTests, LDFBoolIntTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=7:3")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_BOOLEAN); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<bool>*)data.get())->GetValue(), true); + ASSERT_EQ(data->GetString(), "KEY=7:1"); +} + +TEST_F(LDFTests, LDFU64Test) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=8:15")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_U64); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<uint64_t>*)data.get())->GetValue(), 15); + ASSERT_EQ(data->GetString(), "KEY=8:15"); +} + +TEST_F(LDFTests, LDFLWOOBJIDTest) { + LdfUniquePtr data(LDFBaseData::DataFromString("KEY=9:15")); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_OBJID); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<uint64_t>*)data.get())->GetValue(), 15); + ASSERT_EQ(data->GetString(), "KEY=9:15"); +} + +TEST_F(LDFTests, LDFUTF8Test) { + std::string testWord = "KEY=13:IAmA weird string with :::: and spac,./;'][\\es that I expect to be parsed correctly...; "; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_8); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::string>*)data.get())->GetValue(), "IAmA weird string with :::: and spac,./;'][\\es that I expect to be parsed correctly...; "); + ASSERT_EQ(data->GetString(), testWord); +} + +TEST_F(LDFTests, LDFUTF8EmptyTest) { + std::string testWord = "KEY=13:"; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_8); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::string>*)data.get())->GetValue(), ""); + ASSERT_EQ(data->GetString(), testWord); +} + +TEST_F(LDFTests, LDFUTF8ColonsTest) { + std::string testWord = "KEY=13:::"; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_8); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::string>*)data.get())->GetValue(), "::"); + ASSERT_EQ(data->GetString(), testWord); +} +TEST_F(LDFTests, LDFUTF8EqualsTest) { + std::string testWord = "KEY=13:=="; + LdfUniquePtr data(LDFBaseData::DataFromString(testWord)); + ASSERT_NE(data, nullptr); + ASSERT_EQ(data->GetValueType(), eLDFType::LDF_TYPE_UTF_8); + ASSERT_EQ(data->GetKey(), u"KEY"); + ASSERT_EQ(((LDFData<std::string>*)data.get())->GetValue(), "=="); + ASSERT_EQ(data->GetString(), testWord); +} + + +TEST_F(LDFTests, LDFParseEdgeCaseTest) { + std::vector<std::string> tests = { + // Test undefined data + "", // Empty + "=", // Only equals sign + ":", // Only colon + "=:", // Only colon and equals sign + + // Test no LDFType + "KEY=:", // No LDF Type + "KEY=:44", // No LDF Type, but has value + + // Test invalid values, but valid types + "key=1:", // no value for int32 + "key=1:banana", // invalid value for int32 + "key=3:", // No value for float + "key=3:banana", // invalid for float + "key=4:", // No value for double + "key=4:banana", // invalid for double + "key=5:", // No value for U32 + "key=5:banana", // invalid for U32 + "key=7:", // No value for bool + "key=7:banana", // invalid for bool + "key=8:", // No value for U64 + "key=8:banana", // invalid for U64 + "key=9:", // No value for LWOOBJID + "key=9:banana", // invalid for LWOOBJID + + // Test invalid LDF types + "key=14:value", // invalid LDF type + "key=-1:value", // invalid LDF type + "key=-2:value", // invalid LDF type (no enum definition) + "key=Garbage:value", // invalid LDF type + }; + for (auto testString : tests) { + Game::logger->Log("LDFTests", "Testing LDF Parsing of invalid string (%s)", testString.c_str()); + EXPECT_NO_THROW(LDFBaseData::DataFromString(testString)); + } +} + +#ifdef PERF_TEST + +TEST_F(LDFTests, LDFSpeedTest) { + std::string keyToTest = "KEY=0:IAmA weird string with :::: and s"; + for (int i = 0; i < 10000; i++) LDFBaseData::DataFromString(keyToTest); +} + +#endif //PERF diff --git a/tests/dCommonTests/TestNiPoint3.cpp b/tests/dCommonTests/TestNiPoint3.cpp new file mode 100644 index 00000000..fbc98eb0 --- /dev/null +++ b/tests/dCommonTests/TestNiPoint3.cpp @@ -0,0 +1,23 @@ +#include <gtest/gtest.h> + +#include "NiPoint3.h" + +/** + * @brief Basic test for NiPoint3 functionality + * + */ +TEST(dCommonTests, NiPoint3Test) { + // Check that Unitize works + ASSERT_EQ(NiPoint3(3, 0, 0).Unitize(), NiPoint3::UNIT_X); + // Check what unitize does to a vector of length 0 + ASSERT_EQ(NiPoint3::ZERO.Unitize(), NiPoint3::ZERO); +} + +TEST(dCommonTests, NiPoint3OperatorTest) { + NiPoint3 a(1, 2, 3); + NiPoint3 b(4, 5, 6); + a += b; + EXPECT_FLOAT_EQ(a.x, 5); + EXPECT_FLOAT_EQ(a.y, 7); + EXPECT_FLOAT_EQ(a.z, 9); +} diff --git a/tests/dCommonTests/dCommonDependencies.cpp b/tests/dCommonTests/dCommonDependencies.cpp new file mode 100644 index 00000000..5b25fd47 --- /dev/null +++ b/tests/dCommonTests/dCommonDependencies.cpp @@ -0,0 +1,7 @@ +#include "Game.h" + +class dLogger; +namespace Game +{ + dLogger* logger; +} // namespace Game diff --git a/tests/dCommonTests/dCommonDependencies.h b/tests/dCommonTests/dCommonDependencies.h new file mode 100644 index 00000000..12aeb938 --- /dev/null +++ b/tests/dCommonTests/dCommonDependencies.h @@ -0,0 +1,26 @@ +#ifndef __DCOMMONDEPENDENCIES__H__ +#define __DCOMMONDEPENDENCIES__H__ + +#include "Game.h" +#include "dLogger.h" +#include "dServer.h" +#include "EntityInfo.h" +#include "EntityManager.h" +#include "dConfig.h" +#include <gtest/gtest.h> + +class dCommonDependenciesTest : public ::testing::Test { +protected: + void SetUpDependencies() { + Game::logger = new dLogger("./testing.log", true, true); + } + + void TearDownDependencies() { + if (Game::logger) { + Game::logger->Flush(); + delete Game::logger; + } + } +}; + +#endif //!__DCOMMONDEPENDENCIES__H__ diff --git a/tests/dGameTests/CMakeLists.txt b/tests/dGameTests/CMakeLists.txt new file mode 100644 index 00000000..b1fdaa07 --- /dev/null +++ b/tests/dGameTests/CMakeLists.txt @@ -0,0 +1,19 @@ +set(DGAMETEST_SOURCES + "GameDependencies.cpp" +) + +add_subdirectory(dComponentsTests) +list(APPEND DGAMETEST_SOURCES ${DCOMPONENTS_TESTS}) + +add_subdirectory(dGameMessagesTests) +list(APPEND DGAMETEST_SOURCES ${DGAMEMESSAGES_TESTS}) + +file(COPY ${GAMEMESSAGE_TESTBITSTREAMS} DESTINATION ${CMAKE_CURRENT_BINARY_DIR}) + +# Add the executable. Remember to add all tests above this! +add_executable(dGameTests ${DGAMETEST_SOURCES}) + +target_link_libraries(dGameTests ${COMMON_LIBRARIES} GTest::gtest_main dGame dZoneManager dPhysics Detour Recast tinyxml2 dWorldServer dChatFilter dNavigation) + +# Discover the tests +gtest_discover_tests(dGameTests) diff --git a/tests/dGameTests/GameDependencies.cpp b/tests/dGameTests/GameDependencies.cpp new file mode 100644 index 00000000..1765bd9b --- /dev/null +++ b/tests/dGameTests/GameDependencies.cpp @@ -0,0 +1,14 @@ +#include "GameDependencies.h" + +namespace Game { + dLogger* logger = nullptr; + dServer* server = nullptr; + dZoneManager* zoneManager = nullptr; + dChatFilter* chatFilter = nullptr; + dConfig* config = nullptr; + std::mt19937 randomEngine; + RakPeerInterface* chatServer = nullptr; + AssetManager* assetManager = nullptr; + SystemAddress chatSysAddr; + EntityManager* entityManager = nullptr; +} diff --git a/tests/dGameTests/GameDependencies.h b/tests/dGameTests/GameDependencies.h new file mode 100644 index 00000000..95ef2f9f --- /dev/null +++ b/tests/dGameTests/GameDependencies.h @@ -0,0 +1,51 @@ +#ifndef __GAMEDEPENDENCIES__H__ +#define __GAMEDEPENDENCIES__H__ + +#include "Game.h" +#include "dLogger.h" +#include "dServer.h" +#include "EntityInfo.h" +#include "EntityManager.h" +#include "dConfig.h" +#include <gtest/gtest.h> + +class dZoneManager; +class AssetManager; + +class dServerMock : public dServer { + RakNet::BitStream* sentBitStream = nullptr; +public: + dServerMock() {}; + ~dServerMock() {}; + RakNet::BitStream* GetMostRecentBitStream() { return sentBitStream; }; + void Send(RakNet::BitStream* bitStream, const SystemAddress& sysAddr, bool broadcast) override { sentBitStream = bitStream; }; +}; + +class GameDependenciesTest : public ::testing::Test { +protected: + void SetUpDependencies() { + info.pos = NiPoint3::ZERO; + info.rot = NiQuaternion::IDENTITY; + info.scale = 1.0f; + info.spawner = nullptr; + info.lot = 999; + Game::logger = new dLogger("./testing.log", true, true); + Game::server = new dServerMock(); + Game::config = new dConfig("worldconfig.ini"); + Game::entityManager = new EntityManager(); + } + + void TearDownDependencies() { + if (Game::server) delete Game::server; + if (Game::entityManager) delete Game::entityManager; + if (Game::logger) { + Game::logger->Flush(); + delete Game::logger; + } + if (Game::config) delete Game::config; + } + + EntityInfo info{}; +}; + +#endif //!__GAMEDEPENDENCIES__H__ diff --git a/tests/dGameTests/dComponentsTests/CMakeLists.txt b/tests/dGameTests/dComponentsTests/CMakeLists.txt new file mode 100644 index 00000000..17e69a2f --- /dev/null +++ b/tests/dGameTests/dComponentsTests/CMakeLists.txt @@ -0,0 +1,10 @@ +set(DCOMPONENTS_TESTS + "DestroyableComponentTests.cpp" +) + +# Get the folder name and prepend it to the files above +get_filename_component(thisFolderName ${CMAKE_CURRENT_SOURCE_DIR} NAME) +list(TRANSFORM DCOMPONENTS_TESTS PREPEND "${thisFolderName}/") + +# Export to parent scope +set(DCOMPONENTS_TESTS ${DCOMPONENTS_TESTS} PARENT_SCOPE) diff --git a/tests/dGameTests/dComponentsTests/DestroyableComponentTests.cpp b/tests/dGameTests/dComponentsTests/DestroyableComponentTests.cpp new file mode 100644 index 00000000..db9c033a --- /dev/null +++ b/tests/dGameTests/dComponentsTests/DestroyableComponentTests.cpp @@ -0,0 +1,533 @@ +#include "GameDependencies.h" +#include <gtest/gtest.h> + +#include "BitStream.h" +#include "DestroyableComponent.h" +#include "Entity.h" +#include "eReplicaComponentType.h" +#include "eStateChangeType.h" + +class DestroyableTest : public GameDependenciesTest { +protected: + Entity* baseEntity; + DestroyableComponent* destroyableComponent; + CBITSTREAM + uint32_t flags = 0; + void SetUp() override { + SetUpDependencies(); + baseEntity = new Entity(15, GameDependenciesTest::info); + destroyableComponent = new DestroyableComponent(baseEntity); + baseEntity->AddComponent(eReplicaComponentType::DESTROYABLE, destroyableComponent); + // Initialize some values to be not default + destroyableComponent->SetMaxHealth(12345.0f); + destroyableComponent->SetHealth(23); + destroyableComponent->SetMaxArmor(14.0f); + destroyableComponent->SetArmor(7); + destroyableComponent->SetMaxImagination(14000.0f); + destroyableComponent->SetImagination(6000); + destroyableComponent->SetIsSmashable(true); + destroyableComponent->SetExplodeFactor(1.1f); + destroyableComponent->AddFactionNoLookup(-1); + destroyableComponent->AddFactionNoLookup(6); + } + + void TearDown() override { + delete baseEntity; + TearDownDependencies(); + } +}; + +/** + * Test Construction of a DestroyableComponent + */ +TEST_F(DestroyableTest, DestroyableComponentSerializeConstructionTest) { + destroyableComponent->Serialize(&bitStream, true, flags); + // Assert that the full number of bits are present + ASSERT_EQ(bitStream.GetNumberOfUnreadBits(), 748); + { + // Now read in the full serialized construction BitStream + bool optionStatusImmunityInfo{}; + uint32_t ImmuneToBasicAttackCount{}; + uint32_t ImmuneToDamageOverTimeCount{}; + uint32_t ImmuneToKnockbackCount{}; + uint32_t ImmuneToInterruptCount{}; + uint32_t ImmuneToSpeedCount{}; + uint32_t ImmuneToImaginationGainCount{}; + uint32_t ImmuneToImaginationLossCount{}; + uint32_t ImmuneToQuickbuildInterruptCount{}; + uint32_t ImmuneToPullToPointCount{}; + bool optionStatsInfo{}; + uint32_t currentHealth{}; + float maxHealth{}; + uint32_t currentArmor{}; + float maxArmor{}; + uint32_t currentImagination{}; + float maxImagination{}; + uint32_t damageAbsorptionPoints{}; + bool hasImmunity{}; + bool isGmImmune{}; + bool isShielded{}; + float actualMaxHealth{}; + float actualMaxArmor{}; + float actualMaxImagination{}; + uint32_t factionsSize{}; + std::vector<int32_t> factions{}; + bool isSmashable{}; + bool isDead{}; + bool isSmashed{}; + bool isModuleAssembly{}; + bool optionExplodeFactor{}; + float explodeFactor{}; + bool optionIsOnThreatList{}; + bool isThreatened{}; + bitStream.Read(optionStatusImmunityInfo); + bitStream.Read(ImmuneToBasicAttackCount); + bitStream.Read(ImmuneToDamageOverTimeCount); + bitStream.Read(ImmuneToKnockbackCount); + bitStream.Read(ImmuneToInterruptCount); + bitStream.Read(ImmuneToSpeedCount); + bitStream.Read(ImmuneToImaginationGainCount); + bitStream.Read(ImmuneToImaginationLossCount); + bitStream.Read(ImmuneToQuickbuildInterruptCount); + bitStream.Read(ImmuneToPullToPointCount); + bitStream.Read(optionStatsInfo); + bitStream.Read(currentHealth); + bitStream.Read(maxHealth); + bitStream.Read(currentArmor); + bitStream.Read(maxArmor); + bitStream.Read(currentImagination); + bitStream.Read(maxImagination); + bitStream.Read(damageAbsorptionPoints); + bitStream.Read(hasImmunity); + bitStream.Read(isGmImmune); + bitStream.Read(isShielded); + bitStream.Read(actualMaxHealth); + bitStream.Read(actualMaxArmor); + bitStream.Read(actualMaxImagination); + bitStream.Read(factionsSize); + for (uint32_t i = 0; i < factionsSize; i++) { + int32_t factionID{}; + bitStream.Read(factionID); + factions.push_back(factionID); + } + bitStream.Read(isSmashable); // This is an option later and also a flag at this spot + bitStream.Read(isDead); + bitStream.Read(isSmashed); + // if IsSmashable is true, read the next bits. + bitStream.Read(isModuleAssembly); + bitStream.Read(optionExplodeFactor); + bitStream.Read(explodeFactor); + + bitStream.Read(optionIsOnThreatList); + bitStream.Read(isThreatened); + EXPECT_EQ(optionStatusImmunityInfo, true); + EXPECT_EQ(ImmuneToBasicAttackCount, 0); + EXPECT_EQ(ImmuneToDamageOverTimeCount, 0); + EXPECT_EQ(ImmuneToKnockbackCount, 0); + EXPECT_EQ(ImmuneToInterruptCount, 0); + EXPECT_EQ(ImmuneToSpeedCount, 0); + EXPECT_EQ(ImmuneToImaginationGainCount, 0); + EXPECT_EQ(ImmuneToImaginationLossCount, 0); + EXPECT_EQ(ImmuneToQuickbuildInterruptCount, 0); + EXPECT_EQ(ImmuneToPullToPointCount, 0); + + EXPECT_EQ(optionStatsInfo, true); + EXPECT_EQ(currentHealth, 23); + EXPECT_EQ(maxHealth, 12345.0f); + EXPECT_EQ(currentArmor, 7); + EXPECT_EQ(maxArmor, 14.0f); + EXPECT_EQ(currentImagination, 6000); + EXPECT_EQ(maxImagination, 14000.0f); + EXPECT_EQ(damageAbsorptionPoints, 0.0f); + EXPECT_EQ(hasImmunity, false); + EXPECT_EQ(isGmImmune, false); + EXPECT_EQ(isShielded, false); + EXPECT_EQ(actualMaxHealth, 12345.0f); + EXPECT_EQ(actualMaxArmor, 14.0f); + EXPECT_EQ(actualMaxImagination, 14000.0f); + EXPECT_EQ(factionsSize, 2); + EXPECT_NE(std::find(factions.begin(), factions.end(), -1), factions.end()); + EXPECT_NE(std::find(factions.begin(), factions.end(), 6), factions.end()); + EXPECT_EQ(isSmashable, true); + EXPECT_EQ(isDead, false); + EXPECT_EQ(isSmashed, false); + EXPECT_EQ(isSmashable, true); // For the sake of readability with the struct viewers, we will test this twice since its used as an option here, but as a bool above. + EXPECT_EQ(isModuleAssembly, false); + EXPECT_EQ(optionExplodeFactor, true); + EXPECT_EQ(explodeFactor, 1.1f); + + EXPECT_EQ(optionIsOnThreatList, true); + EXPECT_EQ(isThreatened, false); + } + bitStream.Reset(); +} + +/** + * Test serialization of a DestroyableComponent + */ +TEST_F(DestroyableTest, DestroyableComponentSerializeTest) { + bitStream.Reset(); + // Initialize some values to be not default so we can test a full serialization + destroyableComponent->SetMaxHealth(1233.0f); + + // Now we test a serialization for correctness. + destroyableComponent->Serialize(&bitStream, false, flags); + ASSERT_EQ(bitStream.GetNumberOfUnreadBits(), 422); + { + // Now read in the full serialized BitStream + bool optionStatsInfo{}; + uint32_t currentHealth{}; + float maxHealth{}; + uint32_t currentArmor{}; + float maxArmor{}; + uint32_t currentImagination{}; + float maxImagination{}; + uint32_t damageAbsorptionPoints{}; + bool hasImmunity{}; + bool isGmImmune{}; + bool isShielded{}; + float actualMaxHealth{}; + float actualMaxArmor{}; + float actualMaxImagination{}; + uint32_t factionsSize{}; + std::vector<int32_t> factions{}; + bool isSmashable{}; + bool optionIsOnThreatList{}; + bitStream.Read(optionStatsInfo); + bitStream.Read(currentHealth); + bitStream.Read(maxHealth); + bitStream.Read(currentArmor); + bitStream.Read(maxArmor); + bitStream.Read(currentImagination); + bitStream.Read(maxImagination); + bitStream.Read(damageAbsorptionPoints); + bitStream.Read(hasImmunity); + bitStream.Read(isGmImmune); + bitStream.Read(isShielded); + bitStream.Read(actualMaxHealth); + bitStream.Read(actualMaxArmor); + bitStream.Read(actualMaxImagination); + bitStream.Read(factionsSize); + for (uint32_t i = 0; i < factionsSize; i++) { + int32_t factionID{}; + bitStream.Read(factionID); + factions.push_back(factionID); + } + bitStream.Read(isSmashable); + + bitStream.Read(optionIsOnThreatList); + + EXPECT_EQ(optionStatsInfo, true); + EXPECT_EQ(currentHealth, 23); + EXPECT_EQ(maxHealth, 1233.0f); + EXPECT_EQ(currentArmor, 7); + EXPECT_EQ(maxArmor, 14.0f); + EXPECT_EQ(currentImagination, 6000); + EXPECT_EQ(maxImagination, 14000.0f); + EXPECT_EQ(damageAbsorptionPoints, 0.0f); + EXPECT_EQ(hasImmunity, false); + EXPECT_EQ(isGmImmune, false); + EXPECT_EQ(isShielded, false); + EXPECT_EQ(actualMaxHealth, 1233.0f); + EXPECT_EQ(actualMaxArmor, 14.0f); + EXPECT_EQ(actualMaxImagination, 14000.0f); + EXPECT_EQ(factionsSize, 2); + EXPECT_NE(std::find(factions.begin(), factions.end(), -1), factions.end()); + EXPECT_NE(std::find(factions.begin(), factions.end(), 6), factions.end()); + EXPECT_EQ(isSmashable, true); + + EXPECT_EQ(optionIsOnThreatList, false); // Always zero for now on serialization + } +} + +/** + * Test the Damage method of DestroyableComponent + */ +TEST_F(DestroyableTest, DestroyableComponentDamageTest) { + // Do some actions + destroyableComponent->SetMaxHealth(100.0f); + destroyableComponent->SetHealth(100); + destroyableComponent->SetMaxArmor(0.0f); + destroyableComponent->Damage(10, LWOOBJID_EMPTY); + // Check that we take damage + ASSERT_EQ(destroyableComponent->GetHealth(), 90); + // Check that if we have armor, we take the correct amount of damage + destroyableComponent->SetMaxArmor(10.0f); + destroyableComponent->SetArmor(5); + destroyableComponent->Damage(10, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 85); + // Check that if we have damage absorption we take the correct damage + destroyableComponent->SetDamageToAbsorb(10); + destroyableComponent->Damage(9, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 85); + ASSERT_EQ(destroyableComponent->GetDamageToAbsorb(), 1); + destroyableComponent->Damage(6, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 80); + // Check that we take the correct reduced damage if we take reduced damage + destroyableComponent->SetDamageReduction(2); + destroyableComponent->Damage(7, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 75); + destroyableComponent->Damage(2, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 74); + ASSERT_EQ(destroyableComponent->GetDamageReduction(), 2); + destroyableComponent->SetDamageReduction(0); + // Check that blocking works + destroyableComponent->SetAttacksToBlock(1); + destroyableComponent->Damage(UINT32_MAX, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 74); + destroyableComponent->Damage(4, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 70); + // Check that immunity works + destroyableComponent->SetIsImmune(true); + destroyableComponent->Damage(UINT32_MAX, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 70); + ASSERT_TRUE(destroyableComponent->IsImmune()); + destroyableComponent->SetIsImmune(false); + destroyableComponent->SetIsGMImmune(true); + destroyableComponent->Damage(UINT32_MAX, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 70); + ASSERT_TRUE(destroyableComponent->IsImmune()); + destroyableComponent->SetIsGMImmune(false); + // Check knockback immunity + destroyableComponent->SetIsShielded(true); + ASSERT_TRUE(destroyableComponent->IsKnockbackImmune()); + // Finally deal enough damage to kill the Entity + destroyableComponent->Damage(71, LWOOBJID_EMPTY); + ASSERT_EQ(destroyableComponent->GetHealth(), 0); + // Now lets heal some stats + destroyableComponent->Heal(15); + ASSERT_EQ(destroyableComponent->GetHealth(), 15); + destroyableComponent->Heal(15000); + ASSERT_EQ(destroyableComponent->GetHealth(), 100); + destroyableComponent->Repair(10); + ASSERT_EQ(destroyableComponent->GetArmor(), 10); + destroyableComponent->Repair(15000); + ASSERT_EQ(destroyableComponent->GetArmor(), 10); + destroyableComponent->SetMaxImagination(100.0f); + destroyableComponent->SetImagination(0); + destroyableComponent->Imagine(99); + ASSERT_EQ(destroyableComponent->GetImagination(), 99); + destroyableComponent->Imagine(4); + ASSERT_EQ(destroyableComponent->GetImagination(), 100); +} + +TEST_F(DestroyableTest, DestroyableComponentFactionTest) { + ASSERT_TRUE(destroyableComponent->HasFaction(-1)); + ASSERT_TRUE(destroyableComponent->HasFaction(6)); +} + +TEST_F(DestroyableTest, DestroyableComponentValiditiyTest) { + auto* enemyEntity = new Entity(19, info); + auto* enemyDestroyableComponent = new DestroyableComponent(enemyEntity); + enemyEntity->AddComponent(eReplicaComponentType::DESTROYABLE, enemyDestroyableComponent); + enemyDestroyableComponent->AddFactionNoLookup(16); + destroyableComponent->AddEnemyFaction(16); + EXPECT_TRUE(destroyableComponent->IsEnemy(enemyEntity)); + EXPECT_FALSE(destroyableComponent->IsFriend(enemyEntity)); + delete enemyEntity; +} + +TEST_F(DestroyableTest, DestroyableComponentImmunityTest) { + // assert to show that they are empty + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + // set them all to true (count 1) and check + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, true, true, true, true, true, true, true, true, true); + ASSERT_TRUE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_TRUE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_TRUE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_TRUE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_TRUE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_TRUE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_TRUE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_TRUE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_TRUE(destroyableComponent->GetImmuneToPullToPoint()); + + // remove them to check that they get removed properly + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, true, true, true, true, true, true, true, true, true); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + + // should not crash to remove them again + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, true, true, true, true, true, true, true, true, true); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + + // just do one + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, true); + ASSERT_TRUE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + // now stack it to 2 on basic attack + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, true); + ASSERT_TRUE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + // remove one and still shoudl be true + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, true); + ASSERT_TRUE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + // go back to 0 + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, true); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + // check individual ones now + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, true, false, false, false, false, false, false, false, false); + ASSERT_TRUE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, true, false, false, false, false, false, false, false, false); + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, true, false, false, false, false, false, false, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_TRUE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, true, false, false, false, false, false, false, false); + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, true, false, false, false, false, false, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_TRUE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, true, false, false, false, false, false, false); + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, false, true, false, false, false, false, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_TRUE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, false, true, false, false, false, false, false); + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, false, false, true, false, false, false, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_TRUE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, false, false, true, false, false, false, false); + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, false, false, false, true, false, false, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_TRUE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, false, false, false, true, false, false, false); + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, false, false, false, false, true, false, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_TRUE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, false, false, false, false, true, false, false); + + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, false, false, false, false, false, true, false); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_TRUE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, false, false, false, false, false, true, false); + + + destroyableComponent->SetStatusImmunity(eStateChangeType::PUSH, false, false, false, false, false, false, false, false, true); + ASSERT_FALSE(destroyableComponent->GetImmuneToBasicAttack()); + ASSERT_FALSE(destroyableComponent->GetImmuneToDamageOverTime()); + ASSERT_FALSE(destroyableComponent->GetImmuneToKnockback()); + ASSERT_FALSE(destroyableComponent->GetImmuneToInterrupt()); + ASSERT_FALSE(destroyableComponent->GetImmuneToSpeed()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationGain()); + ASSERT_FALSE(destroyableComponent->GetImmuneToImaginationLoss()); + ASSERT_FALSE(destroyableComponent->GetImmuneToQuickbuildInterrupt()); + ASSERT_TRUE(destroyableComponent->GetImmuneToPullToPoint()); + destroyableComponent->SetStatusImmunity(eStateChangeType::POP, false, false, false, false, false, false, false, false, true); + +} + diff --git a/tests/dGameTests/dGameMessagesTests/CMakeLists.txt b/tests/dGameTests/dGameMessagesTests/CMakeLists.txt new file mode 100644 index 00000000..e1c4884c --- /dev/null +++ b/tests/dGameTests/dGameMessagesTests/CMakeLists.txt @@ -0,0 +1,15 @@ +SET(DGAMEMESSAGES_TESTS + "GameMessageTests.cpp" + "LegacyGameMessageTests.cpp") + +# Get the folder name and prepend it to the files above +get_filename_component(thisFolderName ${CMAKE_CURRENT_SOURCE_DIR} NAME) +list(TRANSFORM DGAMEMESSAGES_TESTS PREPEND "${thisFolderName}/") + +# Copy test files to testing directory +add_subdirectory(TestBitStreams) +list(TRANSFORM GAMEMESSAGE_TESTBITSTREAMS PREPEND "${thisFolderName}/") +set(GAMEMESSAGE_TESTBITSTREAMS ${GAMEMESSAGE_TESTBITSTREAMS} PARENT_SCOPE) + +# Export to parent scope +set(DGAMEMESSAGES_TESTS ${DGAMEMESSAGES_TESTS} PARENT_SCOPE) diff --git a/tests/dGameTests/dGameMessagesTests/GameMessageTests.cpp b/tests/dGameTests/dGameMessagesTests/GameMessageTests.cpp new file mode 100644 index 00000000..047f56d6 --- /dev/null +++ b/tests/dGameTests/dGameMessagesTests/GameMessageTests.cpp @@ -0,0 +1,226 @@ +#include "Action.h" +#include "Amf3.h" +#include "AMFDeserialize.h" +#include "GameMessages.h" +#include "GameDependencies.h" + +#include <gtest/gtest.h> + +// Message includes +#include "AddActionMessage.h" +#include "AddStripMessage.h" +#include "AddMessage.h" +#include "MigrateActionsMessage.h" +#include "MoveToInventoryMessage.h" +#include "MergeStripsMessage.h" +#include "RearrangeStripMessage.h" +#include "RemoveActionsMessage.h" +#include "RemoveStripMessage.h" +#include "RenameMessage.h" +#include "SplitStripMessage.h" +#include "UpdateActionMessage.h" +#include "UpdateStripUiMessage.h" + +class GameMessageTests: public GameDependenciesTest { +protected: + void SetUp() override { + SetUpDependencies(); + } + void TearDown() override { + TearDownDependencies(); + } + std::string ReadFromFile(std::string filename) { + std::ifstream file(filename, std::ios::binary); + std::string readFile; + while (file.good()) { + char readCharacter = file.get(); + readFile.push_back(readCharacter); + } + return readFile; + } + AMFArrayValue* ReadArrayFromBitStream(RakNet::BitStream* inStream) { + AMFDeserialize des; + AMFBaseValue* readArray = des.Read(inStream); + EXPECT_EQ(readArray->GetValueType(), eAmf::Array); + return static_cast<AMFArrayValue*>(readArray); + } +}; + +/** + * @brief Tests that the serialization struct BlueprintLoadItemResponse is serialized correctly + * + */ +TEST_F(GameMessageTests, SendBlueprintLoadItemResponse) { + GameMessages::SendBlueprintLoadItemResponse(UNASSIGNED_SYSTEM_ADDRESS, true, 515, 990); + auto* bitStream = static_cast<dServerMock*>(Game::server)->GetMostRecentBitStream(); + ASSERT_NE(bitStream, nullptr); + ASSERT_EQ(bitStream->GetNumberOfUnreadBits(), 200); + // First read in the packets' header + uint8_t rakNetPacketId{}; + uint16_t remoteConnectionType{}; + uint32_t packetId{}; + uint8_t always0{}; + + bitStream->Read(rakNetPacketId); + bitStream->Read(remoteConnectionType); + bitStream->Read(packetId); + bitStream->Read(always0); + ASSERT_EQ(rakNetPacketId, 0x53); + ASSERT_EQ(remoteConnectionType, 0x05); + ASSERT_EQ(packetId, 0x17); + ASSERT_EQ(always0, 0x00); + + // Next read in packet data + + uint8_t bSuccess{}; // unsigned bool + LWOOBJID previousId{}; + LWOOBJID newId{}; + bitStream->Read(bSuccess); + bitStream->Read(previousId); + bitStream->Read(newId); + ASSERT_EQ(bSuccess, static_cast<uint8_t>(true)); + ASSERT_EQ(previousId, 515); + ASSERT_EQ(newId, 990); + + ASSERT_EQ(bitStream->GetNumberOfUnreadBits(), 0); +} + +TEST_F(GameMessageTests, ControlBehaviorAddStrip) { + auto data = ReadFromFile("addStrip"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + AddStripMessage addStrip(ReadArrayFromBitStream(&inStream)); + ASSERT_FLOAT_EQ(addStrip.GetPosition().GetX(), 50.65); + ASSERT_FLOAT_EQ(addStrip.GetPosition().GetY(), 178.05); + ASSERT_EQ(addStrip.GetActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(addStrip.GetActionContext().GetStateId()), 0); + ASSERT_EQ(addStrip.GetBehaviorId(), -1); + ASSERT_EQ(addStrip.GetActionsToAdd().front().GetType(), "DropImagination"); + ASSERT_EQ(addStrip.GetActionsToAdd().front().GetValueParameterName(), "Amount"); + ASSERT_EQ(addStrip.GetActionsToAdd().front().GetValueParameterString(), ""); + ASSERT_FLOAT_EQ(addStrip.GetActionsToAdd().front().GetValueParameterDouble(), 1.0); +} + +TEST_F(GameMessageTests, ControlBehaviorRemoveStrip) { + auto data = ReadFromFile("removeStrip"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + RemoveStripMessage removeStrip(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(static_cast<int32_t>(removeStrip.GetActionContext().GetStripId()), 1); + ASSERT_EQ(static_cast<int32_t>(removeStrip.GetActionContext().GetStateId()), 0); + ASSERT_EQ(removeStrip.GetBehaviorId(), -1); +} + +TEST_F(GameMessageTests, ControlBehaviorMergeStrips) { + auto data = ReadFromFile("mergeStrips"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + MergeStripsMessage mergeStrips(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(mergeStrips.GetSourceActionContext().GetStripId(), 2); + ASSERT_EQ(mergeStrips.GetDestinationActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(mergeStrips.GetSourceActionContext().GetStateId()), 0); + ASSERT_EQ(static_cast<uint32_t>(mergeStrips.GetDestinationActionContext().GetStateId()), 0); + ASSERT_EQ(mergeStrips.GetDstActionIndex(), 0); + ASSERT_EQ(mergeStrips.GetBehaviorId(), -1); +} + +TEST_F(GameMessageTests, ControlBehaviorSplitStrip) { + auto data = ReadFromFile("splitStrip"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + SplitStripMessage splitStrip(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(splitStrip.GetBehaviorId(), -1); + + ASSERT_FLOAT_EQ(splitStrip.GetPosition().GetX(), 275.65); + ASSERT_FLOAT_EQ(splitStrip.GetPosition().GetY(), 28.7); + ASSERT_EQ(splitStrip.GetSourceActionContext().GetStripId(), 0); + ASSERT_EQ(splitStrip.GetDestinationActionContext().GetStripId(), 2); + ASSERT_EQ(static_cast<uint32_t>(splitStrip.GetSourceActionContext().GetStateId()), 0); + ASSERT_EQ(static_cast<uint32_t>(splitStrip.GetDestinationActionContext().GetStateId()), 0); + ASSERT_EQ(splitStrip.GetSrcActionIndex(), 1); +} + +TEST_F(GameMessageTests, ControlBehaviorUpdateStripUI) { + auto data = ReadFromFile("updateStripUI"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + UpdateStripUiMessage updateStripUi(ReadArrayFromBitStream(&inStream)); + ASSERT_FLOAT_EQ(updateStripUi.GetPosition().GetX(), 116.65); + ASSERT_FLOAT_EQ(updateStripUi.GetPosition().GetY(), 35.35); + ASSERT_EQ(updateStripUi.GetActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(updateStripUi.GetActionContext().GetStateId()), 0); + ASSERT_EQ(updateStripUi.GetBehaviorId(), -1); +} + +TEST_F(GameMessageTests, ControlBehaviorAddAction) { + auto data = ReadFromFile("addAction"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + AddActionMessage addAction(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(addAction.GetActionIndex(), 3); + ASSERT_EQ(addAction.GetActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(addAction.GetActionContext().GetStateId()), 0); + ASSERT_EQ(addAction.GetAction().GetType(), "DoDamage"); + ASSERT_EQ(addAction.GetAction().GetValueParameterName(), ""); + ASSERT_EQ(addAction.GetAction().GetValueParameterString(), ""); + ASSERT_EQ(addAction.GetAction().GetValueParameterDouble(), 0.0); + ASSERT_EQ(addAction.GetBehaviorId(), -1); +} + +TEST_F(GameMessageTests, ControlBehaviorMigrateActions) { + auto data = ReadFromFile("migrateActions"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + MigrateActionsMessage migrateActions(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(migrateActions.GetSrcActionIndex(), 1); + ASSERT_EQ(migrateActions.GetDstActionIndex(), 2); + ASSERT_EQ(migrateActions.GetSourceActionContext().GetStripId(), 1); + ASSERT_EQ(migrateActions.GetDestinationActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(migrateActions.GetSourceActionContext().GetStateId()), 0); + ASSERT_EQ(static_cast<uint32_t>(migrateActions.GetDestinationActionContext().GetStateId()), 0); + ASSERT_EQ(migrateActions.GetBehaviorId(), -1); +} + +TEST_F(GameMessageTests, ControlBehaviorRearrangeStrip) { + auto data = ReadFromFile("rearrangeStrip"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + RearrangeStripMessage rearrangeStrip(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(rearrangeStrip.GetSrcActionIndex(), 2); + ASSERT_EQ(rearrangeStrip.GetDstActionIndex(), 1); + ASSERT_EQ(rearrangeStrip.GetActionContext().GetStripId(), 0); + ASSERT_EQ(rearrangeStrip.GetBehaviorId(), -1); + ASSERT_EQ(static_cast<uint32_t>(rearrangeStrip.GetActionContext().GetStateId()), 0); +} + +TEST_F(GameMessageTests, ControlBehaviorAdd) { + auto data = ReadFromFile("add"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + AddMessage add(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(add.GetBehaviorId(), 10446); + ASSERT_EQ(add.GetBehaviorIndex(), 0); +} + +TEST_F(GameMessageTests, ControlBehaviorRemoveActions) { + auto data = ReadFromFile("removeActions"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + RemoveActionsMessage removeActions(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(removeActions.GetBehaviorId(), -1); + ASSERT_EQ(removeActions.GetActionIndex(), 1); + ASSERT_EQ(removeActions.GetActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(removeActions.GetActionContext().GetStateId()), 0); +} + +TEST_F(GameMessageTests, ControlBehaviorRename) { + auto data = ReadFromFile("rename"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + RenameMessage rename(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(rename.GetName(), "test"); + ASSERT_EQ(rename.GetBehaviorId(), -1); +} + +TEST_F(GameMessageTests, ControlBehaviorUpdateAction) { + auto data = ReadFromFile("updateAction"); + RakNet::BitStream inStream((unsigned char*)data.c_str(), data.length(), true); + UpdateActionMessage updateAction(ReadArrayFromBitStream(&inStream)); + ASSERT_EQ(updateAction.GetAction().GetType(), "FlyDown"); + ASSERT_EQ(updateAction.GetAction().GetValueParameterName(), "Distance"); + ASSERT_EQ(updateAction.GetAction().GetValueParameterString(), ""); + ASSERT_EQ(updateAction.GetAction().GetValueParameterDouble(), 50.0); + ASSERT_EQ(updateAction.GetBehaviorId(), -1); + ASSERT_EQ(updateAction.GetActionIndex(), 1); + ASSERT_EQ(updateAction.GetActionContext().GetStripId(), 0); + ASSERT_EQ(static_cast<uint32_t>(updateAction.GetActionContext().GetStateId()), 0); +} diff --git a/tests/dGameTests/dGameMessagesTests/LegacyGameMessageTests.cpp b/tests/dGameTests/dGameMessagesTests/LegacyGameMessageTests.cpp new file mode 100644 index 00000000..0226b0ad --- /dev/null +++ b/tests/dGameTests/dGameMessagesTests/LegacyGameMessageTests.cpp @@ -0,0 +1,298 @@ +#include <gtest/gtest.h> +#include "eGameMessageType.h" + +TEST(LegacyGameMessageTests, AssertLegacyGmValues) { + EXPECT_EQ(eGameMessageType::TELEPORT, (eGameMessageType)19); + EXPECT_EQ(eGameMessageType::SET_PLAYER_CONTROL_SCHEME, (eGameMessageType)26); + EXPECT_EQ(eGameMessageType::DROP_CLIENT_LOOT, (eGameMessageType)30); + EXPECT_EQ(eGameMessageType::DIE, (eGameMessageType)37); + EXPECT_EQ(eGameMessageType::REQUEST_DIE, (eGameMessageType)38); + EXPECT_EQ(eGameMessageType::PLAY_EMOTE, (eGameMessageType)41); + EXPECT_EQ(eGameMessageType::PLAY_ANIMATION, (eGameMessageType)43); + EXPECT_EQ(eGameMessageType::CONTROL_BEHAVIORS, (eGameMessageType)48); + EXPECT_EQ(eGameMessageType::SET_NAME, (eGameMessageType)72); + EXPECT_EQ(eGameMessageType::ECHO_START_SKILL, (eGameMessageType)118); + EXPECT_EQ(eGameMessageType::START_SKILL, (eGameMessageType)119); + EXPECT_EQ(eGameMessageType::VERIFY_ACK, (eGameMessageType)121); + EXPECT_EQ(eGameMessageType::ADD_SKILL, (eGameMessageType)127); + EXPECT_EQ(eGameMessageType::REMOVE_SKILL, (eGameMessageType)128); + EXPECT_EQ(eGameMessageType::SET_CURRENCY, (eGameMessageType)133); + EXPECT_EQ(eGameMessageType::PICKUP_CURRENCY, (eGameMessageType)137); + EXPECT_EQ(eGameMessageType::PICKUP_ITEM, (eGameMessageType)139); + EXPECT_EQ(eGameMessageType::TEAM_PICKUP_ITEM, (eGameMessageType)140); + EXPECT_EQ(eGameMessageType::PLAY_FX_EFFECT, (eGameMessageType)154); + EXPECT_EQ(eGameMessageType::STOP_FX_EFFECT, (eGameMessageType)155); + EXPECT_EQ(eGameMessageType::REQUEST_RESURRECT, (eGameMessageType)159); + EXPECT_EQ(eGameMessageType::RESURRECT, (eGameMessageType)160); + EXPECT_EQ(eGameMessageType::PUSH_EQUIPPED_ITEMS_STATE, (eGameMessageType)191); + EXPECT_EQ(eGameMessageType::POP_EQUIPPED_ITEMS_STATE, (eGameMessageType)192); + EXPECT_EQ(eGameMessageType::SET_GM_LEVEL, (eGameMessageType)193); + EXPECT_EQ(eGameMessageType::SET_STUNNED, (eGameMessageType)198); + EXPECT_EQ(eGameMessageType::SET_STUN_IMMUNITY, (eGameMessageType)200); + EXPECT_EQ(eGameMessageType::KNOCKBACK, (eGameMessageType)202); + EXPECT_EQ(eGameMessageType::REBUILD_CANCEL, (eGameMessageType)209); + EXPECT_EQ(eGameMessageType::ENABLE_REBUILD, (eGameMessageType)213); + EXPECT_EQ(eGameMessageType::MOVE_ITEM_IN_INVENTORY, (eGameMessageType)224); + EXPECT_EQ(eGameMessageType::ADD_ITEM_TO_INVENTORY_CLIENT_SYNC, (eGameMessageType)227); + EXPECT_EQ(eGameMessageType::REMOVE_ITEM_FROM_INVENTORY, (eGameMessageType)230); + EXPECT_EQ(eGameMessageType::EQUIP_INVENTORY, (eGameMessageType)231); + EXPECT_EQ(eGameMessageType::UN_EQUIP_INVENTORY, (eGameMessageType)233); + EXPECT_EQ(eGameMessageType::OFFER_MISSION, (eGameMessageType)248); + EXPECT_EQ(eGameMessageType::RESPOND_TO_MISSION, (eGameMessageType)249); + EXPECT_EQ(eGameMessageType::NOTIFY_MISSION, (eGameMessageType)254); + EXPECT_EQ(eGameMessageType::NOTIFY_MISSION_TASK, (eGameMessageType)255); + EXPECT_EQ(eGameMessageType::REBUILD_NOTIFY_STATE, (eGameMessageType)336); + EXPECT_EQ(eGameMessageType::TERMINATE_INTERACTION, (eGameMessageType)357); + EXPECT_EQ(eGameMessageType::SERVER_TERMINATE_INTERACTION, (eGameMessageType)358); + EXPECT_EQ(eGameMessageType::REQUEST_USE, (eGameMessageType)364); + EXPECT_EQ(eGameMessageType::VENDOR_OPEN_WINDOW, (eGameMessageType)369); + EXPECT_EQ(eGameMessageType::BUY_FROM_VENDOR, (eGameMessageType)373); + EXPECT_EQ(eGameMessageType::SELL_TO_VENDOR, (eGameMessageType)374); + EXPECT_EQ(eGameMessageType::TEAM_SET_OFF_WORLD_FLAG, (eGameMessageType)383); + EXPECT_EQ(eGameMessageType::SET_INVENTORY_SIZE, (eGameMessageType)389); + EXPECT_EQ(eGameMessageType::ACKNOWLEDGE_POSSESSION, (eGameMessageType)391); + EXPECT_EQ(eGameMessageType::SET_SHOOTING_GALLERY_PARAMS, (eGameMessageType)400); + EXPECT_EQ(eGameMessageType::REQUEST_ACTIVITY_START_STOP, (eGameMessageType)402); + EXPECT_EQ(eGameMessageType::REQUEST_ACTIVITY_ENTER, (eGameMessageType)403); + EXPECT_EQ(eGameMessageType::REQUEST_ACTIVITY_EXIT, (eGameMessageType)404); + EXPECT_EQ(eGameMessageType::ACTIVITY_ENTER, (eGameMessageType)405); + EXPECT_EQ(eGameMessageType::ACTIVITY_EXIT, (eGameMessageType)406); + EXPECT_EQ(eGameMessageType::ACTIVITY_START, (eGameMessageType)407); + EXPECT_EQ(eGameMessageType::ACTIVITY_STOP, (eGameMessageType)408); + EXPECT_EQ(eGameMessageType::SHOOTING_GALLERY_CLIENT_AIM_UPDATE, (eGameMessageType)409); + EXPECT_EQ(eGameMessageType::SHOOTING_GALLERY_FIRE, (eGameMessageType)411); + EXPECT_EQ(eGameMessageType::REQUEST_VENDOR_STATUS_UPDATE, (eGameMessageType)416); + EXPECT_EQ(eGameMessageType::VENDOR_STATUS_UPDATE, (eGameMessageType)417); + EXPECT_EQ(eGameMessageType::NOTIFY_CLIENT_SHOOTING_GALLERY_SCORE, (eGameMessageType)425); + EXPECT_EQ(eGameMessageType::CONSUME_CLIENT_ITEM, (eGameMessageType)427); + EXPECT_EQ(eGameMessageType::CLIENT_ITEM_CONSUMED, (eGameMessageType)428); + EXPECT_EQ(eGameMessageType::UPDATE_SHOOTING_GALLERY_ROTATION, (eGameMessageType)448); + EXPECT_EQ(eGameMessageType::SET_FLAG, (eGameMessageType)471); + EXPECT_EQ(eGameMessageType::NOTIFY_CLIENT_FLAG_CHANGE, (eGameMessageType)472); + EXPECT_EQ(eGameMessageType::VENDOR_TRANSACTION_RESULT, (eGameMessageType)476); + EXPECT_EQ(eGameMessageType::HAS_BEEN_COLLECTED, (eGameMessageType)486); + EXPECT_EQ(eGameMessageType::DISPLAY_CHAT_BUBBLE, (eGameMessageType)495); + EXPECT_EQ(eGameMessageType::SPAWN_PET, (eGameMessageType)498); + EXPECT_EQ(eGameMessageType::DESPAWN_PET, (eGameMessageType)499); + EXPECT_EQ(eGameMessageType::PLAYER_LOADED, (eGameMessageType)505); + EXPECT_EQ(eGameMessageType::PLAYER_READY, (eGameMessageType)509); + EXPECT_EQ(eGameMessageType::REQUEST_LINKED_MISSION, (eGameMessageType)515); + EXPECT_EQ(eGameMessageType::INVALID_ZONE_TRANSFER_LIST, (eGameMessageType)519); + EXPECT_EQ(eGameMessageType::MISSION_DIALOGUE_OK, (eGameMessageType)520); + EXPECT_EQ(eGameMessageType::DISPLAY_MESSAGE_BOX, (eGameMessageType)529); + EXPECT_EQ(eGameMessageType::MESSAGE_BOX_RESPOND, (eGameMessageType)530); + EXPECT_EQ(eGameMessageType::CHOICE_BOX_RESPOND, (eGameMessageType)531); + EXPECT_EQ(eGameMessageType::SMASH, (eGameMessageType)537); + EXPECT_EQ(eGameMessageType::UN_SMASH, (eGameMessageType)538); + EXPECT_EQ(eGameMessageType::PLACE_MODEL_RESPONSE, (eGameMessageType)547); + EXPECT_EQ(eGameMessageType::SET_SHOOTING_GALLERY_RETICULE_EFFECT, (eGameMessageType)546); + EXPECT_EQ(eGameMessageType::SET_JET_PACK_MODE, (eGameMessageType)561); + EXPECT_EQ(eGameMessageType::REGISTER_PET_ID, (eGameMessageType)565); + EXPECT_EQ(eGameMessageType::REGISTER_PET_DBID, (eGameMessageType)566); + EXPECT_EQ(eGameMessageType::SHOW_ACTIVITY_COUNTDOWN, (eGameMessageType)568); + EXPECT_EQ(eGameMessageType::START_ACTIVITY_TIME, (eGameMessageType)576); + EXPECT_EQ(eGameMessageType::ACTIVITY_PAUSE, (eGameMessageType)602); + EXPECT_EQ(eGameMessageType::USE_NON_EQUIPMENT_ITEM, (eGameMessageType)603); + EXPECT_EQ(eGameMessageType::USE_ITEM_RESULT, (eGameMessageType)607); + EXPECT_EQ(eGameMessageType::COMMAND_PET, (eGameMessageType)640); + EXPECT_EQ(eGameMessageType::PET_RESPONSE, (eGameMessageType)641); + EXPECT_EQ(eGameMessageType::REQUEST_ACTIVITY_SUMMARY_LEADERBOARD_DATA, (eGameMessageType)648); + EXPECT_EQ(eGameMessageType::SEND_ACTIVITY_SUMMARY_LEADERBOARD_DATA, (eGameMessageType)649); + EXPECT_EQ(eGameMessageType::NOTIFY_OBJECT, (eGameMessageType)656); + EXPECT_EQ(eGameMessageType::CLIENT_NOTIFY_PET, (eGameMessageType)659); + EXPECT_EQ(eGameMessageType::NOTIFY_PET, (eGameMessageType)660); + EXPECT_EQ(eGameMessageType::NOTIFY_PET_TAMING_MINIGAME, (eGameMessageType)661); + EXPECT_EQ(eGameMessageType::START_SERVER_PET_MINIGAME_TIMER, (eGameMessageType)662); + EXPECT_EQ(eGameMessageType::CLIENT_EXIT_TAMING_MINIGAME, (eGameMessageType)663); + EXPECT_EQ(eGameMessageType::PET_NAME_CHANGED, (eGameMessageType)686); + EXPECT_EQ(eGameMessageType::PET_TAMING_MINIGAME_RESULT, (eGameMessageType)667); + EXPECT_EQ(eGameMessageType::PET_TAMING_TRY_BUILD_RESULT, (eGameMessageType)668); + EXPECT_EQ(eGameMessageType::NOTIFY_TAMING_BUILD_SUCCESS, (eGameMessageType)673); + EXPECT_EQ(eGameMessageType::NOTIFY_TAMING_MODEL_LOADED_ON_SERVER, (eGameMessageType)674); + EXPECT_EQ(eGameMessageType::ACTIVATE_BUBBLE_BUFF, (eGameMessageType)678); + EXPECT_EQ(eGameMessageType::DECTIVATE_BUBBLE_BUFF, (eGameMessageType)679); + EXPECT_EQ(eGameMessageType::ADD_PET_TO_PLAYER, (eGameMessageType)681); + EXPECT_EQ(eGameMessageType::REQUEST_SET_PET_NAME, (eGameMessageType)683); + EXPECT_EQ(eGameMessageType::SET_PET_NAME, (eGameMessageType)684); + EXPECT_EQ(eGameMessageType::NOTIFY_TAMING_PUZZLE_SELECTED, (eGameMessageType)675); + EXPECT_EQ(eGameMessageType::SHOW_PET_ACTION_BUTTON, (eGameMessageType)692); + EXPECT_EQ(eGameMessageType::SET_EMOTE_LOCK_STATE, (eGameMessageType)693); + EXPECT_EQ(eGameMessageType::USE_ITEM_REQUIREMENTS_RESPONSE, (eGameMessageType)703); + EXPECT_EQ(eGameMessageType::PLAY_EMBEDDED_EFFECT_ON_ALL_CLIENTS_NEAR_OBJECT, (eGameMessageType)713); + EXPECT_EQ(eGameMessageType::DOWNLOAD_PROPERTY_DATA, (eGameMessageType)716); + EXPECT_EQ(eGameMessageType::QUERY_PROPERTY_DATA, (eGameMessageType)717); + EXPECT_EQ(eGameMessageType::PROPERTY_EDITOR_BEGIN, (eGameMessageType)724); + EXPECT_EQ(eGameMessageType::PROPERTY_EDITOR_END, (eGameMessageType)725); + EXPECT_EQ(eGameMessageType::IS_MINIFIG_IN_A_BUBBLE, (eGameMessageType)729); + EXPECT_EQ(eGameMessageType::START_PATHING, (eGameMessageType)733); + EXPECT_EQ(eGameMessageType::ACTIVATE_BUBBLE_BUFF_FROM_SERVER, (eGameMessageType)734); + EXPECT_EQ(eGameMessageType::DEACTIVATE_BUBBLE_BUFF_FROM_SERVER, (eGameMessageType)735); + EXPECT_EQ(eGameMessageType::NOTIFY_CLIENT_ZONE_OBJECT, (eGameMessageType)737); + EXPECT_EQ(eGameMessageType::UPDATE_REPUTATION, (eGameMessageType)746); + EXPECT_EQ(eGameMessageType::PROPERTY_RENTAL_RESPONSE, (eGameMessageType)750); + EXPECT_EQ(eGameMessageType::REQUEST_PLATFORM_RESYNC, (eGameMessageType)760); + EXPECT_EQ(eGameMessageType::PLATFORM_RESYNC, (eGameMessageType)761); + EXPECT_EQ(eGameMessageType::PLAY_CINEMATIC, (eGameMessageType)762); + EXPECT_EQ(eGameMessageType::END_CINEMATIC, (eGameMessageType)763); + EXPECT_EQ(eGameMessageType::CINEMATIC_UPDATE, (eGameMessageType)764); + EXPECT_EQ(eGameMessageType::TOGGLE_GHOST_REFERENCE_OVERRIDE, (eGameMessageType)767); + EXPECT_EQ(eGameMessageType::SET_GHOST_REFERENCE_POSITION, (eGameMessageType)768); + EXPECT_EQ(eGameMessageType::FIRE_EVENT_SERVER_SIDE, (eGameMessageType)770); + EXPECT_EQ(eGameMessageType::SCRIPT_NETWORK_VAR_UPDATE, (eGameMessageType)781); + EXPECT_EQ(eGameMessageType::UPDATE_MODEL_FROM_CLIENT, (eGameMessageType)793); + EXPECT_EQ(eGameMessageType::DELETE_MODEL_FROM_CLIENT, (eGameMessageType)794); + EXPECT_EQ(eGameMessageType::PLAY_ND_AUDIO_EMITTER, (eGameMessageType)821); + EXPECT_EQ(eGameMessageType::PLAY2_D_AMBIENT_SOUND, (eGameMessageType)831); + EXPECT_EQ(eGameMessageType::ENTER_PROPERTY1, (eGameMessageType)840); + EXPECT_EQ(eGameMessageType::ENTER_PROPERTY2, (eGameMessageType)841); + EXPECT_EQ(eGameMessageType::PROPERTY_ENTRANCE_SYNC, (eGameMessageType)842); + EXPECT_EQ(eGameMessageType::PROPERTY_SELECT_QUERY, (eGameMessageType)845); + EXPECT_EQ(eGameMessageType::PARSE_CHAT_MESSAGE, (eGameMessageType)850); + EXPECT_EQ(eGameMessageType::BROADCAST_TEXT_TO_CHATBOX, (eGameMessageType)858); + EXPECT_EQ(eGameMessageType::OPEN_PROPERTY_MANAGEMENT, (eGameMessageType)860); + EXPECT_EQ(eGameMessageType::OPEN_PROPERTY_VENDOR, (eGameMessageType)861); + EXPECT_EQ(eGameMessageType::UPDATE_PROPERTY_OR_MODEL_FOR_FILTER_CHECK, (eGameMessageType)863); + EXPECT_EQ(eGameMessageType::CLIENT_TRADE_REQUEST, (eGameMessageType)868); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_REQUEST, (eGameMessageType)869); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_INVITE, (eGameMessageType)870); + EXPECT_EQ(eGameMessageType::CLIENT_TRADE_REPLY, (eGameMessageType)871); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_REPLY, (eGameMessageType)872); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_INITIAL_REPLY, (eGameMessageType)873); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_FINAL_REPLY, (eGameMessageType)874); + EXPECT_EQ(eGameMessageType::CLIENT_TRADE_UPDATE, (eGameMessageType)875); + EXPECT_EQ(eGameMessageType::SERVER_SIDE_TRADE_UPDATE, (eGameMessageType)876); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_UPDATE, (eGameMessageType)877); + EXPECT_EQ(eGameMessageType::CLIENT_TRADE_CANCEL, (eGameMessageType)878); + EXPECT_EQ(eGameMessageType::CLIENT_SIDE_TRADE_CANCEL, (eGameMessageType)879); + EXPECT_EQ(eGameMessageType::CLIENT_TRADE_ACCEPT, (eGameMessageType)880); + EXPECT_EQ(eGameMessageType::SERVER_SIDE_TRADE_ACCEPT, (eGameMessageType)881); + EXPECT_EQ(eGameMessageType::SERVER_SIDE_TRADE_CANCEL, (eGameMessageType)882); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_CANCEL, (eGameMessageType)883); + EXPECT_EQ(eGameMessageType::SERVER_TRADE_ACCEPT, (eGameMessageType)884); + EXPECT_EQ(eGameMessageType::READY_FOR_UPDATES, (eGameMessageType)888); + EXPECT_EQ(eGameMessageType::ORIENT_TO_OBJECT, (eGameMessageType)905); + EXPECT_EQ(eGameMessageType::ORIENT_TO_POSITION, (eGameMessageType)906); + EXPECT_EQ(eGameMessageType::ORIENT_TO_ANGLE, (eGameMessageType)907); + EXPECT_EQ(eGameMessageType::BOUNCER_ACTIVE_STATUS, (eGameMessageType)942); + EXPECT_EQ(eGameMessageType::UN_USE_BBB_MODEL, (eGameMessageType)999); + EXPECT_EQ(eGameMessageType::BBB_LOAD_ITEM_REQUEST, (eGameMessageType)1000); + EXPECT_EQ(eGameMessageType::BBB_SAVE_REQUEST, (eGameMessageType)1001); + EXPECT_EQ(eGameMessageType::BBB_SAVE_RESPONSE, (eGameMessageType)1005); + EXPECT_EQ(eGameMessageType::NOTIFY_CLIENT_OBJECT, (eGameMessageType)1042); + EXPECT_EQ(eGameMessageType::DISPLAY_ZONE_SUMMARY, (eGameMessageType)1043); + EXPECT_EQ(eGameMessageType::ZONE_SUMMARY_DISMISSED, (eGameMessageType)1044); + EXPECT_EQ(eGameMessageType::ACTIVITY_STATE_CHANGE_REQUEST, (eGameMessageType)1053); + EXPECT_EQ(eGameMessageType::MODIFY_PLAYER_ZONE_STATISTIC, (eGameMessageType)1046); + EXPECT_EQ(eGameMessageType::START_BUILDING_WITH_ITEM, (eGameMessageType)1057); + EXPECT_EQ(eGameMessageType::START_ARRANGING_WITH_ITEM, (eGameMessageType)1061); + EXPECT_EQ(eGameMessageType::FINISH_ARRANGING_WITH_ITEM, (eGameMessageType)1062); + EXPECT_EQ(eGameMessageType::DONE_ARRANGING_WITH_ITEM, (eGameMessageType)1063); + EXPECT_EQ(eGameMessageType::SET_BUILD_MODE, (eGameMessageType)1068); + EXPECT_EQ(eGameMessageType::BUILD_MODE_SET, (eGameMessageType)1069); + EXPECT_EQ(eGameMessageType::SET_BUILD_MODE_CONFIRMED, (eGameMessageType)1073); + EXPECT_EQ(eGameMessageType::NOTIFY_CLIENT_FAILED_PRECONDITION, (eGameMessageType)1081); + EXPECT_EQ(eGameMessageType::MOVE_ITEM_BETWEEN_INVENTORY_TYPES, (eGameMessageType)1093); + EXPECT_EQ(eGameMessageType::MODULAR_BUILD_BEGIN, (eGameMessageType)1094); + EXPECT_EQ(eGameMessageType::MODULAR_BUILD_END, (eGameMessageType)1095); + EXPECT_EQ(eGameMessageType::MODULAR_BUILD_MOVE_AND_EQUIP, (eGameMessageType)1096); + EXPECT_EQ(eGameMessageType::MODULAR_BUILD_FINISH, (eGameMessageType)1097); + EXPECT_EQ(eGameMessageType::REPORT_BUG, (eGameMessageType)1198); + EXPECT_EQ(eGameMessageType::MISSION_DIALOGUE_CANCELLED, (eGameMessageType)1129); + EXPECT_EQ(eGameMessageType::ECHO_SYNC_SKILL, (eGameMessageType)1144); + EXPECT_EQ(eGameMessageType::SYNC_SKILL, (eGameMessageType)1145); + EXPECT_EQ(eGameMessageType::REQUEST_SERVER_PROJECTILE_IMPACT, (eGameMessageType)1148); + EXPECT_EQ(eGameMessageType::DO_CLIENT_PROJECTILE_IMPACT, (eGameMessageType)1151); + EXPECT_EQ(eGameMessageType::MODULAR_BUILD_CONVERT_MODEL, (eGameMessageType)1155); + EXPECT_EQ(eGameMessageType::SET_PLAYER_ALLOWED_RESPAWN, (eGameMessageType)1165); + EXPECT_EQ(eGameMessageType::UI_MESSAGE_SERVER_TO_SINGLE_CLIENT, (eGameMessageType)1184); + EXPECT_EQ(eGameMessageType::UI_MESSAGE_SERVER_TO_ALL_CLIENTS, (eGameMessageType)1185); + EXPECT_EQ(eGameMessageType::PET_TAMING_TRY_BUILD, (eGameMessageType)1197); + EXPECT_EQ(eGameMessageType::REQUEST_SMASH_PLAYER, (eGameMessageType)1202); + EXPECT_EQ(eGameMessageType::FIRE_EVENT_CLIENT_SIDE, (eGameMessageType)1213); + EXPECT_EQ(eGameMessageType::TOGGLE_GM_INVIS, (eGameMessageType)1218); + EXPECT_EQ(eGameMessageType::CHANGE_OBJECT_WORLD_STATE, (eGameMessageType)1223); + EXPECT_EQ(eGameMessageType::VEHICLE_LOCK_INPUT, (eGameMessageType)1230); + EXPECT_EQ(eGameMessageType::VEHICLE_UNLOCK_INPUT, (eGameMessageType)1231); + EXPECT_EQ(eGameMessageType::RACING_RESET_PLAYER_TO_LAST_RESET, (eGameMessageType)1252); + EXPECT_EQ(eGameMessageType::RACING_SERVER_SET_PLAYER_LAP_AND_PLANE, (eGameMessageType)1253); + EXPECT_EQ(eGameMessageType::RACING_SET_PLAYER_RESET_INFO, (eGameMessageType)1254); + EXPECT_EQ(eGameMessageType::RACING_PLAYER_INFO_RESET_FINISHED, (eGameMessageType)1255); + EXPECT_EQ(eGameMessageType::LOCK_NODE_ROTATION, (eGameMessageType)1260); + EXPECT_EQ(eGameMessageType::VEHICLE_SET_WHEEL_LOCK_STATE, (eGameMessageType)1273); + EXPECT_EQ(eGameMessageType::NOTIFY_VEHICLE_OF_RACING_OBJECT, (eGameMessageType)1276); + EXPECT_EQ(eGameMessageType::SET_NAME_BILLBOARD_STATE, (eGameMessageType)1284); + EXPECT_EQ(eGameMessageType::PLAYER_REACHED_RESPAWN_CHECKPOINT, (eGameMessageType)1296); + EXPECT_EQ(eGameMessageType::HANDLE_UGC_POST_DELETE_BASED_ON_EDIT_MODE, (eGameMessageType)1300); + EXPECT_EQ(eGameMessageType::HANDLE_UGC_POST_CREATE_BASED_ON_EDIT_MODE, (eGameMessageType)1301); + EXPECT_EQ(eGameMessageType::PROPERTY_CONTENTS_FROM_CLIENT, (eGameMessageType)1305); + EXPECT_EQ(eGameMessageType::GET_MODELS_ON_PROPERTY, (eGameMessageType)1306); + EXPECT_EQ(eGameMessageType::MATCH_REQUEST, (eGameMessageType)1308); + EXPECT_EQ(eGameMessageType::MATCH_RESPONSE, (eGameMessageType)1309); + EXPECT_EQ(eGameMessageType::MATCH_UPDATE, (eGameMessageType)1310); + EXPECT_EQ(eGameMessageType::MODULE_ASSEMBLY_DB_DATA_FOR_CLIENT, (eGameMessageType)1131); + EXPECT_EQ(eGameMessageType::MODULE_ASSEMBLY_QUERY_DATA, (eGameMessageType)1132); + EXPECT_EQ(eGameMessageType::SHOW_BILLBOARD_INTERACT_ICON, (eGameMessageType)1337); + EXPECT_EQ(eGameMessageType::CHANGE_IDLE_FLAGS, (eGameMessageType)1338); + EXPECT_EQ(eGameMessageType::VEHICLE_ADD_PASSIVE_BOOST_ACTION, (eGameMessageType)1340); + EXPECT_EQ(eGameMessageType::VEHICLE_REMOVE_PASSIVE_BOOST_ACTION, (eGameMessageType)1341); + EXPECT_EQ(eGameMessageType::NOTIFY_SERVER_VEHICLE_ADD_PASSIVE_BOOST_ACTION, (eGameMessageType)1342); + EXPECT_EQ(eGameMessageType::NOTIFY_SERVER_VEHICLE_REMOVE_PASSIVE_BOOST_ACTION, (eGameMessageType)1343); + EXPECT_EQ(eGameMessageType::VEHICLE_ADD_SLOWDOWN_ACTION, (eGameMessageType)1344); + EXPECT_EQ(eGameMessageType::VEHICLE_REMOVE_SLOWDOWN_ACTION, (eGameMessageType)1345); + EXPECT_EQ(eGameMessageType::NOTIFY_SERVER_VEHICLE_ADD_SLOWDOWN_ACTION, (eGameMessageType)1346); + EXPECT_EQ(eGameMessageType::NOTIFY_SERVER_VEHICLE_REMOVE_SLOWDOWN_ACTION, (eGameMessageType)1347); + EXPECT_EQ(eGameMessageType::BUYBACK_FROM_VENDOR, (eGameMessageType)1350); + EXPECT_EQ(eGameMessageType::SET_PROPERTY_ACCESS, (eGameMessageType)1366); + EXPECT_EQ(eGameMessageType::ZONE_PROPERTY_MODEL_PLACED, (eGameMessageType)1369); + EXPECT_EQ(eGameMessageType::ZONE_PROPERTY_MODEL_ROTATED, (eGameMessageType)1370); + EXPECT_EQ(eGameMessageType::ZONE_PROPERTY_MODEL_REMOVED_WHILE_EQUIPPED, (eGameMessageType)1371); + EXPECT_EQ(eGameMessageType::ZONE_PROPERTY_MODEL_EQUIPPED, (eGameMessageType)1372); + EXPECT_EQ(eGameMessageType::ZONE_PROPERTY_MODEL_PICKED_UP, (eGameMessageType)1373); + EXPECT_EQ(eGameMessageType::ZONE_PROPERTY_MODEL_REMOVED, (eGameMessageType)1374); + EXPECT_EQ(eGameMessageType::NOTIFY_RACING_CLIENT, (eGameMessageType)1390); + EXPECT_EQ(eGameMessageType::RACING_PLAYER_HACK_CAR, (eGameMessageType)1391); + EXPECT_EQ(eGameMessageType::RACING_PLAYER_LOADED, (eGameMessageType)1392); + EXPECT_EQ(eGameMessageType::RACING_CLIENT_READY, (eGameMessageType)1393); + EXPECT_EQ(eGameMessageType::UPDATE_CHAT_MODE, (eGameMessageType)1395); + EXPECT_EQ(eGameMessageType::VEHICLE_NOTIFY_FINISHED_RACE, (eGameMessageType)1396); + EXPECT_EQ(eGameMessageType::SET_CONSUMABLE_ITEM, (eGameMessageType)1409); + EXPECT_EQ(eGameMessageType::SET_STATUS_IMMUNITY, (eGameMessageType)1435); + EXPECT_EQ(eGameMessageType::SET_PET_NAME_MODERATED, (eGameMessageType)1448); + EXPECT_EQ(eGameMessageType::MODIFY_LEGO_SCORE, (eGameMessageType)1459); + EXPECT_EQ(eGameMessageType::RESTORE_TO_POST_LOAD_STATS, (eGameMessageType)1468); + EXPECT_EQ(eGameMessageType::SET_RAIL_MOVEMENT, (eGameMessageType)1471); + EXPECT_EQ(eGameMessageType::START_RAIL_MOVEMENT, (eGameMessageType)1472); + EXPECT_EQ(eGameMessageType::CANCEL_RAIL_MOVEMENT, (eGameMessageType)1474); + EXPECT_EQ(eGameMessageType::CLIENT_RAIL_MOVEMENT_READY, (eGameMessageType)1476); + EXPECT_EQ(eGameMessageType::PLAYER_RAIL_ARRIVED_NOTIFICATION, (eGameMessageType)1477); + EXPECT_EQ(eGameMessageType::UPDATE_PLAYER_STATISTIC, (eGameMessageType)1481); + EXPECT_EQ(eGameMessageType::MODULAR_ASSEMBLY_NIF_COMPLETED, (eGameMessageType)1498); + EXPECT_EQ(eGameMessageType::NOTIFY_NOT_ENOUGH_INV_SPACE, (eGameMessageType)1516); + EXPECT_EQ(eGameMessageType::TEAM_SET_LEADER, (eGameMessageType)1557); + EXPECT_EQ(eGameMessageType::TEAM_INVITE_CONFIRM, (eGameMessageType)1558); + EXPECT_EQ(eGameMessageType::TEAM_GET_STATUS_RESPONSE, (eGameMessageType)1559); + EXPECT_EQ(eGameMessageType::TEAM_ADD_PLAYER, (eGameMessageType)1562); + EXPECT_EQ(eGameMessageType::TEAM_REMOVE_PLAYER, (eGameMessageType)1563); + EXPECT_EQ(eGameMessageType::START_CELEBRATION_EFFECT, (eGameMessageType)1618); + EXPECT_EQ(eGameMessageType::ADD_BUFF, (eGameMessageType)1647); + EXPECT_EQ(eGameMessageType::SERVER_DONE_LOADING_ALL_OBJECTS, (eGameMessageType)1642); + EXPECT_EQ(eGameMessageType::PLACE_PROPERTY_MODEL, (eGameMessageType)1170); + EXPECT_EQ(eGameMessageType::VEHICLE_NOTIFY_HIT_IMAGINATION_SERVER, (eGameMessageType)1606); + EXPECT_EQ(eGameMessageType::ADD_RUN_SPEED_MODIFIER, (eGameMessageType)1505); + EXPECT_EQ(eGameMessageType::GET_HOT_PROPERTY_DATA, (eGameMessageType)1511); + EXPECT_EQ(eGameMessageType::SEND_HOT_PROPERTY_DATA, (eGameMessageType)1510); + EXPECT_EQ(eGameMessageType::REMOVE_RUN_SPEED_MODIFIER, (eGameMessageType)1506); + EXPECT_EQ(eGameMessageType::UPDATE_PROPERTY_PERFORMANCE_COST, (eGameMessageType)1547); + EXPECT_EQ(eGameMessageType::PROPERTY_ENTRANCE_BEGIN, (eGameMessageType)1553); + EXPECT_EQ(eGameMessageType::SET_RESURRECT_RESTORE_VALUES, (eGameMessageType)1591); + EXPECT_EQ(eGameMessageType::VEHICLE_STOP_BOOST, (eGameMessageType)1617); + EXPECT_EQ(eGameMessageType::REMOVE_BUFF, (eGameMessageType)1648); + EXPECT_EQ(eGameMessageType::REQUEST_MOVE_ITEM_BETWEEN_INVENTORY_TYPES, (eGameMessageType)1666); + EXPECT_EQ(eGameMessageType::RESPONSE_MOVE_ITEM_BETWEEN_INVENTORY_TYPES, (eGameMessageType)1667); + EXPECT_EQ(eGameMessageType::PLAYER_SET_CAMERA_CYCLING_MODE, (eGameMessageType)1676); + EXPECT_EQ(eGameMessageType::SET_MOUNT_INVENTORY_ID, (eGameMessageType)1727); + EXPECT_EQ(eGameMessageType::NOTIFY_SERVER_LEVEL_PROCESSING_COMPLETE, (eGameMessageType)1734); + EXPECT_EQ(eGameMessageType::NOTIFY_LEVEL_REWARDS, (eGameMessageType)1735); + EXPECT_EQ(eGameMessageType::DISMOUNT_COMPLETE, (eGameMessageType)1756); + EXPECT_EQ(eGameMessageType::MARK_INVENTORY_ITEM_AS_ACTIVE, (eGameMessageType)1767); +} diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/CMakeLists.txt b/tests/dGameTests/dGameMessagesTests/TestBitStreams/CMakeLists.txt new file mode 100644 index 00000000..e32ed3ef --- /dev/null +++ b/tests/dGameTests/dGameMessagesTests/TestBitStreams/CMakeLists.txt @@ -0,0 +1,24 @@ +set(GAMEMESSAGE_TESTBITSTREAMS +"sendBehaviorListToClient" +"modelTypeChanged" +"toggleExecutionUpdates" +"addStrip" +"removeStrip" +"mergeStrips" +"splitStrip" +"updateStripUI" +"addAction" +"migrateActions" +"rearrangeStrip" +"add" +"removeActions" +"rename" +"updateAction" +) + +# Get the folder name and prepend it to the files above +get_filename_component(thisFolderName ${CMAKE_CURRENT_SOURCE_DIR} NAME) +list(TRANSFORM GAMEMESSAGE_TESTBITSTREAMS PREPEND "${thisFolderName}/") + +# Export our list of files +set(GAMEMESSAGE_TESTBITSTREAMS ${GAMEMESSAGE_TESTBITSTREAMS} PARENT_SCOPE) diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/add b/tests/dGameTests/dGameMessagesTests/TestBitStreams/add new file mode 100644 index 00000000..13c0dd92 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/add differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/addAction b/tests/dGameTests/dGameMessagesTests/TestBitStreams/addAction new file mode 100644 index 00000000..d91d0ee4 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/addAction differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/addStrip b/tests/dGameTests/dGameMessagesTests/TestBitStreams/addStrip new file mode 100644 index 00000000..60ba6521 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/addStrip differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/mergeStrips b/tests/dGameTests/dGameMessagesTests/TestBitStreams/mergeStrips new file mode 100644 index 00000000..062fd10b Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/mergeStrips differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/migrateActions b/tests/dGameTests/dGameMessagesTests/TestBitStreams/migrateActions new file mode 100644 index 00000000..217f44d9 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/migrateActions differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/modelTypeChanged b/tests/dGameTests/dGameMessagesTests/TestBitStreams/modelTypeChanged new file mode 100644 index 00000000..ef282ce2 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/modelTypeChanged differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/rearrangeStrip b/tests/dGameTests/dGameMessagesTests/TestBitStreams/rearrangeStrip new file mode 100644 index 00000000..06dda90b Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/rearrangeStrip differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/removeActions b/tests/dGameTests/dGameMessagesTests/TestBitStreams/removeActions new file mode 100644 index 00000000..56e158e5 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/removeActions differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/removeStrip b/tests/dGameTests/dGameMessagesTests/TestBitStreams/removeStrip new file mode 100644 index 00000000..46ca0640 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/removeStrip differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/rename b/tests/dGameTests/dGameMessagesTests/TestBitStreams/rename new file mode 100644 index 00000000..bc8827dc Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/rename differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/sendBehaviorListToClient b/tests/dGameTests/dGameMessagesTests/TestBitStreams/sendBehaviorListToClient new file mode 100644 index 00000000..fcca696d Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/sendBehaviorListToClient differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/splitStrip b/tests/dGameTests/dGameMessagesTests/TestBitStreams/splitStrip new file mode 100644 index 00000000..a23c1682 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/splitStrip differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/toggleExecutionUpdates b/tests/dGameTests/dGameMessagesTests/TestBitStreams/toggleExecutionUpdates new file mode 100644 index 00000000..02a72181 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/toggleExecutionUpdates differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/updateAction b/tests/dGameTests/dGameMessagesTests/TestBitStreams/updateAction new file mode 100644 index 00000000..e007d5e6 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/updateAction differ diff --git a/tests/dGameTests/dGameMessagesTests/TestBitStreams/updateStripUI b/tests/dGameTests/dGameMessagesTests/TestBitStreams/updateStripUI new file mode 100644 index 00000000..7d0eed92 Binary files /dev/null and b/tests/dGameTests/dGameMessagesTests/TestBitStreams/updateStripUI differ diff --git a/thirdparty/CMakeMariaDBLists.txt b/thirdparty/CMakeMariaDBLists.txt index 55e95c4d..81b15a6f 100644 --- a/thirdparty/CMakeMariaDBLists.txt +++ b/thirdparty/CMakeMariaDBLists.txt @@ -10,30 +10,42 @@ if(WIN32 AND NOT MARIADB_BUILD_SOURCE) file(MAKE_DIRECTORY "${MARIADB_MSI_DIR}") file(MAKE_DIRECTORY "${MARIADB_CONNECTOR_DIR}") - if(NOT EXISTS "${MARIADB_MSI_DIR}/mariadb-connector-c-3.2.5-win64.msi" ) + # These values need to be updated whenever a new minor release replaces an old one + # Go to https://mariadb.com/downloads/connectors/ to find the up-to-date URL parts + set(MARIADB_CONNECTOR_C_VERSION "3.2.7") + set(MARIADB_CONNECTOR_C_BUCKET "2319651") + set(MARIADB_CONNECTOR_C_MD5 "f8636d733f1d093af9d4f22f3239f885") + set(MARIADB_CONNECTOR_CPP_VERSION "1.0.2") + set(MARIADB_CONNECTOR_CPP_BUCKET "2531525") + set(MARIADB_CONNECTOR_CPP_MD5 "3034bbd6ca00a0125345f9fd1a178401") + + set(MARIADB_CONNECTOR_C_MSI "mariadb-connector-c-${MARIADB_CONNECTOR_C_VERSION}-win64.msi") + set(MARIADB_CONNECTOR_CPP_MSI "mariadb-connector-cpp-${MARIADB_CONNECTOR_CPP_VERSION}-win64.msi") + + if(NOT EXISTS "${MARIADB_MSI_DIR}/${MARIADB_CONNECTOR_C_MSI}" ) message("Downloading mariadb connector/c") - file(DOWNLOAD https://dlm.mariadb.com/1936366/connectors/c/connector-c-3.2.5/mariadb-connector-c-3.2.5-win64.msi - "${MARIADB_MSI_DIR}/mariadb-connector-c-3.2.5-win64.msi" - EXPECTED_HASH MD5=09d418c290109068a5bea136dafca36b) + file(DOWNLOAD https://dlm.mariadb.com/${MARIADB_CONNECTOR_C_BUCKET}/Connectors/c/connector-c-${MARIADB_CONNECTOR_C_VERSION}/${MARIADB_CONNECTOR_C_MSI} + "${MARIADB_MSI_DIR}/${MARIADB_CONNECTOR_C_MSI}" + EXPECTED_HASH MD5=${MARIADB_CONNECTOR_C_MD5}) endif() - if(NOT EXISTS "${MARIADB_MSI_DIR}/mariadb-connector-cpp-1.0.1-win64.msi" ) + if(NOT EXISTS "${MARIADB_MSI_DIR}/${MARIADB_CONNECTOR_CPP_MSI}" ) message("Downloading mariadb connector/c++") - file(DOWNLOAD https://dlm.mariadb.com/1683453/connectors/cpp/connector-cpp-1.0.1/mariadb-connector-cpp-1.0.1-win64.msi - "${MARIADB_MSI_DIR}/mariadb-connector-cpp-1.0.1-win64.msi" - EXPECTED_HASH MD5=548e743fbf067d21d42b81d958bf4ed7) + file(DOWNLOAD https://dlm.mariadb.com/${MARIADB_CONNECTOR_CPP_BUCKET}/Connectors/cpp/connector-cpp-${MARIADB_CONNECTOR_CPP_VERSION}/${MARIADB_CONNECTOR_CPP_MSI} + "${MARIADB_MSI_DIR}/${MARIADB_CONNECTOR_CPP_MSI}" + EXPECTED_HASH MD5=${MARIADB_CONNECTOR_CPP_MD5}) endif() file(TO_NATIVE_PATH "${MARIADB_CONNECTOR_DIR}" MSIEXEC_TARGETDIR) # extract msi files without installing to users system if(NOT EXISTS "${MARIADB_C_CONNECTOR_DIR}") - file(TO_NATIVE_PATH "${MARIADB_MSI_DIR}/mariadb-connector-c-3.2.5-win64.msi" MSI_DIR) + file(TO_NATIVE_PATH "${MARIADB_MSI_DIR}/${MARIADB_CONNECTOR_C_MSI}" MSI_DIR) execute_process(COMMAND msiexec /a ${MSI_DIR} /qn TARGETDIR=${MSIEXEC_TARGETDIR}) endif() if(NOT EXISTS "${MARIADB_CPP_CONNECTOR_DIR}") - file(TO_NATIVE_PATH "${MARIADB_MSI_DIR}/mariadb-connector-cpp-1.0.1-win64.msi" MSI_DIR) + file(TO_NATIVE_PATH "${MARIADB_MSI_DIR}/${MARIADB_CONNECTOR_CPP_MSI}" MSI_DIR) execute_process(COMMAND msiexec /a ${MSI_DIR} /qn TARGETDIR=${MSIEXEC_TARGETDIR}) endif() @@ -43,7 +55,7 @@ if(WIN32 AND NOT MARIADB_BUILD_SOURCE) add_custom_target(mariadb_connector_cpp) add_custom_command(TARGET mariadb_connector_cpp POST_BUILD - COMMAND ${CMAKE_COMMAND} -E copy_if_different + COMMAND ${CMAKE_COMMAND} -E copy_if_different "${MARIADB_CPP_CONNECTOR_DIR}/mariadbcpp.dll" "${MARIADB_C_CONNECTOR_DIR}/lib/libmariadb.dll" "${PROJECT_BINARY_DIR}") @@ -118,7 +130,7 @@ else() # Build from source ${BINARY_DIR}/mariadbcpp/plugin ${MARIADB_SHARED_LIBRARY_COPY_LOCATION} - COMMAND ${CMAKE_COMMAND} -E copy_if_different + COMMAND ${CMAKE_COMMAND} -E copy_if_different ${MARIADB_SHARED_LIBRARY_LOCATION} ${MARIADB_SHARED_LIBRARY_COPY_LOCATION} diff --git a/thirdparty/LUnpack b/thirdparty/LUnpack deleted file mode 160000 index f8d7e442..00000000 --- a/thirdparty/LUnpack +++ /dev/null @@ -1 +0,0 @@ -Subproject commit f8d7e442a78910b298fe1cd5780f07c9c9285b8c diff --git a/thirdparty/docker-utils b/thirdparty/docker-utils deleted file mode 160000 index 3f0129e0..00000000 --- a/thirdparty/docker-utils +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 3f0129e0939ce5ccf41f0808dcbbe71a6243e37f diff --git a/thirdparty/raknet/Source/SocketLayer.cpp b/thirdparty/raknet/Source/SocketLayer.cpp index b6af8751..6cb4f7c6 100644 --- a/thirdparty/raknet/Source/SocketLayer.cpp +++ b/thirdparty/raknet/Source/SocketLayer.cpp @@ -417,8 +417,14 @@ int SocketLayer::RecvFrom( const SOCKET s, RakPeer *rakPeer, int *errorCode, uns assert( 0 ); #endif - *errorCode = -1; - return -1; + //*errorCode = -1; +#ifdef __linux__ + char str[INET_ADDRSTRLEN]; + inet_ntop(AF_INET, &(sa.sin_addr), str, INET_ADDRSTRLEN); + printf("[RakNet] SocketLayer::RecvFrom: empty datagram from %s", str); +#endif + //return -1; + return 0; } if ( len > 0 ) @@ -479,6 +485,47 @@ int SocketLayer::RecvFrom( const SOCKET s, RakPeer *rakPeer, int *errorCode, uns } #endif } +#elif defined(__linux__) + if (len < -1) + { + printf("[RakNet] SocketLayer::RecvFrom: Unexpected return value."); + return -1; + } + + int local_errno = errno; + if (local_errno == EAGAIN || local_errno == EWOULDBLOCK) + { + return 0; // no data + } + + if (local_errno == EINTR) + { + printf("[RakNet] SocketLayer::RecvFrom: The receive was interrupted by delivery of a signal before any data were available."); + return 0; // log, but ignore + } + + char str[INET_ADDRSTRLEN]; + inet_ntop(AF_INET, &(sa.sin_addr), str, INET_ADDRSTRLEN); + printf("[RakNet] SocketLayer::RecvFrom: Error receiving data from %s", str); + + switch (local_errno) { + case EINVAL: + printf("[RakNet] SocketLayer::RecvFrom: Invalid argument passed."); break; + case ENOMEM: + case ENOBUFS: + printf("[RakNet] SocketLayer::RecvFrom: Could not allocate memory for recvmsg()."); break; + case EFAULT: + printf("[RakNet] SocketLayer::RecvFrom: The receive buffer pointer(s) point outside the process's address space."); break; + case EBADF: + printf("[RakNet] SocketLayer::RecvFrom: The argument sockfd is an invalid descriptor."); break; + case ENOTSOCK: + printf("[RakNet] SocketLayer::RecvFrom: The argument sockfd does not refer to a socket."); break; + case EPIPE: + printf("[RakNet] SocketLayer::RecvFrom: The connection was unexpectedly closed or shut down by the other end. "); break; + default: + printf("[RakNet] SocketLayer::RecvFrom: Unknown Error %d", local_errno); break; + } + return -1; #endif } diff --git a/vanity/CREDITS.md b/vanity/CREDITS.md index e130cfb3..6780cff7 100644 --- a/vanity/CREDITS.md +++ b/vanity/CREDITS.md @@ -1,18 +1,17 @@ # CREDITS -## Developers +## DLU Team DarwinAnim8or (Max) Wincent01 Mick averysumner (codeshaunted) Jon002 Jonny -EmosewaMC -Jettford - -## Research & Tooling Xiphoseer lcdr - -## Community Management +Aaron K. Neal + +## Active Contributors +EmosewaMC +Jettford \ No newline at end of file diff --git a/vanity/NPC.xml b/vanity/NPC.xml index 3bb5ae9f..2311ab46 100644 --- a/vanity/NPC.xml +++ b/vanity/NPC.xml @@ -17,6 +17,19 @@ </locations> </zone> </npc> + <npc name="EmosewaMC - Quickbuilder" lot="6738"> + <equipment>12947, 12949, 12962, 12963</equipment> + <phrases> + <phrase>I hope quickbulds are still working!</phrase> + <phrase>Be careful crossing the gap!</phrase> + <phrase>Have The Maelstrom stopped going invisible?</phrase> + </phrases> + <zone id="1800"> + <locations> + <location x="745.756" y="75.262" z="-207.989" rw="0.838565" rx="0.0" ry="0.544801" rz="0.0" /> + </locations> + </zone> + </npc> <npc name="Neal - Paradox Scout" lot="6738"> <equipment>9950, 9944, 14102, 14092</equipment> <phrases>