axmol/extensions/scripting/lua-bindings/manual/LuaBasicConversions.cpp

3198 lines
88 KiB
C++

/****************************************************************************
Copyright (c) 2013-2016 Chukong Technologies Inc.
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
Copyright (c) 2019-present Axmol Engine contributors (see AUTHORS.md).
https://axmolengine.github.io/
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
****************************************************************************/
#include "lua-bindings/manual/LuaBasicConversions.h"
#include "lua-bindings/manual/tolua_fix.h"
#include "base/Utils.h"
#include <sstream>
std::unordered_map<uintptr_t, const char*> g_luaType;
std::unordered_map<cxx17::string_view, const char*> g_typeCast;
#if _AX_DEBUG >= 1
void luaval_to_native_err(lua_State* L, const char* msg, tolua_Error* err, const char* funcName)
{
if (NULL == L || NULL == err || NULL == msg || 0 == strlen(msg))
return;
if (msg[0] == '#')
{
const char* expected = err->type;
const char* provided = tolua_typename(L, err->index);
if (msg[1] == 'f')
{
int narg = err->index;
if (err->array)
AXLOG("%s\n %s argument #%d is array of '%s'; array of '%s' expected.\n", msg + 2, funcName, narg,
provided, expected);
else
AXLOG("%s\n %s argument #%d is '%s'; '%s' expected.\n", msg + 2, funcName, narg, provided,
expected);
}
else if (msg[1] == 'v')
{
if (err->array)
AXLOG("%s\n %s value is array of '%s'; array of '%s' expected.\n", funcName, msg + 2, provided,
expected);
else
AXLOG("%s\n %s value is '%s'; '%s' expected.\n", msg + 2, funcName, provided, expected);
}
}
}
#endif
#ifdef __cplusplus
extern "C" {
#endif
extern int lua_isusertype(lua_State* L, int lo, const char* type);
#ifdef __cplusplus
}
#endif
bool luaval_is_usertype(lua_State* L, int lo, const char* type, int def)
{
if (def && lua_gettop(L) < std::abs(lo))
return true;
if (lua_isnil(L, lo) || lua_isusertype(L, lo, type))
return true;
return false;
}
bool luaval_to_ushort(lua_State* L, int lo, unsigned short* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (unsigned short)tolua_tonumber(L, lo, 0);
}
return ok;
}
bool luaval_to_float(lua_State* L, int lo, float* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (float)lua_tonumber(L, lo);
}
return ok;
}
bool luaval_to_int32(lua_State* L, int lo, int* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
/**
When we want to convert the number value from the Lua to int, we would call lua_tonumber to implement.It would
experience two phase conversion: int -> double, double->int.But,for the 0x80000000 which the min value of int,
the int cast may return an undefined result,like 0x7fffffff.So we must use the (int)(unsigned
int)lua_tonumber() to get predictable results for 0x80000000.In this place,we didn't use lua_tointeger, because
it may produce different results depending on the compiler,e.g:for iPhone4s,it also get wrong value for
0x80000000.
*/
unsigned int estimateValue = (unsigned int)lua_tonumber(L, lo);
if (estimateValue == std::numeric_limits<int>::min())
{
*outValue = (int)estimateValue;
}
else
{
*outValue = (int)lua_tonumber(L, lo);
}
}
return ok;
}
bool luaval_to_uint32(lua_State* L, int lo, unsigned int* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (unsigned int)tolua_tonumber(L, lo, 0);
}
return ok;
}
bool luaval_to_uint16(lua_State* L, int lo, uint16_t* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (unsigned char)tolua_tonumber(L, lo, 0);
}
return ok;
}
bool luaval_to_boolean(lua_State* L, int lo, bool* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isboolean(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (bool)tolua_toboolean(L, lo, 0);
}
return ok;
}
bool luaval_to_number(lua_State* L, int lo, double* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = tolua_tonumber(L, lo, 0);
}
return ok;
}
bool luaval_to_long_long(lua_State* L, int lo, long long* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (long long)tolua_tonumber(L, lo, 0);
}
return ok;
}
bool luaval_to_std_string(lua_State* L, int lo, std::string* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_iscppstring(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t size;
auto rawString = lua_tolstring(L, lo, &size);
*outValue = std::string(rawString, size);
}
return ok;
}
bool luaval_to_std_string_view(lua_State* L, int lo, cxx17::string_view* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_iscppstring(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t size;
auto rawString = lua_tolstring(L, lo, &size);
*outValue = cxx17::string_view(rawString, size);
}
return ok;
}
bool luaval_to_vec2(lua_State* L, int lo, ax::Vec2* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
// assertion: since we only have vec2, you should never passing rect as vec2 to native
const auto objlen = lua_objlen(L, lo);
assert(objlen != 4);
if (ok)
{
lua_pushstring(L, "x");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
lua_pop(L, 1);
lua_pushstring(L, "width");
lua_gettable(L, lo);
}
outValue->x = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "y");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
lua_pop(L, 1);
lua_pushstring(L, "height");
lua_gettable(L, lo);
}
outValue->y = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_vec3(lua_State* L, int lo, ax::Vec3* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "x");
lua_gettable(L, lo);
outValue->x = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "y");
lua_gettable(L, lo);
outValue->y = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "z");
lua_gettable(L, lo);
outValue->z = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_vec4(lua_State* L, int lo, ax::Vec4* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "x");
lua_gettable(L, lo);
outValue->x = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "y");
lua_gettable(L, lo);
outValue->y = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "z");
lua_gettable(L, lo);
outValue->z = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "w");
lua_gettable(L, lo);
outValue->w = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_blendfunc(lua_State* L, int lo, ax::BlendFunc* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "src");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
outValue->src = static_cast<ax::backend::BlendFactor>(lua_tointeger(L, -1));
}
lua_pop(L, 1);
lua_pushstring(L, "dst");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
outValue->dst = static_cast<ax::backend::BlendFactor>(lua_tointeger(L, -1));
}
lua_pop(L, 1);
}
return ok;
}
#if AX_USE_PHYSICS
bool luaval_to_physics_material(lua_State* L, int lo, PhysicsMaterial* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
# if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
# endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "density");
lua_gettable(L, lo);
outValue->density = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "restitution");
lua_gettable(L, lo);
outValue->restitution = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "friction");
lua_gettable(L, lo);
outValue->friction = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
#endif //#if AX_USE_PHYSICS
bool luaval_to_ssize_t(lua_State* L, int lo, ssize_t* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (ssize_t)tolua_tointeger(L, lo, 0);
}
return ok;
}
bool luaval_to_size_t(lua_State* L, int lo, size_t* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_isnumber(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
*outValue = (size_t)tolua_tointeger(L, lo, 0);
}
return ok;
}
bool luaval_to_size(lua_State* L, int lo, Size* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "width"); /* L: paramStack key */
lua_gettable(L, lo); /* L: paramStack paramStack[lo][key] */
outValue->width = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1); /* L: paramStack*/
lua_pushstring(L, "height");
lua_gettable(L, lo);
outValue->height = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_rect(lua_State* L, int lo, Rect* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "x");
lua_gettable(L, lo);
outValue->origin.x = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "y");
lua_gettable(L, lo);
outValue->origin.y = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "width");
lua_gettable(L, lo);
outValue->size.width = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "height");
lua_gettable(L, lo);
outValue->size.height = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_color4b(lua_State* L, int lo, Color4B* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "r");
lua_gettable(L, lo);
outValue->r = lua_isnil(L, -1) ? 0 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "g");
lua_gettable(L, lo);
outValue->g = lua_isnil(L, -1) ? 0 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "b");
lua_gettable(L, lo);
outValue->b = lua_isnil(L, -1) ? 0 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "a");
lua_gettable(L, lo);
outValue->a = lua_isnil(L, -1) ? 255 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_color4f(lua_State* L, int lo, Color4F* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "r");
lua_gettable(L, lo);
outValue->r = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "g");
lua_gettable(L, lo);
outValue->g = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "b");
lua_gettable(L, lo);
outValue->b = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "a");
lua_gettable(L, lo);
outValue->a = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_color3b(lua_State* L, int lo, Color3B* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "r");
lua_gettable(L, lo);
outValue->r = lua_isnil(L, -1) ? 0 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "g");
lua_gettable(L, lo);
outValue->g = lua_isnil(L, -1) ? 0 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "b");
lua_gettable(L, lo);
outValue->b = lua_isnil(L, -1) ? 0 : static_cast<uint8_t>(lua_tonumber(L, -1));
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_affinetransform(lua_State* L, int lo, AffineTransform* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
do
{
lua_pushstring(L, "a");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
ok = false;
lua_pop(L, 1);
break;
}
outValue->a = (float)lua_tonumber(L, -1);
lua_pushstring(L, "b");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
ok = false;
lua_pop(L, 1);
break;
}
outValue->b = (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "c");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
ok = false;
lua_pop(L, 1);
break;
}
outValue->c = (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "d");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
ok = false;
lua_pop(L, 1);
break;
}
outValue->d = (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "tx");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
ok = false;
lua_pop(L, 1);
break;
}
outValue->tx = lua_isnil(L, -1) ? 0 : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "ty");
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
ok = false;
lua_pop(L, 1);
break;
}
outValue->ty = lua_isnil(L, -1) ? 0 : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
} while (0);
}
return ok;
}
bool luaval_to_fontdefinition(lua_State* L, int lo, FontDefinition* outValue, const char* funcName)
{
if (NULL == L || NULL == outValue)
return false;
tolua_Error tolua_err;
bool ok = !!tolua_istable(L, lo, 0, &tolua_err);
if (ok)
{
// default values
const char* defautlFontName = "Arial";
const int defaultFontSize = 32;
TextHAlignment defaultTextAlignment = TextHAlignment::LEFT;
TextVAlignment defaultTextVAlignment = TextVAlignment::TOP;
// by default shadow and stroke are off
outValue->_shadow._shadowEnabled = false;
outValue->_stroke._strokeEnabled = false;
// white text by default
outValue->_fontFillColor = Color3B::WHITE;
lua_pushstring(L, "fontName");
lua_gettable(L, lo);
outValue->_fontName = tolua_tocppstring(L, lua_gettop(L), defautlFontName);
lua_pop(L, 1);
lua_pushstring(L, "fontSize");
lua_gettable(L, lo);
outValue->_fontSize = lua_isnil(L, -1) ? defaultFontSize : (int)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "fontAlignmentH");
lua_gettable(L, lo);
outValue->_alignment = lua_isnil(L, -1) ? defaultTextAlignment : (TextHAlignment)(int)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "fontAlignmentV");
lua_gettable(L, lo);
outValue->_vertAlignment = lua_isnil(L, -1) ? defaultTextVAlignment : (TextVAlignment)(int)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "fontFillColor");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
luaval_to_color3b(L, lua_gettop(L), &outValue->_fontFillColor);
}
lua_pop(L, 1);
lua_pushstring(L, "fontDimensions");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
luaval_to_size(L, lua_gettop(L), &outValue->_dimensions);
}
lua_pop(L, 1);
lua_pushstring(L, "shadowEnabled");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
luaval_to_boolean(L, -1, &outValue->_shadow._shadowEnabled);
if (outValue->_shadow._shadowEnabled)
{
// default shadow values
outValue->_shadow._shadowOffset = Size(5, 5);
outValue->_shadow._shadowBlur = 1;
outValue->_shadow._shadowOpacity = 1;
}
lua_pushstring(L, "shadowOffset");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
luaval_to_size(L, lua_gettop(L), &outValue->_shadow._shadowOffset);
}
lua_pop(L, 1);
lua_pushstring(L, "shadowBlur");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
outValue->_shadow._shadowBlur = (float)lua_tonumber(L, -1);
}
lua_pop(L, 1);
lua_pushstring(L, "shadowOpacity");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
outValue->_shadow._shadowOpacity = (float)lua_tonumber(L, -1);
}
lua_pop(L, 1);
}
lua_pop(L, 1);
lua_pushstring(L, "strokeEnabled");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
luaval_to_boolean(L, -1, &outValue->_stroke._strokeEnabled);
if (outValue->_stroke._strokeEnabled)
{
// default stroke values
outValue->_stroke._strokeSize = 1;
outValue->_stroke._strokeColor = Color3B::BLUE;
lua_pushstring(L, "strokeColor");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
luaval_to_color3b(L, lua_gettop(L), &outValue->_stroke._strokeColor);
}
lua_pop(L, 1);
lua_pushstring(L, "strokeSize");
lua_gettable(L, lo);
if (!lua_isnil(L, -1))
{
outValue->_stroke._strokeSize = (float)lua_tonumber(L, -1);
}
lua_pop(L, 1);
}
}
lua_pop(L, 1);
}
#if _AX_DEBUG >= 1
else
{
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
}
#endif
return ok;
}
bool luaval_to_ttfconfig(lua_State* L, int lo, ax::TTFConfig* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "fontFilePath"); /* L: paramStack key */
lua_gettable(L, lo); /* L: paramStack paramStack[lo][key] */
outValue->fontFilePath = lua_isstring(L, -1) ? lua_tostring(L, -1) : "";
lua_pop(L, 1); /* L: paramStack*/
lua_pushstring(L, "fontSize");
lua_gettable(L, lo);
outValue->fontSize = lua_isnumber(L, -1) ? (float)lua_tointeger(L, -1) : 0.0f;
lua_pop(L, 1);
lua_pushstring(L, "glyphs");
lua_gettable(L, lo);
outValue->glyphs =
lua_isnumber(L, -1) ? static_cast<GlyphCollection>(lua_tointeger(L, -1)) : GlyphCollection::NEHE;
lua_pop(L, 1);
lua_pushstring(L, "customGlyphs");
lua_gettable(L, lo);
outValue->customGlyphs = lua_isstring(L, -1) ? lua_tostring(L, -1) : "";
lua_pop(L, 1);
lua_pushstring(L, "distanceFieldEnabled");
lua_gettable(L, lo);
outValue->distanceFieldEnabled = lua_isboolean(L, -1) ? lua_toboolean(L, -1) : false;
lua_pop(L, 1);
lua_pushstring(L, "outlineSize");
lua_gettable(L, lo);
outValue->outlineSize = lua_isnumber(L, -1) ? (int)lua_tointeger(L, -1) : 0;
lua_pop(L, 1);
return true;
}
return false;
}
bool luaval_to_mat4(lua_State* L, int lo, ax::Mat4* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
ok = false;
#endif
}
if (ok)
{
do
{
size_t len = lua_objlen(L, lo);
if (len != 16)
{
ok = false;
break;
}
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (tolua_isnumber(L, -1, 0, &tolua_err))
{
outValue->m[i] = (float)tolua_tonumber(L, -1, 0);
}
else
{
outValue->m[i] = 0;
}
lua_pop(L, 1);
}
} while (0);
}
return ok;
}
bool luaval_to_array_of_vec2(lua_State* L, int lo, ax::Vec2** points, int* numPoints, const char* funcName)
{
if (NULL == L)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
if (len > 0)
{
ax::Vec2* array = (ax::Vec2*)new Vec2[len];
if (NULL == array)
return false;
for (size_t i = 0; i < len; ++i)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (!tolua_istable(L, -1, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
lua_pop(L, 1);
AX_SAFE_DELETE_ARRAY(array);
return false;
}
ok &= luaval_to_vec2(L, lua_gettop(L), &array[i]);
if (!ok)
{
lua_pop(L, 1);
AX_SAFE_DELETE_ARRAY(array);
return false;
}
lua_pop(L, 1);
}
*numPoints = (int)len;
*points = array;
}
}
return ok;
}
bool luavals_variadic_to_ccvaluevector(lua_State* L, int argc, ax::ValueVector* ret)
{
if (nullptr == L || argc == 0)
return false;
for (int i = 0; i < argc; i++)
{
if (lua_istable(L, i + 2))
{
lua_pushnumber(L, 1);
lua_gettable(L, i + 2);
if (lua_isnil(L, -1))
{
lua_pop(L, 1);
ValueMap dictVal;
if (luaval_to_ccvaluemap(L, i + 2, &dictVal))
{
ret->emplace_back(Value(dictVal));
}
}
else
{
lua_pop(L, 1);
ValueVector arrVal;
if (luaval_to_ccvaluevector(L, i + 2, &arrVal))
{
ret->emplace_back(Value(arrVal));
}
}
}
else if (lua_type(L, i + 2) == LUA_TSTRING)
{
std::string stringValue = "";
if (luaval_to_std_string(L, i + 2, &stringValue))
{
ret->emplace_back(Value(stringValue));
}
}
else if (lua_isboolean(L, i + 2))
{
bool boolVal = false;
if (luaval_to_boolean(L, i + 2, &boolVal))
{
ret->emplace_back(Value(boolVal));
}
}
else if (lua_type(L, i + 2) == LUA_TNUMBER)
{
ret->emplace_back(Value(tolua_tonumber(L, i + 2, 0)));
}
else
{
AXASSERT(false, "not supported type");
}
}
return true;
}
bool luaval_to_ccvalue(lua_State* L, int lo, ax::Value* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret)
return false;
bool ok = true;
tolua_Error tolua_err;
if (tolua_istable(L, lo, 0, &tolua_err))
{
lua_pushnumber(L, 1);
lua_gettable(L, lo);
if (lua_isnil(L, -1)) /** if table[1] = nil,we don't think it is a pure array */
{
lua_pop(L, 1);
ValueMap dictVal;
if (luaval_to_ccvaluemap(L, lo, &dictVal))
{
*ret = Value(dictVal);
}
}
else
{
lua_pop(L, 1);
ValueVector arrVal;
if (luaval_to_ccvaluevector(L, lo, &arrVal))
{
*ret = Value(arrVal);
}
}
}
else if ((lua_type(L, lo) == LUA_TSTRING) && tolua_isstring(L, lo, 0, &tolua_err))
{
std::string stringValue = "";
if (luaval_to_std_string(L, lo, &stringValue))
{
*ret = Value(stringValue);
}
}
else if ((lua_type(L, lo) == LUA_TBOOLEAN) && tolua_isboolean(L, lo, 0, &tolua_err))
{
bool boolVal = false;
if (luaval_to_boolean(L, lo, &boolVal))
{
*ret = Value(boolVal);
}
}
else if ((lua_type(L, lo) == LUA_TNUMBER) && tolua_isnumber(L, lo, 0, &tolua_err))
{
*ret = Value(tolua_tonumber(L, lo, 0));
}
return ok;
}
bool luaval_to_ccvaluemap(lua_State* L, int lo, ax::ValueMap* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
std::string stringKey = "";
std::string stringValue = "";
bool boolVal = false;
ValueMap& dict = *ret;
lua_pushnil(L); /* first key L: lotable ..... nil */
while (0 != lua_next(L, lo)) /* L: lotable ..... key value */
{
if (!lua_isstring(L, -2))
{
lua_pop(L, 1); /* removes 'value'; keep 'key' for next iteration*/
continue;
}
if (luaval_to_std_string(L, -2, &stringKey))
{
if (lua_istable(L, -1))
{
lua_pushnumber(L, 1);
lua_gettable(L, -2);
if (lua_isnil(L, -1)) /** if table[1] = nil,we don't think it is a pure array */
{
lua_pop(L, 1);
ValueMap dictVal;
if (luaval_to_ccvaluemap(L, lua_gettop(L), &dictVal))
{
dict[stringKey] = Value(dictVal);
}
}
else
{
lua_pop(L, 1);
ValueVector arrVal;
if (luaval_to_ccvaluevector(L, lua_gettop(L), &arrVal))
{
dict[stringKey] = Value(arrVal);
}
}
}
else if (lua_type(L, -1) == LUA_TSTRING)
{
if (luaval_to_std_string(L, -1, &stringValue))
{
dict[stringKey] = Value(stringValue);
}
}
else if (lua_type(L, -1) == LUA_TBOOLEAN)
{
if (luaval_to_boolean(L, -1, &boolVal))
{
dict[stringKey] = Value(boolVal);
}
}
else if (lua_type(L, -1) == LUA_TNUMBER)
{
dict[stringKey] = Value(tolua_tonumber(L, -1, 0));
}
else
{
AXASSERT(false, "not supported type");
}
}
lua_pop(L, 1); /* L: lotable ..... key */
}
}
return ok;
}
bool luaval_to_ccvaluemapintkey(lua_State* L, int lo, ax::ValueMapIntKey* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
std::string stringKey = "";
std::string stringValue = "";
int intKey = 0;
bool boolVal = false;
ValueMapIntKey& dict = *ret;
lua_pushnil(L); /* first key L: lotable ..... nil */
while (0 != lua_next(L, lo)) /* L: lotable ..... key value */
{
if (!lua_isstring(L, -2))
{
lua_pop(L, 1); /* removes 'value'; keep 'key' for next iteration*/
continue;
}
if (luaval_to_std_string(L, -2, &stringKey))
{
intKey = atoi(stringKey.c_str());
if (lua_istable(L, -1))
{
lua_pushnumber(L, 1);
lua_gettable(L, -2);
if (lua_isnil(L, -1)) /** if table[1] = nil,we don't think it is a pure array */
{
lua_pop(L, 1);
ValueMap dictVal;
if (luaval_to_ccvaluemap(L, lua_gettop(L), &dictVal))
{
dict[intKey] = Value(dictVal);
}
}
else
{
lua_pop(L, 1);
ValueVector arrVal;
if (luaval_to_ccvaluevector(L, lua_gettop(L), &arrVal))
{
dict[intKey] = Value(arrVal);
}
}
}
else if (lua_type(L, -1) == LUA_TSTRING)
{
if (luaval_to_std_string(L, -1, &stringValue))
{
dict[intKey] = Value(stringValue);
}
}
else if (lua_type(L, -1) == LUA_TBOOLEAN)
{
if (luaval_to_boolean(L, -1, &boolVal))
{
dict[intKey] = Value(boolVal);
}
}
else if (lua_type(L, -1) == LUA_TNUMBER)
{
dict[intKey] = Value(tolua_tonumber(L, -1, 0));
}
else
{
AXASSERT(false, "not supported type");
}
}
lua_pop(L, 1); /* L: lotable ..... key */
}
}
return ok;
}
bool luaval_to_ccvaluevector(lua_State* L, int lo, ax::ValueVector* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_isnil(L, -1))
{
lua_pop(L, 1);
continue;
}
if (lua_istable(L, -1))
{
lua_pushnumber(L, 1);
lua_gettable(L, -2);
if (lua_isnil(L, -1))
{
lua_pop(L, 1);
ValueMap dictVal;
if (luaval_to_ccvaluemap(L, lua_gettop(L), &dictVal))
{
ret->emplace_back(Value(dictVal));
}
}
else
{
lua_pop(L, 1);
ValueVector arrVal;
if (luaval_to_ccvaluevector(L, lua_gettop(L), &arrVal))
{
ret->emplace_back(Value(arrVal));
}
}
}
else if (lua_type(L, -1) == LUA_TSTRING)
{
std::string stringValue = "";
if (luaval_to_std_string(L, -1, &stringValue))
{
ret->emplace_back(Value(stringValue));
}
}
else if (lua_type(L, -1) == LUA_TBOOLEAN)
{
bool boolVal = false;
if (luaval_to_boolean(L, -1, &boolVal))
{
ret->emplace_back(Value(boolVal));
}
}
else if (lua_type(L, -1) == LUA_TNUMBER)
{
ret->emplace_back(Value(tolua_tonumber(L, -1, 0)));
}
else
{
AXASSERT(false, "not supported type");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_std_vector_string(lua_State* L, int lo, std::vector<std::string>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
std::string value = "";
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_isstring(L, -1))
{
ok = luaval_to_std_string(L, -1, &value);
if (ok)
ret->emplace_back(value);
}
else
{
AXASSERT(false, "string type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_std_vector_string_view(lua_State* L, int lo, std::vector<std::string_view>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
std::string_view value = "";
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_isstring(L, -1))
{
ok = luaval_to_std_string_view(L, -1, &value);
if (ok)
ret->emplace_back(value);
}
else
{
AXASSERT(false, "string type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_std_vector_int(lua_State* L, int lo, std::vector<int>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_isnumber(L, -1))
{
ret->emplace_back((int)tolua_tonumber(L, -1, 0));
}
else
{
AXASSERT(false, "int type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_mesh_vertex_attrib(lua_State* L, int lo, ax::MeshVertexAttrib* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "type"); /* L: paramStack key */
lua_gettable(L, lo); /* L: paramStack paramStack[lo][key] */
ret->type = (backend::VertexFormat)(int)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "vertexAttrib"); /* L: paramStack key */
lua_gettable(L, lo); /* L: paramStack paramStack[lo][key] */
ret->vertexAttrib = (shaderinfos::VertexKey)(int)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_std_vector_float(lua_State* L, int lo, std::vector<float>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_isnumber(L, -1))
{
ret->emplace_back((float)tolua_tonumber(L, -1, 0));
}
else
{
AXASSERT(false, "float type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_std_vector_ushort(lua_State* L, int lo, std::vector<unsigned short>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_isnumber(L, -1))
{
ret->emplace_back((unsigned short)tolua_tonumber(L, -1, 0));
}
else
{
AXASSERT(false, "unsigned short type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_quaternion(lua_State* L, int lo, ax::Quaternion* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "x");
lua_gettable(L, lo);
outValue->x = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "y");
lua_gettable(L, lo);
outValue->y = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "z");
lua_gettable(L, lo);
outValue->z = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "w");
lua_gettable(L, lo);
outValue->w = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_texparams(lua_State* L, int lo, ax::Texture2D::TexParams* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "minFilter");
lua_gettable(L, lo);
outValue->minFilter = static_cast<backend::SamplerFilter>(lua_isnil(L, -1) ? 0 : lua_tointeger(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "magFilter");
lua_gettable(L, lo);
outValue->magFilter = static_cast<backend::SamplerFilter>(lua_isnil(L, -1) ? 0 : lua_tointeger(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "wrapS");
lua_gettable(L, lo);
outValue->sAddressMode = static_cast<backend::SamplerAddressMode>(lua_isnil(L, -1) ? 0 : lua_tointeger(L, -1));
lua_pop(L, 1);
lua_pushstring(L, "wrapT");
lua_gettable(L, lo);
outValue->tAddressMode = static_cast<backend::SamplerAddressMode>(lua_isnil(L, -1) ? 0 : lua_tointeger(L, -1));
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_tex2f(lua_State* L, int lo, ax::Tex2F* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "u");
lua_gettable(L, lo);
outValue->u = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
lua_pushstring(L, "v");
lua_gettable(L, lo);
outValue->v = lua_isnil(L, -1) ? 0.0f : (float)lua_tonumber(L, -1);
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_v3f_c4b_t2f(lua_State* L, int lo, ax::V3F_C4B_T2F* outValue, const char* funcName)
{
if (nullptr == L || nullptr == outValue)
return false;
bool ok = true;
tolua_Error tolua_err;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
lua_pushstring(L, "vertices");
lua_gettable(L, lo);
if (!tolua_istable(L, lua_gettop(L), 0, &tolua_err))
{
lua_pop(L, 1);
return false;
}
ok &= luaval_to_vec3(L, lua_gettop(L), &outValue->vertices);
if (!ok)
{
lua_pop(L, 1);
return false;
}
lua_pop(L, 1);
lua_pushstring(L, "colors");
lua_gettable(L, lo);
if (!tolua_istable(L, lua_gettop(L), 0, &tolua_err))
{
lua_pop(L, 1);
return false;
}
ok &= luaval_to_color4b(L, lua_gettop(L), &outValue->colors);
if (!ok)
{
lua_pop(L, 1);
return false;
}
lua_pop(L, 1);
lua_pushstring(L, "texCoords");
lua_gettable(L, lo);
if (!tolua_istable(L, lua_gettop(L), 0, &tolua_err))
{
lua_pop(L, 1);
return false;
}
ok &= luaval_to_tex2f(L, lua_gettop(L), &outValue->texCoords);
if (!ok)
{
lua_pop(L, 1);
return false;
}
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_std_vector_vec2(lua_State* L, int lo, std::vector<ax::Vec2>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
ax::Vec2 value;
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_istable(L, lua_gettop(L)))
{
ok &= luaval_to_vec2(L, lua_gettop(L), &value);
if (ok)
{
ret->emplace_back(value);
}
}
else
{
AXASSERT(false, "vec2 type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_std_vector_vec3(lua_State* L, int lo, std::vector<ax::Vec3>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
ax::Vec3 value;
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_istable(L, lua_gettop(L)))
{
ok &= luaval_to_vec3(L, lua_gettop(L), &value);
if (ok)
{
ret->emplace_back(value);
}
}
else
{
AXASSERT(false, "vec3 type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
bool luaval_to_std_vector_v3f_c4b_t2f(lua_State* L,
int lo,
std::vector<ax::V3F_C4B_T2F>* ret,
const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (ok)
{
size_t len = lua_objlen(L, lo);
ax::V3F_C4B_T2F value;
for (size_t i = 0; i < len; i++)
{
lua_pushnumber(L, i + 1);
lua_gettable(L, lo);
if (lua_istable(L, lua_gettop(L)))
{
ok &= luaval_to_v3f_c4b_t2f(L, lua_gettop(L), &value);
if (ok)
{
ret->emplace_back(value);
}
}
else
{
AXASSERT(false, "V3F_C4B_T2F type is needed");
}
lua_pop(L, 1);
}
}
return ok;
}
void vec2_array_to_luaval(lua_State* L, const ax::Vec2* points, int count)
{
if (NULL == L)
return;
lua_newtable(L);
for (int i = 1; i <= count; ++i)
{
lua_pushnumber(L, i);
vec2_to_luaval(L, points[i - 1]);
lua_rawset(L, -3);
}
}
static int vec2_index(lua_State* L)
{ // t k
const char signature = lua_tostring(L, 2)[0];
int n = 0;
if (signature == 'x' || signature == 'w' || signature == 'u')
{
lua_pop(L, 1); // pop the string key
n = 1;
}
else if (signature == 'y' || signature == 'h' || signature == 'v')
{
lua_pop(L, 1); // pop the string key
n = 2;
}
if (n)
lua_rawgeti(L, -1, n);
else
lua_pushnil(L);
return 1;
}
static int vec2_newindex(lua_State* L)
{ // t k v
const char signature = lua_tostring(L, 2)[0];
int n = 0;
if (signature == 'x' || signature == 'w' || signature == 'u')
{
lua_remove(L, 2); // remove the string key
n = 1;
}
else if (signature == 'y' || signature == 'h' || signature == 'v')
{
lua_remove(L, 2); // remove the string key
n = 2;
}
if (n)
lua_rawseti(L, -2, n);
return 0;
}
int vec2_to_luaval(lua_State* L, const ax::Vec2& vec2)
{
lua_createtable(L, 2, 0); /* L: table */
lua_pushnumber(L, (lua_Number)vec2.x); /* L: table key value*/
lua_rawseti(L, -2, 1); /* table[key] = value, L: table */
lua_pushnumber(L, (lua_Number)vec2.y); /* L: table key value*/
lua_rawseti(L, -2, 2);
int top = lua_gettop(L);
luaL_getmetatable(L, "_vec2mt");
if (!lua_istable(L, -1))
{
lua_settop(L, top); // restore stack
luaL_newmetatable(L, "_vec2mt");
lua_pushcfunction(L, vec2_index);
lua_setfield(L, -2, "__index");
lua_pushcfunction(L, vec2_newindex);
lua_setfield(L, -2, "__newindex");
}
lua_setmetatable(L, -2);
return 1;
}
static int vec3_index(lua_State* L)
{ // t k
const char signature = lua_tostring(L, 2)[0];
assert(signature >= 'x' || signature <= 'z');
lua_pop(L, 1); // pop the string key
const int n = signature - 'x' + 1;
lua_rawgeti(L, -1, n);
return 1;
}
static int vec3_newindex(lua_State* L)
{ // t k v
const char signature = lua_tostring(L, 2)[0];
assert(signature >= 'x' || signature <= 'z');
lua_remove(L, 2); // remove the string key
const int n = signature - 'x' + 1;
lua_rawseti(L, -2, n);
return 0;
}
int vec3_to_luaval(lua_State* L, const ax::Vec3& vec3)
{
lua_createtable(L, 3, 0); /* L: table */
lua_pushnumber(L, (lua_Number)vec3.x); /* L: table key value*/
lua_rawseti(L, -2, 1); /* table[key] = value, L: table */
lua_pushnumber(L, (lua_Number)vec3.y); /* L: table key value*/
lua_rawseti(L, -2, 2);
lua_pushnumber(L, (lua_Number)vec3.z); /* L: table key value*/
lua_rawseti(L, -2, 3);
int top = lua_gettop(L);
luaL_getmetatable(L, "_vec3mt");
if (!lua_istable(L, -1))
{
lua_settop(L, top); // restore stack
luaL_newmetatable(L, "_vec3mt");
lua_pushcfunction(L, vec3_index);
lua_setfield(L, -2, "__index");
lua_pushcfunction(L, vec3_newindex);
lua_setfield(L, -2, "__newindex");
}
lua_setmetatable(L, -2);
return 1;
}
static int vec4_index(lua_State* L)
{ // t k
const char signature = lua_tostring(L, 2)[0];
assert(signature >= 'w' || signature <= 'z');
lua_pop(L, 1); // pop the string key
const int n = signature != 'w' ? signature - 'x' + 1 : 4;
lua_rawgeti(L, -1, n);
return 1;
}
static int vec4_newindex(lua_State* L)
{ // t k v
const char signature = lua_tostring(L, 2)[0];
assert(signature >= 'w' || signature <= 'z');
lua_remove(L, 2); // remove the string key
const int n = signature != 'w' ? signature - 'x' + 1 : 4;
lua_rawseti(L, -2, n);
return 0;
}
int vec4_to_luaval(lua_State* L, const ax::Vec4& vec4)
{
lua_createtable(L, 4, 0); /* L: table */
lua_pushnumber(L, (lua_Number)vec4.x); /* L: table key value*/
lua_rawseti(L, -2, 1); /* table[key] = value, L: table */
lua_pushnumber(L, (lua_Number)vec4.y); /* L: table key value*/
lua_rawseti(L, -2, 2);
lua_pushnumber(L, (lua_Number)vec4.z); /* L: table key value*/
lua_rawseti(L, -2, 3);
lua_pushnumber(L, (lua_Number)vec4.w); /* L: table key value*/
lua_rawseti(L, -2, 4);
int top = lua_gettop(L);
luaL_getmetatable(L, "_vec4mt");
if (!lua_istable(L, -1))
{
lua_settop(L, top); // restore stack
luaL_newmetatable(L, "_vec4mt");
lua_pushcfunction(L, vec4_index);
lua_setfield(L, -2, "__index");
lua_pushcfunction(L, vec4_newindex);
lua_setfield(L, -2, "__newindex");
}
lua_setmetatable(L, -2);
return 1;
}
#if AX_USE_PHYSICS
void physics_material_to_luaval(lua_State* L, const PhysicsMaterial& pm)
{
if (nullptr == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "density"); /* L: table key */
lua_pushnumber(L, (lua_Number)pm.density); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "restitution"); /* L: table key */
lua_pushnumber(L, (lua_Number)pm.restitution); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "friction"); /* L: table key */
lua_pushnumber(L, (lua_Number)pm.friction); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void physics_raycastinfo_to_luaval(lua_State* L, const PhysicsRayCastInfo& info)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "shape"); /* L: table key */
PhysicsShape* shape = info.shape;
if (shape == nullptr)
{
lua_pushnil(L);
}
else
{
int ID = (int)(shape->_ID);
int* luaID = &(shape->_luaID);
toluafix_pushusertype_ccobject(L, ID, luaID, (void*)shape, "ax.PhysicsShape");
}
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "start"); /* L: table key */
vec2_to_luaval(L, info.start);
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "ended"); /* L: table key */
vec2_to_luaval(L, info.end);
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "contact"); /* L: table key */
vec2_to_luaval(L, info.contact);
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "normal"); /* L: table key */
vec2_to_luaval(L, info.normal);
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "fraction"); /* L: table key */
lua_pushnumber(L, (lua_Number)info.fraction); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void physics_contactdata_to_luaval(lua_State* L, const PhysicsContactData* data)
{
if (nullptr == L || nullptr == data)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "points");
vec2_array_to_luaval(L, data->points, data->count);
lua_rawset(L, -3);
lua_pushstring(L, "normal");
vec2_to_luaval(L, data->normal);
lua_rawset(L, -3);
lua_pushstring(L, "POINT_MAX");
lua_pushnumber(L, data->POINT_MAX);
lua_rawset(L, -3);
}
#endif //#if AX_USE_PHYSICS
void size_to_luaval(lua_State* L, const Size& sz)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "width"); /* L: table key */
lua_pushnumber(L, (lua_Number)sz.width); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "height"); /* L: table key */
lua_pushnumber(L, (lua_Number)sz.height); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void rect_to_luaval(lua_State* L, const Rect& rt)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "x"); /* L: table key */
lua_pushnumber(L, (lua_Number)rt.origin.x); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "y"); /* L: table key */
lua_pushnumber(L, (lua_Number)rt.origin.y); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "width"); /* L: table key */
lua_pushnumber(L, (lua_Number)rt.size.width); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "height"); /* L: table key */
lua_pushnumber(L, (lua_Number)rt.size.height); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void color4b_to_luaval(lua_State* L, const Color4B& color)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "r"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.r); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "g"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.g); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "b"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.b); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "a"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.a); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void color4f_to_luaval(lua_State* L, const Color4F& color)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "r"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.r); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "g"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.g); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "b"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.b); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "a"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.a); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void std_thread_id_to_luaval(lua_State* L, const std::thread::id& value) {
auto threadHash = std::hash<std::thread::id>{}(value);
lua_pushinteger(L, threadHash);
}
void color3b_to_luaval(lua_State* L, const Color3B& color)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "r"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.r); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "g"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.g); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "b"); /* L: table key */
lua_pushnumber(L, (lua_Number)color.b); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void affinetransform_to_luaval(lua_State* L, const AffineTransform& inValue)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "a"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.a); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "b"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.b); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "c"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.c); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "d"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.d); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "tx"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.tx); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "ty"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.ty); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void fontdefinition_to_luaval(lua_State* L, const FontDefinition& inValue)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "fontName"); /* L: table key */
tolua_pushcppstring(L, inValue._fontName); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "fontSize"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue._fontSize); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "fontAlignmentH"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue._alignment); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "fontAlignmentV"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue._vertAlignment); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "fontFillColor"); /* L: table key */
color3b_to_luaval(L, inValue._fontFillColor); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "fontDimensions"); /* L: table key */
size_to_luaval(L, inValue._dimensions); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
// Shadow
lua_pushstring(L, "shadowEnabled"); /* L: table key */
lua_pushboolean(L, inValue._shadow._shadowEnabled); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "shadowOffset"); /* L: table key */
size_to_luaval(L, inValue._shadow._shadowOffset); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "shadowBlur"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue._shadow._shadowBlur); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "shadowOpacity"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue._shadow._shadowOpacity); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
// Stroke
lua_pushstring(L, "shadowEnabled"); /* L: table key */
lua_pushboolean(L, inValue._stroke._strokeEnabled); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "strokeColor"); /* L: table key */
color3b_to_luaval(L, inValue._stroke._strokeColor); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "strokeSize"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue._stroke._strokeSize); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
}
void ccvalue_to_luaval(lua_State* L, const ax::Value& inValue)
{
const Value& obj = inValue;
switch (obj.getTypeFamily())
{
case Value::Type::BOOLEAN:
lua_pushboolean(L, obj.asBool());
break;
case Value::Type::FLOAT:
case Value::Type::DOUBLE:
lua_pushnumber(L, obj.asDouble());
break;
case Value::Type::INTEGER:
lua_pushinteger(L, obj.asInt64());
break;
case Value::Type::STRING:
lua_pushstring(L, obj.asStringRef().data());
break;
case Value::Type::VECTOR:
ccvaluevector_to_luaval(L, obj.asValueVector());
break;
case Value::Type::MAP:
ccvaluemap_to_luaval(L, obj.asValueMap());
break;
case Value::Type::INT_KEY_MAP:
ccvaluemapintkey_to_luaval(L, obj.asIntKeyMap());
break;
default:
break;
}
}
void ccvaluemap_to_luaval(lua_State* L, const ax::ValueMap& inValue)
{
lua_newtable(L);
if (nullptr == L)
return;
for (auto iter = inValue.begin(); iter != inValue.end(); ++iter)
{
std::string key = iter->first;
const Value& obj = iter->second;
switch (obj.getTypeFamily())
{
case Value::Type::BOOLEAN:
{
lua_pushstring(L, key.c_str());
lua_pushboolean(L, obj.asBool());
lua_rawset(L, -3);
}
break;
case Value::Type::FLOAT:
case Value::Type::DOUBLE:
{
lua_pushstring(L, key.c_str());
lua_pushnumber(L, obj.asDouble());
lua_rawset(L, -3);
}
break;
case Value::Type::INTEGER:
{
lua_pushstring(L, key.c_str());
lua_pushinteger(L, obj.asInt64());
lua_rawset(L, -3);
}
break;
case Value::Type::STRING:
{
lua_pushstring(L, key.c_str());
lua_pushstring(L, obj.asString().c_str());
lua_rawset(L, -3);
}
break;
case Value::Type::VECTOR:
{
lua_pushstring(L, key.c_str());
ccvaluevector_to_luaval(L, obj.asValueVector());
lua_rawset(L, -3);
}
break;
case Value::Type::MAP:
{
lua_pushstring(L, key.c_str());
ccvaluemap_to_luaval(L, obj.asValueMap());
lua_rawset(L, -3);
}
break;
case Value::Type::INT_KEY_MAP:
{
lua_pushstring(L, key.c_str());
ccvaluemapintkey_to_luaval(L, obj.asIntKeyMap());
lua_rawset(L, -3);
}
break;
default:
break;
}
}
}
void ccvaluemapintkey_to_luaval(lua_State* L, const ax::ValueMapIntKey& inValue)
{
lua_newtable(L);
if (nullptr == L)
return;
for (auto iter = inValue.begin(); iter != inValue.end(); ++iter)
{
std::stringstream keyss;
keyss << iter->first;
std::string key = keyss.str();
const Value& obj = iter->second;
switch (obj.getTypeFamily())
{
case Value::Type::BOOLEAN:
{
lua_pushstring(L, key.c_str());
lua_pushboolean(L, obj.asBool());
lua_rawset(L, -3);
}
break;
case Value::Type::FLOAT:
case Value::Type::DOUBLE:
{
lua_pushstring(L, key.c_str());
lua_pushnumber(L, obj.asDouble());
lua_rawset(L, -3);
}
break;
case Value::Type::INTEGER:
{
lua_pushstring(L, key.c_str());
lua_pushinteger(L, obj.asInt64());
lua_rawset(L, -3);
}
break;
case Value::Type::STRING:
{
lua_pushstring(L, key.c_str());
lua_pushstring(L, obj.asStringRef().data());
lua_rawset(L, -3);
}
break;
case Value::Type::VECTOR:
{
lua_pushstring(L, key.c_str());
ccvaluevector_to_luaval(L, obj.asValueVector());
lua_rawset(L, -3);
}
break;
case Value::Type::MAP:
{
lua_pushstring(L, key.c_str());
ccvaluemap_to_luaval(L, obj.asValueMap());
lua_rawset(L, -3);
}
break;
case Value::Type::INT_KEY_MAP:
{
lua_pushstring(L, key.c_str());
ccvaluemapintkey_to_luaval(L, obj.asIntKeyMap());
lua_rawset(L, -3);
}
break;
default:
break;
}
}
}
void ccvaluevector_to_luaval(lua_State* L, const ax::ValueVector& inValue)
{
lua_newtable(L);
if (nullptr == L)
return;
int index = 1;
for (const auto& obj : inValue)
{
switch (obj.getTypeFamily())
{
case Value::Type::BOOLEAN:
{
lua_pushnumber(L, (lua_Number)index);
lua_pushboolean(L, obj.asBool());
lua_rawset(L, -3);
++index;
}
break;
case Value::Type::FLOAT:
case Value::Type::DOUBLE:
{
lua_pushnumber(L, (lua_Number)index);
lua_pushnumber(L, obj.asDouble());
lua_rawset(L, -3);
++index;
}
break;
case Value::Type::INTEGER:
{
lua_pushnumber(L, (lua_Number)index);
lua_pushnumber(L, obj.asInt64());
lua_rawset(L, -3);
++index;
}
break;
case Value::Type::STRING:
{
lua_pushnumber(L, (lua_Number)index);
lua_pushstring(L, obj.asString().c_str());
lua_rawset(L, -3);
++index;
}
break;
case Value::Type::VECTOR:
{
lua_pushnumber(L, (lua_Number)index);
ccvaluevector_to_luaval(L, obj.asValueVector());
lua_rawset(L, -3);
++index;
}
break;
case Value::Type::MAP:
{
lua_pushnumber(L, (lua_Number)index);
ccvaluemap_to_luaval(L, obj.asValueMap());
lua_rawset(L, -3);
++index;
}
break;
case Value::Type::INT_KEY_MAP:
{
lua_pushnumber(L, (lua_Number)index);
ccvaluemapintkey_to_luaval(L, obj.asIntKeyMap());
lua_rawset(L, -3);
++index;
}
break;
default:
break;
}
}
}
void mat4_to_luaval(lua_State* L, const ax::Mat4& mat)
{
if (nullptr == L)
return;
lua_newtable(L); /* L: table */
int indexTable = 1;
for (int i = 0; i < 16; i++)
{
lua_pushnumber(L, (lua_Number)indexTable);
lua_pushnumber(L, (lua_Number)mat.m[i]);
lua_rawset(L, -3);
++indexTable;
}
}
void blendfunc_to_luaval(lua_State* L, const ax::BlendFunc& func)
{
if (nullptr == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "src"); /* L: table key */
lua_pushnumber(L, (lua_Number)func.src); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "dst"); /* L: table key */
lua_pushnumber(L, (lua_Number)func.dst); /* L: table key value*/
lua_rawset(L, -3);
}
void ttfconfig_to_luaval(lua_State* L, const ax::TTFConfig& config)
{
if (nullptr == L)
return;
lua_newtable(L);
lua_pushstring(L, "fontFilePath");
lua_pushstring(L, config.fontFilePath.c_str());
lua_rawset(L, -3);
lua_pushstring(L, "fontSize");
lua_pushnumber(L, (lua_Number)config.fontSize);
lua_rawset(L, -3);
lua_pushstring(L, "glyphs");
lua_pushnumber(L, (lua_Number)config.glyphs);
lua_rawset(L, -3);
lua_pushstring(L, "customGlyphs");
lua_pushlstring(L, config.customGlyphs.c_str(), config.customGlyphs.length());
lua_rawset(L, -3);
lua_pushstring(L, "distanceFieldEnabled");
lua_pushboolean(L, config.distanceFieldEnabled);
lua_rawset(L, -3);
lua_pushstring(L, "outlineSize");
lua_pushnumber(L, (lua_Number)config.outlineSize);
lua_rawset(L, -3);
}
void mesh_vertex_attrib_to_luaval(lua_State* L, const ax::MeshVertexAttrib& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
lua_pushstring(L, "type");
lua_pushnumber(L, (lua_Number)inValue.type);
lua_rawset(L, -3);
lua_pushstring(L, "vertexAttrib");
lua_pushnumber(L, (lua_Number)inValue.vertexAttrib);
lua_rawset(L, -3);
lua_pushstring(L, "attribSizeBytes");
lua_pushnumber(L, (lua_Number)inValue.getAttribSizeBytes());
lua_rawset(L, -3);
}
void ccvector_std_string_to_luaval(lua_State* L, const std::vector<std::string>& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
int index = 1;
for (std::string_view value : inValue)
{
lua_pushnumber(L, (lua_Number)index);
lua_pushlstring(L, value.data(), value.length());
lua_rawset(L, -3);
++index;
}
}
void ccvector_int_to_luaval(lua_State* L, const std::vector<int>& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
int index = 1;
for (const int value : inValue)
{
lua_pushnumber(L, (lua_Number)index);
lua_pushnumber(L, (lua_Number)value);
lua_rawset(L, -3);
++index;
}
}
void ccvector_float_to_luaval(lua_State* L, const std::vector<float>& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
int index = 1;
for (const float value : inValue)
{
lua_pushnumber(L, (lua_Number)index);
lua_pushnumber(L, (lua_Number)value);
lua_rawset(L, -3);
++index;
}
}
void ccvector_ushort_to_luaval(lua_State* L, const std::vector<unsigned short>& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
int index = 1;
for (const unsigned short value : inValue)
{
lua_pushnumber(L, (lua_Number)index);
lua_pushnumber(L, (lua_Number)value);
lua_rawset(L, -3);
++index;
}
}
void quaternion_to_luaval(lua_State* L, const ax::Quaternion& inValue)
{
if (NULL == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "x"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.x); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "y"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.y); /* L: table key value*/
lua_rawset(L, -3);
lua_pushstring(L, "z"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.z); /* L: table key value*/
lua_rawset(L, -3);
lua_pushstring(L, "w"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.w); /* L: table key value*/
lua_rawset(L, -3);
}
void texParams_to_luaval(lua_State* L, const ax::Texture2D::TexParams& inValue)
{
if (nullptr == L)
return;
lua_newtable(L); /* L: table */
lua_pushstring(L, "minFilter"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.minFilter); /* L: table key value*/
lua_rawset(L, -3); /* table[key] = value, L: table */
lua_pushstring(L, "magFilter"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.magFilter); /* L: table key value*/
lua_rawset(L, -3);
lua_pushstring(L, "sAddressMode"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.sAddressMode); /* L: table key value*/
lua_rawset(L, -3);
lua_pushstring(L, "sAddressMode"); /* L: table key */
lua_pushnumber(L, (lua_Number)inValue.tAddressMode); /* L: table key value*/
lua_rawset(L, -3);
}
void std_vector_vec3_to_luaval(lua_State* L, const std::vector<ax::Vec3>& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
int index = 1;
for (const ax::Vec3& value : inValue)
{
lua_pushnumber(L, (lua_Number)index);
vec3_to_luaval(L, value);
lua_rawset(L, -3);
++index;
}
}
void std_map_string_string_to_luaval(lua_State* L, const std::map<std::string, std::string>& inValue)
{
if (nullptr == L)
return;
lua_newtable(L);
for (auto iter = inValue.begin(); iter != inValue.end(); ++iter)
{
lua_pushstring(L, iter->first.c_str());
lua_pushstring(L, iter->second.c_str());
lua_rawset(L, -3);
}
}
bool luaval_to_std_map_string_string(lua_State* L, int lo, hlookup::string_map<std::string>* ret, const char* funcName)
{
if (nullptr == L || nullptr == ret || lua_gettop(L) < lo)
return false;
tolua_Error tolua_err;
bool ok = true;
if (!tolua_istable(L, lo, 0, &tolua_err))
{
#if _AX_DEBUG >= 1
luaval_to_native_err(L, "#ferror:", &tolua_err, funcName);
#endif
ok = false;
}
if (!ok)
return ok;
lua_pushnil(L);
std::string_view key;
std::string_view value;
while (lua_next(L, lo) != 0)
{
if (lua_isstring(L, -2) && lua_isstring(L, -1))
{
if (luaval_to_std_string_view(L, -2, &key) && luaval_to_std_string_view(L, -1, &value))
{
ret->emplace(key, value); // (*ret)[key] = value;
}
}
else
{
AXASSERT(false, "string type is needed");
}
lua_pop(L, 1);
}
return ok;
}
bool luaval_to_node(lua_State* L, int lo, const char* type, ax::Node** node)
{
return luaval_to_object<ax::Node>(L, lo, type, node);
}
void node_to_luaval(lua_State* L, const char* type, ax::Node* node)
{
object_to_luaval<ax::Node>(L, type, node);
}
bool luaval_to_vertexLayout(lua_State* L, int pos, ax::backend::VertexLayout& outLayout, const char* message)
{
if (L == nullptr)
return false;
ax::backend::VertexLayout* tmp = nullptr;
auto ret = luaval_to_object<ax::backend::VertexLayout>(L, pos, "ax.VertexLayout", &tmp, message);
if (!tmp)
{
return false;
}
outLayout = *tmp;
return ret;
}
bool luaval_to_samplerDescriptor(lua_State* L,
int pos,
ax::backend::SamplerDescriptor& output,
const char* message)
{
if (L == nullptr)
return false;
lua_pushstring(L, "magFilter");
lua_gettable(L, pos);
if (!lua_isnil(L, -1))
{
output.magFilter = static_cast<ax::backend::SamplerFilter>(lua_tointeger(L, -1));
}
lua_pop(L, 1);
lua_pushstring(L, "minFilter");
lua_gettable(L, pos);
if (!lua_isnil(L, -1))
{
output.minFilter = static_cast<ax::backend::SamplerFilter>(lua_tointeger(L, -1));
}
lua_pop(L, 1);
lua_pushstring(L, "sAddressMode");
lua_gettable(L, pos);
if (!lua_isnil(L, -1))
{
output.sAddressMode = static_cast<ax::backend::SamplerAddressMode>(lua_tointeger(L, -1));
}
lua_pop(L, 1);
lua_pushstring(L, "tAddressMode");
lua_gettable(L, pos);
if (!lua_isnil(L, -1))
{
output.tAddressMode = static_cast<ax::backend::SamplerAddressMode>(lua_tointeger(L, -1));
}
lua_pop(L, 1);
return true;
}
bool luaval_to_uniformLocation(lua_State* L, int pos, ax::backend::UniformLocation& loc, const char* message)
{
if (L == nullptr)
return false;
lua_pushstring(L, "location");
lua_gettable(L, pos);
if (lua_isnil(L, -1))
{
AXASSERT(false, "invalidate UniformLocation value");
}
int len = lua_objlen(L, -1);
for (int i = 0; i < len; i++)
{
lua_rawgeti(L, -1, i + 1);
loc.location[i] = lua_tointeger(L, -1);
lua_pop(L, 1);
}
lua_pop(L, 1);
lua_pushstring(L, "shaderStage");
lua_gettable(L, pos);
if (lua_isnil(L, -1))
{
AXASSERT(false, "invalidate UniformLocation value");
}
loc.shaderStage = static_cast<ax::backend::ShaderStage>(lua_tointeger(L, -1));
lua_pop(L, 1);
return true;
}
void uniformLocation_to_luaval(lua_State* L, const ax::backend::UniformLocation& loc)
{
if (L == nullptr)
return;
lua_newtable(L);
lua_pushstring(L, "location");
lua_newtable(L);
for (int i = 1; i <= 2; i++)
{
lua_pushnumber(L, i);
lua_pushinteger(L, static_cast<int>(loc.location[i - 1]));
lua_rawset(L, -3);
}
lua_rawset(L, -3);
lua_pushstring(L, "shaderStage");
lua_pushinteger(L, static_cast<int>(loc.shaderStage));
lua_rawset(L, -3);
}
void program_activeattrs_to_luaval(lua_State* L, const hlookup::string_map<ax::backend::AttributeBindInfo>& attrs)
{
if (L == nullptr)
return;
lua_newtable(L);
for (auto& p : attrs)
{
if (p.first.empty())
continue;
lua_newtable(L);
lua_pushstring(L, "attributeName");
lua_pushstring(L, p.first.c_str());
lua_rawset(L, -3);
lua_pushstring(L, "location");
lua_pushinteger(L, p.second.location);
lua_rawset(L, -3);
lua_pushstring(L, "size");
lua_pushinteger(L, p.second.size);
lua_rawset(L, -3);
lua_pushstring(L, "type");
lua_pushinteger(L, p.second.type);
lua_rawset(L, -3);
lua_pushstring(L, p.first.c_str());
lua_insert(L, -2);
lua_rawset(L, -3);
}
}
void resourceData_to_luaval(lua_State* L, const ax::ResourceData& data)
{
if (L == nullptr)
return;
lua_newtable(L);
lua_pushstring(L, "type");
lua_pushinteger(L, data.type);
lua_rawset(L, -3);
lua_pushstring(L, "file");
lua_pushstring(L, data.file.c_str());
lua_rawset(L, -3);
lua_pushstring(L, "plist");
lua_pushstring(L, data.plist.c_str());
lua_rawset(L, -3);
}