aboutsummaryrefslogtreecommitdiffstats
path: root/lib/sol2/examples/source/pairs_with_raw_functions.cpp
diff options
context:
space:
mode:
authorAndy <drumsetmonkey@gmail.com>2019-08-29 13:07:45 -0400
committerAndy <drumsetmonkey@gmail.com>2019-08-29 13:07:45 -0400
commit4ac4b280abf2ffa28caa5a532353115a3033444f (patch)
tree2a13d658bb454360b2faf401244bb0321d3460d4 /lib/sol2/examples/source/pairs_with_raw_functions.cpp
parente9758416b18b27a65337c28d9641afc0ee89b34b (diff)
parent7a46fa2dd3dad3f038bf8e7339bc67abca428ae6 (diff)
Started creating scripting library/namespace and added sol2 for interfacing
Diffstat (limited to 'lib/sol2/examples/source/pairs_with_raw_functions.cpp')
-rw-r--r--lib/sol2/examples/source/pairs_with_raw_functions.cpp87
1 files changed, 87 insertions, 0 deletions
diff --git a/lib/sol2/examples/source/pairs_with_raw_functions.cpp b/lib/sol2/examples/source/pairs_with_raw_functions.cpp
new file mode 100644
index 0000000..cfb5af5
--- /dev/null
+++ b/lib/sol2/examples/source/pairs_with_raw_functions.cpp
@@ -0,0 +1,87 @@
+#define SOL_ALL_SAFETIES_ON 1
+#include <sol/sol.hpp>
+
+#include <map>
+#include <iostream>
+
+struct my_thing {
+ std::map<std::string, int> m{
+ { "bark", 20 },
+ { "woof", 60 },
+ { "borf", 30 },
+ { "awoo", 5 },
+ };
+
+ my_thing() {
+
+ }
+};
+
+struct lua_iterator_state {
+ typedef std::map<std::string, int>::iterator it_t;
+ it_t it;
+ it_t last;
+
+ lua_iterator_state(my_thing& mt) : it(mt.m.begin()), last(mt.m.end()) {}
+};
+
+int my_next(lua_State* L) {
+ // this gets called
+ // to start the first iteration, and every
+ // iteration there after
+ // the state you passed in pairs is argument 1
+ // the key value is argument 2
+ // we do not care about the key value here
+ lua_iterator_state& it_state = sol::stack::get<sol::user<lua_iterator_state>>(L, 1);
+ auto& it = it_state.it;
+ if (it == it_state.last) {
+ return sol::stack::push(L, sol::lua_nil);
+ }
+ auto itderef = *it;
+ // 2 values are returned (pushed onto the stack):
+ // the key and the value
+ // the state is left alone
+ int pushed = sol::stack::push(L, itderef.first);
+ pushed += sol::stack::push_reference(L, itderef.second);
+ std::advance(it, 1);
+ return pushed;
+}
+
+int my_pairs(lua_State* L) {
+ my_thing& mt = sol::stack::get<my_thing>(L, 1);
+ lua_iterator_state it_state(mt);
+ // pairs expects 3 returns:
+ // the "next" function on how to advance,
+ // the "table" itself or some state,
+ // and an initial key value (can be nil)
+
+ // next function controls iteration
+ int pushed = sol::stack::push(L, my_next);
+ pushed += sol::stack::push<sol::user<lua_iterator_state>>(L, std::move(it_state));
+ pushed += sol::stack::push(L, sol::lua_nil);
+ return pushed;
+}
+
+int main(int, char*[]) {
+ std::cout << "===== pairs (using raw Lua C functions) (advanced) =====" << std::endl;
+
+ sol::state lua;
+ lua.open_libraries(sol::lib::base);
+
+ lua.new_usertype<my_thing>("my_thing",
+ sol::meta_function::pairs, &my_pairs
+ );
+
+#if SOL_LUA_VERSION > 501
+ lua.safe_script(R"(
+local mt = my_thing.new()
+for k, v in pairs(mt) do
+ print(k, v)
+end
+)");
+#endif // Does not work on Lua 5.1 and below
+
+ std::cout << std::endl;
+
+ return 0;
+}