Не уверен, что понимаете, что именно вы хотите, но ...
А как насчет передачи массивов в стиле C вместо списков инициализаторов?
Требовалось немного помощников ( возможно, можно немного упростить)
template <typename T, std::size_t>
using getType = T;
template <typename T, std::size_t ... Is>
auto getY (std::index_sequence<Is...>)
-> y<getType<T, Is>...>;
template <typename T, std::size_t N>
using proY = decltype(getY<T>(std::make_index_sequence<N>{}));
, но это руководство по выводу должно работать для случая вложенного однородного состояния
template <std::size_t ... Dims, typename ... Ts>
y ( Ts const (&...arr)[Dims] ) -> y<proY<Ts, Dims>...>;
Ниже приведен полный пример компиляции
#include <tuple>
#include <iostream>
template <typename... Ts>
struct y
{
using values_t = std::tuple<Ts...>;
values_t values;
constexpr y (Ts const & ... vs) : values({ vs... })
{ }
};
template <typename T, std::size_t>
using getType = T;
template <typename T, std::size_t ... Is>
auto getY (std::index_sequence<Is...>)
-> y<getType<T, Is>...>;
template <typename T, std::size_t N>
using proY = decltype(getY<T>(std::make_index_sequence<N>{}));
template <std::size_t ... Dims, typename ... Ts>
y ( Ts const (&...arr)[Dims] ) -> y<proY<Ts, Dims>...>;
int main()
{
using T0 = decltype(y{ { 1, 2 }, { 'a', 'b', 'c' },
{1.0, 2.0, 3.0, 4.0} });
using T1 = y<y<int, int>, y<char, char, char>,
y<double, double, double, double>>;
static_assert( std::is_same_v<T0, T1> );
}