Add calendar example and supporting functions
This commit is contained in:
parent
cad9f96739
commit
90d93d2583
4 changed files with 192 additions and 0 deletions
143
examples/calendar/main.cpp
Normal file
143
examples/calendar/main.cpp
Normal file
|
@ -0,0 +1,143 @@
|
||||||
|
/* Copyright 2020, Michele Santullo
|
||||||
|
* This file is part of wrenpp.
|
||||||
|
*
|
||||||
|
* Wrenpp is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Wrenpp is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with wrenpp. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "wrenpp/def_configuration.hpp"
|
||||||
|
#include "wrenpp/vm_fun.hpp"
|
||||||
|
#include <string_view>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <ctime>
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
#include <cstring>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
constexpr const char g_calendar_src[] =
|
||||||
|
"foreign class Calendar {" R"script(
|
||||||
|
construct new() {}
|
||||||
|
foreign static today()
|
||||||
|
foreign add_appointment(desc)
|
||||||
|
}
|
||||||
|
)script";
|
||||||
|
|
||||||
|
constexpr const char g_test_script[] =
|
||||||
|
R"script(import "calendar" for Calendar
|
||||||
|
var cale = Calendar.new()
|
||||||
|
System.print("Today is %(Calendar.today())")
|
||||||
|
cale.add_appointment("go get a haircut")
|
||||||
|
)script";
|
||||||
|
|
||||||
|
auto today_unique_str() {
|
||||||
|
std::time_t now = std::time(nullptr);
|
||||||
|
constexpr const std::size_t buff_sz = 1024;
|
||||||
|
auto mem(std::make_unique<char[]>(buff_sz));
|
||||||
|
const std::size_t sz = std::strftime(mem.get(), buff_sz, "%A %-e %B %Y", std::localtime(&now));
|
||||||
|
if (not sz)
|
||||||
|
std::strcpy(mem.get(), "n/a");
|
||||||
|
return mem;
|
||||||
|
}
|
||||||
|
|
||||||
|
class Calendar {
|
||||||
|
public:
|
||||||
|
Calendar() {
|
||||||
|
//std::cout << "Calendar initialized\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
~Calendar() {
|
||||||
|
try {
|
||||||
|
//std::cout << "Calendar destroyed\n";
|
||||||
|
}
|
||||||
|
catch (...) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void today (wren::VM& vm) {
|
||||||
|
vm.set_slot_string(0, today_unique_str().get());
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_appointment (wren::VM& vm) {
|
||||||
|
m_appointments.push_back(wren::get<std::string>(vm, 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
void print_appointments() const {
|
||||||
|
std::cout << "Appointments for " << today_unique_str().get() << ":\n";
|
||||||
|
if (m_appointments.empty()) {
|
||||||
|
std::cout << "\t[none]\n";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
for (const auto& app : m_appointments) {
|
||||||
|
std::cout << '\t' << app << '\n';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<std::string> m_appointments;
|
||||||
|
};
|
||||||
|
|
||||||
|
class MyConf : public wren::DefConfiguration {
|
||||||
|
public:
|
||||||
|
char* load_module_fn(wren::VM* vm, const char* module_name) {
|
||||||
|
if (std::string_view{module_name} == "calendar") {
|
||||||
|
constexpr const std::size_t buff_sz = sizeof(g_calendar_src);
|
||||||
|
char* const buff = static_cast<char*>(MyConf::reallocate_fn(nullptr, buff_sz));
|
||||||
|
std::copy(g_calendar_src, g_calendar_src + buff_sz / sizeof(g_calendar_src[0]), buff);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
wren::foreign_method_t foreign_method_fn(wren::VM* vm, const char* m, const char* c, bool is_static, const char* s) {
|
||||||
|
std::string_view module(m);
|
||||||
|
std::string_view class_name(c);
|
||||||
|
std::string_view signature(s);
|
||||||
|
|
||||||
|
if (module == "calendar" and class_name == "Calendar") {
|
||||||
|
if (is_static and signature == "today()")
|
||||||
|
return &Calendar::today;
|
||||||
|
else if (not is_static and signature == "add_appointment(_)")
|
||||||
|
return wren::make_method_bindable<&Calendar::add_appointment>();
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
wren::foreign_class_t foreign_class_fn(wren::VM* vm, const char* m, const char* c) {
|
||||||
|
std::string_view module(m);
|
||||||
|
std::string_view class_name(c);
|
||||||
|
|
||||||
|
if (module == "calendar" and class_name == "Calendar") {
|
||||||
|
return wren::make_foreign_class<Calendar>();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return {nullptr, nullptr};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} //unnamed namespace
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
MyConf conf;
|
||||||
|
wren::VM vm(&conf, nullptr);
|
||||||
|
vm.interpret("main", g_test_script);
|
||||||
|
|
||||||
|
vm.ensure_slots(1);
|
||||||
|
wren::variable(vm, {"main", "cale"}, 0);
|
||||||
|
const auto cale = wren::foreign<Calendar>(vm, 0);
|
||||||
|
cale->print_appointments();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
5
examples/calendar/meson.build
Normal file
5
examples/calendar/meson.build
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
executable('calendar',
|
||||||
|
'main.cpp',
|
||||||
|
dependencies: wrenpp_dep,
|
||||||
|
install: false,
|
||||||
|
)
|
|
@ -1,2 +1,3 @@
|
||||||
subdir('dieroll')
|
subdir('dieroll')
|
||||||
subdir('greet')
|
subdir('greet')
|
||||||
|
subdir('calendar')
|
||||||
|
|
|
@ -56,6 +56,12 @@ namespace wren {
|
||||||
template <typename R, std::size_t N, typename... Args>
|
template <typename R, std::size_t N, typename... Args>
|
||||||
R call (VM& vm, const ModuleAndName& object, const char (&method)[N], const Args&... args);
|
R call (VM& vm, const ModuleAndName& object, const char (&method)[N], const Args&... args);
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
foreign_class_t make_foreign_class();
|
||||||
|
|
||||||
|
template <auto V>
|
||||||
|
foreign_method_t make_method_bindable();
|
||||||
|
|
||||||
void interpret (VM& vm, const std::string& module_name, const std::string& script);
|
void interpret (VM& vm, const std::string& module_name, const std::string& script);
|
||||||
void set (VM& vm, int slot_num, const char* value);
|
void set (VM& vm, int slot_num, const char* value);
|
||||||
void set (VM& vm, int slot_num, double value);
|
void set (VM& vm, int slot_num, double value);
|
||||||
|
@ -69,6 +75,7 @@ namespace wren {
|
||||||
void set (VM& vm, int slot_num, int value);
|
void set (VM& vm, int slot_num, int value);
|
||||||
std::string_view slot_string_view (VM& vm, int slot_num);
|
std::string_view slot_string_view (VM& vm, int slot_num);
|
||||||
template <typename T> T get (VM& vm, int slot_num);
|
template <typename T> T get (VM& vm, int slot_num);
|
||||||
|
template <typename T> T* foreign (VM& vm, int slot_num);
|
||||||
void variable(VM& vm, const ModuleAndName& mod_and_name, int slot);
|
void variable(VM& vm, const ModuleAndName& mod_and_name, int slot);
|
||||||
void variable(VM& vm, const Handle& handle, int slot);
|
void variable(VM& vm, const Handle& handle, int slot);
|
||||||
|
|
||||||
|
@ -116,6 +123,16 @@ namespace wren {
|
||||||
vm.ensure_slots(sizeof...(Args));
|
vm.ensure_slots(sizeof...(Args));
|
||||||
(set_single_for_call(vm, Indices + 1, args), ...);
|
(set_single_for_call(vm, Indices + 1, args), ...);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <auto V> struct MakeMethodBindable;
|
||||||
|
template <typename T, void(T::*Method)(VM&)> struct MakeMethodBindable<Method> {
|
||||||
|
static foreign_method_t make() {
|
||||||
|
return [](VM& vm) {
|
||||||
|
T* obj = foreign<T>(vm, 0);
|
||||||
|
(obj->*Method)(vm);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
} //namespace detail
|
} //namespace detail
|
||||||
|
|
||||||
#if __cpp_concepts >= 201907
|
#if __cpp_concepts >= 201907
|
||||||
|
@ -167,6 +184,12 @@ namespace wren {
|
||||||
vm.set_slot_double(slot_num, static_cast<double>(value));
|
vm.set_slot_double(slot_num, static_cast<double>(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
inline T* foreign (VM& vm, int slot_num) {
|
||||||
|
T* obj = static_cast<T*>(vm.slot_foreign(slot_num));
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
template <typename R, typename... Args>
|
template <typename R, typename... Args>
|
||||||
inline R call (VM& vm, const Handle& object, const Handle& method, const Args&... args) {
|
inline R call (VM& vm, const Handle& object, const Handle& method, const Args&... args) {
|
||||||
detail::set_for_call<Args...>(std::make_integer_sequence<int, sizeof...(Args)>(), vm, args...);
|
detail::set_for_call<Args...>(std::make_integer_sequence<int, sizeof...(Args)>(), vm, args...);
|
||||||
|
@ -205,4 +228,24 @@ namespace wren {
|
||||||
Handle obj_handle = vm.slot_handle(0);
|
Handle obj_handle = vm.slot_handle(0);
|
||||||
return call<R, N, Args...>(vm, obj_handle, method, args...);
|
return call<R, N, Args...>(vm, obj_handle, method, args...);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
inline foreign_class_t make_foreign_class() {
|
||||||
|
foreign_class_t ret;
|
||||||
|
ret.allocate = [](VM& vm) {
|
||||||
|
void* const mem = vm.set_slot_new_foreign(0, 0, sizeof(T));
|
||||||
|
new(mem) T;
|
||||||
|
};
|
||||||
|
ret.finalize = [](void* mem) {
|
||||||
|
const auto cale = static_cast<T*>(mem);
|
||||||
|
cale->~T();
|
||||||
|
};
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <auto V>
|
||||||
|
inline foreign_method_t make_method_bindable() {
|
||||||
|
return detail::MakeMethodBindable<V>::make();
|
||||||
|
}
|
||||||
} //namespace wren
|
} //namespace wren
|
||||||
|
|
Loading…
Reference in a new issue