Sparse Configuration in C++ using Variadic Templates and Strong Types

Let's consider data structure with well-defined default values. This structure is to be partially modified with different values on multiple places in the code base (especially in test code, where the same structure needs to be initialized with different data for each test case).

struct ShapeParameters
{
    unsigned stroke_color = 0x000000;
    double stroke_width = 1.0;
    unsigned fill_color = 0xffffff;
    double scale_x = 1.0;
    double scale_y = 1.0;
    double rotate = 0.0;
};

Strong Types

So called strong types (inspired by this post) can be used to strictly constrain parameters of a function. As the quoted article states:

A strong type is a type used in place of another type to carry specific meaning through its name.

We can define following strong types for the above configuration structure.

template <typename T, typename Dummy>
struct NamedType
{
    explicit NamedType(const T & v): value(v) { }
    T value;
};

using StrokeColor = NamedType<unsigned, struct StrokeCollorDummy>;
using StrokeWidth = NamedType<double, struct StrokeWidthDummy>;
using FillColor = NamedType<unsigned, struct FillCollorDummy>;
using ScaleX = NamedType<double, struct ScaleXDummy>;
using ScaleY = NamedType<double, struct ScaleYDummy>;
using Rotate = NamedType<double, struct RotateDummy>;

Variadic Template Configurator

Now ve can define define variadic sparse configurator function as follows.

inline void setParameter(ShapeParameters & sp, StrokeColor v)
    { sp.stroke_color = v.value; }
inline void setParameter(ShapeParameters & sp, StrokeWidth v)
    { sp.stroke_width = v.value; }
inline void setParameter(ShapeParameters & sp, FillColor v)
    { sp.fill_color = v.value; }
inline void setParameter(ShapeParameters & sp, ScaleX v)
    { sp.scale_x = v.value; }
inline void setParameter(ShapeParameters & sp, ScaleY v)
    { sp.scale_y = v.value; }
inline void setParameter(ShapeParameters & sp, Rotate v)
    { sp.rotate = v.value; }

template <typename T, typename ...Args>
inline void setParameter(
        ShapeParameters & sp,
        T parameter,
        Args... others)
{
    setParameter(sp, parameter);
    setParameter(sp, others...);
}

template <typename ...Args>
inline void configure(ShapeParameters & sp, Args... parameters)
{
    // do some initialization of the 'sp' here
    setParameter(sp, parameters...);
}

Trying it Out

By putting the above code together we can configure the parameters in single function call.

int main()
{
    ShapeParameters sp;

    configure(sp, StrokeColor(0xff0000), StrokeWidth(2.33), ScaleY(1.5));

    std::cout << "Stroke color: " << sp.stroke_color << std::endl;
    std::cout << "Stroke width: " << sp.stroke_width << std::endl;
    std::cout << "Fill color:   " << sp.fill_color << std::endl;
    std::cout << "Scale X:      " << sp.scale_x << std::endl;
    std::cout << "Scale Y:      " << sp.scale_y << std::endl;
    std::cout << "Rotate:       " << sp.rotate << std::endl;
    return 0;
}