Add calendar example and supporting functions

This commit is contained in:
King_DuckZ 2020-05-03 01:54:00 +02:00
parent cad9f96739
commit 90d93d2583
4 changed files with 192 additions and 0 deletions

143
examples/calendar/main.cpp Normal file
View 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;
}

View file

@ -0,0 +1,5 @@
executable('calendar',
'main.cpp',
dependencies: wrenpp_dep,
install: false,
)

View file

@ -1,2 +1,3 @@
subdir('dieroll')
subdir('greet')
subdir('calendar')

View file

@ -56,6 +56,12 @@ namespace wren {
template <typename R, std::size_t N, typename... 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 set (VM& vm, int slot_num, const char* 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);
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* foreign (VM& vm, int slot_num);
void variable(VM& vm, const ModuleAndName& mod_and_name, int slot);
void variable(VM& vm, const Handle& handle, int slot);
@ -116,6 +123,16 @@ namespace wren {
vm.ensure_slots(sizeof...(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
#if __cpp_concepts >= 201907
@ -167,6 +184,12 @@ namespace wren {
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>
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...);
@ -205,4 +228,24 @@ namespace wren {
Handle obj_handle = vm.slot_handle(0);
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