34 #define BOOST_FT_AUTODETECT_CALLING_CONVENTIONS
35 #include <boost/function_types/components.hpp>
36 #include <boost/function_types/is_nonmember_callable_builtin.hpp>
38 #include <boost/mpl/begin_end.hpp>
39 #include <boost/mpl/deref.hpp>
40 #include <boost/unordered_map.hpp>
41 #include <boost/utility/enable_if.hpp>
51 namespace mpl = boost::mpl;
52 namespace ft = boost::function_types;
89 #define XLKIT_TYPEINFO(TYPE, CODE, HELP) \
91 struct TypeInfo<TYPE> { \
92 static size_t size() { return sizeof(TYPE); } \
93 static char code() { return CODE; } \
94 static const char* name() { return #TYPE; } \
95 static const char* help() { return HELP; } \
99 XLKIT_TYPEINFO(
double,
'B',
"Number")
100 XLKIT_TYPEINFO(const
char*, 'C', "String")
101 XLKIT_TYPEINFO(uint16_t, 'H', "Unsigned Integer")
102 XLKIT_TYPEINFO(int16_t, 'I', "Signed Integer")
103 XLKIT_TYPEINFO(int32_t, 'J', "Signed Integer")
104 XLKIT_TYPEINFO(
xlOperand*, 'P', "Cell or Cell Range")
105 XLKIT_TYPEINFO(const xlOperand*, 'P', "Cell or Cell Range")
106 XLKIT_TYPEINFO(ResultOperandPtr, 'P', "Cell or Cell Range")
108 #undef XLKIT_TYPEINFO
111 template <
typename T>
116 template <
typename S>
117 static T value(S s) {
122 struct ErrorResult<ResultOperandPtr> {
123 static ResultOperandPtr value() {
124 return ResultOperandPtr();
126 template <
typename S>
127 static ResultOperandPtr value(S s) {
133 template <
typename T>
135 static const char* name() {
138 static const char* help() {
146 template <
typename T,
typename PARM_HELP = detail::empty_>
150 typedef PARM_HELP parm_help_tag;
154 return detail::ParmHelp<PARM_HELP>::name();
158 return detail::ParmHelp<PARM_HELP>::help();
166 operator const T&()
const {
180 template <
typename T,
typename PARM_HELP>
181 struct TypeInfo< Parm<T, PARM_HELP> > {
182 static size_t size() {
183 return TypeInfo<T>::size();
186 return TypeInfo<T>::code();
188 static const char* name() {
191 return TypeInfo<T>::name();
194 static const char* help() {
197 return TypeInfo<T>::help();
216 void setAddinLabel(
const char* label);
221 template <
typename F>
222 void addFunction(
const std::string& name, F f,
const char* help) {
223 myFunctions[name] = makeWrapper<F>(name, f, help);
226 template <
typename F>
228 const std::string& name, F f,
const char* help) {
229 myFunctions[name] = makeWrapper<F>(excel_name, f, help);
235 for (
const auto& i : myFunctions) {
236 printf(
"'%s' -> '%s' [", i.first.c_str(), i.second.myTypes.c_str());
237 for (
int j = 0, n = i.second.myParmHelp.size(); j < n; ++j) {
240 printf(
"%s", i.second.myParmHelp[j].c_str());
250 Wrapper(
const std::string& func_name,
251 const std::string& sig,
252 const std::string& func_help,
253 const std::string& arg_names,
254 const std::vector<std::string>& parm_help)
255 : myFuncName(func_name)
257 , myFuncHelp(func_help)
258 , myArgNames(arg_names)
259 , myParmHelp(parm_help) {
262 std::string myFuncName;
264 std::string myFuncHelp;
265 std::string myArgNames;
266 std::vector<std::string> myParmHelp;
269 typedef boost::unordered_map<std::string, Wrapper> NameMap;
281 template <
typename F>
283 makeWrapper(
const std::string&name, F f,
const char* func_help,
284 typename boost::enable_if< ft::is_nonmember_callable_builtin<F, ft::stdcall_cc>
285 >::type *dummy = 0) {
287 typedef Func<F> FuncT;
290 Func<F>::getTypes(types);
293 typedef typename mpl::next<mpl::begin< ft::components<F> >::type>::type Second;
294 typedef typename mpl::end< ft::components<F> >::type End;
296 std::string arg_names;
297 Func<F, Second, End>::getArgNames(arg_names);
299 std::vector<std::string> parm_help;
300 Func<F, Second, End>::getParmHelp(parm_help);
302 return Wrapper(name, types, func_help, arg_names, parm_help);
308 ,
class Beg =
typename mpl::begin< ft::components<F> >::type
309 ,
class End =
typename mpl::end< ft::components<F> >::type
317 static Registry* theInstance;
328 template <
typename F,
typename Curr,
typename End>
329 struct Registry::Func {
331 static void getTypes(std::string& types) {
332 typedef typename mpl::next<Curr>::type Next;
333 typedef typename mpl::deref<Curr>::type Type;
335 types += detail::TypeInfo<Type>::code();
336 Func<F, Next, End>::getTypes(types);
339 static void getArgNames(std::string& arg_names) {
340 typedef typename mpl::next<Curr>::type Next;
341 typedef typename mpl::deref<Curr>::type Type;
343 if (arg_names.size() > 0)
345 arg_names += detail::TypeInfo<Type>::name();
346 Func<F, Next, End>::getArgNames(arg_names);
349 static void getParmHelp(std::vector<std::string>& parm_help) {
350 typedef typename mpl::next<Curr>::type Next;
351 typedef typename mpl::deref<Curr>::type Type;
353 parm_help.push_back(detail::TypeInfo<Type>::help());
354 Func<F, Next, End>::getParmHelp(parm_help);
358 template <
typename F,
typename End>
359 struct Registry::Func<F, End, End> {
360 static void getTypes(std::string&) { }
361 static void getArgNames(std::string&) { }
362 static void getParmHelp(std::vector<std::string>&) { }
374 template <
typename T,
typename PARM_HELP = xlkit::detail::empty_>
375 using xlParm = xlkit::Parm<T, PARM_HELP>;
389 #define XLKIT_INIT_ADDIN_LABEL(LABEL) \
390 struct XLInitAddinLabel { \
391 XLInitAddinLabel() { \
392 xlkit::Registry::instance().setAddinLabel(LABEL); \
395 static XLInitAddinLabel theInitAddinLabel; \
401 #define XLKIT_PARM(VALUE_TYPE, NAME, HELP) \
402 struct HELP_FOR_##NAME { }; \
404 XLKIT_USE_VERSION_NAMESPACE \
405 namespace XLKIT_VERSION_NAME { \
407 template <> struct ParmHelp<HELP_FOR_##NAME> { \
408 static const char* name() { return #NAME; } \
409 static const char* help() { return HELP; } \
412 typedef xlParm<VALUE_TYPE, HELP_FOR_##NAME> xlParm##NAME;
416 #define XLKIT_REGISTER(FUNC, HELP) \
417 struct FUNC##Registrar { \
418 FUNC##Registrar() { \
419 xlkit::Registry::instance().addFunction(#FUNC,FUNC,HELP); \
422 static FUNC##Registrar the##FUNC##Registrar; \
427 #define XLKIT_REGISTER_AS(XLNAME, FUNC, HELP) \
428 struct FUNC##Registrar { \
429 FUNC##Registrar() { \
430 xlkit::Registry::instance().addFunction(XLNAME,#FUNC,FUNC,HELP); \
433 static FUNC##Registrar the##FUNC##Registrar; \
437 #define XLKIT_API __stdcall
440 #define XLKIT_PRAGMA_DLL_EXPORT \
441 __pragma(comment(linker, "/EXPORT:" __FUNCTION__ "=" __FUNCDNAME__))
444 #define XLKIT_BEGIN_FUNCTION \
445 XLKIT_PRAGMA_DLL_EXPORT \
451 #define XLKIT_END_FUNCTION(RESULT_T) \
452 } catch (xlkit::xlException& err) { \
453 XLDBG("Exception caught: %s", err.what()); \
454 return xlkit::detail::ErrorResult<RESULT_T>::value(); \
455 } catch (std::exception& err){ \
456 XLDBG("Exception caught: %s", err.what()); \
457 return xlkit::detail::ErrorResult<RESULT_T>::value(err.what()); \
458 } catch (xlkit::xlError& err){ \
459 XLDBG("Exception caught: %s", err.str().c_str()); \
460 return xlkit::detail::ErrorResult<RESULT_T>::value(err); \
462 XLDBG("Unknown exception caught"); \
463 return xlkit::detail::ErrorResult<RESULT_T>::value(); \
Stores all registry of functions for the XLL.
Definition: xlkit.hpp:205
ResultOperandPtr()
Get pointer to TLS copy and default initialize it.
Definition: xlkit.cpp:332
xlkit::ResultOperandPtr xlResultOperandPtr
Return value for XLL functions. See xlResultOperandPtr.
Definition: xlkit.hpp:379
xlkit::xlOperand xlOperand
C++ wrapper for XLOPER struct. See xlOper4.
Definition: xlOperand.hpp:547
void addFunction(const std::string &name, F f, const char *help)
Register a new function.Name of the Excel function is the same as the C++ function name...
Definition: xlkit.hpp:222
Return value for XLL functions.
Definition: xlkit.hpp:55
void dump()
Print a list of all registered functions for debugging.
Definition: xlkit.hpp:234
#define XLKIT_VERSION_NAME
Version namespace for this library.
Definition: xlversion.hpp:33
Information for a registered function.
Definition: xlkit.hpp:247
C++ methods that operate on top of an XLOPER struct.
Definition: xlOperand.hpp:130
static Registry & instance()
Get the singleton instance.
Definition: xlkit.hpp:210
XLKit debugging facility for the XLDBG() macro.
const NameMap & functions() const
List of registered functions.
Definition: xlkit.hpp:272
#define XLKIT_USE_VERSION_NAMESPACE
Macro used to pull the versioned namespace into the main xlkit namepsace.
Definition: xlversion.hpp:52
xlkit::Parm< T, PARM_HELP > xlParm
An xlkit function parameter of type T, with optional help for convenience. See Parm.
Definition: xlkit.hpp:375
static const char * help()
Help of the parameter for Excel.
Definition: xlkit.hpp:157
Function parameter with associated help.
Definition: xlkit.hpp:147
void addFunction(const std::string &excel_name, const std::string &name, F f, const char *help)
Name of the Excel function is different from the the C++ function name.
Definition: xlkit.hpp:227
const T & value() const
Value of the parameter.
Definition: xlkit.hpp:163
static const char * name()
Name of the parameter for Excel.
Definition: xlkit.hpp:153
Provides XLKIT_THROW() macro and xlException class.