From 0afe3c9cd8d67d21ac1115576f88cb86d9ef2134 Mon Sep 17 00:00:00 2001
From: ManifoldFR <wilson.jallet@polytechnique.org>
Date: Thu, 6 Apr 2023 19:00:01 +0200
Subject: [PATCH] optional: add support for std::optional under cpp17

---
 include/eigenpy/optional.hpp | 33 +++++++++++++++++++++++++++++++--
 1 file changed, 31 insertions(+), 2 deletions(-)

diff --git a/include/eigenpy/optional.hpp b/include/eigenpy/optional.hpp
index 38f167db..949b8419 100644
--- a/include/eigenpy/optional.hpp
+++ b/include/eigenpy/optional.hpp
@@ -8,17 +8,28 @@
 #include "eigenpy/fwd.hpp"
 #include "eigenpy/eigen-from-python.hpp"
 #include <boost/optional.hpp>
+#ifdef EIGENPY_WITH_CXX17_SUPPORT
+#include <optional>
+#endif
 
+#ifndef EIGENPY_DEFAULT_OPTIONAL
 #define EIGENPY_DEFAULT_OPTIONAL boost::optional
+#endif
 
 namespace boost {
 namespace python {
 namespace converter {
 
 template <typename T>
-struct expected_pytype_for_arg<EIGENPY_DEFAULT_OPTIONAL<T> >
+struct expected_pytype_for_arg<boost::optional<T> >
     : expected_pytype_for_arg<T> {};
 
+#ifdef EIGENPY_WITH_CXX17_SUPPORT
+template <typename T>
+struct expected_pytype_for_arg<std::optional<T> >
+    : expected_pytype_for_arg<T> {};
+#endif
+
 }  // namespace converter
 }  // namespace python
 }  // namespace boost
@@ -26,6 +37,24 @@ struct expected_pytype_for_arg<EIGENPY_DEFAULT_OPTIONAL<T> >
 namespace eigenpy {
 namespace detail {
 
+/// Helper struct to decide which type is the "none" type for a specific optional<T> implementation.
+template<template <typename> class OptionalTpl>
+struct nullopt_helper {};
+
+template<>
+struct nullopt_helper<boost::optional> {
+  typedef boost::none_t type;
+  static type value() { return boost::none; }
+};
+
+#ifdef EIGENPY_WITH_CXX17_SUPPORT
+template<>
+struct nullopt_helper<std::optional> {
+  typedef std::nullopt_t type;
+  static type value() { return std::nullopt; }
+};
+#endif
+
 template <typename T,
           template <typename> class OptionalTpl = EIGENPY_DEFAULT_OPTIONAL>
 struct OptionalToPython {
@@ -80,7 +109,7 @@ void OptionalFromPython<T, OptionalTpl>::construct(
           ->storage.bytes;
 
   if (obj_ptr == Py_None) {
-    new (storage) OptionalTpl<T>(boost::none);
+    new (storage) OptionalTpl<T>(nullopt_helper<OptionalTpl>::value());
   } else {
     const T value = bp::extract<T>(obj_ptr);
     new (storage) OptionalTpl<T>(value);
-- 
GitLab