mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2025-01-10 23:07:07 +00:00
225 lines
8.9 KiB
C++
225 lines
8.9 KiB
C++
|
#include "NsConcertQuickBuild.h"
|
||
|
#include "EntityManager.h"
|
||
|
#include "NsConcertChoiceBuildManager.h"
|
||
|
#include "DestroyableComponent.h"
|
||
|
#include "RenderComponent.h"
|
||
|
#include "GameMessages.h"
|
||
|
#include "MovingPlatformComponent.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");
|
||
|
}
|
||
|
}
|
||
|
}
|