简体   繁体   English

Lua可以访问和调用成员函数的C ++对象数组

[英]Array of c++ objects that Lua can access and call member functions

I am pulling my hair out here, I can share an array between Lua and c++, I can even create an array of objects in Lua (using the code below) and access their member functions (eg obj[10]:setPosition(0,0,0) ). 我在这里拉头发,我可以在Lua和c ++之间共享一个数组,甚至可以在Lua中创建一个对象数组(使用下面的代码)并访问其成员函数(例如obj [10]:setPosition(0, 0,0))。 What I can't do is send c++ objects to Lua that already exist and have Lua call their respective member functions. 我不能做的是将c ++对象发送到已经存在的Lua,并让Lua调用其各自的成员函数。 For example: 例如:

objects = Scene.getAllObjects()
objects[5]:setPosition(0,0,0)

...doesn't work, however, The code below works ...无效,但是下面的代码有效

for i=1,10 do
  objects[i] = Object.new("Box")
  objects[i]:setPosition(0,0,0)
end

...which calls the c++ function below ...将调用下面的c ++函数

int luaAddNewEditableObject(lua_State * L)
{
    const char * name = luaL_checkstring(L, 1);

    EditableObject ** udata = (EditableObject **)lua_newuserdata(L, sizeof(void*));
    EditableObject *obj = scene->addNewEditableObject(name);
    *udata = obj;

    luaL_getmetatable(L, "luaL_EditableObject");

    lua_setmetatable(L, -2);

    return 1;
}

So basically, if it's created in Lua then no problems, but if there are objects that already exist in c++ then I need to get it into a Lua table/array so Lua can do magic to them 因此,基本上,如果它是在Lua中创建的,那么就没问题,但是,如果在c ++中已经存在对象,那么我需要将其放入Lua表/数组中,以便Lua可以对它们进行魔术处理

Please Help 请帮忙

struct ObjectArray 
{
    int size;
    EditableObject *objects;  /* The cpp objects */
};

static int getAllObjects (lua_State *L) 
{
    //This creates the metatable for array notation
    size_t nbytes = sizeof(ObjectArray) + numObjects * sizeof(ObjectArray*);
    ObjectArray *objectArray = (ObjectArray*)lua_newuserdata(L, nbytes);
    objectArray->size = numObjects;

    for (int i = 0; i < numFoos; i++)
    {
            //This sets the c++ pointers to the lua_newuserdata 
                objectArray->objects[i] = objects[i];

        //So maybe here I need to assign the 'luaL_EditableObject' metatable for each object
        //so I can call it's member functions ??
    }
    luaL_getmetatable(L, "ObjectArray");
    lua_setmetatable(L, -2);

    return 1; 
}

Ok I figured it out and will answer my own question in case anyone else has had this problem 好吧,我弄清楚了,如果有人遇到这个问题,我会回答我自己的问题

extern "C"
{
#include <lua.h>
#include <lauxlib.h>
#include <lualib.h>
}

#include <map>

class Object 
{
public:
    Object()
    {
        x = 0;
        y = 0;
    }
    Object(int x, int y)
    {
        this->x = x;
        this->y = y;
    }
    int x, y;
};


std::map<std::string,Object> objects;

static int luaGetObjects(lua_State *L)
{
    lua_newtable(L);
    int i=0;
    for (auto it = objects.begin(); it != objects.end(); it++, i++) 
    {

        lua_pushstring(L, it->first.c_str());

        luaL_getmetatable(L, "luaL_Object");
        lua_setmetatable(L, -2);

        lua_rawseti(L, -2, i+1);
        stackDump(L);
    }

    return 1;   
}

static int luaSetPosition(lua_State* L) 
{
    const char* key = luaL_checkstring(L,1);

    int x = luaL_checknumber(L,2);
    int y = luaL_checknumber(L,3);

    objects[key].x = x;
    objects[key].y = y;

    return 0; 
}

static int luaGetPosition(lua_State* L) 
{
    const char* key = luaL_checkstring(L,1);

    lua_pushnumber(L, objects[key].x);
    lua_pushnumber(L, objects[key].y);

    return 2; 
}

void registerObject(lua_State *L)
{
    luaL_Reg regs[] =
    {
        { "setPosition", luaSetPosition },
        { "getPosition", luaGetPosition },
        { NULL, NULL }
    };

    luaL_newmetatable(L, "luaL_Object");
    luaL_register(L, NULL, regs);
    lua_pushvalue(L, -1);
    lua_setfield(L, -1, "__index");
}

int main()
{
    lua_State * L = luaL_newstate();
    luaL_openlibs(L);

    lua_pushcfunction(L, luaGetObjects);
    lua_setglobal(L, "getObjects");

    registerObject(L);

    objects["id001"] = Object(1,2);
    objects["id002"] = Object(3,4);
    objects["id003"] = Object(5,6);

    int erred = luaL_dofile(L, "hello.lua");
    if(erred)
        std::cout << "Lua error: " << luaL_checkstring(L, -1) << std::endl;

    lua_close(L); 

    return 0;
}

The 'hello.lua' code: 'hello.lua'代码:

objects = getObjects()

for i=1,#objects do
   x,y = objects[i]:getPosition();
   print(string.format("object[%d] x = %d y = %d",i,x,y))

   objects[i]:setPosition(x*100, y*100);

   x,y = objects[i]:getPosition();
   print(string.format("object[%d] x = %d y = %d after mul",i,x,y))
end

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM