Skip to content

Commit 3655c92

Browse files
authored
noise: implement 2d perlin noise (#42) (#87)
1 parent f2972f5 commit 3655c92

File tree

4 files changed

+105
-2
lines changed

4 files changed

+105
-2
lines changed

.github/workflows/ci.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -45,8 +45,8 @@ jobs:
4545

4646
- name: Validate Docs
4747
run: |
48-
cd ./vsl
49-
v check-md .
48+
cd ~
49+
v check-md ~/.vmodules/vsl
5050
5151
ci:
5252
runs-on: ${{ matrix.os }}

noise/perlin.v

+69
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
module noise
2+
3+
import math
4+
import rand
5+
6+
// implemented according to the example implementation on wikipedia
7+
// https://en.wikipedia.org/wiki/Perlin_noise#Implementation
8+
9+
struct Vec2 {
10+
x f32
11+
y f32
12+
}
13+
14+
[inline]
15+
fn random_gradient() ?Vec2 {
16+
nr := rand.f32_in_range(0.0, math.pi * 2)?
17+
return Vec2{
18+
x: math.cosf(nr)
19+
y: math.sinf(nr)
20+
}
21+
}
22+
23+
[inline]
24+
fn interpolate(a f32, b f32, w f32) f32 {
25+
return ((a - b) * w) + a
26+
}
27+
28+
[inline]
29+
fn dot(ix int, iy int, x f32, y f32) ?f32 {
30+
vec := random_gradient()?
31+
dx := x - f32(ix)
32+
dy := y - f32(iy)
33+
return dx * vec.x + dy * vec.y
34+
}
35+
36+
// gets the noise value at coordinate (x, y)
37+
[inline]
38+
pub fn perlin(x f32, y f32) ?f32 {
39+
x1 := int(math.floor(x))
40+
y1 := int(math.floor(y))
41+
x2 := x1 + 1
42+
y2 := y1 + 1
43+
44+
sx := x - f32(x1)
45+
sy := y - f32(y2)
46+
47+
n0 := dot(x1, y1, x, y)?
48+
n1 := dot(x2, y1, x, y)?
49+
first := interpolate(n0, n1, sx)
50+
51+
n2 := dot(x1, y2, x, y)?
52+
n3 := dot(x2, y2, x, y)?
53+
second := interpolate(n2, n3, sx)
54+
55+
return interpolate(first, second, sy)
56+
}
57+
58+
// creates a 2d array of perlin noise of size `w` times `h`
59+
pub fn perlin_many(w int, h int) ?[][]f32 {
60+
mut res := [][]f32{len: h, init: []f32{len: w}}
61+
for i, a in res {
62+
for j, _ in a {
63+
val := perlin(j, i)?
64+
println(f64(val))
65+
res[i][j] = val
66+
}
67+
}
68+
return res
69+
}

noise/perlin_test.v

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
module noise
2+
3+
import rand
4+
5+
fn test_perlin() {
6+
rand.seed([u32(1), 0])
7+
assert perlin(0.0, 0.0) or { assert false } == -0.6457864046096802
8+
}
9+
10+
fn test_perlin_many() {
11+
rand.seed([u32(1), 0])
12+
assert perlin_many(2, 2) or { assert false } == [
13+
[f32(-0.6457864046096802), 0.0279612485319376],
14+
[f32(0.2286834865808487), -0.5146363377571106],
15+
]
16+
}

noise/readme.md

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
# noise
2+
3+
This module aims to to implement noise algorithms.
4+
5+
It uses the `rand` module in vlib to generate random numbers,
6+
so you may seed the generator as you see fit.
7+
8+
Example:
9+
```v
10+
import vsl.noise
11+
import rand
12+
13+
fn main() {
14+
rand.seed([u32(1), 0])
15+
println(noise.perlin(0.0, 0.0)?)
16+
}
17+
```
18+
Output: `-0.6457864`

0 commit comments

Comments
 (0)