24#include "Encoding.hpp"
25#include "StringTools.hpp"
39constexpr enabler dummy = {};
47template <
bool B,
class T =
void>
using enable_if_t =
typename std::enable_if<B, T>::type;
55template <
typename... Ts>
using void_t =
typename make_void<Ts...>::type;
58template <
bool B,
class T,
class F>
using conditional_t =
typename std::conditional<B, T, F>::type;
61template <
typename T>
struct is_bool : std::false_type {};
64template <>
struct is_bool<bool> : std::true_type {};
70template <
typename T>
struct is_shared_ptr<std::shared_ptr<T>> : std::true_type {};
73template <
typename T>
struct is_shared_ptr<const std::shared_ptr<T>> : std::true_type {};
87 using type = std::string;
97 template <
typename TT,
typename SS>
98 static auto test(
int) ->
decltype(lexical_cast(std::declval<const SS &>(), std::declval<TT &>()), std::true_type());
100 template <
typename,
typename>
static auto test(...) -> std::false_type;
103 static constexpr bool value =
decltype(test<T, S>(0))::value;
119template <
typename T>
struct element_type<T, typename std::enable_if<is_copyable_ptr<T>::value>::type> {
120 using type =
typename std::pointer_traits<T>::element_type;
130template <
typename T,
typename _ =
void>
struct pair_adaptor : std::false_type {
131 using value_type =
typename T::value_type;
132 using first_type =
typename std::remove_const<value_type>::type;
133 using second_type =
typename std::remove_const<value_type>::type;
136 template <
typename Q>
static auto first(Q &&pair_value) ->
decltype(std::forward<Q>(pair_value)) {
137 return std::forward<Q>(pair_value);
140 template <
typename Q>
static auto second(Q &&pair_value) ->
decltype(std::forward<Q>(pair_value)) {
141 return std::forward<Q>(pair_value);
150 conditional_t<false, void_t<typename T::value_type::first_type, typename T::value_type::second_type>, void>>
152 using value_type =
typename T::value_type;
153 using first_type =
typename std::remove_const<typename value_type::first_type>::type;
154 using second_type =
typename std::remove_const<typename value_type::second_type>::type;
157 template <
typename Q>
static auto first(Q &&pair_value) ->
decltype(std::get<0>(std::forward<Q>(pair_value))) {
158 return std::get<0>(std::forward<Q>(pair_value));
161 template <
typename Q>
static auto second(Q &&pair_value) ->
decltype(std::get<1>(std::forward<Q>(pair_value))) {
162 return std::get<1>(std::forward<Q>(pair_value));
173#pragma GCC diagnostic push
174#pragma GCC diagnostic ignored "-Wnarrowing"
178 template <
typename TT,
typename CC>
179 static auto test(
int, std::true_type) ->
decltype(
182#ifdef __NVCC_DIAG_PRAGMA_SUPPORT__
183#pragma nv_diag_suppress 2361
185#pragma diag_suppress 2361
188 TT{std::declval<CC>()}
190#ifdef __NVCC_DIAG_PRAGMA_SUPPORT__
191#pragma nv_diag_default 2361
193#pragma diag_default 2361
197 std::is_move_assignable<TT>());
199 template <
typename TT,
typename CC>
static auto test(
int, std::false_type) -> std::false_type;
201 template <
typename,
typename>
static auto test(...) -> std::false_type;
204 static constexpr bool value =
decltype(test<T, C>(0,
typename std::is_constructible<T, C>::type()))::value;
207#pragma GCC diagnostic pop
214 template <
typename TT,
typename SS>
215 static auto test(
int) ->
decltype(std::declval<SS &>() << std::declval<TT>(), std::true_type());
217 template <
typename,
typename>
static auto test(...) -> std::false_type;
220 static constexpr bool value =
decltype(test<T, S>(0))::value;
225 template <
typename TT,
typename SS>
226 static auto test(
int) ->
decltype(std::declval<SS &>() >> std::declval<TT &>(), std::true_type());
228 template <
typename,
typename>
static auto test(...) -> std::false_type;
231 static constexpr bool value =
decltype(test<T, S>(0))::value;
236 template <
typename TT>
237 static auto test(
int) ->
decltype(std::declval<TT>().real(), std::declval<TT>().imag(), std::true_type());
239 template <
typename>
static auto test(...) -> std::false_type;
242 static constexpr bool value =
decltype(test<T>(0))::value;
246template <typename T, enable_if_t<is_istreamable<T>::value, detail::enabler> = detail::dummy>
247bool from_stream(
const std::string &istring, T &obj) {
248 std::istringstream is;
251 return !is.fail() && !is.rdbuf()->in_avail();
254template <typename T, enable_if_t<!is_istreamable<T>::value, detail::enabler> = detail::dummy>
255bool from_stream(
const std::string & , T & ) {
269 void_t<typename T::value_type,
270 decltype(std::declval<T>().end()),
271 decltype(std::declval<T>().clear()),
272 decltype(std::declval<T>().insert(std::declval<decltype(std::declval<T>().end())>(),
273 std::declval<const typename T::value_type &>()))>,
274 void>> :
public conditional_t<std::is_constructible<T, std::string>::value ||
275 std::is_constructible<T, std::wstring>::value,
288 conditional_t<false, void_t<decltype(std::declval<T>().end()), decltype(std::declval<T>().begin())>, void>>
289 :
public std::true_type {};
292template <
typename T,
typename _ =
void>
struct is_wrapper : std::false_type {};
296struct is_wrapper<T, conditional_t<false, void_t<typename T::value_type>, void>> :
public std::true_type {};
300 template <
typename SS>
303 static auto test(
int) ->
decltype(std::tuple_size<typename std::decay<SS>::type>::value, std::true_type{});
304 template <
typename>
static auto test(...) -> std::false_type;
307 static constexpr bool value =
decltype(test<S>(0))::value;
311template <typename T, enable_if_t<std::is_convertible<T, std::string>::value, detail::enabler> = detail::dummy>
312auto to_string(T &&value) ->
decltype(std::forward<T>(value)) {
313 return std::forward<T>(value);
318 enable_if_t<std::is_constructible<std::string, T>::value && !std::is_convertible<T, std::string>::value,
319 detail::enabler> = detail::dummy>
320std::string to_string(
const T &value) {
321 return std::string(value);
326 enable_if_t<!std::is_convertible<std::string, T>::value && !std::is_constructible<std::string, T>::value &&
327 is_ostreamable<T>::value,
328 detail::enabler> = detail::dummy>
329std::string to_string(T &&value) {
330 std::stringstream stream;
337 enable_if_t<!std::is_constructible<std::string, T>::value && !is_ostreamable<T>::value &&
338 !is_readable_container<typename std::remove_const<T>::type>::value,
339 detail::enabler> = detail::dummy>
340std::string to_string(T &&) {
346 enable_if_t<!std::is_constructible<std::string, T>::value && !is_ostreamable<T>::value &&
347 is_readable_container<T>::value,
348 detail::enabler> = detail::dummy>
349std::string to_string(T &&variable) {
350 auto cval = variable.begin();
351 auto end = variable.end();
355 std::vector<std::string> defaults;
357 defaults.emplace_back(CLI::detail::to_string(*cval));
360 return {
"[" + detail::join(defaults) +
"]"};
364template <
typename T1,
367 enable_if_t<std::is_same<T1, T2>::value, detail::enabler> = detail::dummy>
368auto checked_to_string(T &&value) ->
decltype(to_string(std::forward<T>(value))) {
369 return to_string(std::forward<T>(value));
373template <
typename T1,
376 enable_if_t<!std::is_same<T1, T2>::value, detail::enabler> = detail::dummy>
377std::string checked_to_string(T &&) {
378 return std::string{};
381template <typename T, enable_if_t<std::is_arithmetic<T>::value, detail::enabler> = detail::dummy>
382std::string value_string(
const T &value) {
383 return std::to_string(value);
386template <typename T, enable_if_t<std::is_enum<T>::value, detail::enabler> = detail::dummy>
387std::string value_string(
const T &value) {
388 return std::to_string(
static_cast<typename std::underlying_type<T>::type
>(value));
392 enable_if_t<!std::is_enum<T>::value && !std::is_arithmetic<T>::value, detail::enabler> = detail::dummy>
393auto value_string(
const T &value) ->
decltype(to_string(value)) {
394 return to_string(value);
398template <
typename T,
typename def,
typename Enable =
void>
struct wrapped_type {
403template <
typename T,
typename def>
struct wrapped_type<T, def, typename std::enable_if<is_wrapper<T>::value>::type> {
404 using type =
typename T::value_type;
409 static const int value{0};
415 typename std::enable_if<!is_tuple_like<T>::value && !is_mutable_container<T>::value &&
416 !std::is_void<T>::value>::type> {
417 static constexpr int value{1};
422struct type_count_base<T, typename std::enable_if<is_tuple_like<T>::value && !is_mutable_container<T>::value>::type> {
423 static constexpr int value{std::tuple_size<T>::value};
427template <
typename T>
struct type_count_base<T, typename std::enable_if<is_mutable_container<T>::value>::type> {
440template <
typename T,
typename Enable =
void>
struct type_count {
441 static const int value{0};
447 typename std::enable_if<!is_wrapper<T>::value && !is_tuple_like<T>::value && !is_complex<T>::value &&
448 !std::is_void<T>::value>::type> {
449 static constexpr int value{1};
453template <
typename T>
struct type_count<T, typename std::enable_if<is_complex<T>::value>::type> {
454 static constexpr int value{2};
458template <
typename T>
struct type_count<T, typename std::enable_if<is_mutable_container<T>::value>::type> {
465 typename std::enable_if<is_wrapper<T>::value && !is_complex<T>::value && !is_tuple_like<T>::value &&
466 !is_mutable_container<T>::value>::type> {
471template <
typename T, std::
size_t I>
472constexpr typename std::enable_if<I == type_count_base<T>::value,
int>::type tuple_type_size() {
477template <
typename T, std::
size_t I>
478 constexpr typename std::enable_if < I<type_count_base<T>::value,
int>::type tuple_type_size() {
479 return subtype_count<typename std::tuple_element<I, T>::type>::value + tuple_type_size<T, I + 1>();
483template <
typename T>
struct type_count<T, typename std::enable_if<is_tuple_like<T>::value>::type> {
484 static constexpr int value{tuple_type_size<T, 0>()};
488template <
typename T>
struct subtype_count {
489 static constexpr int value{is_mutable_container<T>::value ? expected_max_vector_size : type_count<T>::value};
493template <
typename T,
typename Enable =
void>
struct type_count_min {
494 static const int value{0};
499struct type_count_min<
501 typename std::enable_if<!is_mutable_container<T>::value && !is_tuple_like<T>::value && !is_wrapper<T>::value &&
502 !is_complex<T>::value && !std::is_void<T>::value>::type> {
503 static constexpr int value{type_count<T>::value};
507template <
typename T>
struct type_count_min<T, typename std::enable_if<is_complex<T>::value>::type> {
508 static constexpr int value{1};
513struct type_count_min<
515 typename std::enable_if<is_wrapper<T>::value && !is_complex<T>::value && !is_tuple_like<T>::value>::type> {
516 static constexpr int value{subtype_count_min<typename T::value_type>::value};
520template <
typename T, std::
size_t I>
521constexpr typename std::enable_if<I == type_count_base<T>::value,
int>::type tuple_type_size_min() {
526template <
typename T, std::
size_t I>
527 constexpr typename std::enable_if < I<type_count_base<T>::value,
int>::type tuple_type_size_min() {
528 return subtype_count_min<typename std::tuple_element<I, T>::type>::value + tuple_type_size_min<T, I + 1>();
532template <
typename T>
struct type_count_min<T, typename std::enable_if<is_tuple_like<T>::value>::type> {
533 static constexpr int value{tuple_type_size_min<T, 0>()};
537template <
typename T>
struct subtype_count_min {
538 static constexpr int value{is_mutable_container<T>::value
539 ? ((type_count<T>::value < expected_max_vector_size) ? type_count<T>::value : 0)
540 : type_count_min<T>::value};
544template <
typename T,
typename Enable =
void>
struct expected_count {
545 static const int value{0};
550struct expected_count<T,
551 typename std::enable_if<!is_mutable_container<T>::value && !is_wrapper<T>::value &&
552 !std::is_void<T>::value>::type> {
553 static constexpr int value{1};
556template <
typename T>
struct expected_count<T, typename std::enable_if<is_mutable_container<T>::value>::type> {
557 static constexpr int value{expected_max_vector_size};
562struct expected_count<T, typename std::enable_if<!is_mutable_container<T>::value && is_wrapper<T>::value>::type> {
563 static constexpr int value{expected_count<typename T::value_type>::value};
567enum class object_category :
int {
570 unsigned_integral = 4,
574 number_constructible = 12,
575 double_constructible = 14,
576 integer_constructible = 16,
578 string_assignable = 23,
579 string_constructible = 24,
580 wstring_assignable = 25,
581 wstring_constructible = 26,
587 container_value = 80,
594template <
typename T,
typename Enable =
void>
struct classify_object {
595 static constexpr object_category value{object_category::other};
600struct classify_object<
602 typename std::enable_if<std::is_integral<T>::value && !std::is_same<T, char>::value && std::is_signed<T>::value &&
603 !is_bool<T>::value && !std::is_enum<T>::value>::type> {
604 static constexpr object_category value{object_category::integral_value};
609struct classify_object<T,
610 typename std::enable_if<std::is_integral<T>::value && std::is_unsigned<T>::value &&
611 !std::is_same<T, char>::value && !is_bool<T>::value>::type> {
612 static constexpr object_category value{object_category::unsigned_integral};
617struct classify_object<T, typename std::enable_if<std::is_same<T, char>::value && !std::is_enum<T>::value>::type> {
618 static constexpr object_category value{object_category::char_value};
622template <
typename T>
struct classify_object<T, typename std::enable_if<is_bool<T>::value>::type> {
623 static constexpr object_category value{object_category::boolean_value};
627template <
typename T>
struct classify_object<T, typename std::enable_if<std::is_floating_point<T>::value>::type> {
628 static constexpr object_category value{object_category::floating_point};
633#define WIDE_STRING_CHECK \
634 !std::is_assignable<T &, std::wstring>::value && !std::is_constructible<T, std::wstring>::value
635#define STRING_CHECK true
637#define WIDE_STRING_CHECK true
638#define STRING_CHECK !std::is_assignable<T &, std::string>::value && !std::is_constructible<T, std::string>::value
643struct classify_object<
645 typename std::enable_if<!std::is_floating_point<T>::value && !std::is_integral<T>::value && WIDE_STRING_CHECK &&
646 std::is_assignable<T &, std::string>::value>::type> {
647 static constexpr object_category value{object_category::string_assignable};
652struct classify_object<
654 typename std::enable_if<!std::is_floating_point<T>::value && !std::is_integral<T>::value &&
655 !std::is_assignable<T &, std::string>::value && (type_count<T>::value == 1) &&
656 WIDE_STRING_CHECK && std::is_constructible<T, std::string>::value>::type> {
657 static constexpr object_category value{object_category::string_constructible};
662struct classify_object<T,
663 typename std::enable_if<!std::is_floating_point<T>::value && !std::is_integral<T>::value &&
664 STRING_CHECK && std::is_assignable<T &, std::wstring>::value>::type> {
665 static constexpr object_category value{object_category::wstring_assignable};
669struct classify_object<
671 typename std::enable_if<!std::is_floating_point<T>::value && !std::is_integral<T>::value &&
672 !std::is_assignable<T &, std::wstring>::value && (type_count<T>::value == 1) &&
673 STRING_CHECK && std::is_constructible<T, std::wstring>::value>::type> {
674 static constexpr object_category value{object_category::wstring_constructible};
678template <
typename T>
struct classify_object<T, typename std::enable_if<std::is_enum<T>::value>::type> {
679 static constexpr object_category value{object_category::enumeration};
682template <
typename T>
struct classify_object<T, typename std::enable_if<is_complex<T>::value>::type> {
683 static constexpr object_category value{object_category::complex_number};
688template <
typename T>
struct uncommon_type {
689 using type =
typename std::conditional<
690 !std::is_floating_point<T>::value && !std::is_integral<T>::value &&
691 !std::is_assignable<T &, std::string>::value && !std::is_constructible<T, std::string>::value &&
692 !std::is_assignable<T &, std::wstring>::value && !std::is_constructible<T, std::wstring>::value &&
693 !is_complex<T>::value && !is_mutable_container<T>::value && !std::is_enum<T>::value,
695 std::false_type>::type;
696 static constexpr bool value = type::value;
701struct classify_object<T,
702 typename std::enable_if<(!is_mutable_container<T>::value && is_wrapper<T>::value &&
703 !is_tuple_like<T>::value && uncommon_type<T>::value)>::type> {
704 static constexpr object_category value{object_category::wrapper_value};
709struct classify_object<T,
710 typename std::enable_if<uncommon_type<T>::value && type_count<T>::value == 1 &&
711 !is_wrapper<T>::value && is_direct_constructible<T, double>::value &&
712 is_direct_constructible<T, int>::value>::type> {
713 static constexpr object_category value{object_category::number_constructible};
718struct classify_object<T,
719 typename std::enable_if<uncommon_type<T>::value && type_count<T>::value == 1 &&
720 !is_wrapper<T>::value && !is_direct_constructible<T, double>::value &&
721 is_direct_constructible<T, int>::value>::type> {
722 static constexpr object_category value{object_category::integer_constructible};
727struct classify_object<T,
728 typename std::enable_if<uncommon_type<T>::value && type_count<T>::value == 1 &&
729 !is_wrapper<T>::value && is_direct_constructible<T, double>::value &&
730 !is_direct_constructible<T, int>::value>::type> {
731 static constexpr object_category value{object_category::double_constructible};
736struct classify_object<
738 typename std::enable_if<is_tuple_like<T>::value &&
739 ((type_count<T>::value >= 2 && !is_wrapper<T>::value) ||
740 (uncommon_type<T>::value && !is_direct_constructible<T, double>::value &&
741 !is_direct_constructible<T, int>::value) ||
742 (uncommon_type<T>::value && type_count<T>::value >= 2))>::type> {
743 static constexpr object_category value{object_category::tuple_value};
752template <
typename T>
struct classify_object<T, typename std::enable_if<is_mutable_container<T>::value>::type> {
753 static constexpr object_category value{object_category::container_value};
763 enable_if_t<classify_object<T>::value == object_category::char_value, detail::enabler> = detail::dummy>
764constexpr const char *type_name() {
769 enable_if_t<classify_object<T>::value == object_category::integral_value ||
770 classify_object<T>::value == object_category::integer_constructible,
771 detail::enabler> = detail::dummy>
772constexpr const char *type_name() {
777 enable_if_t<classify_object<T>::value == object_category::unsigned_integral, detail::enabler> = detail::dummy>
778constexpr const char *type_name() {
783 enable_if_t<classify_object<T>::value == object_category::floating_point ||
784 classify_object<T>::value == object_category::number_constructible ||
785 classify_object<T>::value == object_category::double_constructible,
786 detail::enabler> = detail::dummy>
787constexpr const char *type_name() {
793 enable_if_t<classify_object<T>::value == object_category::enumeration, detail::enabler> = detail::dummy>
794constexpr const char *type_name() {
800 enable_if_t<classify_object<T>::value == object_category::boolean_value, detail::enabler> = detail::dummy>
801constexpr const char *type_name() {
807 enable_if_t<classify_object<T>::value == object_category::complex_number, detail::enabler> = detail::dummy>
808constexpr const char *type_name() {
814 enable_if_t<classify_object<T>::value >= object_category::string_assignable &&
815 classify_object<T>::value <= object_category::other,
816 detail::enabler> = detail::dummy>
817constexpr const char *type_name() {
822 enable_if_t<classify_object<T>::value == object_category::tuple_value && type_count_base<T>::value >= 2,
823 detail::enabler> = detail::dummy>
824std::string type_name();
828 enable_if_t<classify_object<T>::value == object_category::container_value ||
829 classify_object<T>::value == object_category::wrapper_value,
830 detail::enabler> = detail::dummy>
831std::string type_name();
835 enable_if_t<classify_object<T>::value == object_category::tuple_value && type_count_base<T>::value == 1,
836 detail::enabler> = detail::dummy>
837inline std::string type_name() {
838 return type_name<typename std::decay<typename std::tuple_element<0, T>::type>::type>();
842template <
typename T, std::
size_t I>
843inline typename std::enable_if<I == type_count_base<T>::value, std::string>::type tuple_name() {
844 return std::string{};
848template <
typename T, std::
size_t I>
849inline typename std::enable_if<(I < type_count_base<T>::value), std::string>::type tuple_name() {
850 auto str = std::string{type_name<typename std::decay<typename std::tuple_element<I, T>::type>::type>()} +
',' +
851 tuple_name<T, I + 1>();
852 if(str.back() ==
',')
859 enable_if_t<classify_object<T>::value == object_category::tuple_value && type_count_base<T>::value >= 2,
861inline std::string type_name() {
862 auto tname = std::string(1,
'[') + tuple_name<T, 0>();
863 tname.push_back(
']');
869 enable_if_t<classify_object<T>::value == object_category::container_value ||
870 classify_object<T>::value == object_category::wrapper_value,
872inline std::string type_name() {
873 return type_name<typename T::value_type>();
879template <typename T, enable_if_t<std::is_unsigned<T>::value, detail::enabler> = detail::dummy>
880bool integral_conversion(
const std::string &input, T &output)
noexcept {
881 if(input.empty() || input.front() ==
'-') {
886 std::uint64_t output_ll = std::strtoull(input.c_str(), &val, 0);
887 if(errno == ERANGE) {
890 output =
static_cast<T
>(output_ll);
891 if(val == (input.c_str() + input.size()) &&
static_cast<std::uint64_t
>(output) == output_ll) {
895 std::int64_t output_sll = std::strtoll(input.c_str(), &val, 0);
896 if(val == (input.c_str() + input.size())) {
897 output = (output_sll < 0) ? static_cast<T>(0) : static_cast<T>(output_sll);
898 return (
static_cast<std::int64_t
>(output) == output_sll);
901 if(input.find_first_of(
"_'") != std::string::npos) {
902 std::string nstring = input;
903 nstring.erase(std::remove(nstring.begin(), nstring.end(),
'_'), nstring.end());
904 nstring.erase(std::remove(nstring.begin(), nstring.end(),
'\''), nstring.end());
905 return integral_conversion(nstring, output);
907 if(input.compare(0, 2,
"0o") == 0) {
910 output_ll = std::strtoull(input.c_str() + 2, &val, 8);
911 if(errno == ERANGE) {
914 output =
static_cast<T
>(output_ll);
915 return (val == (input.c_str() + input.size()) &&
static_cast<std::uint64_t
>(output) == output_ll);
917 if(input.compare(0, 2,
"0b") == 0) {
920 output_ll = std::strtoull(input.c_str() + 2, &val, 2);
921 if(errno == ERANGE) {
924 output =
static_cast<T
>(output_ll);
925 return (val == (input.c_str() + input.size()) &&
static_cast<std::uint64_t
>(output) == output_ll);
931template <typename T, enable_if_t<std::is_signed<T>::value, detail::enabler> = detail::dummy>
932bool integral_conversion(
const std::string &input, T &output)
noexcept {
938 std::int64_t output_ll = std::strtoll(input.c_str(), &val, 0);
939 if(errno == ERANGE) {
942 output =
static_cast<T
>(output_ll);
943 if(val == (input.c_str() + input.size()) &&
static_cast<std::int64_t
>(output) == output_ll) {
946 if(input ==
"true") {
948 output =
static_cast<T
>(1);
952 if(input.find_first_of(
"_'") != std::string::npos) {
953 std::string nstring = input;
954 nstring.erase(std::remove(nstring.begin(), nstring.end(),
'_'), nstring.end());
955 nstring.erase(std::remove(nstring.begin(), nstring.end(),
'\''), nstring.end());
956 return integral_conversion(nstring, output);
958 if(input.compare(0, 2,
"0o") == 0) {
961 output_ll = std::strtoll(input.c_str() + 2, &val, 8);
962 if(errno == ERANGE) {
965 output =
static_cast<T
>(output_ll);
966 return (val == (input.c_str() + input.size()) &&
static_cast<std::int64_t
>(output) == output_ll);
968 if(input.compare(0, 2,
"0b") == 0) {
971 output_ll = std::strtoll(input.c_str() + 2, &val, 2);
972 if(errno == ERANGE) {
975 output =
static_cast<T
>(output_ll);
976 return (val == (input.c_str() + input.size()) &&
static_cast<std::int64_t
>(output) == output_ll);
982inline std::int64_t to_flag_value(std::string val)
noexcept {
983 static const std::string trueString(
"true");
984 static const std::string falseString(
"false");
985 if(val == trueString) {
988 if(val == falseString) {
991 val = detail::to_lower(val);
992 std::int64_t ret = 0;
993 if(val.size() == 1) {
994 if(val[0] >=
'1' && val[0] <=
'9') {
995 return (
static_cast<std::int64_t
>(val[0]) -
'0');
1015 if(val == trueString || val ==
"on" || val ==
"yes" || val ==
"enable") {
1017 }
else if(val == falseString || val ==
"off" || val ==
"no" || val ==
"disable") {
1020 char *loc_ptr{
nullptr};
1021 ret = std::strtoll(val.c_str(), &loc_ptr, 0);
1022 if(loc_ptr != (val.c_str() + val.size()) && errno == 0) {
1030template <
typename T,
1031 enable_if_t<classify_object<T>::value == object_category::integral_value ||
1032 classify_object<T>::value == object_category::unsigned_integral,
1033 detail::enabler> = detail::dummy>
1034bool lexical_cast(
const std::string &input, T &output) {
1035 return integral_conversion(input, output);
1039template <
typename T,
1040 enable_if_t<classify_object<T>::value == object_category::char_value, detail::enabler> = detail::dummy>
1041bool lexical_cast(
const std::string &input, T &output) {
1042 if(input.size() == 1) {
1043 output =
static_cast<T
>(input[0]);
1046 return integral_conversion(input, output);
1050template <
typename T,
1051 enable_if_t<classify_object<T>::value == object_category::boolean_value, detail::enabler> = detail::dummy>
1052bool lexical_cast(
const std::string &input, T &output) {
1054 auto out = to_flag_value(input);
1057 }
else if(errno == ERANGE) {
1058 output = (input[0] !=
'-');
1066template <
typename T,
1067 enable_if_t<classify_object<T>::value == object_category::floating_point, detail::enabler> = detail::dummy>
1068bool lexical_cast(
const std::string &input, T &output) {
1072 char *val =
nullptr;
1073 auto output_ld = std::strtold(input.c_str(), &val);
1074 output =
static_cast<T
>(output_ld);
1075 if(val == (input.c_str() + input.size())) {
1079 if(input.find_first_of(
"_'") != std::string::npos) {
1080 std::string nstring = input;
1081 nstring.erase(std::remove(nstring.begin(), nstring.end(),
'_'), nstring.end());
1082 nstring.erase(std::remove(nstring.begin(), nstring.end(),
'\''), nstring.end());
1083 return lexical_cast(nstring, output);
1089template <
typename T,
1090 enable_if_t<classify_object<T>::value == object_category::complex_number, detail::enabler> = detail::dummy>
1091bool lexical_cast(
const std::string &input, T &output) {
1092 using XC =
typename wrapped_type<T, double>::type;
1095 bool worked =
false;
1096 auto nloc = str1.find_last_of(
"+-");
1097 if(nloc != std::string::npos && nloc > 0) {
1098 worked = lexical_cast(str1.substr(0, nloc), x);
1099 str1 = str1.substr(nloc);
1100 if(str1.back() ==
'i' || str1.back() ==
'j')
1102 worked = worked && lexical_cast(str1, y);
1104 if(str1.back() ==
'i' || str1.back() ==
'j') {
1106 worked = lexical_cast(str1, y);
1109 worked = lexical_cast(str1, x);
1117 return from_stream(input, output);
1121template <
typename T,
1122 enable_if_t<classify_object<T>::value == object_category::string_assignable, detail::enabler> = detail::dummy>
1123bool lexical_cast(
const std::string &input, T &output) {
1131 enable_if_t<classify_object<T>::value == object_category::string_constructible, detail::enabler> = detail::dummy>
1132bool lexical_cast(
const std::string &input, T &output) {
1140 enable_if_t<classify_object<T>::value == object_category::wstring_assignable, detail::enabler> = detail::dummy>
1141bool lexical_cast(
const std::string &input, T &output) {
1142 output = widen(input);
1148 enable_if_t<classify_object<T>::value == object_category::wstring_constructible, detail::enabler> = detail::dummy>
1149bool lexical_cast(
const std::string &input, T &output) {
1150 output = T{widen(input)};
1155template <
typename T,
1156 enable_if_t<classify_object<T>::value == object_category::enumeration, detail::enabler> = detail::dummy>
1157bool lexical_cast(
const std::string &input, T &output) {
1158 typename std::underlying_type<T>::type val;
1159 if(!integral_conversion(input, val)) {
1162 output =
static_cast<T
>(val);
1167template <
typename T,
1168 enable_if_t<classify_object<T>::value == object_category::wrapper_value &&
1169 std::is_assignable<T &, typename T::value_type>::value,
1170 detail::enabler> = detail::dummy>
1171bool lexical_cast(
const std::string &input, T &output) {
1172 typename T::value_type val;
1173 if(lexical_cast(input, val)) {
1177 return from_stream(input, output);
1180template <
typename T,
1181 enable_if_t<classify_object<T>::value == object_category::wrapper_value &&
1182 !std::is_assignable<T &, typename T::value_type>::value && std::is_assignable<T &, T>::value,
1183 detail::enabler> = detail::dummy>
1184bool lexical_cast(
const std::string &input, T &output) {
1185 typename T::value_type val;
1186 if(lexical_cast(input, val)) {
1190 return from_stream(input, output);
1196 enable_if_t<classify_object<T>::value == object_category::number_constructible, detail::enabler> = detail::dummy>
1197bool lexical_cast(
const std::string &input, T &output) {
1199 if(integral_conversion(input, val)) {
1205 if(lexical_cast(input, dval)) {
1210 return from_stream(input, output);
1216 enable_if_t<classify_object<T>::value == object_category::integer_constructible, detail::enabler> = detail::dummy>
1217bool lexical_cast(
const std::string &input, T &output) {
1219 if(integral_conversion(input, val)) {
1223 return from_stream(input, output);
1229 enable_if_t<classify_object<T>::value == object_category::double_constructible, detail::enabler> = detail::dummy>
1230bool lexical_cast(
const std::string &input, T &output) {
1232 if(lexical_cast(input, val)) {
1236 return from_stream(input, output);
1240template <
typename T,
1241 enable_if_t<classify_object<T>::value == object_category::other && std::is_assignable<T &, int>::value,
1242 detail::enabler> = detail::dummy>
1243bool lexical_cast(
const std::string &input, T &output) {
1245 if(integral_conversion(input, val)) {
1247#pragma warning(push)
1248#pragma warning(disable : 4800)
1261 return from_stream(input, output);
1266template <
typename T,
1267 enable_if_t<classify_object<T>::value == object_category::other && !std::is_assignable<T &, int>::value &&
1268 is_istreamable<T>::value,
1269 detail::enabler> = detail::dummy>
1270bool lexical_cast(
const std::string &input, T &output) {
1271 return from_stream(input, output);
1276template <
typename T,
1277 enable_if_t<classify_object<T>::value == object_category::other && !std::is_assignable<T &, int>::value &&
1278 !is_istreamable<T>::value && !adl_detail::is_lexical_castable<T>::value,
1279 detail::enabler> = detail::dummy>
1280bool lexical_cast(
const std::string & , T & ) {
1281 static_assert(!std::is_same<T, T>::value,
1282 "option object type must have a lexical cast overload or streaming input operator(>>) defined, if it "
1283 "is convertible from another type use the add_option<T, XC>(...) with XC being the known type");
1289template <
typename AssignTo,
1291 enable_if_t<std::is_same<AssignTo, ConvertTo>::value &&
1292 (classify_object<AssignTo>::value == object_category::string_assignable ||
1293 classify_object<AssignTo>::value == object_category::string_constructible ||
1294 classify_object<AssignTo>::value == object_category::wstring_assignable ||
1295 classify_object<AssignTo>::value == object_category::wstring_constructible),
1296 detail::enabler> = detail::dummy>
1297bool lexical_assign(
const std::string &input, AssignTo &output) {
1298 return lexical_cast(input, output);
1302template <
typename AssignTo,
1304 enable_if_t<std::is_same<AssignTo, ConvertTo>::value && std::is_assignable<AssignTo &, AssignTo>::value &&
1305 classify_object<AssignTo>::value != object_category::string_assignable &&
1306 classify_object<AssignTo>::value != object_category::string_constructible &&
1307 classify_object<AssignTo>::value != object_category::wstring_assignable &&
1308 classify_object<AssignTo>::value != object_category::wstring_constructible,
1309 detail::enabler> = detail::dummy>
1310bool lexical_assign(
const std::string &input, AssignTo &output) {
1312 output = AssignTo{};
1316 return lexical_cast(input, output);
1320template <
typename AssignTo,
1322 enable_if_t<std::is_same<AssignTo, ConvertTo>::value && !std::is_assignable<AssignTo &, AssignTo>::value &&
1323 classify_object<AssignTo>::value == object_category::wrapper_value,
1324 detail::enabler> = detail::dummy>
1325bool lexical_assign(
const std::string &input, AssignTo &output) {
1327 typename AssignTo::value_type emptyVal{};
1331 return lexical_cast(input, output);
1336template <
typename AssignTo,
1338 enable_if_t<std::is_same<AssignTo, ConvertTo>::value && !std::is_assignable<AssignTo &, AssignTo>::value &&
1339 classify_object<AssignTo>::value != object_category::wrapper_value &&
1340 std::is_assignable<AssignTo &, int>::value,
1341 detail::enabler> = detail::dummy>
1342bool lexical_assign(
const std::string &input, AssignTo &output) {
1348 if(lexical_cast(input, val)) {
1349#if defined(__clang__)
1351#pragma clang diagnostic push
1352#pragma clang diagnostic ignored "-Wsign-conversion"
1355#if defined(__clang__)
1356#pragma clang diagnostic pop
1364template <
typename AssignTo,
1366 enable_if_t<!std::is_same<AssignTo, ConvertTo>::value && std::is_assignable<AssignTo &, ConvertTo &>::value,
1367 detail::enabler> = detail::dummy>
1368bool lexical_assign(
const std::string &input, AssignTo &output) {
1370 bool parse_result = (!input.empty()) ? lexical_cast(input, val) :
true;
1374 return parse_result;
1381 enable_if_t<!std::is_same<AssignTo, ConvertTo>::value && !std::is_assignable<AssignTo &, ConvertTo &>::value &&
1382 std::is_move_assignable<AssignTo>::value,
1383 detail::enabler> = detail::dummy>
1384bool lexical_assign(
const std::string &input, AssignTo &output) {
1386 bool parse_result = input.empty() ? true : lexical_cast(input, val);
1388 output = AssignTo(val);
1390 return parse_result;
1394template <
typename AssignTo,
1396 enable_if_t<classify_object<ConvertTo>::value <= object_category::other &&
1397 classify_object<AssignTo>::value <= object_category::wrapper_value,
1398 detail::enabler> = detail::dummy>
1399bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1400 return lexical_assign<AssignTo, ConvertTo>(strings[0], output);
1405template <
typename AssignTo,
1407 enable_if_t<(type_count<AssignTo>::value <= 2) && expected_count<AssignTo>::value == 1 &&
1408 is_tuple_like<ConvertTo>::value && type_count_base<ConvertTo>::value == 2,
1409 detail::enabler> = detail::dummy>
1410bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1412 using FirstType =
typename std::remove_const<typename std::tuple_element<0, ConvertTo>::type>::type;
1413 using SecondType =
typename std::tuple_element<1, ConvertTo>::type;
1416 bool retval = lexical_assign<FirstType, FirstType>(strings[0], v1);
1417 retval = retval && lexical_assign<SecondType, SecondType>((strings.size() > 1) ? strings[1] : std::string{}, v2);
1419 output = AssignTo{v1, v2};
1425template <
class AssignTo,
1427 enable_if_t<is_mutable_container<AssignTo>::value && is_mutable_container<ConvertTo>::value &&
1428 type_count<ConvertTo>::value == 1,
1429 detail::enabler> = detail::dummy>
1430bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1431 output.erase(output.begin(), output.end());
1432 if(strings.empty()) {
1435 if(strings.size() == 1 && strings[0] ==
"{}") {
1438 bool skip_remaining =
false;
1439 if(strings.size() == 2 && strings[0] ==
"{}" && is_separator(strings[1])) {
1440 skip_remaining =
true;
1442 for(
const auto &elem : strings) {
1443 typename AssignTo::value_type out;
1444 bool retval = lexical_assign<typename AssignTo::value_type, typename ConvertTo::value_type>(elem, out);
1448 output.insert(output.end(), std::move(out));
1449 if(skip_remaining) {
1453 return (!output.empty());
1457template <class AssignTo, class ConvertTo, enable_if_t<is_complex<ConvertTo>::value, detail::enabler> = detail::dummy>
1458bool lexical_conversion(
const std::vector<std::string> &strings, AssignTo &output) {
1460 if(strings.size() >= 2 && !strings[1].empty()) {
1461 using XC2 =
typename wrapped_type<ConvertTo, double>::type;
1463 auto str1 = strings[1];
1464 if(str1.back() ==
'i' || str1.back() ==
'j') {
1467 auto worked = lexical_cast(strings[0], x) && lexical_cast(str1, y);
1469 output = ConvertTo{x, y};
1473 return lexical_assign<AssignTo, ConvertTo>(strings[0], output);
1477template <
class AssignTo,
1479 enable_if_t<is_mutable_container<AssignTo>::value && (expected_count<ConvertTo>::value == 1) &&
1480 (type_count<ConvertTo>::value == 1),
1481 detail::enabler> = detail::dummy>
1482bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1485 output.reserve(strings.size());
1486 for(
const auto &elem : strings) {
1488 output.emplace_back();
1489 retval = retval && lexical_assign<typename AssignTo::value_type, ConvertTo>(elem, output.back());
1491 return (!output.empty()) && retval;
1497template <
class AssignTo,
1499 enable_if_t<is_mutable_container<AssignTo>::value && is_mutable_container<ConvertTo>::value &&
1500 type_count_base<ConvertTo>::value == 2,
1501 detail::enabler> = detail::dummy>
1502bool lexical_conversion(std::vector<std::string> strings, AssignTo &output);
1505template <
class AssignTo,
1507 enable_if_t<is_mutable_container<AssignTo>::value && is_mutable_container<ConvertTo>::value &&
1508 type_count_base<ConvertTo>::value != 2 &&
1509 ((type_count<ConvertTo>::value > 2) ||
1510 (type_count<ConvertTo>::value > type_count_base<ConvertTo>::value)),
1511 detail::enabler> = detail::dummy>
1512bool lexical_conversion(
const std::vector<std::string> &strings, AssignTo &output);
1515template <
class AssignTo,
1517 enable_if_t<is_tuple_like<AssignTo>::value && is_tuple_like<ConvertTo>::value &&
1518 (type_count_base<ConvertTo>::value != type_count<ConvertTo>::value ||
1519 type_count<ConvertTo>::value > 2),
1520 detail::enabler> = detail::dummy>
1521bool lexical_conversion(
const std::vector<std::string> &strings, AssignTo &output);
1525template <
typename AssignTo,
1527 enable_if_t<!is_tuple_like<AssignTo>::value && !is_mutable_container<AssignTo>::value &&
1528 classify_object<ConvertTo>::value != object_category::wrapper_value &&
1529 (is_mutable_container<ConvertTo>::value || type_count<ConvertTo>::value > 2),
1530 detail::enabler> = detail::dummy>
1531bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1533 if(strings.size() > 1 || (!strings.empty() && !(strings.front().empty()))) {
1535 auto retval = lexical_conversion<ConvertTo, ConvertTo>(strings, val);
1536 output = AssignTo{val};
1539 output = AssignTo{};
1544template <
class AssignTo,
class ConvertTo, std::
size_t I>
1545inline typename std::enable_if<(I >= type_count_base<AssignTo>::value),
bool>::type
1546tuple_conversion(
const std::vector<std::string> &, AssignTo &) {
1551template <
class AssignTo,
class ConvertTo>
1552inline typename std::enable_if<!is_mutable_container<ConvertTo>::value && type_count<ConvertTo>::value == 1,
bool>::type
1553tuple_type_conversion(std::vector<std::string> &strings, AssignTo &output) {
1554 auto retval = lexical_assign<AssignTo, ConvertTo>(strings[0], output);
1555 strings.erase(strings.begin());
1560template <
class AssignTo,
class ConvertTo>
1561inline typename std::enable_if<!is_mutable_container<ConvertTo>::value && (type_count<ConvertTo>::value > 1) &&
1562 type_count<ConvertTo>::value == type_count_min<ConvertTo>::value,
1564tuple_type_conversion(std::vector<std::string> &strings, AssignTo &output) {
1565 auto retval = lexical_conversion<AssignTo, ConvertTo>(strings, output);
1566 strings.erase(strings.begin(), strings.begin() + type_count<ConvertTo>::value);
1571template <
class AssignTo,
class ConvertTo>
1572inline typename std::enable_if<is_mutable_container<ConvertTo>::value ||
1573 type_count<ConvertTo>::value != type_count_min<ConvertTo>::value,
1575tuple_type_conversion(std::vector<std::string> &strings, AssignTo &output) {
1577 std::size_t index{subtype_count_min<ConvertTo>::value};
1578 const std::size_t mx_count{subtype_count<ConvertTo>::value};
1579 const std::size_t mx{(std::min)(mx_count, strings.size() - 1)};
1582 if(is_separator(strings[index])) {
1587 bool retval = lexical_conversion<AssignTo, ConvertTo>(
1588 std::vector<std::string>(strings.begin(), strings.begin() +
static_cast<std::ptrdiff_t
>(index)), output);
1589 if(strings.size() > index) {
1590 strings.erase(strings.begin(), strings.begin() +
static_cast<std::ptrdiff_t
>(index) + 1);
1598template <
class AssignTo,
class ConvertTo, std::
size_t I>
1599inline typename std::enable_if<(I < type_count_base<AssignTo>::value),
bool>::type
1600tuple_conversion(std::vector<std::string> strings, AssignTo &output) {
1602 using ConvertToElement =
typename std::
1603 conditional<is_tuple_like<ConvertTo>::value,
typename std::tuple_element<I, ConvertTo>::type, ConvertTo>::type;
1604 if(!strings.empty()) {
1605 retval = retval && tuple_type_conversion<typename std::tuple_element<I, AssignTo>::type, ConvertToElement>(
1606 strings, std::get<I>(output));
1608 retval = retval && tuple_conversion<AssignTo, ConvertTo, I + 1>(std::move(strings), output);
1613template <
class AssignTo,
1615 enable_if_t<is_mutable_container<AssignTo>::value && is_mutable_container<ConvertTo>::value &&
1616 type_count_base<ConvertTo>::value == 2,
1618bool lexical_conversion(std::vector<std::string> strings, AssignTo &output) {
1620 while(!strings.empty()) {
1622 typename std::remove_const<typename std::tuple_element<0, typename ConvertTo::value_type>::type>::type v1;
1623 typename std::tuple_element<1, typename ConvertTo::value_type>::type v2;
1624 bool retval = tuple_type_conversion<decltype(v1), decltype(v1)>(strings, v1);
1625 if(!strings.empty()) {
1626 retval = retval && tuple_type_conversion<decltype(v2), decltype(v2)>(strings, v2);
1629 output.insert(output.end(),
typename AssignTo::value_type{v1, v2});
1634 return (!output.empty());
1638template <
class AssignTo,
1640 enable_if_t<is_tuple_like<AssignTo>::value && is_tuple_like<ConvertTo>::value &&
1641 (type_count_base<ConvertTo>::value != type_count<ConvertTo>::value ||
1642 type_count<ConvertTo>::value > 2),
1644bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1646 !is_tuple_like<ConvertTo>::value || type_count_base<AssignTo>::value == type_count_base<ConvertTo>::value,
1647 "if the conversion type is defined as a tuple it must be the same size as the type you are converting to");
1648 return tuple_conversion<AssignTo, ConvertTo, 0>(strings, output);
1652template <
class AssignTo,
1654 enable_if_t<is_mutable_container<AssignTo>::value && is_mutable_container<ConvertTo>::value &&
1655 type_count_base<ConvertTo>::value != 2 &&
1656 ((type_count<ConvertTo>::value > 2) ||
1657 (type_count<ConvertTo>::value > type_count_base<ConvertTo>::value)),
1659bool lexical_conversion(
const std::vector<std ::string> &strings, AssignTo &output) {
1662 std::vector<std::string> temp;
1664 std::size_t icount{0};
1665 std::size_t xcm{type_count<ConvertTo>::value};
1666 auto ii_max = strings.size();
1667 while(ii < ii_max) {
1668 temp.push_back(strings[ii]);
1671 if(icount == xcm || is_separator(temp.back()) || ii == ii_max) {
1672 if(
static_cast<int>(xcm) > type_count_min<ConvertTo>::value && is_separator(temp.back())) {
1675 typename AssignTo::value_type temp_out;
1677 lexical_conversion<typename AssignTo::value_type, typename ConvertTo::value_type>(temp, temp_out);
1682 output.insert(output.end(), std::move(temp_out));
1690template <
typename AssignTo,
1692 enable_if_t<classify_object<ConvertTo>::value == object_category::wrapper_value &&
1693 std::is_assignable<ConvertTo &, ConvertTo>::value,
1694 detail::enabler> = detail::dummy>
1695bool lexical_conversion(
const std::vector<std::string> &strings, AssignTo &output) {
1696 if(strings.empty() || strings.front().empty()) {
1697 output = ConvertTo{};
1700 typename ConvertTo::value_type val;
1701 if(lexical_conversion<typename ConvertTo::value_type, typename ConvertTo::value_type>(strings, val)) {
1702 output = ConvertTo{val};
1709template <
typename AssignTo,
1711 enable_if_t<classify_object<ConvertTo>::value == object_category::wrapper_value &&
1712 !std::is_assignable<AssignTo &, ConvertTo>::value,
1713 detail::enabler> = detail::dummy>
1714bool lexical_conversion(
const std::vector<std::string> &strings, AssignTo &output) {
1715 using ConvertType =
typename ConvertTo::value_type;
1716 if(strings.empty() || strings.front().empty()) {
1717 output = ConvertType{};
1721 if(lexical_conversion<typename ConvertTo::value_type, typename ConvertTo::value_type>(strings, val)) {
1729inline std::string sum_string_vector(
const std::vector<std::string> &values) {
1733 for(
const auto &arg : values) {
1735 auto comp = lexical_cast(arg, tv);
1738 auto fv = detail::to_flag_value(arg);
1739 fail = (errno != 0);
1743 tv =
static_cast<double>(fv);
1748 for(
const auto &arg : values) {
1752 std::ostringstream out;
Definition TypeTools.hpp:96
Check for complex.
Definition TypeTools.hpp:235
Definition TypeTools.hpp:177
Check for input streamability.
Definition TypeTools.hpp:224
Definition TypeTools.hpp:213
Definition TypeTools.hpp:299
This can be specialized to override the type deduction for IsMember.
Definition TypeTools.hpp:81
not a pointer
Definition TypeTools.hpp:115
Definition TypeTools.hpp:125
Definition TypeTools.hpp:260
Definition TypeTools.hpp:280
Definition TypeTools.hpp:292
static auto first(Q &&pair_value) -> decltype(std::get< 0 >(std::forward< Q >(pair_value)))
Get the first value (really just the underlying value)
Definition TypeTools.hpp:157
static auto second(Q &&pair_value) -> decltype(std::get< 1 >(std::forward< Q >(pair_value)))
Get the second value (really just the underlying value)
Definition TypeTools.hpp:161
Adaptor for set-like structure: This just wraps a normal container in a few utilities that do almost ...
Definition TypeTools.hpp:130
static auto second(Q &&pair_value) -> decltype(std::forward< Q >(pair_value))
Get the second value (really just the underlying value)
Definition TypeTools.hpp:140
static auto first(Q &&pair_value) -> decltype(std::forward< Q >(pair_value))
Get the first value (really just the underlying value)
Definition TypeTools.hpp:136
forward declare the subtype_count_min structure
Definition TypeTools.hpp:437
Set of overloads to get the type size of an object.
Definition TypeTools.hpp:434
This will only trigger for actual void type.
Definition TypeTools.hpp:408
This will only trigger for actual void type.
Definition TypeTools.hpp:440
template to get the underlying value type if it exists or use a default
Definition TypeTools.hpp:398
Check to see if something is bool (fail check by default)
Definition TypeTools.hpp:61
Check to see if something is copyable pointer.
Definition TypeTools.hpp:76
Check to see if something is a shared pointer.
Definition TypeTools.hpp:67
A copy of std::void_t from C++17 (helper for C++11 and C++14)
Definition TypeTools.hpp:50