1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77
|
// SPDX-FileCopyrightText: Copyright © DUNE Project contributors, see file LICENSE.md in module root
// SPDX-License-Identifier: LicenseRef-GPL-2.0-only-with-DUNE-exception
#ifndef DUNE_PYTHON_COMMON_TVECTOR_HH
#define DUNE_PYTHON_COMMON_TVECTOR_HH
/**
* @file tuplevector.hh
* @brief Python bindings for TupleVector.
*/
#include <dune/python/pybind11/pybind11.h>
#include <dune/python/pybind11/cast.h>
#include <dune/common/tuplevector.hh>
#include <dune/common/hybridutilities.hh>
namespace Dune {
namespace Python {
/**
* @brief Register TupleVector bindings.
*
* This function registers Python bindings for TupleVector.
*
* @tparam TV The type of the TupleVector.
* @tparam options Additional options for pybind11 class.
* @param scope The scope to register the bindings.
* @param cls The pybind11 class to register the bindings.
*/
template <class TV, class... options>
void registerTupleVector(pybind11::handle scope, pybind11::class_<TV, options...> cls) {
namespace py = pybind11;
using py::operator"" _a;
cls.def(py::init([](py::tuple x) {
assert(std::tuple_size_v<TV> == x.size());
return Dune::unpackIntegerSequence([&](auto... i) {
return new TV((x[i].template cast<std::tuple_element_t<i, TV> >())...); },
std::make_index_sequence<std::tuple_size_v<TV> >{});
}));
cls.def("assign", [](TV &self, const TV &x) { self = x; }, "x"_a);
cls.def("copy", [](const TV &self) { return new TV(self); });
cls.def("__getitem__", [](const TV &self, size_t index) {
if (index >= self.size())
throw py::index_error();
return Dune::Hybrid::switchCases(Dune::Hybrid::integralRange(Dune::index_constant<std::tuple_size_v<TV> >()),index, [&](auto i) {
return py::cast(self[i],py::return_value_policy::reference);
},[]() {return py::object{};});
},py::return_value_policy::reference_internal);
cls.def("__setitem__", [&](TV &self, size_t index, const py::object &value) {
if (index >= self.size())
throw py::index_error();
Dune::Hybrid::switchCases(Dune::Hybrid::integralRange(Dune::index_constant<std::tuple_size_v<TV> >()),index, [&](auto i) {
try
{
self[i] = value.cast<std::tuple_element_t<i, TV>>();
}
catch(const py::cast_error & e)
{
std::cerr << "Your provide value is not of the correct type, which should be " << Dune::className<std::tuple_element_t<i, TV>>()<<" for your provided index "<< index<< std::endl;
throw e;
}
});
});
cls.def("__len__", [](const TV &self) { return self.size(); });
}
} // namespace Python
} // namespace Dune
#endif // #ifndef DUNE_PYTHON_COMMON_TVECTOR_HH
|