Advertisement
Guest User

Untitled

a guest
Jan 21st, 2017
93
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 5.91 KB | None | 0 0
  1. public class Perlin {
  2.  
  3. public static double OctavePerlin(double x, double y, double z, int octaves, double persistence) {
  4. double total = 0;
  5. double frequency = 1;
  6. double amplitude = 1;
  7. for(int i=0;i<octaves;i++) {
  8. total += perlin(x * frequency, y * frequency, z * frequency) * amplitude;
  9.  
  10. amplitude *= persistence;
  11. frequency *= 2;
  12. }
  13.  
  14. return total;
  15. }
  16.  
  17. private static readonly int[] permutation = { 151,160,137,91,90,15, // Hash lookup table as defined by Ken Perlin. This is a randomly
  18. 131,13,201,95,96,53,194,233,7,225,140,36,103,30,69,142,8,99,37,240,21,10,23, // arranged array of all numbers from 0-255 inclusive.
  19. 190, 6,148,247,120,234,75,0,26,197,62,94,252,219,203,117,35,11,32,57,177,33,
  20. 88,237,149,56,87,174,20,125,136,171,168, 68,175,74,165,71,134,139,48,27,166,
  21. 77,146,158,231,83,111,229,122,60,211,133,230,220,105,92,41,55,46,245,40,244,
  22. 102,143,54, 65,25,63,161, 1,216,80,73,209,76,132,187,208, 89,18,169,200,196,
  23. 135,130,116,188,159,86,164,100,109,198,173,186, 3,64,52,217,226,250,124,123,
  24. 5,202,38,147,118,126,255,82,85,212,207,206,59,227,47,16,58,17,182,189,28,42,
  25. 223,183,170,213,119,248,152, 2,44,154,163, 70,221,153,101,155,167, 43,172,9,
  26. 129,22,39,253, 19,98,108,110,79,113,224,232,178,185, 112,104,218,246,97,228,
  27. 251,34,242,193,238,210,144,12,191,179,162,241, 81,51,145,235,249,14,239,107,
  28. 49,192,214, 31,181,199,106,157,184, 84,204,176,115,121,50,45,127, 4,150,254,
  29. 138,236,205,93,222,114,67,29,24,72,243,141,128,195,78,66,215,61,156,180
  30. };
  31.  
  32. private static readonly int[] p; // Doubled permutation to avoid overflow
  33.  
  34. static Perlin() {
  35. p = new int[512];
  36. for(int x=0;x<512;x++) {
  37. p[x] = permutation[x%256];
  38. }
  39. }
  40.  
  41. public static double perlin(double x, double y, double z) {
  42. if(repeat > 0) { // If we have any repeat on, change the coordinates to their "local" repetitions
  43. x = x%repeat;
  44. y = y%repeat;
  45. z = z%repeat;
  46. }
  47.  
  48. int xi = (int)x & 255; // Calculate the "unit cube" that the point asked will be located in
  49. int yi = (int)y & 255; // The left bound is ( |_x_|,|_y_|,|_z_| ) and the right bound is that
  50. int zi = (int)z & 255; // plus 1. Next we calculate the location (from 0.0 to 1.0) in that cube.
  51. double xf = x-(int)x; // We also fade the location to smooth the result.
  52. double yf = y-(int)y;
  53. double zf = z-(int)z;
  54. double u = fade(xf);
  55. double v = fade(yf);
  56. double w = fade(zf);
  57.  
  58. int a = p[xi] +yi; // This here is Perlin's hash function. We take our x value (remember,
  59. int aa = p[a] +zi; // between 0 and 255) and get a random value (from our p[] array above) between
  60. int ab = p[a+1] +zi; // 0 and 255. We then add y to it and plug that into p[], and add z to that.
  61. int b = p[xi+1]+yi; // Then, we get another random value by adding 1 to that and putting it into p[]
  62. int ba = p[b] +zi; // and add z to it. We do the whole thing over again starting with x+1. Later
  63. int bb = p[b+1] +zi; // we plug aa, ab, ba, and bb back into p[] along with their +1's to get another set.
  64. // in the end we have 8 values between 0 and 255 - one for each vertex on the unit cube.
  65. // These are all interpolated together using u, v, and w below.
  66.  
  67. double x1, x2, y1, y2;
  68. x1 = lerp( grad (p[aa ], xf , yf , zf), // This is where the "magic" happens. We calculate a new set of p[] values and use that to get
  69. grad (p[ba ], xf-1, yf , zf), // our final gradient values. Then, we interpolate between those gradients with the u value to get
  70. u); // 4 x-values. Next, we interpolate between the 4 x-values with v to get 2 y-values. Finally,
  71. x2 = lerp( grad (p[ab ], xf , yf-1, zf), // we interpolate between the y-values to get a z-value.
  72. grad (p[bb ], xf-1, yf-1, zf),
  73. u); // When calculating the p[] values, remember that above, p[a+1] expands to p[xi]+yi+1 -- so you are
  74. y1 = lerp(x1, x2, v); // essentially adding 1 to yi. Likewise, p[ab+1] expands to p[p[xi]+yi+1]+zi+1] -- so you are adding
  75. // to zi. The other 3 parameters are your possible return values (see grad()), which are actually
  76. x1 = lerp( grad (p[aa+1], xf , yf , zf-1), // the vectors from the edges of the unit cube to the point in the unit cube itself.
  77. grad (p[ba+1], xf-1, yf , zf-1),
  78. u);
  79. x2 = lerp( grad (p[ab+1], xf , yf-1, zf-1),
  80. grad (p[bb+1], xf-1, yf-1, zf-1),
  81. u);
  82. y2 = lerp (x1, x2, v);
  83.  
  84. return (lerp (y1, y2, w)+1)/2; // For convenience we bound it to 0 - 1 (theoretical min/max before is -1 - 1)
  85. }
  86.  
  87. public static double grad(int hash, double x, double y, double z) {
  88. int h = hash & 15; // Take the hashed value and take the first 4 bits of it (15 == 0b1111)
  89. double u = h < 8 /* 0b1000 */ ? x : y; // If the most signifigant bit (MSB) of the hash is 0 then set u = x. Otherwise y.
  90.  
  91. double v; // In Ken Perlin's original implementation this was another conditional operator (?:). I
  92. // expanded it for readability.
  93.  
  94. if(h < 4 /* 0b0100 */) // If the first and second signifigant bits are 0 set v = y
  95. v = y;
  96. else if(h == 12 /* 0b1100 */ || h == 14 /* 0b1110*/)// If the first and second signifigant bits are 1 set v = x
  97. v = x;
  98. else // If the first and second signifigant bits are not equal (0/1, 1/0) set v = z
  99. v = z;
  100.  
  101. return ((h&1) == 0 ? u : -u)+((h&2) == 0 ? v : -v); // Use the last 2 bits to decide if u and v are positive or negative. Then return their addition.
  102. }
  103.  
  104. public static double fade(double t) {
  105. // Fade function as defined by Ken Perlin. This eases coordinate values
  106. // so that they will "ease" towards integral values. This ends up smoothing
  107. // the final output.
  108. return t * t * t * (t * (t * 6 - 15) + 10); // 6t^5 - 15t^4 + 10t^3
  109. }
  110.  
  111. public static double lerp(double a, double b, double x) {
  112. return a + x * (b - a);
  113. }
  114. }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement