mstch.hpp 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. #pragma once
  2. #include <vector>
  3. #include <map>
  4. #include <string>
  5. #include <memory>
  6. #include <functional>
  7. #include <boost/variant.hpp>
  8. namespace mstch {
  9. struct config {
  10. static std::function<std::string(const std::string&)> escape;
  11. };
  12. namespace internal {
  13. template<class N>
  14. class object_t {
  15. public:
  16. const N& at(const std::string& name) const {
  17. cache[name] = (methods.at(name))();
  18. return cache[name];
  19. }
  20. bool has(const std::string name) const {
  21. return methods.count(name) != 0;
  22. }
  23. protected:
  24. template<class S>
  25. void register_methods(S* s, std::map<std::string,N(S::*)()> methods) {
  26. for(auto& item: methods)
  27. this->methods.insert({item.first, std::bind(item.second, s)});
  28. }
  29. private:
  30. std::map<std::string, std::function<N()>> methods;
  31. mutable std::map<std::string, N> cache;
  32. };
  33. template<class T, class N>
  34. class is_fun {
  35. private:
  36. using not_fun = char;
  37. using fun_without_args = char[2];
  38. using fun_with_args = char[3];
  39. template <typename U, U> struct really_has;
  40. template <typename C> static fun_without_args& test(
  41. really_has<N(C::*)() const, &C::operator()>*);
  42. template <typename C> static fun_with_args& test(
  43. really_has<N(C::*)(const std::string&) const,
  44. &C::operator()>*);
  45. template <typename> static not_fun& test(...);
  46. public:
  47. static bool const no_args = sizeof(test<T>(0)) == sizeof(fun_without_args);
  48. static bool const has_args = sizeof(test<T>(0)) == sizeof(fun_with_args);
  49. };
  50. template<class N>
  51. using node_renderer = std::function<std::string(const N& n)>;
  52. template<class N>
  53. class lambda_t {
  54. public:
  55. template<class F>
  56. lambda_t(F f, typename std::enable_if<is_fun<F, N>::no_args>::type* = 0):
  57. fun([f](node_renderer<N> renderer, const std::string&) {
  58. return renderer(f());
  59. })
  60. {
  61. }
  62. template<class F>
  63. lambda_t(F f, typename std::enable_if<is_fun<F, N>::has_args>::type* = 0):
  64. fun([f](node_renderer<N> renderer, const std::string& text) {
  65. return renderer(f(text));
  66. })
  67. {
  68. }
  69. std::string operator()(node_renderer<N> renderer,
  70. const std::string& text = "") const
  71. {
  72. return fun(renderer, text);
  73. }
  74. private:
  75. std::function<std::string(node_renderer<N> renderer, const std::string&)> fun;
  76. };
  77. }
  78. using node = boost::make_recursive_variant<
  79. std::nullptr_t, std::string, int, double, bool,
  80. internal::lambda_t<boost::recursive_variant_>,
  81. std::shared_ptr<internal::object_t<boost::recursive_variant_>>,
  82. std::map<const std::string, boost::recursive_variant_>,
  83. std::vector<boost::recursive_variant_>>::type;
  84. using object = internal::object_t<node>;
  85. using lambda = internal::lambda_t<node>;
  86. using map = std::map<const std::string, node>;
  87. using array = std::vector<node>;
  88. std::string render(
  89. const std::string& tmplt,
  90. const node& root,
  91. const std::map<std::string,std::string>& partials =
  92. std::map<std::string,std::string>());
  93. }