// // Lol Engine - Graphing tutorial // // Copyright: (c) 2012-2013 Sam Hocevar <sam@hocevar.net> // This program is free software; you can redistribute it and/or // modify it under the terms of the Do What The Fuck You Want To // Public License, Version 2, as published by Sam Hocevar. See // http://www.wtfpl.net/ for more details. // #if HAVE_CONFIG_H # include "config.h" #endif #include <lol/engine.h> #include "loldebug.h" #include <cstdio> using namespace lol; //----------------------------------------------------------------------------- class FooTest { public: FooTest() { } static int New(Lolua::State* l) { int n_args = lua_gettop(l); // Number of arguments // We could actually allocate Foo itself as a user data but // since user data can be GC'ed and we gain unity by using CRT's heap // all along. FooTest** udata = (FooTest**)lua_newuserdata(l, sizeof(FooTest*)); *udata = new FooTest(); // Usually, we'll just use "Foo" as the second parameter, but I // say luaL_Foo here to distinguish the difference: // // This 2nd parameter here is an _internal label_ for luaL, it is // _not_ exposed to Lua by default. // // Effectively, this metatable is not accessible by Lua by default. //luaL_getmetatable(l, m_table_name); // The Lua stack at this point looks like this: // // 3| metatable "luaL_foo" |-1 // 2| userdata |-2 // 1| string parameter |-3 // // So the following line sets the metatable for the user data to the luaL_Foo // metatable // // We must set the metatable here because Lua prohibits setting // the metatable of a userdata in Lua. The only way to set a metatable // of a userdata is to do it in C. //lua_setmetatable(l, -2); luaL_getmetatable(l, m_table2_name); lua_setmetatable(l, -2); // The Lua stack at this point looks like this: // // 2| userdata |-1 // 1| string parameter |-2 // // We return 1 so Lua callsite will get the user data and // Lua will clean the stack after that. return 1; } static FooTest* Check(Lolua::State* l, int n) { // This checks that the argument is a userdata // with the metatable "luaL_Foo" return *(FooTest**)luaL_testudata(l, n, m_table2_name); } static int Del(Lolua::State * l) { FooTest* foo = Check(l, 1); delete foo; return 0; } static void Register(Lolua::State * l) { // Create a luaL metatable. This metatable is not // exposed to Lua. The "luaL_Foo" label is used by luaL // internally to identity things. luaL_newmetatable(l, m_table2_name); luaL_setfuncs(l, FooTest::m_statics2, 0); lua_pushvalue(l, -1); lua_setfield(l, -1, "__index"); //lua_setglobal(l, m_class2_name); lua_pop(l, 1); // Create a luaL metatable. This metatable is not // exposed to Lua. The "luaL_Foo" label is used by luaL // internally to identity things. luaL_newmetatable(l, m_table_name); // Register the C functions _into_ the metatable we just created. luaL_setfuncs(l, FooTest::m_statics, 0); // The Lua stack at this point looks like this: // // 1| metatable "luaL_Foo" |-1 lua_pushvalue(l, -1); // The Lua stack at this point looks like this: // // 2| metatable "luaL_Foo" |-1 // 1| metatable "luaL_Foo" |-2 // Set the "__index" field of the metatable to point to itself // This pops the stack lua_setfield(l, -1, "__index"); // The Lua stack at this point looks like this: // // 1| metatable "luaL_Foo" |-1 // The luaL_Foo metatable now has the following fields // - __gc // - __index // - new // Now we use setglobal to officially expose the luaL_Foo metatable // to Lua. And we use the name "Foo". // // This allows Lua scripts to _override_ the metatable of Foo. // For high security code this may not be called for but // we'll do this to get greater flexibility. lua_setglobal(l, m_class_name); } static int Test1(Lolua::State* L) { Lolua::Var<int> i1(L, 1); i1 += 5; return i1.Return(L); } static int Test2(Lolua::State* L) { FooTest* foo = Check(L, 1); Lolua::Var<float> i1(L, 2); i1 = foo->Test2Inst(i1.V()); return i1.Return(L); } float Test2Inst(float f) { return (f + 10); } static const char m_class_name[]; static const char m_class2_name[]; static const char m_table_name[]; static const char m_table2_name[]; static const luaL_Reg m_statics[]; static const luaL_Reg m_statics2[]; static const luaL_Reg m_methods[]; }; const luaL_Reg FooTest::m_statics[] = { { "New", &FooTest::New }, { "__gc", &FooTest::Del }, { "Test1", &FooTest::Test1 }, { NULL, NULL } }; const luaL_Reg FooTest::m_statics2[] = { { "Test2", &FooTest::Test2 }, { NULL, NULL } }; const luaL_Reg FooTest::m_methods[] = { { NULL, NULL } }; const char FooTest::m_class_name[] = "FooTest"; const char FooTest::m_class2_name[] = "FooTest2"; const char FooTest::m_table_name[] = "FooTest_mt"; const char FooTest::m_table2_name[] = "FooTest_mt2"; typedef Lolua::ObjectDef<FooTest> FooTestLuaBase; class FooTestLua : public FooTestLuaBase { public: FooTestLua() : FooTestLuaBase() {} virtual ~FooTestLua() {} static FooTestLua* New(Lolua::State* l, int arg_nb) { return new FooTestLua(); } static const char* GetClassName(); static const char* GetClassLibName(); static const char* GetClassInstName(); static const Lolua::ClassMethod* GetStaticMethods(); static const Lolua::ClassMethod* GetInstanceMethods(); static int Test1(Lolua::State* L) { Lolua::Var<int> i1(L, 1); i1 += 5; return i1.Return(L); } static int Test2(Lolua::State* L) { Lolua::VarPtr<FooTestLua> foo(L, 1); Lolua::Var<float> i1(L, 2); i1 = foo.V()->Test2Inst(i1.V()); return i1.Return(L); } float Test2Inst(float f) { return (f + 10); } }; static const Lolua::ClassMethod FooTest_statics[] = { { "Test1", &FooTestLua::Test1 }, { NULL, NULL } }; const Lolua::ClassMethod FooTest_methods[] = { { "Test2", &FooTestLua::Test2 }, { NULL, NULL } }; const char* FooTestLua::GetClassName() { static const char name[] = "FooTest"; return name; } const char* FooTestLua::GetClassLibName() { static const char name[] = "FooTestLib"; return name; } const char* FooTestLua::GetClassInstName() { static const char name[] = "FooTestInst"; return name; } const Lolua::ClassMethod* FooTestLua::GetStaticMethods() { return FooTest_statics; } const Lolua::ClassMethod* FooTestLua::GetInstanceMethods() { return FooTest_methods; } //----------------------------------------------------------------------------- class LoluaDemoLoader : public Lolua::Loader { public: LoluaDemoLoader() : Lolua::Loader() { Lolua::State* l = GetLuaState(); Lolua::Object::Register<FooTestLua>(l); ExecLua("14_lol_lua.lua"); } virtual ~LoluaDemoLoader() { } }; //----------------------------------------------------------------------------- class LoluaDemo : public WorldEntity { public: LoluaDemo() { } virtual void TickGame(float seconds) { WorldEntity::TickGame(seconds); LoluaDemoLoader* demo_loader = new LoluaDemoLoader(); float TestValueNum = demo_loader->GetVar<float>("TestValueNum"); int32_t TestValueInt = demo_loader->GetVar<int32_t>("TestValueInt"); uint32_t TestValueUint = demo_loader->GetVar<uint32_t>("TestValueUint"); String TestValueStr = demo_loader->GetVar<String>("TestValueStr"); //String test_return2 = demo_loader->GetVar<String>("test_return2"); //String test_return3 = demo_loader->GetVar<String>("test_return3"); int32_t testtruc_return = demo_loader->GetVar<int32_t>("footest_return"); float testtruc_return2 = demo_loader->GetVar<float>("footest_return2"); //int32_t testtruc2_return = demo_loader->GetVar<int32_t>("testtruc_return2"); String::Printf("Lua Vars: TestValueNum: %.2f, TestValueInt: %i, TestValueUint: %i, TestValueStr: %s.", TestValueNum, TestValueInt, TestValueUint, TestValueStr); delete demo_loader; Ticker::Shutdown(); } virtual void TickDraw(float seconds, Scene &scene) { WorldEntity::TickDraw(seconds, scene); } }; //----------------------------------------------------------------------------- int main(int argc, char **argv) { System::Init(argc, argv); Application app("Tutorial 14: Lolua Demo", ivec2(800, 600), 60.0f); new LoluaDemo(); app.Run(); return EXIT_SUCCESS; }