mirror of
				https://github.com/pocketpy/pocketpy
				synced 2025-10-26 14:30:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			230 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #pragma once
 | |
| 
 | |
| #include "module.h"
 | |
| #include "type_traits.h"
 | |
| 
 | |
| namespace pkbind {
 | |
| 
 | |
| struct dynamic_attr {};
 | |
| 
 | |
| template <typename T, typename Base = void>
 | |
| class class_ : public type {
 | |
| protected:
 | |
|     handle m_scope;
 | |
| 
 | |
| public:
 | |
|     using type::type;
 | |
|     using underlying_type = T;
 | |
| 
 | |
|     template <typename... Args>
 | |
|     class_(handle scope, const char* name, const Args&... args) :
 | |
|         type(py_newtype(name,
 | |
|                         std::is_same_v<Base, void> ? tp_object : type::of<Base>().index(),
 | |
|                         scope.ptr(),
 | |
|                         [](void* data) {
 | |
|                             static_cast<instance*>(data)->~instance();
 | |
|                         })),
 | |
|         m_scope(scope) {
 | |
|         m_type_map.try_emplace(typeid(T), this->index());
 | |
| 
 | |
|         auto& info = type_info::of<T>();
 | |
|         info.name = name;
 | |
| 
 | |
|         py_newfunction(
 | |
|             py_tpgetmagic(this->index(), __new__),
 | |
|             "__new__(type, *args, **kwargs)",
 | |
|             [](int, py_Ref stack) {
 | |
|                 auto cls = py_offset(stack, 0);
 | |
|                 [[maybe_unused]] auto args = py_offset(stack, 1);
 | |
|                 [[maybe_unused]] auto kwargs = py_offset(stack, 2);
 | |
| 
 | |
|                 auto info = &type_info::of<T>();
 | |
|                 int slot = ((std::is_same_v<dynamic_attr, Args> || ...) ? -1 : 0);
 | |
|                 void* data =
 | |
|                     py_newobject(py_retval(), steal<type>(cls).index(), slot, sizeof(instance));
 | |
|                 new (data) instance{instance::Flag::Own, operator new (info->size), info};
 | |
|                 return true;
 | |
|             },
 | |
|             nullptr,
 | |
|             0);
 | |
|     }
 | |
| 
 | |
|     /// bind constructor
 | |
|     template <typename... Args, typename... Extra>
 | |
|     class_& def(impl::constructor<Args...>, const Extra&... extra) {
 | |
|         if constexpr(!std::is_constructible_v<T, Args...>) {
 | |
|             static_assert(std::is_constructible_v<T, Args...>, "Invalid constructor arguments");
 | |
|         } else {
 | |
|             impl::bind_function<true, false>(
 | |
|                 *this,
 | |
|                 "__init__",
 | |
|                 [](T* self, Args... args) {
 | |
|                     new (self) T(args...);
 | |
|                 },
 | |
|                 extra...);
 | |
|             return *this;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     template <typename Fn, typename... Extra>
 | |
|     class_& def(impl::factory<Fn> factory, const Extra&... extra) {
 | |
|         using ret = callable_return_t<Fn>;
 | |
|         if constexpr(!std::is_same_v<T, ret>) {
 | |
|             static_assert(std::is_same_v<T, ret>, "Factory function must return the class type");
 | |
|         } else {
 | |
|             impl::bind_function<true, false>(*this, "__init__", factory.make(), extra...);
 | |
|             return *this;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// bind member function
 | |
|     template <typename Fn, typename... Extra>
 | |
|     class_& def(const char* name, Fn&& f, const Extra&... extra) {
 | |
|         using first = remove_cvref_t<std::tuple_element_t<0, callable_args_t<remove_cvref_t<Fn>>>>;
 | |
|         constexpr bool is_first_base_of_v = std::is_base_of_v<first, T> || std::is_same_v<first, T>;
 | |
| 
 | |
|         if constexpr(!is_first_base_of_v) {
 | |
|             static_assert(
 | |
|                 is_first_base_of_v,
 | |
|                 "If you want to bind member function, the first argument must be the base class");
 | |
|         } else {
 | |
|             impl::bind_function<true, false>(*this, name, std::forward<Fn>(f), extra...);
 | |
|         }
 | |
| 
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     /// bind operators
 | |
|     template <typename Operator, typename... Extras>
 | |
|     class_& def(Operator op, const Extras&... extras) {
 | |
|         op.execute(*this, extras...);
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     /// bind static function
 | |
|     template <typename Fn, typename... Extra>
 | |
|     class_& def_static(const char* name, Fn&& f, const Extra&... extra) {
 | |
|         impl::bind_function<false, true>(*this, name, std::forward<Fn>(f), extra...);
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename MP, typename... Extras>
 | |
|     class_& def_readwrite(const char* name, MP mp, const Extras&... extras) {
 | |
|         if constexpr(!std::is_member_object_pointer_v<MP>) {
 | |
|             static_assert(std::is_member_object_pointer_v<MP>,
 | |
|                           "def_readwrite only supports pointer to data member");
 | |
|         } else {
 | |
|             impl::bind_property(
 | |
|                 *this,
 | |
|                 name,
 | |
|                 [mp](class_type_t<MP>& self) -> auto& {
 | |
|                     return self.*mp;
 | |
|                 },
 | |
|                 [mp](class_type_t<MP>& self, const member_type_t<MP>& value) {
 | |
|                     self.*mp = value;
 | |
|                 },
 | |
|                 extras...);
 | |
|         }
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename MP, typename... Extras>
 | |
|     class_& def_readonly(const char* name, MP mp, const Extras&... extras) {
 | |
|         if constexpr(!std::is_member_object_pointer_v<MP>) {
 | |
|             static_assert(std::is_member_object_pointer_v<MP>,
 | |
|                           "def_readonly only supports pointer to data member");
 | |
|         } else {
 | |
|             impl::bind_property(
 | |
|                 *this,
 | |
|                 name,
 | |
|                 [mp](class_type_t<MP>& self) -> auto& {
 | |
|                     return self.*mp;
 | |
|                 },
 | |
|                 nullptr,
 | |
|                 extras...);
 | |
|         }
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename Getter, typename Setter, typename... Extras>
 | |
|     class_& def_property(const char* name, Getter&& g, Setter&& s, const Extras&... extras) {
 | |
|         impl::bind_property(*this,
 | |
|                             name,
 | |
|                             std::forward<Getter>(g),
 | |
|                             std::forward<Setter>(s),
 | |
|                             extras...);
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename Getter, typename... Extras>
 | |
|     class_& def_property_readonly(const char* name, Getter&& mp, const Extras&... extras) {
 | |
|         impl::bind_property(*this, name, std::forward<Getter>(mp), nullptr, extras...);
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename Var, typename... Extras>
 | |
|     class_& def_readwrite_static(const char* name, Var& mp, const Extras&... extras) {
 | |
|         static_assert(
 | |
|             dependent_false<Var>,
 | |
|             "define static properties requires metaclass. This is a complex feature with few use cases, so it may never be implemented.");
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename Var, typename... Extras>
 | |
|     class_& def_readonly_static(const char* name, Var& mp, const Extras&... extras) {
 | |
|         static_assert(
 | |
|             dependent_false<Var>,
 | |
|             "define static properties requires metaclass. This is a complex feature with few use cases, so it may never be implemented.");
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     template <typename Getter, typename Setter, typename... Extras>
 | |
|     class_& def_property_static(const char* name, Getter&& g, Setter&& s, const Extras&... extras) {
 | |
|         static_assert(
 | |
|             dependent_false<Getter>,
 | |
|             "define static properties requires metaclass. This is a complex feature with few use cases, so it may never be implemented.");
 | |
|         return *this;
 | |
|     }
 | |
| };
 | |
| 
 | |
| template <typename T, typename... Others>
 | |
| class enum_ : public class_<T, Others...> {
 | |
|     std::vector<std::pair<const char*, object>> m_values;
 | |
| 
 | |
| public:
 | |
|     using Base = class_<T, Others...>;
 | |
| 
 | |
|     template <typename... Args>
 | |
|     enum_(const handle& scope, const char* name, Args&&... args) :
 | |
|         class_<T, Others...>(scope, name, std::forward<Args>(args)...) {
 | |
| 
 | |
|         Base::def(init([](int value) {
 | |
|             return static_cast<T>(value);
 | |
|         }));
 | |
| 
 | |
|         Base::def("__eq__", [](T& self, T& other) {
 | |
|             return self == other;
 | |
|         });
 | |
| 
 | |
|         Base::def_property_readonly("value", [](T& self) {
 | |
|             return int_(static_cast<std::underlying_type_t<T>>(self));
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     enum_& value(const char* name, T value) {
 | |
|         auto var = pkbind::cast(value, return_value_policy::copy);
 | |
|         setattr(*this, name, var);
 | |
|         m_values.emplace_back(name, std::move(var));
 | |
|         return *this;
 | |
|     }
 | |
| 
 | |
|     enum_& export_values() {
 | |
|         for(auto& [name, value]: m_values) {
 | |
|             setattr(Base::m_scope, name, value);
 | |
|         }
 | |
|         return *this;
 | |
|     }
 | |
| };
 | |
| 
 | |
| }  // namespace pkbind
 |