-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathterrain_gen.py
112 lines (81 loc) · 2.35 KB
/
terrain_gen.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
from noise import noise2, noise3
from random import random
from settings import *
@njit
def get_height(x, z):
# island mask: tapers off edges of world
island = 1 / (pow(0.0025 * math.hypot(x - CENTER_XZ, z - CENTER_XZ), 20) + 0.0001)
island = min(island, 1)
# amplitude
a1 = CENTER_Y
a2, a4, a8 = a1 * 0.5, a1 * 0.25, a1 * 0.125
# frequency
f1 = 0.005
f2, f4, f8 = f1 * 2, f1 * 4, f1 * 8
if noise2(0.1 * x, 0.1 * z) < 0:
a1 /= 1.07
height = 0
# octaves
height += noise2(x * f1, z * f1) * a1 + a1
height += noise2(x * f2, z * f2) * a2 - a2
height += noise2(x * f4, z * f4) * a4 + a4
height += noise2(x * f8, z * f8) * a8 - a8
height = max(height, 1)
height *= island
return int(height)
@njit
def get_index(x, y, z):
return x + CHUNK_SIZE * z + CHUNK_AREA * y
@njit
def set_voxel_id(voxels, x, y, z, wx, wy, wz, world_height):
voxel_id = 0
if wy < world_height - 1:
# creates caves
if (noise3(wx * 0.09, wy * 0.09, wz * 0.09) > 0 and noise2(wx * 0.1, wz * 0.1) * 3 + 3 < wy < world_height - 10):
voxel_id = 0
else:
voxel_id = STONE
else:
rng = int(7 * random())
ry = wy - rng
if SNOW_LVL <= ry < world_height:
voxel_id = SNOW
elif STONE_LVL <= ry < SNOW_LVL:
voxel_id = STONE
elif DIRT_LVL <= ry < STONE_LVL:
voxel_id = DIRT
elif GRASS_LVL <= ry < DIRT_LVL:
voxel_id = GRASS
else:
voxel_id = SAND
voxels[get_index(x, y, z)] = voxel_id
# place trees
if wy < DIRT_LVL:
place_tree(voxels, x, y, z, voxel_id)
@njit
def place_tree(voxels, x, y, z, voxel_id):
rnd = random()
if voxel_id != GRASS or rnd > TREE_PROBABILITY:
return None
if y + TREE_HEIGHT >= CHUNK_SIZE:
return None
if x - TREE_H_WIDTH < 0 or x + TREE_H_WIDTH >= CHUNK_SIZE:
return None
if z - TREE_H_WIDTH < 0 or z + TREE_H_WIDTH >= CHUNK_SIZE:
return None
voxels[get_index(x, y, z)] = DIRT
# places leaves
m = 0
for n, iy in enumerate(range(TREE_H_HEIGHT, TREE_HEIGHT - 1)):
k = iy % 2
rng = int(random() * 2)
for ix in range(-TREE_H_WIDTH + m, TREE_H_WIDTH - m * rng):
for iz in range(-TREE_H_WIDTH + m * rng, TREE_H_WIDTH - m):
if (ix + iz) % 4:
voxels[get_index(x + ix + k, y + iy, z + iz + k)] = LEAVES
m += 1 if n > 0 else 3 if n > 1 else 0
# tree trunk
for iy in range(1, TREE_HEIGHT - 2):
voxels[get_index(x, y + iy, z)] = WOOD
# top of trunk
voxels[get_index(x, y + TREE_HEIGHT - 2, z)] = LEAVES