#pragma once #include "../utils/addresser.hpp" #include "Traits.hpp" #include "../loader/Log.hpp" namespace geode::modifier { /** * A helper struct that generates a static function that calls the given function. */ #define GEODE_AS_STATIC_FUNCTION(FunctionName_) \ template \ struct AsStaticFunction_##FunctionName_ { \ template \ struct Impl {}; \ template \ struct Impl { \ static Return GEODE_CDECL_CALL function(Params... params) { \ return Class2::FunctionName_(params...); \ } \ }; \ template \ struct Impl { \ static Return GEODE_CDECL_CALL function(Class* self, Params... params) { \ auto self2 = addresser::rthunkAdjust( \ Resolve::func(&Class2::FunctionName_), self \ ); \ return self2->Class2::FunctionName_(params...); \ } \ }; \ template \ struct Impl { \ static Return GEODE_CDECL_CALL function(Class const* self, Params... params) { \ auto self2 = addresser::rthunkAdjust( \ Resolve::func(&Class2::FunctionName_), self \ ); \ return self2->Class2::FunctionName_(params...); \ } \ }; \ static constexpr auto value = &Impl::function; \ }; GEODE_AS_STATIC_FUNCTION(constructor) GEODE_AS_STATIC_FUNCTION(destructor) #define GEODE_CONCEPT_FUNCTION_CHECK(FunctionName_) \ template \ concept FunctionExists_##FunctionName_ = requires(Class* self, Args... args) { \ self->FunctionName_(args...); \ }; }