-
Notifications
You must be signed in to change notification settings - Fork 47
/
mountain.gd
94 lines (66 loc) · 2.91 KB
/
mountain.gd
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
extends Node3D
@export var top_colors : Gradient
@export var base_colors : Gradient
@export var heightmap_texture : NoiseTexture2D
@export var source_mesh : Mesh
@export var target_mesh_instance : MeshInstance3D
@onready var collision_shape_3d: CollisionShape3D= %StaticBody3D/CollisionShape3D
@onready var static_body_3d: StaticBody3D = %StaticBody3D
@onready var texture_size := heightmap_texture.get_size()
@onready var texture_image : Image
var mdt := MeshDataTool.new()
func _ready() -> void:
_generate()
func _input(event: InputEvent) -> void:
if event is InputEventMouseButton \
and event.button_index == MOUSE_BUTTON_LEFT \
and event.is_pressed():
_generate()
func _generate() -> void:
heightmap_texture.noise.frequency = randfn(0.0035, 0.0025)
heightmap_texture.noise.seed = randi()
await heightmap_texture.changed
texture_image = heightmap_texture.get_image()
_compute_heightmap()
var color_offset := randf()
target_mesh_instance.material_override.set_shader_parameter("top_color", top_colors.sample(color_offset))
target_mesh_instance.material_override.set_shader_parameter("base_color", base_colors.sample(color_offset))
func _sample_xy(uv: Vector2) -> float:
var pixel: Color = texture_image.get_pixel(
clamp(uv.x * texture_size.x, 0.0, texture_size.x - 1.0),
clamp(uv.y * texture_size.y, 0.0, texture_size.y - 1.0)
)
var pixel_luminance := pixel.get_luminance()
var uv_distance := uv.distance_to(Vector2(0.5, 0.5))
uv_distance = smoothstep(0.5, 0.0, uv_distance)
return pixel_luminance * uv_distance * 0.6
func _compute_heightmap() -> void:
var base_resolution := 46
var heightmap_shape := HeightMapShape3D.new()
heightmap_shape.map_width = base_resolution
heightmap_shape.map_depth = base_resolution
for p_index in base_resolution * base_resolution:
var x := (p_index % base_resolution) / float(base_resolution - 1.0)
var y: float = floor(p_index / float(base_resolution)) / float(base_resolution - 1.0)
var uv := Vector2(x,y)
var uv_distance := uv.distance_to(Vector2(0.5, 0.5))
var uv_mask := smoothstep(0.45, 1.0, uv_distance)
heightmap_shape.map_data[p_index] += base_resolution * _sample_xy(uv)
heightmap_shape.map_data[p_index] -= uv_mask * 100.0
collision_shape_3d.scale = Vector3.ONE * (1.0 / float(base_resolution))
collision_shape_3d.shape = heightmap_shape
var mesh := ArrayMesh.new()
mdt.create_from_surface(source_mesh, 0)
for vertex_index in mdt.get_vertex_count():
var vertex_normal := mdt.get_vertex_normal(vertex_index)
if vertex_normal.dot(Vector3.UP) < 0.1: continue
var vertex_uv := mdt.get_vertex_uv(vertex_index)
var vertex := mdt.get_vertex(vertex_index)
var sample := _sample_xy(vertex_uv)
vertex.y += sample
mdt.set_vertex(vertex_index, vertex)
mdt.commit_to_surface(mesh)
var surface_tool := SurfaceTool.new()
surface_tool.create_from(mesh, 0)
surface_tool.generate_normals()
target_mesh_instance.mesh = surface_tool.commit()