2023-06-20 14:33:09 +10:00
|
|
|
#include "StarLiquidsDatabase.hpp"
|
|
|
|
#include "StarLexicalCast.hpp"
|
|
|
|
#include "StarAssets.hpp"
|
|
|
|
#include "StarJsonExtra.hpp"
|
|
|
|
#include "StarMaterialDatabase.hpp"
|
|
|
|
#include "StarRoot.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
LiquidSettings::LiquidSettings() : id(EmptyLiquidId) {}
|
|
|
|
|
|
|
|
LiquidsDatabase::LiquidsDatabase() {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
auto materialDatabase = Root::singleton().materialDatabase();
|
|
|
|
|
|
|
|
auto config = assets->json("/liquids.config");
|
|
|
|
|
|
|
|
auto liquidEngineParameters = config.get("liquidEngineParameters");
|
|
|
|
m_liquidEngineParameters.lateralMoveFactor = liquidEngineParameters.getFloat("lateralMoveFactor");
|
|
|
|
m_liquidEngineParameters.spreadOverfillUpFactor = liquidEngineParameters.getFloat("spreadOverfillUpFactor");
|
|
|
|
m_liquidEngineParameters.spreadOverfillLateralFactor = liquidEngineParameters.getFloat("spreadOverfillLateralFactor");
|
|
|
|
m_liquidEngineParameters.spreadOverfillDownFactor = liquidEngineParameters.getFloat("spreadOverfillDownFactor");
|
|
|
|
m_liquidEngineParameters.pressureEqualizeFactor = liquidEngineParameters.getFloat("pressureEqualizeFactor");
|
|
|
|
m_liquidEngineParameters.pressureMoveFactor = liquidEngineParameters.getFloat("pressureMoveFactor");
|
|
|
|
m_liquidEngineParameters.maximumPressureLevelImbalance = liquidEngineParameters.getFloat("maximumPressureLevelImbalance");
|
|
|
|
m_liquidEngineParameters.minimumLivenPressureChange = liquidEngineParameters.getFloat("minimumLivenPressureChange");
|
|
|
|
m_liquidEngineParameters.minimumLivenLevelChange = liquidEngineParameters.getFloat("minimumLivenLevelChange");
|
|
|
|
m_liquidEngineParameters.minimumLiquidLevel = liquidEngineParameters.getFloat("minimumLiquidLevel");
|
|
|
|
m_liquidEngineParameters.interactTransformationLevel = liquidEngineParameters.getFloat("interactTransformationLevel");
|
|
|
|
|
|
|
|
m_backgroundDrain = config.getFloat("backgroundDrain");
|
|
|
|
|
|
|
|
m_liquidNames["empty"] = EmptyLiquidId;
|
|
|
|
|
2024-03-15 21:28:11 +11:00
|
|
|
auto& liquids = assets->scanExtension("liquid");
|
2023-06-20 14:33:09 +10:00
|
|
|
assets->queueJsons(liquids);
|
|
|
|
|
2024-03-15 21:28:11 +11:00
|
|
|
for (auto& file : liquids) {
|
2023-06-20 14:33:09 +10:00
|
|
|
try {
|
|
|
|
auto liquidConfig = assets->json(file);
|
|
|
|
|
|
|
|
unsigned id = liquidConfig.getUInt("liquidId");
|
|
|
|
if (id != (LiquidId)id)
|
2023-06-27 20:23:44 +10:00
|
|
|
throw LiquidException(strf("Liquid id {} does not fall in the valid range, wrapped to {}\n", id, (LiquidId)id));
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
auto entry = make_shared<LiquidSettings>();
|
|
|
|
entry->id = (LiquidId)id;
|
|
|
|
entry->name = liquidConfig.getString("name");
|
|
|
|
entry->path = file;
|
|
|
|
entry->config = liquidConfig;
|
|
|
|
|
|
|
|
JsonObject descriptions;
|
|
|
|
for (auto pair : liquidConfig.iterateObject())
|
|
|
|
if (pair.first.endsWith("Description"))
|
|
|
|
descriptions[pair.first] = pair.second;
|
|
|
|
descriptions["description"] = liquidConfig.getString("description", "");
|
|
|
|
entry->descriptions = descriptions;
|
|
|
|
|
|
|
|
entry->tickDelta = liquidConfig.getUInt("tickDelta");
|
|
|
|
entry->liquidColor = jsonToVec4B(liquidConfig.get("color"));
|
|
|
|
|
|
|
|
if (liquidConfig.contains("radiantLight")) {
|
|
|
|
auto lightLevel = jsonToVec3B(liquidConfig.get("radiantLight"));
|
|
|
|
entry->radiantLightLevel = Color::rgb(lightLevel).toRgbF();
|
|
|
|
}
|
|
|
|
|
|
|
|
entry->itemDrop = ItemDescriptor(liquidConfig.get("itemDrop", {}));
|
|
|
|
|
|
|
|
entry->statusEffects = liquidConfig.getArray("statusEffects", {});
|
|
|
|
|
|
|
|
for (auto const& interaction : liquidConfig.getArray("interactions", {})) {
|
|
|
|
LiquidId liquid = interaction.getUInt("liquid");
|
|
|
|
LiquidInteractionResult interactionResult;
|
|
|
|
if (auto materialResult = interaction.optString("materialResult"))
|
|
|
|
interactionResult = makeLeft<MaterialId>(materialDatabase->materialId(*materialResult));
|
|
|
|
else if (auto liquidResult = interaction.optUInt("liquidResult"))
|
|
|
|
interactionResult = makeRight<LiquidId>(*liquidResult);
|
|
|
|
else
|
|
|
|
throw LiquidException::format(
|
2023-06-27 20:23:44 +10:00
|
|
|
"Neither resultMaterial or resultLiquid specified in liquid interaction of liquid {}", entry->id);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
entry->interactions[liquid] = interactionResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_settings.set(entry->id, entry);
|
|
|
|
m_liquidNames.add(entry->name, entry->id);
|
|
|
|
} catch (StarException const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
throw LiquidException(strf("Error loading liquid file {}", file), e);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LiquidCellEngineParameters LiquidsDatabase::liquidEngineParameters() const {
|
|
|
|
return m_liquidEngineParameters;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList LiquidsDatabase::liquidNames() const {
|
|
|
|
return m_liquidNames.keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
List<LiquidSettingsConstPtr> LiquidsDatabase::allLiquidSettings() const {
|
|
|
|
return filtered<List<LiquidSettingsConstPtr>>(m_settings, mem_fn(&LiquidSettingsConstPtr::operator bool));
|
|
|
|
}
|
|
|
|
|
|
|
|
LiquidId LiquidsDatabase::liquidId(String const& str) const {
|
|
|
|
return m_liquidNames.get(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
String LiquidsDatabase::liquidName(LiquidId liquidId) const {
|
|
|
|
if (liquidId == EmptyLiquidId)
|
|
|
|
return "empty";
|
|
|
|
else if (auto settings = liquidSettings(liquidId))
|
|
|
|
return settings->name;
|
2023-06-27 20:23:44 +10:00
|
|
|
throw LiquidException::format("invalid liquid id {}", liquidId);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
String LiquidsDatabase::liquidDescription(LiquidId liquidId, String const& species) const {
|
|
|
|
if (liquidId == EmptyLiquidId)
|
|
|
|
return String();
|
|
|
|
else if (auto settings = liquidSettings(liquidId))
|
|
|
|
return settings->descriptions.getString(
|
2023-06-27 20:23:44 +10:00
|
|
|
strf("{}Description", species), settings->descriptions.getString("description"));
|
|
|
|
throw LiquidException::format("invalid liquid id {}", liquidId);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
String LiquidsDatabase::liquidDescription(LiquidId liquidId) const {
|
|
|
|
if (liquidId == EmptyLiquidId)
|
|
|
|
return String();
|
|
|
|
else if (auto settings = liquidSettings(liquidId))
|
|
|
|
return settings->descriptions.getString("description");
|
2023-06-27 20:23:44 +10:00
|
|
|
throw LiquidException::format("invalid liquid id {}", liquidId);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<String> LiquidsDatabase::liquidPath(LiquidId liquidId) const {
|
|
|
|
if (liquidId == EmptyLiquidId)
|
|
|
|
return {};
|
|
|
|
else if (auto settings = liquidSettings(liquidId))
|
|
|
|
return settings->path;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<Json> LiquidsDatabase::liquidConfig(LiquidId liquidId) const {
|
|
|
|
if (liquidId == EmptyLiquidId)
|
|
|
|
return {};
|
|
|
|
else if (auto settings = liquidSettings(liquidId))
|
|
|
|
return settings->config;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<LiquidInteractionResult> LiquidsDatabase::interact(LiquidId target, LiquidId other) const {
|
|
|
|
if (auto settings = liquidSettings(target))
|
|
|
|
return settings->interactions.value(other);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|