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