Fixed handling of decltype cxxmember type aliases with dependend parameters
This commit is contained in:
@@ -1,28 +1,53 @@
|
||||
#include <cstddef>
|
||||
#include <tuple>
|
||||
|
||||
namespace clanguml {
|
||||
namespace t00064 {
|
||||
|
||||
// Loosely based on
|
||||
template <typename... Ts> struct type_list { };
|
||||
|
||||
template <typename Ret, typename Arg, typename... Ts>
|
||||
struct type_list<Ret (*)(Arg &&arg), Ts...> { };
|
||||
struct type_list<Ret (*)(Arg &&), Ts...> { };
|
||||
|
||||
template <typename T> struct head;
|
||||
template <typename T, typename... Ts> struct type_list<const T, Ts...> { };
|
||||
|
||||
template <typename> struct head;
|
||||
template <typename Head, typename... Tail>
|
||||
struct head<type_list<Head, Tail...>> {
|
||||
using type = Head;
|
||||
};
|
||||
|
||||
template <typename T> using head_t = typename head<T>::type;
|
||||
template <typename... Type> using first_t = type_list<Type...>;
|
||||
|
||||
template <typename... Type> using second_t = type_list<Type...>;
|
||||
|
||||
template <typename, typename> class type_group_pair;
|
||||
template <typename... First, typename... Second>
|
||||
class type_group_pair<type_list<First...>, type_list<Second...>> {
|
||||
template <typename Type>
|
||||
class type_group_pair<first_t<First...>, second_t<Second...>> {
|
||||
static constexpr size_t size = sizeof...(First) + sizeof...(Second);
|
||||
};
|
||||
|
||||
template <typename T> struct optional_ref { };
|
||||
|
||||
template <typename, typename, typename> class type_group_pair_it;
|
||||
template <typename It, typename... First, typename... Second>
|
||||
class type_group_pair_it<It, first_t<First...>, second_t<Second...>> {
|
||||
public:
|
||||
using value_type =
|
||||
decltype(std::tuple_cat(std::make_tuple(*std::declval<It>()),
|
||||
std::declval<First>().get_as_tuple({})...,
|
||||
std::declval<Second>().get_as_tuple({})...));
|
||||
|
||||
using ref_t = optional_ref<value_type>;
|
||||
|
||||
ref_t get(unsigned i) { return {}; }
|
||||
|
||||
const value_type *getp(unsigned i) { return nullptr; }
|
||||
|
||||
constexpr unsigned find(value_type const &v) { return 0; }
|
||||
};
|
||||
|
||||
struct A { };
|
||||
struct B { };
|
||||
struct C { };
|
||||
|
||||
@@ -37,9 +37,37 @@ TEST_CASE("t00064", "[test-case][class]")
|
||||
|
||||
REQUIRE_THAT(puml, !Contains("type-parameter-"));
|
||||
|
||||
// Check if all classes exist
|
||||
REQUIRE_THAT(puml, IsClass(_A("A")));
|
||||
REQUIRE_THAT(puml, IsClass(_A("B")));
|
||||
REQUIRE_THAT(puml, IsClass(_A("C")));
|
||||
REQUIRE_THAT(puml, IsClass(_A("R")));
|
||||
|
||||
REQUIRE_THAT(puml, IsClassTemplate("type_list", "Ts..."));
|
||||
REQUIRE_THAT(puml, IsClassTemplate("type_list", "Ret(Arg &&),Ts..."));
|
||||
REQUIRE_THAT(puml, IsClassTemplate("type_list", "T const,Ts..."));
|
||||
|
||||
REQUIRE_THAT(puml, IsClassTemplate("head", "typename"));
|
||||
REQUIRE_THAT(puml, IsClassTemplate("head", "type_list<Head,Tail...>"));
|
||||
REQUIRE_THAT(
|
||||
puml, IsClassTemplate("type_group_pair", "typename,typename"));
|
||||
REQUIRE_THAT(puml,
|
||||
IsClassTemplate(
|
||||
"type_group_pair", "type_list<First...>,type_list<Second...>"));
|
||||
REQUIRE_THAT(puml,
|
||||
IsClassTemplate(
|
||||
"type_group_pair", "type_list<float,double>,type_list<A,B,C>"));
|
||||
|
||||
REQUIRE_THAT(puml, IsClassTemplate("optional_ref", "T"));
|
||||
|
||||
REQUIRE_THAT(puml,
|
||||
IsClassTemplate("type_group_pair_it",
|
||||
"It,type_list<First...>,type_list<Second...>"));
|
||||
REQUIRE_THAT(
|
||||
puml, (IsMethod<Public>("get", "ref_t", "unsigned int i")));
|
||||
REQUIRE_THAT(puml,
|
||||
(IsMethod<Public>("getp", "value_type const*", "unsigned int i")));
|
||||
REQUIRE_THAT(puml,
|
||||
(IsMethod<Public>("find", "unsigned int", "value_type const& v")));
|
||||
|
||||
save_puml(
|
||||
config.output_directory() + "/" + diagram->name + ".puml", puml);
|
||||
|
||||
Reference in New Issue
Block a user