2023-06-20 14:33:09 +10:00
|
|
|
#include "StarProjectileDatabase.hpp"
|
|
|
|
#include "StarProjectile.hpp"
|
|
|
|
#include "StarJsonExtra.hpp"
|
|
|
|
#include "StarAssets.hpp"
|
|
|
|
#include "StarRoot.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
ProjectileDatabase::ProjectileDatabase() {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
2024-03-15 21:28:11 +11:00
|
|
|
auto& files = assets->scanExtension("projectile");
|
2023-06-20 14:33:09 +10:00
|
|
|
assets->queueJsons(files);
|
2024-03-15 21:28:11 +11:00
|
|
|
for (auto& file : files) {
|
2023-06-20 14:33:09 +10:00
|
|
|
try {
|
|
|
|
auto projectileConfig = readConfig(file);
|
|
|
|
if (m_configs.contains(projectileConfig->typeName))
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::error("Duplicate projectile asset typeName {}. configfile {}", projectileConfig->typeName, file);
|
2023-06-20 14:33:09 +10:00
|
|
|
else
|
|
|
|
m_configs[projectileConfig->typeName] = projectileConfig;
|
|
|
|
} catch (std::exception const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::error("Could not read projectile '{}', error: {}", file, outputException(e, false));
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList ProjectileDatabase::allProjectileTypes() const {
|
|
|
|
return m_configs.keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProjectileDatabase::isProjectile(String const& projectileName) const {
|
|
|
|
return m_configs.contains(projectileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
Json ProjectileDatabase::projectileConfig(String const& type) const {
|
|
|
|
if (!m_configs.contains(type))
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ProjectileDatabaseException(strf("Unknown projectile with typeName {}.", type));
|
2023-06-20 14:33:09 +10:00
|
|
|
return m_configs.get(type)->config;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectilePtr ProjectileDatabase::createProjectile(String const& type, Json const& parameters) const {
|
|
|
|
if (!m_configs.contains(type))
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ProjectileDatabaseException(strf("Unknown projectile with typeName {}.", type));
|
2023-06-20 14:33:09 +10:00
|
|
|
return make_shared<Projectile>(m_configs.get(type), parameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
String ProjectileDatabase::damageKindImage(String const& type) const {
|
|
|
|
if (!m_configs.contains(type))
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ProjectileDatabaseException(strf("Unknown projectile with typeName {}.", type));
|
2023-06-20 14:33:09 +10:00
|
|
|
auto& config = m_configs.get(type);
|
|
|
|
return config->damageKindImage;
|
|
|
|
}
|
|
|
|
|
|
|
|
float ProjectileDatabase::gravityMultiplier(String const& type) const {
|
|
|
|
if (!m_configs.contains(type))
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ProjectileDatabaseException(strf("Unknown projectile with typeName {}.", type));
|
2023-06-20 14:33:09 +10:00
|
|
|
auto& config = m_configs.get(type);
|
|
|
|
return config->movementSettings.getFloat("gravityMultiplier", 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectilePtr ProjectileDatabase::netLoadProjectile(ByteArray const& netStore) const {
|
|
|
|
DataStreamBuffer ds(netStore);
|
|
|
|
String typeName = ds.read<String>();
|
|
|
|
return make_shared<Projectile>(m_configs.get(typeName), ds);
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectileConfigPtr ProjectileDatabase::readConfig(String const& path) {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
|
|
|
Json config = assets->json(path);
|
|
|
|
|
|
|
|
auto projectileConfig = make_shared<ProjectileConfig>();
|
|
|
|
projectileConfig->config = config;
|
|
|
|
projectileConfig->typeName = config.getString("projectileName");
|
|
|
|
projectileConfig->directory = AssetPath::directory(path);
|
|
|
|
|
|
|
|
projectileConfig->description = config.getString("description", "");
|
|
|
|
|
|
|
|
projectileConfig->boundBox = config.opt("boundBox").apply(jsonToRectF).value({-5, -5, 5, 5});
|
|
|
|
|
|
|
|
auto physicsType = config.getString("physics", "default");
|
|
|
|
JsonObject movementSettings = config.getObject("movementSettings", JsonObject());
|
|
|
|
projectileConfig->movementSettings =
|
2023-06-27 20:23:44 +10:00
|
|
|
jsonMerge(assets->json(strf("/projectiles/physics.config:{}", physicsType)), movementSettings);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
projectileConfig->initialSpeed = config.getFloat("speed", 50);
|
|
|
|
projectileConfig->acceleration = config.getFloat("acceleration", 0);
|
|
|
|
projectileConfig->power = config.getFloat("power", 1);
|
|
|
|
if (config.contains("damagePoly")) {
|
|
|
|
projectileConfig->damagePoly = jsonToPolyF(config.get("damagePoly"));
|
|
|
|
projectileConfig->damagePoly.scale(1.0f / TilePixels);
|
|
|
|
}
|
|
|
|
projectileConfig->piercing = config.getBool("piercing", false);
|
|
|
|
projectileConfig->falldown = config.getBool("falldown", false);
|
|
|
|
projectileConfig->bounces = config.getInt("bounces", 0);
|
|
|
|
|
|
|
|
projectileConfig->actionOnCollide = config.getArray("actionOnCollide", {});
|
|
|
|
projectileConfig->actionOnReap = config.getArray("actionOnReap", {});
|
|
|
|
projectileConfig->actionOnHit = config.getArray("actionOnHit", {});
|
|
|
|
projectileConfig->actionOnTimeout = config.getArray("actionOnTimeout", {});
|
|
|
|
|
|
|
|
for (auto const& c : config.getArray("periodicActions", {}))
|
|
|
|
projectileConfig->periodicActions.append(make_tuple(c.getFloat("time"), c.getBool("repeat", true), c));
|
|
|
|
|
|
|
|
projectileConfig->image = AssetPath::relativeTo(path, config.getString("image"));
|
|
|
|
projectileConfig->frameNumber = config.getUInt("frameNumber", 1);
|
|
|
|
projectileConfig->animationCycle = config.getFloat("animationCycle", 1.0);
|
|
|
|
projectileConfig->animationLoops = config.getBool("animationLoops", true);
|
|
|
|
projectileConfig->windupFrames = config.getUInt("windupFrames", 0);
|
|
|
|
projectileConfig->intangibleWindup = config.getBool("intangibleWindup", false);
|
|
|
|
projectileConfig->winddownFrames = config.getUInt("winddownFrames", 0);
|
|
|
|
projectileConfig->intangibleWinddown = config.getBool("intangibleWinddown", false);
|
|
|
|
projectileConfig->flippable = config.getBool("flippable", false);
|
|
|
|
projectileConfig->orientationLocked = config.getBool("orientationLocked", false);
|
|
|
|
|
|
|
|
projectileConfig->fullbright = config.getBool("fullbright", false);
|
|
|
|
projectileConfig->renderLayer = parseRenderLayer(config.getString("renderLayer", "Projectile"));
|
|
|
|
|
2024-03-20 01:53:34 +11:00
|
|
|
projectileConfig->lightColor = jsonToColor(config.get("lightColor", JsonArray{0, 0, 0}));
|
2023-06-20 14:33:09 +10:00
|
|
|
projectileConfig->lightPosition = jsonToVec2F(config.get("lightPosition", JsonArray{0, 0}));
|
2024-03-26 07:31:33 +11:00
|
|
|
if (auto lightType = config.optString("lightType"))
|
|
|
|
projectileConfig->lightType = LightTypeNames.getLeft(*lightType);
|
|
|
|
else
|
|
|
|
projectileConfig->lightType = (LightType)config.getBool("pointLight", false);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
projectileConfig->persistentAudio = config.getString("persistentAudio", "");
|
|
|
|
|
|
|
|
// Initialize timeToLive after animationCycle so we can have the default be
|
|
|
|
// based on animationCycle
|
|
|
|
if (!projectileConfig->animationLoops)
|
|
|
|
projectileConfig->timeToLive = config.getFloat("timeToLive", projectileConfig->animationCycle);
|
|
|
|
else
|
|
|
|
projectileConfig->timeToLive = config.getFloat("timeToLive", 5.0);
|
|
|
|
|
|
|
|
projectileConfig->damageKindImage = config.getString("damageKindImage", "");
|
|
|
|
|
|
|
|
projectileConfig->damageKind = config.getString("damageKind", "");
|
|
|
|
projectileConfig->damageType = config.getString("damageType", "damage");
|
|
|
|
projectileConfig->damageTeam = config.get("damageTeam", Json());
|
|
|
|
projectileConfig->damageRepeatGroup = config.optString("damageRepeatGroup");
|
|
|
|
projectileConfig->damageRepeatTimeout = config.optFloat("damageRepeatTimeout");
|
|
|
|
|
|
|
|
if (!projectileConfig->damageKindImage.empty())
|
|
|
|
projectileConfig->damageKindImage =
|
|
|
|
AssetPath::relativeTo(projectileConfig->directory, projectileConfig->damageKindImage);
|
|
|
|
|
|
|
|
projectileConfig->statusEffects = config.getArray("statusEffects", {}).transformed(jsonToEphemeralStatusEffect);
|
|
|
|
|
|
|
|
projectileConfig->emitters = jsonToStringSet(config.getArray("emitters", JsonArray()));
|
|
|
|
|
|
|
|
projectileConfig->hydrophobic = config.getBool("hydrophobic", false);
|
|
|
|
|
|
|
|
projectileConfig->rayCheckToSource = config.getBool("rayCheckToSource", false);
|
|
|
|
projectileConfig->knockback = config.getFloat("knockback", 0.0f);
|
|
|
|
projectileConfig->knockbackDirectional = config.getBool("knockbackDirectional", false);
|
|
|
|
|
|
|
|
projectileConfig->onlyHitTerrain = config.getBool("onlyHitTerrain", false);
|
|
|
|
|
|
|
|
projectileConfig->clientEntityMode = ClientEntityModeNames.getLeft(config.getString("clientEntityMode", "ClientMasterAllowed"));
|
|
|
|
projectileConfig->masterOnly = config.getBool("masterOnly", false);
|
|
|
|
|
|
|
|
projectileConfig->scripts =
|
|
|
|
jsonToStringList(config.get("scripts", JsonArray())).transformed(bind(AssetPath::relativeTo, path, _1));
|
|
|
|
|
|
|
|
projectileConfig->physicsForces = config.get("physicsForces", JsonObject());
|
|
|
|
projectileConfig->physicsCollisions = config.get("physicsCollisions", JsonObject());
|
|
|
|
|
|
|
|
projectileConfig->persistentStatusEffects = config.getArray("persistentStatusEffects", JsonArray()).transformed(jsonToPersistentStatusEffect);
|
|
|
|
projectileConfig->statusEffectArea = jsonToPolyF(config.get("statusEffectArea", JsonArray()));
|
|
|
|
|
|
|
|
return projectileConfig;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|