1 module gfm.math.vector;
2 
3 import std.traits,
4        std.math,
5        std.conv,
6        std.array,
7        std..string;
8 
9 import gfm.math.funcs;
10 
11 /**
12  * Generic 1D small vector.
13  * Params:
14  *    N = number of elements
15  *    T = type of elements
16  */
17 struct Vector(T, int N)
18 {
19 nothrow:
20     public
21     {
22         static assert(N >= 1);
23 
24         // fields definition
25         union
26         {
27             T[N] v;
28             struct
29             {
30                 static if (N >= 1)
31                 {
32                     T x;
33                     alias x r;
34                 }
35                 static if (N >= 2)
36                 {
37                     T y;
38                     alias y g;
39                 }
40                 static if (N >= 3)
41                 {
42                     T z;
43                     alias z b;
44                 }
45                 static if (N >= 4)
46                 {
47                     T w;
48                     alias w a;
49                 }
50             }
51         }
52 
53         @nogc this(Args...)(Args args) pure nothrow
54         {
55             static if (args.length == 1)
56             {
57                 // Construct a Vector from a single value.
58                 opAssign!(Args[0])(args[0]);
59             }
60             else
61             {
62                 int index = 0;
63                 foreach(arg; args)
64                 {
65                     static if (isAssignable!(T, typeof(arg)))
66                     {
67                         v[index] = arg;
68                         index++; // has to be on its own line (DMD 2.068)
69                     }
70                     else static if (is(typeof(arg._isVector)) && isAssignable!(T, arg._T))
71                     {
72                         mixin(generateLoopCode!("v[index + @] = arg[@];", arg._N)());
73                         index += arg._N;
74                     }
75                     else
76                         static assert(false, "Unrecognized argument in Vector constructor");
77                 }
78                 assert(index == N, "Bad arguments in Vector constructor");
79             }
80         }
81 
82         /// Assign a Vector from a compatible type.
83         @nogc ref Vector opAssign(U)(U x) pure nothrow if (isAssignable!(T, U))
84         {
85             mixin(generateLoopCode!("v[@] = x;", N)()); // copy to each component
86             return this;
87         }
88 
89         /// Assign a Vector with a static array type.
90         @nogc ref Vector opAssign(U)(U arr) pure nothrow if ((isStaticArray!(U) && isAssignable!(T, typeof(arr[0])) && (arr.length == N)))
91         {
92             mixin(generateLoopCode!("v[@] = arr[@];", N)());
93             return this;
94         }
95 
96         /// Assign with a dynamic array.
97         /// Size is checked in debug-mode.
98         @nogc ref Vector opAssign(U)(U arr) pure nothrow if (isDynamicArray!(U) && isAssignable!(T, typeof(arr[0])))
99         {
100             assert(arr.length == N);
101             mixin(generateLoopCode!("v[@] = arr[@];", N)());
102             return this;
103         }
104 
105         /// Assign from a samey Vector.
106         @nogc ref Vector opAssign(U)(U u) pure nothrow if (is(U : Vector))
107         {
108             v[] = u.v[];
109             return this;
110         }
111 
112         /// Assign from other vectors types (same size, compatible type).
113         @nogc ref Vector opAssign(U)(U x) pure nothrow if (is(typeof(U._isVector))
114                                                        && isAssignable!(T, U._T)
115                                                        && (!is(U: Vector))
116                                                        && (U._N == _N))
117         {
118             mixin(generateLoopCode!("v[@] = x.v[@];", N)());
119             return this;
120         }
121 
122         /// Returns: a pointer to content.
123         @nogc inout(T)* ptr() pure inout nothrow @property
124         {
125             return v.ptr;
126         }
127 
128         /// Converts to a pretty string.
129         string toString() const nothrow
130         {
131             try
132                 return format("%s", v);
133             catch (Exception e)
134                 assert(false); // should not happen since format is right
135         }
136 
137         @nogc bool opEquals(U)(U other) pure const nothrow
138             if (is(U : Vector))
139         {
140             for (int i = 0; i < N; ++i)
141             {
142                 if (v[i] != other.v[i])
143                 {
144                     return false;
145                 }
146             }
147             return true;
148         }
149 
150         @nogc bool opEquals(U)(U other) pure const nothrow
151             if (isConvertible!U)
152         {
153             Vector conv = other;
154             return opEquals(conv);
155         }
156 
157         @nogc Vector opUnary(string op)() pure const nothrow
158             if (op == "+" || op == "-" || op == "~" || op == "!")
159         {
160             Vector res = void;
161             mixin(generateLoopCode!("res.v[@] = " ~ op ~ " v[@];", N)());
162             return res;
163         }
164 
165         @nogc ref Vector opOpAssign(string op, U)(U operand) pure nothrow
166             if (is(U : Vector))
167         {
168             mixin(generateLoopCode!("v[@] " ~ op ~ "= operand.v[@];", N)());
169             return this;
170         }
171 
172         @nogc ref Vector opOpAssign(string op, U)(U operand) pure nothrow if (isConvertible!U)
173         {
174             Vector conv = operand;
175             return opOpAssign!op(conv);
176         }
177 
178         @nogc Vector opBinary(string op, U)(U operand) pure const nothrow
179             if (is(U: Vector) || (isConvertible!U))
180         {
181             Vector result = void;
182             static if (is(U: T))
183                 mixin(generateLoopCode!("result.v[@] = cast(T)(v[@] " ~ op ~ " operand);", N)());
184             else
185             {
186                 Vector other = operand;
187                 mixin(generateLoopCode!("result.v[@] = cast(T)(v[@] " ~ op ~ " other.v[@]);", N)());
188             }
189             return result;
190         }
191 
192         @nogc Vector opBinaryRight(string op, U)(U operand) pure const nothrow if (isConvertible!U)
193         {
194             Vector result = void;
195             static if (is(U: T))
196                 mixin(generateLoopCode!("result.v[@] = cast(T)(operand " ~ op ~ " v[@]);", N)());
197             else
198             {
199                 Vector other = operand;
200                 mixin(generateLoopCode!("result.v[@] = cast(T)(other.v[@] " ~ op ~ " v[@]);", N)());
201             }
202             return result;
203         }
204 
205         @nogc ref T opIndex(size_t i) pure nothrow
206         {
207             return v[i];
208         }
209 
210         @nogc ref const(T) opIndex(size_t i) pure const nothrow
211         {
212             return v[i];
213         }
214 
215         @nogc T opIndexAssign(U : T)(U x, size_t i) pure nothrow
216         {
217             return v[i] = x;
218         }
219 
220 
221         /// Implements swizzling.
222         ///
223         /// Example:
224         /// ---
225         /// vec4i vi = [4, 1, 83, 10];
226         /// assert(vi.zxxyw == [83, 4, 4, 1, 10]);
227         /// ---
228         @nogc @property auto opDispatch(string op, U = void)() pure const nothrow if (isValidSwizzle!(op))
229         {
230             alias Vector!(T, op.length) returnType;
231             returnType res = void;
232             enum indexTuple = swizzleTuple!op;
233             foreach(i, index; indexTuple)
234                 res.v[i] = v[index];
235             return res;
236         }
237 
238         /// Support swizzling assignment like in shader languages.
239         ///
240         /// Example:
241         /// ---
242         /// vec3f v = [0, 1, 2];
243         /// v.yz = v.zx;
244         /// assert(v == [0, 2, 0]);
245         /// ---
246         @nogc @property void opDispatch(string op, U)(U x) pure
247             if ((op.length >= 2)
248                 && (isValidSwizzleUnique!op)                   // v.xyy will be rejected
249                 && is(typeof(Vector!(T, op.length)(x)))) // can be converted to a small vector of the right size
250         {
251             Vector!(T, op.length) conv = x;
252             enum indexTuple = swizzleTuple!op;
253             foreach(i, index; indexTuple)
254                 v[index] = conv[i];
255         }
256 
257         /// Casting to small vectors of the same size.
258         /// Example:
259         /// ---
260         /// vec4f vf;
261         /// vec4d vd = cast!(vec4d)vf;
262         /// ---
263         @nogc U opCast(U)() pure const nothrow if (is(typeof(U._isVector)) && (U._N == _N))
264         {
265             U res = void;
266             mixin(generateLoopCode!("res.v[@] = cast(U._T)v[@];", N)());
267             return res;
268         }
269 
270         /// Implement slices operator overloading.
271         /// Allows to go back to slice world.
272         /// Returns: length.
273         @nogc int opDollar() pure const nothrow
274         {
275             return N;
276         }
277 
278         /// Returns: a slice which covers the whole Vector.
279         @nogc T[] opSlice() pure nothrow
280         {
281             return v[];
282         }
283 
284         // vec[a..b]
285         @nogc T[] opSlice(int a, int b) pure nothrow
286         {
287             return v[a..b];
288         }
289 
290         /// Returns: squared length.
291         @nogc T squaredLength() pure const nothrow
292         {
293             T sumSquares = 0;
294             mixin(generateLoopCode!("sumSquares += v[@] * v[@];", N)());
295             return sumSquares;
296         }
297 
298         // Returns: squared Euclidean distance.
299         @nogc T squaredDistanceTo(Vector v) pure const nothrow
300         {
301             return (v - this).squaredLength();
302         }
303 
304         static if (isFloatingPoint!T)
305         {
306             /// Returns: Euclidean length
307             @nogc T length() pure const nothrow
308             {
309                 return sqrt(squaredLength());
310             }
311 
312             /// Returns: Inverse of Euclidean length.
313             @nogc T inverseLength() pure const nothrow
314             {
315                 return 1 / sqrt(squaredLength());
316             }
317 
318             /// Faster but less accurate inverse of Euclidean length.
319             /// Returns: Inverse of Euclidean length.
320             @nogc T fastInverseLength() pure const nothrow
321             {
322                 return inverseSqrt(squaredLength());
323             }
324 
325             /// Returns: Euclidean distance between this and other.
326             @nogc T distanceTo(Vector other) pure const nothrow
327             {
328                 return (other - this).length();
329             }
330 
331             /// In-place normalization.
332             @nogc void normalize() pure nothrow
333             {
334                 auto invLength = inverseLength();
335                 mixin(generateLoopCode!("v[@] *= invLength;", N)());
336             }
337 
338             /// Returns: Normalized vector.
339             @nogc Vector normalized() pure const nothrow
340             {
341                 Vector res = this;
342                 res.normalize();
343                 return res;
344             }
345 
346             /// Faster but less accurate in-place normalization.
347             @nogc void fastNormalize() pure nothrow
348             {
349                 auto invLength = fastInverseLength();
350                 mixin(generateLoopCode!("v[@] *= invLength;", N)());
351             }
352 
353             /// Faster but less accurate vector normalization.
354             /// Returns: Normalized vector.
355             @nogc Vector fastNormalized() pure const nothrow
356             {
357                 Vector res = this;
358                 res.fastNormalize();
359                 return res;
360             }
361 
362             static if (N == 3)
363             {
364                 /// Gets an orthogonal vector from a 3-dimensional vector.
365                 /// Doesn’t normalise the output.
366                 /// Authors: Sam Hocevar
367                 /// See_also: Source at $(WEB lolengine.net/blog/2013/09/21/picking-orthogonal-vector-combing-coconuts).
368                 @nogc Vector getOrthogonalVector() pure const nothrow
369                 {
370                     return abs(x) > abs(z) ? Vector(-y, x, 0.0) : Vector(0.0, -z, y);
371                 }
372             }
373         }
374     }
375 
376     private
377     {
378         enum _isVector = true; // do we really need this? I don't know.
379 
380         enum _N = N;
381         alias T _T;
382 
383         // define types that can be converted to this, but are not the same type
384         template isConvertible(T)
385         {
386             enum bool isConvertible = (!is(T : Vector))
387             && is(typeof(
388                 {
389                     T x;
390                     Vector v = x;
391                 }()));
392         }
393 
394         // define types that can't be converted to this
395         template isForeign(T)
396         {
397             enum bool isForeign = (!isConvertible!T) && (!is(T: Vector));
398         }
399 
400         template isValidSwizzle(string op, int lastSwizzleClass = -1)
401         {
402             static if (op.length == 0)
403                 enum bool isValidSwizzle = true;
404             else
405             {
406                 enum len = op.length;
407                 enum int swizzleClass = swizzleClassify!(op[0]);
408                 enum bool swizzleClassValid = (lastSwizzleClass == -1 || (swizzleClass == lastSwizzleClass));
409                 enum bool isValidSwizzle = (swizzleIndex!(op[0]) != -1)
410                                          && swizzleClassValid
411                                          && isValidSwizzle!(op[1..len], swizzleClass);
412             }
413         }
414 
415         template searchElement(char c, string s)
416         {
417             static if (s.length == 0)
418             {
419                 enum bool result = false;
420             }
421             else
422             {
423                 enum string tail = s[1..s.length];
424                 enum bool result = (s[0] == c) || searchElement!(c, tail).result;
425             }
426         }
427 
428         template hasNoDuplicates(string s)
429         {
430             static if (s.length == 1)
431             {
432                 enum bool result = true;
433             }
434             else
435             {
436                 enum tail = s[1..s.length];
437                 enum bool result = !(searchElement!(s[0], tail).result) && hasNoDuplicates!(tail).result;
438             }
439         }
440 
441         // true if the swizzle has at the maximum one time each letter
442         template isValidSwizzleUnique(string op)
443         {
444             static if (isValidSwizzle!op)
445                 enum isValidSwizzleUnique = hasNoDuplicates!op.result;
446             else
447                 enum bool isValidSwizzleUnique = false;
448         }
449 
450         template swizzleIndex(char c)
451         {
452             static if((c == 'x' || c == 'r') && N >= 1)
453                 enum swizzleIndex = 0;
454             else static if((c == 'y' || c == 'g') && N >= 2)
455                 enum swizzleIndex = 1;
456             else static if((c == 'z' || c == 'b') && N >= 3)
457                 enum swizzleIndex = 2;
458             else static if ((c == 'w' || c == 'a') && N >= 4)
459                 enum swizzleIndex = 3;
460             else
461                 enum swizzleIndex = -1;
462         }
463 
464         template swizzleClassify(char c)
465         {
466             static if(c == 'x' || c == 'y' || c == 'z' || c == 'w')
467                 enum swizzleClassify = 0;
468             else static if(c == 'r' || c == 'g' || c == 'b' || c == 'a')
469                 enum swizzleClassify = 1;
470             else
471                 enum swizzleClassify = -1;
472         }
473 
474         template swizzleTuple(string op)
475         {
476             enum opLength = op.length;
477             static if (op.length == 0)
478                 enum swizzleTuple = [];
479             else
480                 enum swizzleTuple = [ swizzleIndex!(op[0]) ] ~ swizzleTuple!(op[1..op.length]);
481         }
482     }
483 }
484 
485 private string definePostfixAliases(string type)
486 {
487     return "alias " ~ type ~ "!byte "   ~ type ~ "b;\n"
488          ~ "alias " ~ type ~ "!ubyte "  ~ type ~ "ub;\n"
489          ~ "alias " ~ type ~ "!short "  ~ type ~ "s;\n"
490          ~ "alias " ~ type ~ "!ushort " ~ type ~ "us;\n"
491          ~ "alias " ~ type ~ "!int "    ~ type ~ "i;\n"
492          ~ "alias " ~ type ~ "!uint "   ~ type ~ "ui;\n"
493          ~ "alias " ~ type ~ "!long "   ~ type ~ "l;\n"
494          ~ "alias " ~ type ~ "!ulong "  ~ type ~ "ul;\n"
495          ~ "alias " ~ type ~ "!float "  ~ type ~ "f;\n"
496          ~ "alias " ~ type ~ "!double " ~ type ~ "d;\n";
497 }
498 
499 template vec2(T) { alias Vector!(T, 2) vec2; }
500 template vec3(T) { alias Vector!(T, 3) vec3; }
501 template vec4(T) { alias Vector!(T, 4) vec4; }
502 
503 mixin(definePostfixAliases("vec2"));
504 mixin(definePostfixAliases("vec3"));
505 mixin(definePostfixAliases("vec4"));
506 
507 
508 private
509 {
510     static string generateLoopCode(string formatString, int N)() pure nothrow
511     {
512         string result;
513         for (int i = 0; i < N; ++i)
514         {
515             string index = ctIntToString(i);
516             // replace all @ by indices
517             result ~= formatString.replace("@", index);
518         }
519         return result;
520     }
521 
522     // Speed-up CTFE conversions
523     static string ctIntToString(int n) pure nothrow
524     {
525         static immutable string[16] table = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9"];
526         if (n < 10)
527             return table[n];
528         else
529             return to!string(n);
530     }
531 }
532 
533 
534 /// Element-wise minimum.
535 @nogc Vector!(T, N) min(T, int N)(const Vector!(T, N) a, const Vector!(T, N) b) pure nothrow
536 {
537     import std.algorithm: min;
538     Vector!(T, N) res = void;
539     mixin(generateLoopCode!("res.v[@] = min(a.v[@], b.v[@]);", N)());
540     return res;
541 }
542 
543 /// Element-wise maximum.
544 @nogc Vector!(T, N) max(T, int N)(const Vector!(T, N) a, const Vector!(T, N) b) pure nothrow
545 {
546     import std.algorithm: max;
547     Vector!(T, N) res = void;
548     mixin(generateLoopCode!("res.v[@] = max(a.v[@], b.v[@]);", N)());
549     return res;
550 }
551 
552 /// Returns: Dot product.
553 @nogc T dot(T, int N)(const Vector!(T, N) a, const Vector!(T, N) b) pure nothrow
554 {
555     T sum = 0;
556     mixin(generateLoopCode!("sum += a.v[@] * b.v[@];", N)());
557     return sum;
558 }
559 
560 /// Returns: 3D cross product.
561 /// Thanks to vuaru for corrections.
562 @nogc Vector!(T, 3) cross(T)(const Vector!(T, 3) a, const Vector!(T, 3) b) pure nothrow
563 {
564     return Vector!(T, 3)(a.y * b.z - a.z * b.y,
565                          a.z * b.x - a.x * b.z,
566                          a.x * b.y - a.y * b.x);
567 }
568 
569 /// 3D reflect, like the GLSL function.
570 /// Returns: a reflected by normal b.
571 @nogc Vector!(T, 3) reflect(T)(const Vector!(T, 3) a, const Vector!(T, 3) b) pure nothrow
572 {
573     return a - (2 * dot(b, a)) * b;
574 }
575 
576 
577 /// Returns: angle between vectors.
578 /// See_also: "The Right Way to Calculate Stuff" at $(WEB www.plunk.org/~hatch/rightway.php)
579 @nogc T angleBetween(T, int N)(const Vector!(T, N) a, const Vector!(T, N) b) pure nothrow
580 {
581     auto aN = a.normalized();
582     auto bN = b.normalized();
583     auto dp = dot(aN, bN);
584 
585     if (dp < 0)
586         return PI - 2 * asin((-bN-aN).length / 2);
587     else
588         return 2 * asin((bN-aN).length / 2);
589 }
590 
591 static assert(vec2f.sizeof == 8);
592 static assert(vec3d.sizeof == 24);
593 static assert(vec4i.sizeof == 16);
594 
595 
596 unittest
597 {
598     static assert(vec2i.isValidSwizzle!"xyx");
599     static assert(!vec2i.isValidSwizzle!"xyz");
600     static assert(vec4i.isValidSwizzle!"brra");
601     static assert(!vec4i.isValidSwizzle!"rgyz");
602     static assert(vec2i.isValidSwizzleUnique!"xy");
603     static assert(vec2i.isValidSwizzleUnique!"yx");
604     static assert(!vec2i.isValidSwizzleUnique!"xx");
605 
606     assert(vec2l(0, 1) == vec2i(0, 1));
607 
608     int[2] arr = [0, 1];
609     int[] arr2 = new int[2];
610     arr2[] = arr[];
611     vec2i a = vec2i([0, 1]);
612     vec2i a2 = vec2i(0, 1);
613     immutable vec2i b = vec2i(0);
614     assert(b[0] == 0 && b[1] == 0);
615     vec2i c = arr;
616     vec2l d = arr2;
617     assert(a == a2);
618     assert(a == c);
619     assert(vec2l(a) == vec2l(a));
620     assert(vec2l(a) == d);
621 
622     vec4i x = [4, 5, 6, 7];
623     assert(x == x);
624     --x[0];
625     assert(x[0] == 3);
626     ++x[0];
627     assert(x[0] == 4);
628     x[1] &= 1;
629     x[2] = 77 + x[2];
630     x[3] += 3;
631     assert(x == [4, 1, 83, 10]);
632     assert(x.xxywz == [4, 4, 1, 10, 83]);
633     assert(x.xxxxxxx == [4, 4, 4, 4, 4, 4, 4]);
634     assert(x.abgr == [10, 83, 1, 4]);
635     assert(a != b);
636     x = vec4i(x.xyz, 166);
637     assert(x == [4, 1, 83, 166]);
638 
639     vec2l e = a;
640     vec2l f = a + b;
641     assert(f == vec2l(a));
642 
643     vec3ui g = vec3i(78,9,4);
644     g ^= vec3i(78,9,4);
645     assert(g == vec3ui(0));
646     //g[0..2] = 1u;
647     //assert(g == [2, 1, 0]);
648 
649     assert(vec2i(4, 5) + 1 == vec2i(5,6));
650     assert(vec2i(4, 5) - 1 == vec2i(3,4));
651     assert(1 + vec2i(4, 5) == vec2i(5,6));
652     assert(vec3f(1,1,1) * 0 == 0);
653     assert(1.0 * vec3d(4,5,6) == vec3f(4,5.0f,6.0));
654 
655     auto dx = vec2i(1,2);
656     auto dy = vec2i(4,5);
657     auto dp = dot(dx, dy);
658     assert(dp == 14 );
659 
660     vec3i h = cast(vec3i)(vec3d(0.5, 1.1, -2.2));
661     assert(h == [0, 1, -2]);
662     assert(h[] == [0, 1, -2]);
663     assert(h[1..3] == [1, -2]);
664     assert(h.zyx == [-2, 1, 0]);
665 
666     h.yx = vec2i(5, 2); // swizzle assignment
667 
668     assert(h.xy == [2, 5]);
669     assert(-h[1] == -5);
670     assert(++h[0] == 3);
671 
672     //assert(h == [-2, 1, 0]);
673     assert(!__traits(compiles, h.xx = h.yy));
674     vec4ub j;
675 
676     assert(lerp(vec2f(-10, -1), vec2f(10, 1), 0.5) == vec2f(0, 0));
677 
678     // vectors of user-defined types
679     import gfm.math.half;
680     alias Vector!(half, 2) vec2h;
681     vec2h k = vec2h(1.0f, 2.0f);
682 
683     // larger vectors
684     alias Vector!(float, 5) vec5f;
685     vec5f l = vec5f(1, 2.0f, 3.0, k.x.toFloat(), 5.0L);
686     l = vec5f(l.xyz, vec2i(1, 2));
687 }
688