1 /** Deep duplication.
2 
3     Add to Phobos somewhere.
4 
5     TODO: Support inout relevant for const?
6     See: http://forum.dlang.org/thread/hojk95$4mk$1@digitalmars.com
7  */
8 module nxt.deepdup;
9 
10 import core.internal.traits : Unqual;
11 import std.typecons : tuple, Tuple;
12 import std.meta : staticMap;
13 import std.traits : isDynamicArray, isStaticArray, isArray, isPointer;
14 import std.range.primitives : ElementType;
15 
16 alias TypeofDeepdup(T) = typeof(deepdup(T.init));
17 
18 ref Unqual!T deepdup(T)(T t)
19 if (is(T == struct) &&
20     !is(T.Types))
21 {
22     staticMap!(TypeofDeepdup, typeof(t.tupleof)) tup;
23     foreach (const i, Type; tup)
24         tup[i] = t.tupleof[i].deepdup;
25     return Unqual!T(tup);
26 }
27 
28 Tuple!(staticMap!(TypeofDeepdup, T.Types)) deepdup(T)(T t)
29 if (is(T.Types))
30 {
31     staticMap!(TypeofDeepdup, T.Types) tup;
32     foreach (const i, Type; tup)
33         tup[i] = t.field[i].deepdup;
34     return tuple(tup);
35 }
36 
37 Unqual!T deepdup(T)(T t)
38 if (is(T == class))
39 {
40     staticMap!(TypeofDeepdup, typeof(t.tupleof)) tup;
41     foreach (const i, Type; tup)
42         tup[i] = t.tupleof[i].deepdup;
43     return new Unqual!T(tup);
44 }
45 
46 TypeofDeepdup!(ElementType!T)[] deepdup(T)(T t)
47 if (isDynamicArray!T)
48 {
49     auto result = new TypeofDeepdup!(ElementType!T)[](t.length);
50     foreach (const i, elem; t)
51         result[i] = elem.deepdup;
52     return result;
53 }
54 
55 TypeofDeepdup!(ElementType!T)[T.length] deepdup(T)(T t)
56 if (isStaticArray!T)
57 {
58     TypeofDeepdup!(ElementType!T)[T.length] result = t;
59     foreach (ref elem; result)
60         elem = elem.deepdup;
61     return result;
62 }
63 
64 TypeofDeepdup!T* deepdup(T)(T* t)
65 {
66     return &deepdup(*t);
67 }
68 
69 Unqual!T deepdup(T)(T t) if (!is(T == struct) &&
70                              !is(T == class) &&
71                              !isArray!T &&
72                              !is(T.Types) &&
73                              !isPointer!T)
74 {
75     return cast(Unqual!T)t;
76 }
77 
78 @safe pure nothrow:
79 
80 ///
81 unittest
82 {
83     auto x = [1, 2, 3];
84     assert(x == x.dup);
85     auto y = x;
86     assert(&x[0] == &y[0]);
87     assert(&x[0] != &x.dup[0]);
88 }
89 
90 ///
91 unittest
92 {
93     auto x = [[1], [2], [3]];
94     auto y = x.dup;
95     x[0][0] = 11;
96     assert(x[0][0] == 11);
97     assert(y[0][0] == 11);
98 }
99 
100 ///
101 unittest
102 {
103     auto x = [[1], [2], [3]];
104     auto y = x.deepdup;
105     x[0][0] = 11;
106     assert(x[0][0] == 11);
107     assert(y[0][0] == 1);
108 }
109 
110 ///
111 unittest
112 {
113     auto x = [[1], [2], [3]];
114     auto y = x.deepdup;
115     x[0][0] = 11;
116     assert(x[0][0] == 11);
117     assert(y[0][0] == 1);
118 }
119 
120 ///
121 unittest
122 {
123     auto x = [[[1]], [[2]], [[3]]];
124     auto y = x.deepdup;
125     x[0][0][0] = 11;
126     assert(x[0][0][0] == 11);
127     assert(y[0][0][0] == 1);
128 }
129 
130 /// dup of static array
131 unittest
132 {
133     int[3] x = [1, 2, 3];
134     auto y = x.dup;
135     x[0] = 11;
136     assert(x[0] == 11);
137     assert(y[0] == 1);
138 }
139 
140 /// dup of static array of dynamic arrays
141 unittest
142 {
143     int[][3] x = [[1], [2], [3]];
144     auto y = x.dup;
145     x[0][0] = 11;
146     assert(x[0][0] == 11);
147     assert(y[0][0] == 11);
148 }
149 
150 /// deepdup of static array of dynamic arrays
151 unittest
152 {
153     int[][3] x = [[1], [2], [3]];
154     auto y = x.deepdup;
155     x[0][0] = 11;
156     assert(x[0][0] == 11);
157     assert(y[0][0] == 1);
158 }