7 #include "arg_router/policy/display_name.hpp"
8 #include "arg_router/policy/error_name.hpp"
9 #include "arg_router/policy/no_result_value.hpp"
10 #include "arg_router/tree_node.hpp"
14 namespace arg_router::dependency::detail
17 template <
typename ParentDocName,
typename... Params>
18 struct generate_string_of_child_names {
20 template <
typename Prefix,
typename Strings>
21 using prepend_prefix =
22 #ifdef AR_ENABLE_CPP20_STRINGS
23 std::decay_t<decltype(
24 Prefix{} + boost::mp11::mp_fold<Strings, utility::str<>, utility::str_concat>{})>;
26 boost::mp11::mp_flatten<boost::mp11::mp_insert_c<Strings, 0, typename Prefix::array_type>>;
29 template <
typename Child>
32 constexpr
static auto build()
34 constexpr
auto token = parsing::node_token_type<Child>();
38 using type =
typename std::decay_t<decltype(build())>;
41 template <
typename ChildName>
43 #ifdef AR_ENABLE_CPP20_STRINGS
44 typename ChildName::template append_t<utility::str<','>>;
46 boost::mp11::mp_push_back<typename ChildName::array_type, traits::integral_constant<','>>;
49 using children_type = boost::mp11::mp_filter<is_tree_node, std::tuple<Params...>>;
53 using concatenated_string = prepend_prefix<
55 boost::mp11::mp_transform<
57 boost::mp11::mp_transform_q<
59 boost::mp11::mp_bind<build_name, boost::mp11::_1>>,
64 #ifdef AR_ENABLE_CPP20_STRINGS
65 using type = std::decay_t<decltype(
66 concatenated_string{}.template substr<0, concatenated_string::size() - 1>())>;
69 boost::mp11::mp_take_c<concatenated_string, std::tuple_size_v<concatenated_string> - 1>>;
75 template <
typename DefaultString,
typename... Policies>
78 template <
typename Policy,
typename Enable =
void>
79 struct has_display_name_t : std::false_type {
82 template <
typename Policy>
83 struct has_display_name_t<Policy, std::enable_if_t<policy::is_policy_v<Policy>>> {
84 constexpr
static bool value = traits::has_display_name_method_v<Policy>;
87 using policies_tuple = std::tuple<std::decay_t<Policies>...>;
89 using display_index = boost::mp11::mp_find_if<policies_tuple, has_display_name_t>;
92 using display_type =
typename std::tuple_element_t<I::value, policies_tuple>::string_type;
95 constexpr
static auto has_display_name =
96 display_index::value != std::tuple_size_v<policies_tuple>;
99 using display_string =
100 boost::mp11::mp_eval_if_c<!has_display_name, DefaultString, display_type, display_index>;
102 using policies_tuple_with_error_name =
103 boost::mp11::mp_push_front<policies_tuple,
104 policy::error_name_t<
typename generate_string_of_child_names<
106 std::decay_t<Policies>...>::type>>;
109 using type = std::conditional_t<
111 boost::mp11::mp_rename<policies_tuple_with_error_name, tree_node>,
112 boost::mp11::mp_rename<boost::mp11::mp_push_front<policies_tuple_with_error_name,
113 policy::display_name_t<DefaultString>>,
117 template <
typename ParentDocName,
typename... Params>
118 class basic_one_of_t :
public add_names<ParentDocName, Params...>::type
120 using parent_type =
typename add_names<ParentDocName, Params...>::type;
122 static_assert((std::tuple_size_v<typename parent_type::children_type> >= 2),
123 "basic_one_of_t must have at least one two child nodes");
124 static_assert(!traits::has_long_name_method_v<basic_one_of_t>,
125 "basic_one_of_t must not have a long name policy");
126 static_assert(!traits::has_short_name_method_v<basic_one_of_t>,
127 "basic_one_of_t must not have a short name policy");
128 static_assert(!traits::has_none_name_method_v<basic_one_of_t>,
129 "basic_one_of_t must not have a none name policy");
130 static_assert(!traits::has_description_method_v<basic_one_of_t>,
131 "basic_one_of_t must not have a description policy");
134 using typename parent_type::children_type;
135 using typename parent_type::policies_type;
138 boost::mp11::mp_any_of_q<
140 boost::mp11::mp_bind<policy::has_missing_phase_method, boost::mp11::_1, bool>>::value,
141 "basic_one_of_t must have a missing phase method, a "
142 "policy::required or policy::default_value are commonly used");
144 static_assert(!parent_type::template any_phases_v<
bool,
145 policy::has_pre_parse_phase_method,
146 policy::has_parse_phase_method,
147 policy::has_routing_phase_method>,
148 "basic_one_of_t does not support policies with pre-parse, parse, "
149 "or routing phases; as it delegates those to its children");
151 using basic_value_type =
153 boost::mp11::mp_remove_if<
typename parent_type::children_type,
156 static_assert((std::tuple_size_v<basic_value_type> >= 1),
157 "basic_one_of_t must have at least one child with a value_type");
159 template <
bool Flatten>
160 class children_help_data_type
162 template <
typename Child,
typename Prefix>
164 using label =
typename Prefix::template append_t<
165 typename Child::template help_data_type<Flatten>::label>;
166 using description =
typename Child::template help_data_type<Flatten>::description;
167 using children = std::tuple<>;
170 using top_bar = AR_STRING(
"┌ ");
171 using middle_bar = AR_STRING(
"├ ");
172 using bottom_bar = AR_STRING(
"└ ");
174 template <
typename Child>
175 using first_prefixer = prefixer<Child, top_bar>;
177 template <
typename Child>
178 using middle_prefixer = prefixer<Child, middle_bar>;
180 template <
typename Child>
181 using last_prefixer = prefixer<Child, bottom_bar>;
184 using children = boost::mp11::mp_replace_at_c<
185 boost::mp11::mp_replace_first<boost::mp11::mp_transform<middle_prefixer, children_type>,
186 first_prefixer<boost::mp11::mp_first<children_type>>>,
187 std::tuple_size_v<children_type> - 1,
188 last_prefixer<boost::mp11::mp_back<children_type>>>;
190 template <
typename OwnerNode,
typename FilterFn>
191 [[nodiscard]]
static vector<runtime_help_data> runtime_children(
const OwnerNode& owner,
195 auto result = parent_type::template default_leaf_help_data_type<true>::runtime_children(
197 std::forward<FilterFn>(f));
200 if (result.size() <= 1) {
205 result.front().label =
206 utility::dynamic_string_view{top_bar::get()} + result.front().label;
207 for (
auto i = 1u; i < (result.size() - 1); ++i) {
208 result[i].label = utility::dynamic_string_view{middle_bar::get()} + result[i].label;
210 result.back().label =
211 utility::dynamic_string_view{bottom_bar::get()} + result.back().label;
217 template <
auto has_display_name = add_names<ParentDocName, Params...>::has_display_name>
218 constexpr
explicit basic_one_of_t(Params... params,
220 std::enable_if_t<has_display_name>* =
nullptr) noexcept :
221 parent_type{std::tuple_element_t<0, policies_type>{},
222 std::move(params)...}
226 template <
auto has_display_name = add_names<ParentDocName, Params...>::has_display_name>
227 constexpr
explicit basic_one_of_t(Params... params,
229 std::enable_if_t<!has_display_name>* =
nullptr) noexcept :
230 parent_type{std::tuple_element_t<0, policies_type>{},
231 std::tuple_element_t<1, policies_type>{},
232 std::move(params)...}
constexpr std::string_view to_string(prefix_type prefix) noexcept
std::is_base_of< no_result_value<>, T > has_no_result_value
constexpr auto description
typename T::type get_type
typename T::value_type get_value_type
typename convert_to_cts< T >::type convert_to_cts_t