/** * @file script.cpp * Manages all Lua scripts. * * Copyright (C) 2019 Belle-Isle, Andrew * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include /********************* * SYSTEM SPECIFIC * *********************/ void ScriptSystem::configure([[maybe_unused]] entityx::EntityManager& entities, [[maybe_unused]] entityx::EventManager& events) { events.subscribe(*this); // Init after systems.configure() in engine.cpp //init(); lua.open_libraries(sol::lib::base, sol::lib::math, sol::lib::string); scriptExport(); } #include void ScriptSystem::update([[maybe_unused]] entityx::EntityManager& entities, [[maybe_unused]] entityx::EventManager& events, [[maybe_unused]] entityx::TimeDelta dt) { entities.each([](entityx::Entity, Scripted &s){ s.updatePhysics(); }); } void ScriptSystem::receive([[maybe_unused]] const EntitySpawnEvent &toSpawn) { } /********************* * CLASS FUNCTIONS * *********************/ int ScriptSystem::init(void) { doFile(); return 0; } void ScriptSystem::doFile(void) { auto result = lua.script_file("Scripts/init.lua"); if (!result.valid()) { std::cout << "Lua error: " << std::endl; } } /******************** * SCRIPT PARSING * ********************/ #include #include #include #include #include #include #include #include #include void ScriptSystem::scriptExport(void) { std::function entitySpawn = [this](sol::table t){ return spawn(t);}; std::function worldRegister = [this](sol::object t){ return worldSystem.addWorld(t); }; lua.new_usertype("Position", sol::constructors(), "x", &Position::x, "y", &Position::y); lua.new_usertype("Name", sol::constructors(), "value", &Name::name); lua.new_usertype("Render", sol::constructors(), "visible", &Render::visible, "texture", &Render::texture, "flipx", &Render::flipX); lua.new_usertype("Velocity", sol::constructors(), "x", &Velocity::x, "y", &Velocity::y); lua.new_usertype("Player", sol::constructors()); lua.new_usertype("Light", sol::constructors(), "r", &Light::r, "g", &Light::g, "b", &Light::b, "strength", &Light::strength); lua.new_usertype("Physics", sol::constructors(), "standing", &Physics::standing); lua.new_usertype("World", sol::constructors(), "Generate", &World::generate, "Seed", sol::property(&World::setSeed, &World::getSeed), "setData", &World::setData, "registerMaterial", &World::registerMaterial, "setSize", &World::setSize, "getSize", &World::getSize, // New stuff "unitSize", sol::property(&World::setUnitSize, &World::getUnitSize), "createLayer", &World::registerLayer, "createDecoLayer", &World::registerDecoLayer); game = lua["game"].get_or_create(); game.set_function("spawn", entitySpawn); game.set_function("worldRegister", worldRegister); } sol::table ScriptSystem::spawn(sol::object param) { sol::table* toRet; // "Entitiy" table to be returned if (param.get_type() == sol::type::table) { sol::table tab = param; // Cast the generic parameter to a table entityx::Entity e = manager.create(); // Create a new entity auto d = e.assign().get(); // Since this entity was created // via Lua, assign the Scripted // component. d->caller = lua.create_table(); // Create our Entity "table" toRet = &(d->caller); // set our return table to the Entity "table" *toRet = tab; /* Copy table to our new entity to preserve functions * this reduces the amount of work done in this function * as well. We are allowed to do this because lua allows * for the reallocation of types. */ /* * Create the component dependancies (i.e.: position) first */ if (tab["Position"] != nullptr) { (*toRet)["Position"] = e.assign(Position().FromLua(tab["Position"])).get(); } if (tab["Name"] != nullptr) { (*toRet)["Name"] = e.assign(Name().FromLua(tab["Name"])).get(); } if (tab["Render"] != nullptr) { if (!e.has_component()) // Position must exist for render (*toRet)["Position"] = e.assign().get(); (*toRet)["Render"] = e.assign(Render().FromLua(tab["Render"])).get(); } if (tab["Velocity"] != nullptr) { if (!e.has_component()) // Position must exist for vel. (*toRet)["Position"] = e.assign().get(); (*toRet)["Velocity"] = e.assign(Velocity().FromLua(tab["Velocity"])).get(); } if (tab["Physics"] != nullptr) { if (!e.has_component()) // Position must exist for phys. (*toRet)["Position"] = e.assign().get(); if (!e.has_component()) // Velocity must exist for phys. (*toRet)["Velocity"] = e.assign().get(); (*toRet)["Physics"] = e.assign().get(); } if (tab["Light"] != nullptr) { if (!e.has_component()) // Position must exist for vel. (*toRet)["Position"] = e.assign().get(); (*toRet)["Light"] = e.assign(Light().FromLua(tab["Light"])).get(); } // Probably should be last if (tab["EventListeners"] != nullptr) { sol::table listeners = tab["EventListeners"]; e.assign(listeners); } if (tab["Player"] != nullptr) { (*toRet)["Player"] = e.assign().get(); } } else { // TODO better logging std::cerr << "Parameter to spawn() must be a table!" << std::endl; } // If the Entity has an Init function defined, call it if ((*toRet)["Init"].get_type() == sol::type::function) (*toRet)["Init"](*toRet); return *toRet; }