os/ossrv/ossrv_pub/boost_apis/boost/parameter/python.hpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright Daniel Wallin 2006. Use, modification and distribution is
sl@0
     2
// subject to the Boost Software License, Version 1.0. (See accompanying
sl@0
     3
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
sl@0
     4
sl@0
     5
#ifndef BOOST_PARAMETER_PYTHON_060209_HPP
sl@0
     6
# define BOOST_PARAMETER_PYTHON_060209_HPP
sl@0
     7
sl@0
     8
# include <boost/mpl/vector.hpp>
sl@0
     9
# include <boost/mpl/fold.hpp>
sl@0
    10
# include <boost/mpl/prior.hpp>
sl@0
    11
# include <boost/mpl/shift_right.hpp>
sl@0
    12
# include <boost/mpl/shift_left.hpp>
sl@0
    13
# include <boost/mpl/bitand.hpp>
sl@0
    14
# include <boost/mpl/pair.hpp>
sl@0
    15
# include <boost/mpl/size.hpp>
sl@0
    16
# include <boost/mpl/push_back.hpp>
sl@0
    17
# include <boost/mpl/or.hpp>
sl@0
    18
# include <boost/mpl/count_if.hpp>
sl@0
    19
# include <boost/mpl/transform.hpp>
sl@0
    20
# include <boost/mpl/front.hpp>
sl@0
    21
# include <boost/mpl/iterator_range.hpp>
sl@0
    22
# include <boost/mpl/next.hpp>
sl@0
    23
# include <boost/mpl/begin_end.hpp>
sl@0
    24
# include <boost/mpl/not.hpp>
sl@0
    25
# include <boost/mpl/empty.hpp>
sl@0
    26
# include <boost/python/def.hpp>
sl@0
    27
# include <boost/python/make_constructor.hpp>
sl@0
    28
# include <boost/python/init.hpp>
sl@0
    29
# include <boost/python/to_python_converter.hpp>
sl@0
    30
# include <boost/parameter/aux_/maybe.hpp>
sl@0
    31
# include <boost/parameter/aux_/python/invoker.hpp>
sl@0
    32
sl@0
    33
namespace boost { namespace parameter { namespace python 
sl@0
    34
{
sl@0
    35
  namespace python_ = boost::python;
sl@0
    36
}}}
sl@0
    37
sl@0
    38
namespace boost { namespace parameter { namespace python { namespace aux 
sl@0
    39
{
sl@0
    40
sl@0
    41
  inline PyObject* unspecified_type()
sl@0
    42
  {
sl@0
    43
      static PyTypeObject unspecified = {
sl@0
    44
          PyObject_HEAD_INIT(NULL)
sl@0
    45
          0,                                /* ob_size        */
sl@0
    46
          "Boost.Parameter.Unspecified",    /* tp_name        */
sl@0
    47
          PyType_Type.tp_basicsize,         /* tp_basicsize   */
sl@0
    48
          0,                                /* tp_itemsize    */
sl@0
    49
          0,                                /* tp_dealloc     */
sl@0
    50
          0,                                /* tp_print       */
sl@0
    51
          0,                                /* tp_getattr     */
sl@0
    52
          0,                                /* tp_setattr     */
sl@0
    53
          0,                                /* tp_compare     */
sl@0
    54
          0,                                /* tp_repr        */
sl@0
    55
          0,                                /* tp_as_number   */
sl@0
    56
          0,                                /* tp_as_sequence */
sl@0
    57
          0,                                /* tp_as_mapping  */
sl@0
    58
          0,                                /* tp_hash        */
sl@0
    59
          0,                                /* tp_call        */
sl@0
    60
          0,                                /* tp_str         */
sl@0
    61
          0,                                /* tp_getattro    */
sl@0
    62
          0,                                /* tp_setattro    */
sl@0
    63
          0,                                /* tp_as_buffer   */
sl@0
    64
          Py_TPFLAGS_DEFAULT,               /* tp_flags       */
sl@0
    65
          0,                                /* tp_doc         */
sl@0
    66
      };
sl@0
    67
sl@0
    68
      if (unspecified.ob_type == 0)
sl@0
    69
      {
sl@0
    70
          unspecified.ob_type = &PyType_Type;
sl@0
    71
          PyType_Ready(&unspecified);
sl@0
    72
      }
sl@0
    73
sl@0
    74
      return (PyObject*)&unspecified;
sl@0
    75
  }
sl@0
    76
sl@0
    77
  struct empty_tag {};
sl@0
    78
sl@0
    79
  struct empty_tag_to_python
sl@0
    80
  {
sl@0
    81
      static PyObject* convert(empty_tag)
sl@0
    82
      {
sl@0
    83
          return python_::xincref(unspecified_type());
sl@0
    84
      }
sl@0
    85
  };
sl@0
    86
sl@0
    87
}}}} // namespace boost::parameter::python::aux
sl@0
    88
sl@0
    89
namespace boost { namespace python 
sl@0
    90
{
sl@0
    91
sl@0
    92
  // Converts a Python value to a maybe<T>
sl@0
    93
  template <class T>
sl@0
    94
  struct arg_from_python<parameter::aux::maybe<T> >
sl@0
    95
    : arg_from_python<T>
sl@0
    96
  {
sl@0
    97
      arg_from_python(PyObject* p)
sl@0
    98
        : arg_from_python<T>(p)
sl@0
    99
        , empty(parameter::python::aux::unspecified_type() == p)
sl@0
   100
      {}
sl@0
   101
sl@0
   102
      bool convertible() const
sl@0
   103
      {
sl@0
   104
          return empty || arg_from_python<T>::convertible();
sl@0
   105
      }
sl@0
   106
sl@0
   107
      parameter::aux::maybe<T> operator()()
sl@0
   108
      {
sl@0
   109
          if (empty)
sl@0
   110
          {
sl@0
   111
              return parameter::aux::maybe<T>();
sl@0
   112
          }
sl@0
   113
          else
sl@0
   114
          {
sl@0
   115
              return parameter::aux::maybe<T>(
sl@0
   116
                  arg_from_python<T>::operator()()
sl@0
   117
              );
sl@0
   118
          }
sl@0
   119
      }
sl@0
   120
sl@0
   121
      bool empty;
sl@0
   122
  };
sl@0
   123
sl@0
   124
}} // namespace boost::python
sl@0
   125
sl@0
   126
namespace boost { namespace parameter { namespace python {
sl@0
   127
sl@0
   128
namespace aux
sl@0
   129
{
sl@0
   130
sl@0
   131
  template <class K>
sl@0
   132
  struct is_optional
sl@0
   133
    : mpl::not_<
sl@0
   134
          mpl::or_<typename K::required, typename K::optimized_default>
sl@0
   135
      >
sl@0
   136
  {};
sl@0
   137
sl@0
   138
  template <class K, class Required, class Optimized, class T>
sl@0
   139
  struct arg_spec
sl@0
   140
  {
sl@0
   141
      typedef K keyword;
sl@0
   142
      typedef Required required;
sl@0
   143
      typedef T type;
sl@0
   144
      typedef Optimized optimized_default;
sl@0
   145
  };
sl@0
   146
  
sl@0
   147
  template <class K, class T, class Optimized = mpl::false_>
sl@0
   148
  struct make_arg_spec_impl
sl@0
   149
  {
sl@0
   150
      typedef arg_spec<
sl@0
   151
          typename K::first, typename K::second, Optimized, T
sl@0
   152
      > type;
sl@0
   153
  };
sl@0
   154
sl@0
   155
  template <class K, class T>
sl@0
   156
  struct make_arg_spec_impl<K, T, typename K::third>
sl@0
   157
  {
sl@0
   158
      typedef arg_spec<
sl@0
   159
          typename K::first, typename K::second, typename K::third, T
sl@0
   160
      > type;
sl@0
   161
  };
sl@0
   162
sl@0
   163
  template <class K, class T>
sl@0
   164
  struct make_arg_spec
sl@0
   165
    : make_arg_spec_impl<K, T>
sl@0
   166
  {
sl@0
   167
  };
sl@0
   168
sl@0
   169
  template <class Spec, class State>
sl@0
   170
  struct combinations_op
sl@0
   171
  {
sl@0
   172
      typedef typename State::second bits;
sl@0
   173
      typedef typename State::first result0;
sl@0
   174
sl@0
   175
      typedef typename mpl::if_<
sl@0
   176
          mpl::or_<
sl@0
   177
              typename Spec::required
sl@0
   178
            , typename Spec::optimized_default
sl@0
   179
            , mpl::bitand_<bits, mpl::long_<1> >
sl@0
   180
          >
sl@0
   181
        , typename mpl::push_back<result0, Spec>::type
sl@0
   182
        , result0
sl@0
   183
      >::type result;
sl@0
   184
sl@0
   185
      typedef typename mpl::if_<
sl@0
   186
          mpl::or_<
sl@0
   187
              typename Spec::required
sl@0
   188
            , typename Spec::optimized_default
sl@0
   189
          >
sl@0
   190
        , bits
sl@0
   191
        , typename mpl::shift_right<bits, mpl::long_<1> >::type
sl@0
   192
      >::type next_bits;
sl@0
   193
sl@0
   194
      typedef mpl::pair<
sl@0
   195
          result
sl@0
   196
        , next_bits
sl@0
   197
      > type;
sl@0
   198
  };
sl@0
   199
sl@0
   200
  // Used as start value in the recursive arg() composition below.
sl@0
   201
  struct no_keywords
sl@0
   202
  {
sl@0
   203
      template <class T>
sl@0
   204
      T const& operator,(T const& x) const
sl@0
   205
      {
sl@0
   206
          return x;
sl@0
   207
      }
sl@0
   208
  };
sl@0
   209
sl@0
   210
  template <class Def, class F, class Iter, class End, class Keywords>
sl@0
   211
  void def_combination_aux0(
sl@0
   212
      Def def, F f, Iter, End, Keywords const& keywords, mpl::false_)
sl@0
   213
  {
sl@0
   214
      typedef typename mpl::deref<Iter>::type spec;
sl@0
   215
      typedef typename spec::keyword kw;
sl@0
   216
sl@0
   217
      def_combination_aux(
sl@0
   218
          def, f, typename mpl::next<Iter>::type(), End()
sl@0
   219
        , (
sl@0
   220
              keywords, boost::python::arg(kw::keyword_name())
sl@0
   221
          )
sl@0
   222
      );
sl@0
   223
  }
sl@0
   224
sl@0
   225
  template <class Def, class F, class Iter, class End, class Keywords>
sl@0
   226
  void def_combination_aux0(
sl@0
   227
      Def def, F f, Iter, End, Keywords const& keywords, mpl::true_)
sl@0
   228
  {
sl@0
   229
      typedef typename mpl::deref<Iter>::type spec;
sl@0
   230
      typedef typename spec::keyword kw;
sl@0
   231
sl@0
   232
      def_combination_aux(
sl@0
   233
          def, f, typename mpl::next<Iter>::type(), End()
sl@0
   234
        , (
sl@0
   235
              keywords, boost::python::arg(kw::keyword_name()) = empty_tag()
sl@0
   236
          )
sl@0
   237
      );
sl@0
   238
  }
sl@0
   239
sl@0
   240
  inline void initialize_converter()
sl@0
   241
  {
sl@0
   242
      static python_::to_python_converter<empty_tag, empty_tag_to_python> x;
sl@0
   243
  }
sl@0
   244
sl@0
   245
  template <class Def, class F, class Iter, class End, class Keywords>
sl@0
   246
  void def_combination_aux(
sl@0
   247
      Def def, F f, Iter, End, Keywords const& keywords)
sl@0
   248
  {
sl@0
   249
      typedef typename mpl::deref<Iter>::type spec;
sl@0
   250
sl@0
   251
      typedef typename mpl::and_<
sl@0
   252
          typename spec::optimized_default
sl@0
   253
        , mpl::not_<typename spec::required>
sl@0
   254
      >::type optimized_default;
sl@0
   255
      
sl@0
   256
      def_combination_aux0(
sl@0
   257
          def, f, Iter(), End(), keywords, optimized_default()
sl@0
   258
      );
sl@0
   259
  }
sl@0
   260
sl@0
   261
  template <class Def, class F, class End, class Keywords>
sl@0
   262
  void def_combination_aux(
sl@0
   263
      Def def, F f, End, End, Keywords const& keywords)
sl@0
   264
  {
sl@0
   265
      def(f, keywords);
sl@0
   266
  } 
sl@0
   267
sl@0
   268
  template <class Def, class F, class End>
sl@0
   269
  void def_combination_aux(
sl@0
   270
      Def def, F f, End, End, no_keywords const&)
sl@0
   271
  {
sl@0
   272
      def(f);
sl@0
   273
  }
sl@0
   274
sl@0
   275
  template <
sl@0
   276
      class Def, class Specs, class Bits, class Invoker
sl@0
   277
  >
sl@0
   278
  void def_combination(
sl@0
   279
      Def def, Specs*, Bits, Invoker*)
sl@0
   280
  {
sl@0
   281
      typedef typename mpl::fold<
sl@0
   282
          Specs
sl@0
   283
        , mpl::pair<mpl::vector0<>, Bits>
sl@0
   284
        , combinations_op<mpl::_2, mpl::_1>
sl@0
   285
      >::type combination0;
sl@0
   286
sl@0
   287
      typedef typename combination0::first combination;
sl@0
   288
sl@0
   289
      typedef typename mpl::apply_wrap1<
sl@0
   290
          Invoker, combination
sl@0
   291
      >::type invoker;
sl@0
   292
sl@0
   293
      def_combination_aux(
sl@0
   294
          def
sl@0
   295
        , &invoker::execute
sl@0
   296
        , typename mpl::begin<combination>::type()
sl@0
   297
        , typename mpl::end<combination>::type()
sl@0
   298
        , no_keywords()
sl@0
   299
      );
sl@0
   300
  }
sl@0
   301
sl@0
   302
  template <
sl@0
   303
      class Def, class Specs, class Bits, class End, class Invoker
sl@0
   304
  >
sl@0
   305
  void def_combinations(
sl@0
   306
      Def def, Specs*, Bits, End, Invoker*)
sl@0
   307
  {
sl@0
   308
      initialize_converter();
sl@0
   309
sl@0
   310
      def_combination(def, (Specs*)0, Bits(), (Invoker*)0);
sl@0
   311
sl@0
   312
      def_combinations(
sl@0
   313
          def
sl@0
   314
        , (Specs*)0
sl@0
   315
        , mpl::long_<Bits::value + 1>()
sl@0
   316
        , End()
sl@0
   317
        , (Invoker*)0
sl@0
   318
      );
sl@0
   319
  }
sl@0
   320
sl@0
   321
  template <
sl@0
   322
      class Def, class Specs, class End, class Invoker
sl@0
   323
  >
sl@0
   324
  void def_combinations(
sl@0
   325
      Def, Specs*, End, End, Invoker*)
sl@0
   326
  {}
sl@0
   327
sl@0
   328
  struct not_specified {};
sl@0
   329
sl@0
   330
  template <class CallPolicies>
sl@0
   331
  struct call_policies_as_options
sl@0
   332
  {
sl@0
   333
      call_policies_as_options(CallPolicies const& call_policies)
sl@0
   334
        : call_policies(call_policies)
sl@0
   335
      {}
sl@0
   336
sl@0
   337
      CallPolicies const& policies() const
sl@0
   338
      {
sl@0
   339
          return call_policies;
sl@0
   340
      }
sl@0
   341
sl@0
   342
      char const* doc() const
sl@0
   343
      {
sl@0
   344
          return 0;
sl@0
   345
      }
sl@0
   346
sl@0
   347
      CallPolicies call_policies;
sl@0
   348
  };
sl@0
   349
sl@0
   350
  template <class Class, class Options = not_specified>
sl@0
   351
  struct def_class
sl@0
   352
  {
sl@0
   353
      def_class(Class& cl, char const* name, Options options = Options())
sl@0
   354
        : cl(cl)
sl@0
   355
        , name(name)
sl@0
   356
        , options(options)
sl@0
   357
      {}
sl@0
   358
sl@0
   359
      template <class F>
sl@0
   360
      void def(F f, not_specified const*) const
sl@0
   361
      {
sl@0
   362
          cl.def(name, f);
sl@0
   363
      }
sl@0
   364
sl@0
   365
      template <class F>
sl@0
   366
      void def(F f, void const*) const
sl@0
   367
      {
sl@0
   368
          cl.def(name, f, options.doc(), options.policies());
sl@0
   369
      }
sl@0
   370
      
sl@0
   371
      template <class F>
sl@0
   372
      void operator()(F f) const
sl@0
   373
      {
sl@0
   374
          this->def(f, &options);
sl@0
   375
      }
sl@0
   376
sl@0
   377
      template <class F, class Keywords>
sl@0
   378
      void def(F f, Keywords const& keywords, not_specified const*) const
sl@0
   379
      {
sl@0
   380
          cl.def(name, f, keywords);
sl@0
   381
      }
sl@0
   382
sl@0
   383
      template <class F, class Keywords>
sl@0
   384
      void def(F f, Keywords const& keywords, void const*) const
sl@0
   385
      {
sl@0
   386
          cl.def(name, f, keywords, options.doc(), options.policies());
sl@0
   387
      }
sl@0
   388
sl@0
   389
      template <class F, class Keywords>
sl@0
   390
      void operator()(F f, Keywords const& keywords) const
sl@0
   391
      {
sl@0
   392
          this->def(f, keywords, &options);
sl@0
   393
      }
sl@0
   394
sl@0
   395
      Class& cl;
sl@0
   396
      char const* name;
sl@0
   397
      Options options;
sl@0
   398
  };
sl@0
   399
sl@0
   400
  template <class Class, class CallPolicies = boost::python::default_call_policies>
sl@0
   401
  struct def_init
sl@0
   402
  {
sl@0
   403
      def_init(Class& cl, CallPolicies call_policies = CallPolicies())
sl@0
   404
        : cl(cl)
sl@0
   405
        , call_policies(call_policies)
sl@0
   406
      {}
sl@0
   407
sl@0
   408
      template <class F>
sl@0
   409
      void operator()(F f) const
sl@0
   410
      {
sl@0
   411
          cl.def(
sl@0
   412
              "__init__"
sl@0
   413
            , boost::python::make_constructor(f, call_policies)
sl@0
   414
          );
sl@0
   415
      }
sl@0
   416
sl@0
   417
      template <class F, class Keywords>
sl@0
   418
      void operator()(F f, Keywords const& keywords) const
sl@0
   419
      {
sl@0
   420
          cl.def(
sl@0
   421
              "__init__"
sl@0
   422
            , boost::python::make_constructor(f, call_policies, keywords)
sl@0
   423
          );
sl@0
   424
      }
sl@0
   425
sl@0
   426
      Class& cl;
sl@0
   427
      CallPolicies call_policies;
sl@0
   428
  };
sl@0
   429
sl@0
   430
  struct def_function
sl@0
   431
  {
sl@0
   432
      def_function(char const* name)
sl@0
   433
        : name(name)
sl@0
   434
      {}
sl@0
   435
      
sl@0
   436
      template <class F>
sl@0
   437
      void operator()(F f) const
sl@0
   438
      {
sl@0
   439
          boost::python::def(name, f);
sl@0
   440
      }
sl@0
   441
sl@0
   442
      template <class F, class Keywords>
sl@0
   443
      void operator()(F f, Keywords const& keywords) const
sl@0
   444
      {
sl@0
   445
          boost::python::def(name, f, keywords);
sl@0
   446
      }
sl@0
   447
sl@0
   448
      char const* name;
sl@0
   449
  };
sl@0
   450
sl@0
   451
} // namespace aux
sl@0
   452
sl@0
   453
template <class M, class Signature>
sl@0
   454
void def(char const* name, Signature)
sl@0
   455
{
sl@0
   456
    typedef mpl::iterator_range<
sl@0
   457
        typename mpl::next<
sl@0
   458
            typename mpl::begin<Signature>::type
sl@0
   459
        >::type
sl@0
   460
      , typename mpl::end<Signature>::type
sl@0
   461
    > arg_types;
sl@0
   462
sl@0
   463
    typedef typename mpl::transform<
sl@0
   464
        typename M::keywords
sl@0
   465
      , arg_types
sl@0
   466
      , aux::make_arg_spec<mpl::_1, mpl::_2>
sl@0
   467
      , mpl::back_inserter<mpl::vector0<> >
sl@0
   468
    >::type arg_specs;
sl@0
   469
sl@0
   470
    typedef typename mpl::count_if<
sl@0
   471
        arg_specs
sl@0
   472
      , aux::is_optional<mpl::_1>
sl@0
   473
    >::type optional_arity;
sl@0
   474
    
sl@0
   475
    typedef typename mpl::front<Signature>::type result_type;
sl@0
   476
    typedef typename mpl::shift_left<mpl::long_<1>, optional_arity>::type upper;
sl@0
   477
sl@0
   478
    aux::def_combinations(
sl@0
   479
        aux::def_function(name)
sl@0
   480
      , (arg_specs*)0
sl@0
   481
      , mpl::long_<0>()
sl@0
   482
      , mpl::long_<upper::value>()
sl@0
   483
      , (aux::make_invoker<M, result_type>*)0
sl@0
   484
    );
sl@0
   485
}
sl@0
   486
sl@0
   487
template <class M, class Class, class Signature>
sl@0
   488
void def(Class& cl, char const* name, Signature)
sl@0
   489
{
sl@0
   490
    typedef mpl::iterator_range<
sl@0
   491
        typename mpl::next<
sl@0
   492
            typename mpl::begin<Signature>::type
sl@0
   493
        >::type
sl@0
   494
      , typename mpl::end<Signature>::type
sl@0
   495
    > arg_types;
sl@0
   496
sl@0
   497
    typedef typename mpl::transform<
sl@0
   498
        typename M::keywords
sl@0
   499
      , arg_types
sl@0
   500
      , aux::make_arg_spec<mpl::_1, mpl::_2>
sl@0
   501
      , mpl::back_inserter<mpl::vector0<> >
sl@0
   502
    >::type arg_specs;
sl@0
   503
sl@0
   504
    typedef typename mpl::count_if<
sl@0
   505
        arg_specs
sl@0
   506
      , aux::is_optional<mpl::_1>
sl@0
   507
    >::type optional_arity;
sl@0
   508
    
sl@0
   509
    typedef typename mpl::front<Signature>::type result_type;
sl@0
   510
    typedef typename mpl::shift_left<mpl::long_<1>, optional_arity>::type upper;
sl@0
   511
sl@0
   512
    aux::def_combinations(
sl@0
   513
        aux::def_class<Class>(cl, name)
sl@0
   514
      , (arg_specs*)0
sl@0
   515
      , mpl::long_<0>()
sl@0
   516
      , mpl::long_<upper::value>()
sl@0
   517
      , (aux::make_invoker<M, result_type>*)0
sl@0
   518
    );
sl@0
   519
}
sl@0
   520
sl@0
   521
namespace aux
sl@0
   522
{
sl@0
   523
sl@0
   524
  template <class K>
sl@0
   525
  struct keyword
sl@0
   526
  {
sl@0
   527
      typedef K type;
sl@0
   528
  };
sl@0
   529
sl@0
   530
  template <class K>
sl@0
   531
  struct keyword<K*>
sl@0
   532
  {
sl@0
   533
      typedef K type;
sl@0
   534
  };
sl@0
   535
sl@0
   536
  template <class K>
sl@0
   537
  struct keyword<K**>
sl@0
   538
  {
sl@0
   539
      typedef K type;
sl@0
   540
  };
sl@0
   541
sl@0
   542
  template <class K>
sl@0
   543
  struct required
sl@0
   544
  {
sl@0
   545
      typedef mpl::true_ type;
sl@0
   546
  };
sl@0
   547
sl@0
   548
  template <class K>
sl@0
   549
  struct required<K*>
sl@0
   550
  {
sl@0
   551
      typedef mpl::false_ type;
sl@0
   552
  };
sl@0
   553
sl@0
   554
  template <class K>
sl@0
   555
  struct optimized
sl@0
   556
  {
sl@0
   557
      typedef mpl::true_ type;
sl@0
   558
  };
sl@0
   559
sl@0
   560
  template <class K>
sl@0
   561
  struct optimized<K**>
sl@0
   562
  {
sl@0
   563
      typedef mpl::false_ type;
sl@0
   564
  };
sl@0
   565
sl@0
   566
  template <class T>
sl@0
   567
  struct make_kw_spec;
sl@0
   568
sl@0
   569
  template <class K, class T>
sl@0
   570
  struct make_kw_spec<K(T)>
sl@0
   571
  {
sl@0
   572
      typedef arg_spec<
sl@0
   573
          typename keyword<K>::type
sl@0
   574
        , typename required<K>::type
sl@0
   575
        , typename optimized<K>::type
sl@0
   576
        , T
sl@0
   577
      > type;
sl@0
   578
  };
sl@0
   579
sl@0
   580
} // namespace aux
sl@0
   581
sl@0
   582
template <class ParameterSpecs, class CallPolicies = boost::python::default_call_policies>
sl@0
   583
struct init 
sl@0
   584
  : boost::python::def_visitor<init<ParameterSpecs, CallPolicies> >
sl@0
   585
{
sl@0
   586
    init(CallPolicies call_policies = CallPolicies())
sl@0
   587
      : call_policies(call_policies)
sl@0
   588
    {}
sl@0
   589
sl@0
   590
    template <class CallPolicies1>
sl@0
   591
    init<ParameterSpecs, CallPolicies1> 
sl@0
   592
    operator[](CallPolicies1 const& call_policies) const
sl@0
   593
    {
sl@0
   594
        return init<ParameterSpecs, CallPolicies1>(call_policies);
sl@0
   595
    }
sl@0
   596
sl@0
   597
    template <class Class>
sl@0
   598
    void visit_aux(Class& cl, mpl::true_) const
sl@0
   599
    {
sl@0
   600
        cl.def(boost::python::init<>()[call_policies]);
sl@0
   601
    }
sl@0
   602
sl@0
   603
    template <class Class>
sl@0
   604
    void visit_aux(Class& cl, mpl::false_) const
sl@0
   605
    {
sl@0
   606
        typedef typename mpl::transform<
sl@0
   607
            ParameterSpecs
sl@0
   608
          , aux::make_kw_spec<mpl::_>
sl@0
   609
          , mpl::back_inserter<mpl::vector0<> >
sl@0
   610
        >::type arg_specs;
sl@0
   611
sl@0
   612
        typedef typename mpl::count_if<
sl@0
   613
            arg_specs
sl@0
   614
          , aux::is_optional<mpl::_>
sl@0
   615
        >::type optional_arity;
sl@0
   616
sl@0
   617
        typedef typename mpl::shift_left<mpl::long_<1>, optional_arity>::type upper;
sl@0
   618
sl@0
   619
        aux::def_combinations(
sl@0
   620
            aux::def_init<Class, CallPolicies>(cl, call_policies)
sl@0
   621
          , (arg_specs*)0
sl@0
   622
          , mpl::long_<0>()
sl@0
   623
          , mpl::long_<upper::value>()
sl@0
   624
          , (aux::make_init_invoker<typename Class::wrapped_type>*)0
sl@0
   625
        );
sl@0
   626
    }
sl@0
   627
sl@0
   628
    template <class Class>
sl@0
   629
    void visit(Class& cl) const
sl@0
   630
    {
sl@0
   631
        visit_aux(cl, mpl::empty<ParameterSpecs>());
sl@0
   632
    }
sl@0
   633
sl@0
   634
    CallPolicies call_policies;
sl@0
   635
};
sl@0
   636
sl@0
   637
template <class ParameterSpecs, class CallPolicies = boost::python::default_call_policies>
sl@0
   638
struct call 
sl@0
   639
  : boost::python::def_visitor<call<ParameterSpecs, CallPolicies> >
sl@0
   640
{
sl@0
   641
    call(CallPolicies const& call_policies = CallPolicies())
sl@0
   642
      : call_policies(call_policies)
sl@0
   643
    {}
sl@0
   644
sl@0
   645
    template <class CallPolicies1>
sl@0
   646
    call<ParameterSpecs, CallPolicies1>
sl@0
   647
    operator[](CallPolicies1 const& call_policies) const
sl@0
   648
    {
sl@0
   649
        return call<ParameterSpecs, CallPolicies1>(call_policies);
sl@0
   650
    }
sl@0
   651
sl@0
   652
    template <class Class>
sl@0
   653
    void visit(Class& cl) const
sl@0
   654
    {
sl@0
   655
        typedef mpl::iterator_range<
sl@0
   656
            typename mpl::next<
sl@0
   657
                typename mpl::begin<ParameterSpecs>::type
sl@0
   658
            >::type
sl@0
   659
          , typename mpl::end<ParameterSpecs>::type
sl@0
   660
        > arg_types;
sl@0
   661
sl@0
   662
        typedef typename mpl::front<ParameterSpecs>::type result_type;
sl@0
   663
sl@0
   664
        typedef typename mpl::transform<
sl@0
   665
            arg_types
sl@0
   666
          , aux::make_kw_spec<mpl::_>
sl@0
   667
          , mpl::back_inserter<mpl::vector0<> >
sl@0
   668
        >::type arg_specs;
sl@0
   669
sl@0
   670
        typedef typename mpl::count_if<
sl@0
   671
            arg_specs
sl@0
   672
          , aux::is_optional<mpl::_>
sl@0
   673
        >::type optional_arity;
sl@0
   674
sl@0
   675
        typedef typename mpl::shift_left<mpl::long_<1>, optional_arity>::type upper;
sl@0
   676
sl@0
   677
        typedef aux::call_policies_as_options<CallPolicies> options;
sl@0
   678
sl@0
   679
        aux::def_combinations(
sl@0
   680
            aux::def_class<Class, options>(cl, "__call__", options(call_policies))
sl@0
   681
          , (arg_specs*)0
sl@0
   682
          , mpl::long_<0>()
sl@0
   683
          , mpl::long_<upper::value>()
sl@0
   684
          , (aux::make_call_invoker<typename Class::wrapped_type, result_type>*)0
sl@0
   685
        );
sl@0
   686
    }
sl@0
   687
sl@0
   688
    CallPolicies call_policies;
sl@0
   689
};
sl@0
   690
sl@0
   691
template <class Fwd, class ParameterSpecs>
sl@0
   692
struct function 
sl@0
   693
  : boost::python::def_visitor<function<Fwd, ParameterSpecs> >
sl@0
   694
{
sl@0
   695
    template <class Class, class Options>
sl@0
   696
    void visit(Class& cl, char const* name, Options const& options) const
sl@0
   697
    {
sl@0
   698
        typedef mpl::iterator_range<
sl@0
   699
            typename mpl::next<
sl@0
   700
                typename mpl::begin<ParameterSpecs>::type
sl@0
   701
            >::type
sl@0
   702
          , typename mpl::end<ParameterSpecs>::type
sl@0
   703
        > arg_types;
sl@0
   704
sl@0
   705
        typedef typename mpl::front<ParameterSpecs>::type result_type;
sl@0
   706
sl@0
   707
        typedef typename mpl::transform<
sl@0
   708
            arg_types
sl@0
   709
          , aux::make_kw_spec<mpl::_>
sl@0
   710
          , mpl::back_inserter<mpl::vector0<> >
sl@0
   711
        >::type arg_specs;
sl@0
   712
sl@0
   713
        typedef typename mpl::count_if<
sl@0
   714
            arg_specs
sl@0
   715
          , aux::is_optional<mpl::_>
sl@0
   716
        >::type optional_arity;
sl@0
   717
sl@0
   718
        typedef typename mpl::shift_left<mpl::long_<1>, optional_arity>::type upper;
sl@0
   719
sl@0
   720
        aux::def_combinations(
sl@0
   721
            aux::def_class<Class, Options>(cl, name, options)
sl@0
   722
          , (arg_specs*)0
sl@0
   723
          , mpl::long_<0>()
sl@0
   724
          , mpl::long_<upper::value>()
sl@0
   725
          , (aux::make_member_invoker<
sl@0
   726
                Fwd, result_type, typename Class::wrapped_type
sl@0
   727
            >*)0
sl@0
   728
        );
sl@0
   729
    }
sl@0
   730
};
sl@0
   731
sl@0
   732
}}} // namespace boost::parameter::python
sl@0
   733
sl@0
   734
#endif // BOOST_PARAMETER_PYTHON_060209_HPP
sl@0
   735