// sol2 // The MIT License (MIT) // Copyright (c) 2013-2017 Rapptz, ThePhD and contributors // 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. #ifndef SOL_USERTYPE_HPP #define SOL_USERTYPE_HPP #include "stack.hpp" #include "usertype_metatable.hpp" #include "simple_usertype_metatable.hpp" #include "container_usertype_metatable.hpp" #include namespace sol { template class usertype { private: std::unique_ptr metatableregister; template usertype(detail::verified_tag, Args&&... args) : metatableregister(detail::make_unique_deleter, Args...>, detail::deleter>(std::forward(args)...)) { static_assert(detail::has_destructor::value, "this type does not have an explicit destructor declared; please pass a custom destructor function wrapped in sol::destruct, especially if the type does not have an accessible (private) destructor"); } template usertype(detail::add_destructor_tag, Args&&... args) : usertype(detail::verified, std::forward(args)..., "__gc", default_destructor) { } template usertype(detail::check_destructor_tag, Args&&... args) : usertype(meta::condition, meta::neg>>, detail::add_destructor_tag, detail::verified_tag>(), std::forward(args)...) { } public: template usertype(Args&&... args) : usertype(meta::condition, meta::neg>>, decltype(default_constructor), detail::check_destructor_tag>(), std::forward(args)...) { } template usertype(constructors constructorlist, Args&&... args) : usertype(detail::check_destructor_tag(), std::forward(args)..., "new", constructorlist) { } template usertype(constructor_wrapper constructorlist, Args&&... args) : usertype(detail::check_destructor_tag(), std::forward(args)..., "new", constructorlist) { } template usertype(simple_tag, lua_State* L, Args&&... args) : metatableregister(detail::make_unique_deleter, detail::deleter>(L, std::forward(args)...)) { } usertype_detail::registrar* registrar_data() { return metatableregister.get(); } int push(lua_State* L) { int r = metatableregister->push_um(L); metatableregister = nullptr; return r; } }; template class simple_usertype : public usertype { private: typedef usertype base_t; lua_State* state; public: template simple_usertype(lua_State* L, Args&&... args) : base_t(simple, L, std::forward(args)...), state(L) { } template void set(N&& n, F&& f) { auto meta = static_cast*>(base_t::registrar_data()); meta->add(state, std::forward(n), std::forward(f)); } }; namespace stack { template struct pusher> { static int push(lua_State* L, usertype& user) { return user.push(L); } }; } // namespace stack } // namespace sol #endif // SOL_USERTYPE_HPP