Merge pull request #377 from MihailRis/update-scripts-lifetime

Update scripts lifetime
This commit is contained in:
MihailRis 2024-11-22 10:12:55 +03:00 committed by GitHub
commit b40c8c4b4c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 101 additions and 51 deletions

View File

@ -10,6 +10,9 @@ block.index(name: str) -> int
-- Returns the id of the block material.
block.material(blockid: int) -> str
-- Table of materials by their full names (example: base:carpet)
block.materials: table<string, table>
-- Returns the block display name.
block.caption(blockid: int) -> str

View File

@ -10,6 +10,9 @@ block.index(name: str) -> int
-- Возвращает id материала блока.
block.material(blockid: int) -> str
-- Таблица материалов по их полным именам (пример: base:carpet)
block.materials: table<string, table>
-- Возвращает название блока, отображаемое в интерфейсе.
block.caption(blockid: int) -> str

View File

@ -483,16 +483,6 @@ void ContentLoader::loadBlock(
auto configFile = folder / fs::path("blocks/" + name + ".json");
if (fs::exists(configFile)) loadBlock(def, full, configFile);
auto scriptfile = folder / fs::path("scripts/" + def.scriptName + ".lua");
if (fs::is_regular_file(scriptfile)) {
scripting::load_block_script(
env,
full,
scriptfile,
pack->id + ":scripts/" + def.scriptName + ".lua",
def.rt.funcsset
);
}
if (!def.hidden) {
auto& item = builder.items.create(full + BLOCK_ITEM_SUFFIX);
item.generated = true;
@ -514,17 +504,6 @@ void ContentLoader::loadItem(
auto folder = pack->folder;
auto configFile = folder / fs::path("items/" + name + ".json");
if (fs::exists(configFile)) loadItem(def, full, configFile);
auto scriptfile = folder / fs::path("scripts/" + def.scriptName + ".lua");
if (fs::is_regular_file(scriptfile)) {
scripting::load_item_script(
env,
full,
scriptfile,
pack->id + ":scripts/" + def.scriptName + ".lua",
def.rt.funcsset
);
}
}
static std::tuple<std::string, std::string, std::string> create_unit_id(
@ -728,18 +707,6 @@ void ContentLoader::load() {
auto folder = pack->folder;
// Load main world script
fs::path scriptFile = folder / fs::path("scripts/world.lua");
if (fs::is_regular_file(scriptFile)) {
scripting::load_world_script(
env,
pack->id,
scriptFile,
pack->id + ":scripts/world.lua",
runtime->worldfuncsset
);
}
// Load world generators
fs::path generatorsDir = folder / fs::u8path("generators");
foreach_file(generatorsDir, [this](const fs::path& file) {
@ -807,17 +774,6 @@ void ContentLoader::load() {
);
});
// Load entity components
fs::path componentsDir = folder / fs::u8path("scripts/components");
foreach_file(componentsDir, [this](const fs::path& file) {
auto name = pack->id + ":" + file.stem().u8string();
scripting::load_entity_component(
name,
file,
pack->id + ":scripts/components/" + file.filename().u8string()
);
});
// Process content.json and load defined content units
auto contentFile = pack->getContentFile();
if (fs::exists(contentFile)) {
@ -825,6 +781,61 @@ void ContentLoader::load() {
}
}
template <class T>
static void load_scripts(Content& content, ContentUnitDefs<T>& units) {
for (const auto& [name, def] : units.getDefs()) {
size_t pos = name.find(':');
if (pos == std::string::npos) {
throw std::runtime_error("invalid content unit name");
}
const auto runtime = content.getPackRuntime(name.substr(0, pos));
const auto& pack = runtime->getInfo();
const auto& folder = pack.folder;
auto scriptfile = folder / fs::path("scripts/" + def->scriptName + ".lua");
if (fs::is_regular_file(scriptfile)) {
scripting::load_content_script(
runtime->getEnvironment(),
name,
scriptfile,
pack.id + ":scripts/" + def->scriptName + ".lua",
def->rt.funcsset
);
}
}
}
void ContentLoader::loadScripts(Content& content) {
load_scripts(content, content.blocks);
load_scripts(content, content.items);
for (const auto& [packid, runtime] : content.getPacks()) {
const auto& pack = runtime->getInfo();
const auto& folder = pack.folder;
// Load main world script
fs::path scriptFile = folder / fs::path("scripts/world.lua");
if (fs::is_regular_file(scriptFile)) {
scripting::load_world_script(
runtime->getEnvironment(),
pack.id,
scriptFile,
pack.id + ":scripts/world.lua",
runtime->worldfuncsset
);
}
// Load entity components
fs::path componentsDir = folder / fs::u8path("scripts/components");
foreach_file(componentsDir, [&pack](const fs::path& file) {
auto name = pack.id + ":" + file.stem().u8string();
scripting::load_entity_component(
name,
file,
pack.id + ":scripts/components/" + file.filename().u8string()
);
});
}
}
void ContentLoader::loadResources(ResourceType type, const dv::value& list) {
for (size_t i = 0; i < list.size(); i++) {
builder.resourceIndices[static_cast<size_t>(type)].add(

View File

@ -17,6 +17,7 @@ struct ContentPack;
struct GeneratorDef;
class ResPaths;
class Content;
class ContentBuilder;
class ContentPackRuntime;
struct ContentPackStats;
@ -76,4 +77,6 @@ public:
void fixPackIndices();
void load();
static void loadScripts(Content& content);
};

View File

@ -361,8 +361,10 @@ void Engine::loadContent() {
ContentLoader(&pack, contentBuilder, *resPaths).load();
load_configs(pack.folder);
}
content = contentBuilder.build();
scripting::on_content_load(content.get());
ContentLoader::loadScripts(*content);
langs::setup(resdir, langs::current->getId(), contentPacks);
loadAssets();

View File

@ -157,10 +157,25 @@ void scripting::process_post_runnables() {
}
}
void scripting::on_content_load(Content* content) {
scripting::content = content;
scripting::indices = content->getIndices();
auto L = lua::get_main_state();
if (lua::getglobal(L, "block")) {
const auto& materials = content->getBlockMaterials();
lua::createtable(L, 0, materials.size());
for (const auto& [name, material] : materials) {
lua::pushvalue(L, material->serialize());
lua::setfield(L, name);
}
lua::setfield(L, "materials");
lua::pop(L);
}
}
void scripting::on_world_load(LevelController* controller) {
scripting::level = controller->getLevel();
scripting::content = level->content;
scripting::indices = level->content->getIndices();
scripting::blocks = controller->getBlocksController();
scripting::controller = controller;
@ -671,7 +686,7 @@ int scripting::get_values_on_stack() {
return lua::gettop(lua::get_main_state());
}
void scripting::load_block_script(
void scripting::load_content_script(
const scriptenv& senv,
const std::string& prefix,
const fs::path& file,
@ -692,7 +707,7 @@ void scripting::load_block_script(
register_event(env, "on_blocks_tick", prefix + ".blockstick");
}
void scripting::load_item_script(
void scripting::load_content_script(
const scriptenv& senv,
const std::string& prefix,
const fs::path& file,

View File

@ -45,6 +45,8 @@ namespace scripting {
void initialize(Engine* engine);
void on_content_load(Content* content);
bool register_event(
int env, const std::string& name, const std::string& id
);
@ -128,7 +130,7 @@ namespace scripting {
/// @param file item script file
/// @param fileName script file path using the engine format
/// @param funcsset block callbacks set
void load_block_script(
void load_content_script(
const scriptenv& env,
const std::string& prefix,
const fs::path& file,
@ -142,7 +144,7 @@ namespace scripting {
/// @param file item script file
/// @param fileName script file path using the engine format
/// @param funcsset item callbacks set
void load_item_script(
void load_content_script(
const scriptenv& env,
const std::string& prefix,
const fs::path& file,

View File

@ -8,6 +8,15 @@
#include "presets/ParticlesPreset.hpp"
#include "util/stringutil.hpp"
dv::value BlockMaterial::serialize() const {
return dv::object({
{"name", name},
{"stepsSound", stepsSound},
{"placeSound", placeSound},
{"breakSound", breakSound}
});
}
std::string to_string(BlockModel model) {
switch (model) {
case BlockModel::none:

View File

@ -104,6 +104,8 @@ struct BlockMaterial {
std::string stepsSound {""};
std::string placeSound {""};
std::string breakSound {""};
dv::value serialize() const;
};
/// @brief Block properties definition