557 lines
14 KiB
GDScript3
557 lines
14 KiB
GDScript3
|
extends Node
|
||
|
|
||
|
const Painter = preload("./painter.gd")
|
||
|
const HTerrain = preload("../../hterrain.gd")
|
||
|
const HTerrainData = preload("../../hterrain_data.gd")
|
||
|
const Logger = preload("../../util/logger.gd")
|
||
|
|
||
|
const RaiseShader = preload("./shaders/raise.shader")
|
||
|
const SmoothShader = preload("./shaders/smooth.shader")
|
||
|
const LevelShader = preload("./shaders/level.shader")
|
||
|
const FlattenShader = preload("./shaders/flatten.shader")
|
||
|
const ErodeShader = preload("./shaders/erode.shader")
|
||
|
const Splat4Shader = preload("./shaders/splat4.shader")
|
||
|
const Splat16Shader = preload("./shaders/splat16.shader")
|
||
|
const SplatIndexedShader = preload("./shaders/splat_indexed.shader")
|
||
|
const ColorShader = preload("./shaders/color.shader")
|
||
|
const AlphaShader = preload("./shaders/alpha.shader")
|
||
|
|
||
|
const MODE_RAISE = 0
|
||
|
const MODE_LOWER = 1
|
||
|
const MODE_SMOOTH = 2
|
||
|
const MODE_FLATTEN = 3
|
||
|
const MODE_SPLAT = 4
|
||
|
const MODE_COLOR = 5
|
||
|
const MODE_MASK = 6
|
||
|
const MODE_DETAIL = 7
|
||
|
const MODE_LEVEL = 8
|
||
|
const MODE_ERODE = 9
|
||
|
const MODE_COUNT = 10
|
||
|
|
||
|
class ModifiedMap:
|
||
|
var map_type := 0
|
||
|
var map_index := 0
|
||
|
var painter_index := 0
|
||
|
|
||
|
signal changed
|
||
|
|
||
|
var _painters := []
|
||
|
|
||
|
var _brush_size := 32
|
||
|
var _opacity := 1.0
|
||
|
var _color := Color(1, 0, 0, 1)
|
||
|
var _mask_flag := false
|
||
|
var _mode := MODE_RAISE
|
||
|
var _flatten_height := 0.0
|
||
|
var _detail_index := 0
|
||
|
var _detail_density := 1.0
|
||
|
var _texture_index := 0
|
||
|
var _slope_limit_low_angle := 0.0
|
||
|
var _slope_limit_high_angle := PI / 2.0
|
||
|
|
||
|
var _modified_maps := []
|
||
|
var _terrain : HTerrain
|
||
|
var _logger = Logger.get_for(self)
|
||
|
|
||
|
|
||
|
func _init():
|
||
|
for i in 4:
|
||
|
var p = Painter.new()
|
||
|
# The name is just for debugging
|
||
|
p.set_name(str("Painter", i))
|
||
|
p.set_brush_size(_brush_size)
|
||
|
p.connect("texture_region_changed", self, "_on_painter_texture_region_changed", [i])
|
||
|
add_child(p)
|
||
|
_painters.append(p)
|
||
|
|
||
|
|
||
|
func get_brush_size() -> int:
|
||
|
return _brush_size
|
||
|
|
||
|
|
||
|
func set_brush_size(s: int):
|
||
|
if _brush_size == s:
|
||
|
return
|
||
|
_brush_size = s
|
||
|
for p in _painters:
|
||
|
p.set_brush_size(_brush_size)
|
||
|
emit_signal("changed")
|
||
|
|
||
|
|
||
|
func set_brush_texture(texture: Texture):
|
||
|
for p in _painters:
|
||
|
p.set_brush_texture(texture)
|
||
|
|
||
|
|
||
|
func get_opacity() -> float:
|
||
|
return _opacity
|
||
|
|
||
|
|
||
|
func set_opacity(opacity: float):
|
||
|
_opacity = opacity
|
||
|
|
||
|
|
||
|
func set_flatten_height(h: float):
|
||
|
if h == _flatten_height:
|
||
|
return
|
||
|
_flatten_height = h
|
||
|
emit_signal("changed")
|
||
|
|
||
|
|
||
|
func get_flatten_height() -> float:
|
||
|
return _flatten_height
|
||
|
|
||
|
|
||
|
func set_color(c: Color):
|
||
|
_color = c
|
||
|
|
||
|
|
||
|
func get_color() -> Color:
|
||
|
return _color
|
||
|
|
||
|
|
||
|
func set_mask_flag(m: bool):
|
||
|
_mask_flag = m
|
||
|
|
||
|
|
||
|
func get_mask_flag() -> bool:
|
||
|
return _mask_flag
|
||
|
|
||
|
|
||
|
func set_detail_density(d: float):
|
||
|
_detail_density = clamp(d, 0.0, 1.0)
|
||
|
|
||
|
|
||
|
func get_detail_density() -> float:
|
||
|
return _detail_density
|
||
|
|
||
|
|
||
|
func set_detail_index(di: int):
|
||
|
_detail_index = di
|
||
|
|
||
|
|
||
|
func set_texture_index(i: int):
|
||
|
_texture_index = i
|
||
|
|
||
|
|
||
|
func get_texture_index() -> int:
|
||
|
return _texture_index
|
||
|
|
||
|
|
||
|
func get_slope_limit_low_angle() -> float:
|
||
|
return _slope_limit_low_angle
|
||
|
|
||
|
|
||
|
func get_slope_limit_high_angle() -> float:
|
||
|
return _slope_limit_high_angle
|
||
|
|
||
|
|
||
|
func set_slope_limit_angles(low: float, high: float):
|
||
|
_slope_limit_low_angle = low
|
||
|
_slope_limit_high_angle = high
|
||
|
|
||
|
|
||
|
func is_operation_pending() -> bool:
|
||
|
for p in _painters:
|
||
|
if p.is_operation_pending():
|
||
|
return true
|
||
|
return false
|
||
|
|
||
|
|
||
|
func has_modified_chunks() -> bool:
|
||
|
for p in _painters:
|
||
|
if p.has_modified_chunks():
|
||
|
return true
|
||
|
return false
|
||
|
|
||
|
|
||
|
func get_undo_chunk_size() -> int:
|
||
|
return Painter.UNDO_CHUNK_SIZE
|
||
|
|
||
|
|
||
|
func commit() -> Dictionary:
|
||
|
assert(_terrain.get_data() != null)
|
||
|
var terrain_data = _terrain.get_data()
|
||
|
assert(not terrain_data.is_locked())
|
||
|
|
||
|
var changes := []
|
||
|
var chunk_positions : Array
|
||
|
|
||
|
for mm in _modified_maps:
|
||
|
var painter : Painter = _painters[mm.painter_index]
|
||
|
var info := painter.commit()
|
||
|
|
||
|
# Note, positions are always the same for each map
|
||
|
chunk_positions = info.chunk_positions
|
||
|
|
||
|
changes.append({
|
||
|
"map_type": mm.map_type,
|
||
|
"map_index": mm.map_index,
|
||
|
"chunk_initial_datas": info.chunk_initial_datas,
|
||
|
"chunk_final_datas": info.chunk_final_datas
|
||
|
})
|
||
|
|
||
|
var cs := get_undo_chunk_size()
|
||
|
for pos in info.chunk_positions:
|
||
|
var rect = Rect2(pos * cs, Vector2(cs, cs))
|
||
|
# This will update vertical bounds and notify normal map baker,
|
||
|
# since the latter updates out of order for preview
|
||
|
terrain_data.notify_region_change(rect, mm.map_type, mm.map_index, false, true)
|
||
|
|
||
|
assert(not has_modified_chunks())
|
||
|
|
||
|
return {
|
||
|
"chunk_positions": chunk_positions,
|
||
|
"maps": changes
|
||
|
}
|
||
|
|
||
|
|
||
|
func set_mode(mode: int):
|
||
|
assert(mode >= 0 and mode < MODE_COUNT)
|
||
|
_mode = mode
|
||
|
|
||
|
|
||
|
func get_mode() -> int:
|
||
|
return _mode
|
||
|
|
||
|
|
||
|
func set_terrain(terrain: HTerrain):
|
||
|
if terrain == _terrain:
|
||
|
return
|
||
|
_terrain = terrain
|
||
|
# It's important to release resources here,
|
||
|
# otherwise Godot keeps modified terrain maps in memory and "reloads" them like that
|
||
|
# next time we reopen the scene, even if we didn't save it
|
||
|
for p in _painters:
|
||
|
p.set_image(null, null)
|
||
|
p.clear_brush_shader_params()
|
||
|
|
||
|
|
||
|
# This may be called from an `_input` callback
|
||
|
func paint_input(position: Vector2):
|
||
|
assert(_terrain.get_data() != null)
|
||
|
var data = _terrain.get_data()
|
||
|
assert(not data.is_locked())
|
||
|
|
||
|
_modified_maps.clear()
|
||
|
|
||
|
match _mode:
|
||
|
MODE_RAISE:
|
||
|
_paint_height(data, position, 1.0)
|
||
|
|
||
|
MODE_LOWER:
|
||
|
_paint_height(data, position, -1.0)
|
||
|
|
||
|
MODE_SMOOTH:
|
||
|
_paint_smooth(data, position)
|
||
|
|
||
|
MODE_FLATTEN:
|
||
|
_paint_flatten(data, position)
|
||
|
|
||
|
MODE_LEVEL:
|
||
|
_paint_level(data, position)
|
||
|
|
||
|
MODE_ERODE:
|
||
|
_paint_erode(data, position)
|
||
|
|
||
|
MODE_SPLAT:
|
||
|
# TODO Properly support what happens when painting outside of supported index
|
||
|
# var supported_slots_count := terrain.get_cached_ground_texture_slot_count()
|
||
|
# if _texture_index >= supported_slots_count:
|
||
|
# _logger.debug("Painting out of range of supported texture slots: {0}/{1}" \
|
||
|
# .format([_texture_index, supported_slots_count]))
|
||
|
# return
|
||
|
if _terrain.is_using_indexed_splatmap():
|
||
|
_paint_splat_indexed(data, position)
|
||
|
else:
|
||
|
var splatmap_count := _terrain.get_used_splatmaps_count()
|
||
|
match splatmap_count:
|
||
|
1:
|
||
|
_paint_splat4(data, position)
|
||
|
4:
|
||
|
_paint_splat16(data, position)
|
||
|
|
||
|
MODE_COLOR:
|
||
|
_paint_color(data, position)
|
||
|
|
||
|
MODE_MASK:
|
||
|
_paint_mask(data, position)
|
||
|
|
||
|
MODE_DETAIL:
|
||
|
_paint_detail(data, position)
|
||
|
|
||
|
_:
|
||
|
_logger.error("Unknown mode {0}".format([_mode]))
|
||
|
|
||
|
assert(len(_modified_maps) > 0)
|
||
|
|
||
|
|
||
|
func _on_painter_texture_region_changed(rect: Rect2, painter_index: int):
|
||
|
var data = _terrain.get_data()
|
||
|
if data == null:
|
||
|
return
|
||
|
for mm in _modified_maps:
|
||
|
if mm.painter_index == painter_index:
|
||
|
# This will tell auto-baked maps to update (like normals).
|
||
|
data.notify_region_change(rect, mm.map_type, mm.map_index, false, false)
|
||
|
break
|
||
|
|
||
|
|
||
|
func _paint_height(data: HTerrainData, position: Vector2, factor: float):
|
||
|
var image = data.get_image(HTerrainData.CHANNEL_HEIGHT)
|
||
|
var texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0, true)
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_HEIGHT
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
# When using sculpting tools, make it dependent on brush size
|
||
|
var raise_strength := 10.0 + float(_brush_size)
|
||
|
var delta := factor * _opacity * (2.0 / 60.0) * raise_strength
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
p.set_brush_shader(RaiseShader)
|
||
|
p.set_brush_shader_param("u_factor", delta)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_smooth(data: HTerrainData, position: Vector2):
|
||
|
var image = data.get_image(HTerrainData.CHANNEL_HEIGHT)
|
||
|
var texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0, true)
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_HEIGHT
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
p.set_brush_shader(SmoothShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity * (10.0 / 60.0))
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_flatten(data: HTerrainData, position: Vector2):
|
||
|
var image = data.get_image(HTerrainData.CHANNEL_HEIGHT)
|
||
|
var texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0, true)
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_HEIGHT
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
p.set_brush_shader(FlattenShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_flatten_value", _flatten_height)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_level(data: HTerrainData, position: Vector2):
|
||
|
var image = data.get_image(HTerrainData.CHANNEL_HEIGHT)
|
||
|
var texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0, true)
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_HEIGHT
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
p.set_brush_shader(LevelShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity * (10.0 / 60.0))
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_erode(data: HTerrainData, position: Vector2):
|
||
|
var image = data.get_image(HTerrainData.CHANNEL_HEIGHT)
|
||
|
var texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0, true)
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_HEIGHT
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
p.set_brush_shader(ErodeShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_splat4(data: HTerrainData, position: Vector2):
|
||
|
var image = data.get_image(HTerrainData.CHANNEL_SPLAT)
|
||
|
var texture = data.get_texture(HTerrainData.CHANNEL_SPLAT, 0, true)
|
||
|
var heightmap_texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0)
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_SPLAT
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
var splat = Color(0.0, 0.0, 0.0, 0.0)
|
||
|
splat[_texture_index] = 1.0;
|
||
|
p.set_brush_shader(Splat4Shader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_splat", splat)
|
||
|
p.set_brush_shader_param("u_normal_min_y", cos(_slope_limit_high_angle))
|
||
|
p.set_brush_shader_param("u_normal_max_y", cos(_slope_limit_low_angle) + 0.001)
|
||
|
p.set_brush_shader_param("u_heightmap", heightmap_texture)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_splat_indexed(data: HTerrainData, position: Vector2):
|
||
|
var map_types = [
|
||
|
HTerrainData.CHANNEL_SPLAT_INDEX,
|
||
|
HTerrainData.CHANNEL_SPLAT_WEIGHT
|
||
|
]
|
||
|
_modified_maps = []
|
||
|
|
||
|
var textures = []
|
||
|
for mode in 2:
|
||
|
textures.append(data.get_texture(map_types[mode], 0, true))
|
||
|
|
||
|
for mode in 2:
|
||
|
var image = data.get_image(map_types[mode])
|
||
|
|
||
|
var mm = ModifiedMap.new()
|
||
|
mm.map_type = map_types[mode]
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = mode
|
||
|
_modified_maps.append(mm)
|
||
|
|
||
|
var p : Painter = _painters[mode]
|
||
|
|
||
|
p.set_brush_shader(SplatIndexedShader)
|
||
|
p.set_brush_shader_param("u_mode", mode)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_index_map", textures[0])
|
||
|
p.set_brush_shader_param("u_weight_map", textures[1])
|
||
|
p.set_brush_shader_param("u_texture_index", _texture_index)
|
||
|
p.set_image(image, textures[mode])
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_splat16(data: HTerrainData, position: Vector2):
|
||
|
# Make sure required maps are present
|
||
|
while data.get_map_count(HTerrainData.CHANNEL_SPLAT) < 4:
|
||
|
data._edit_add_map(HTerrainData.CHANNEL_SPLAT)
|
||
|
|
||
|
var splats := []
|
||
|
for i in 4:
|
||
|
splats.append(Color(0.0, 0.0, 0.0, 0.0))
|
||
|
splats[_texture_index / 4][_texture_index % 4] = 1.0
|
||
|
|
||
|
var textures := []
|
||
|
for i in 4:
|
||
|
textures.append(data.get_texture(HTerrainData.CHANNEL_SPLAT, i, true))
|
||
|
|
||
|
var heightmap_texture = data.get_texture(HTerrainData.CHANNEL_HEIGHT, 0)
|
||
|
|
||
|
for i in 4:
|
||
|
var image : Image = data.get_image(HTerrainData.CHANNEL_SPLAT, i)
|
||
|
var texture : Texture = textures[i]
|
||
|
|
||
|
var mm := ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_SPLAT
|
||
|
mm.map_index = i
|
||
|
mm.painter_index = i
|
||
|
_modified_maps.append(mm)
|
||
|
|
||
|
var p : Painter = _painters[i]
|
||
|
|
||
|
var other_splatmaps = []
|
||
|
for tex in textures:
|
||
|
if tex != texture:
|
||
|
other_splatmaps.append(tex)
|
||
|
|
||
|
p.set_brush_shader(Splat16Shader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_splat", splats[i])
|
||
|
p.set_brush_shader_param("u_other_splatmap_1", other_splatmaps[0])
|
||
|
p.set_brush_shader_param("u_other_splatmap_2", other_splatmaps[1])
|
||
|
p.set_brush_shader_param("u_other_splatmap_3", other_splatmaps[2])
|
||
|
p.set_brush_shader_param("u_normal_min_y", cos(_slope_limit_high_angle))
|
||
|
p.set_brush_shader_param("u_normal_max_y", cos(_slope_limit_low_angle) + 0.001)
|
||
|
p.set_brush_shader_param("u_heightmap", heightmap_texture)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_color(data: HTerrainData, position: Vector2):
|
||
|
var image := data.get_image(HTerrainData.CHANNEL_COLOR)
|
||
|
var texture := data.get_texture(HTerrainData.CHANNEL_COLOR, 0, true)
|
||
|
|
||
|
var mm := ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_COLOR
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
# There was a problem with painting colors because of sRGB
|
||
|
# https://github.com/Zylann/godot_heightmap_plugin/issues/17#issuecomment-734001879
|
||
|
|
||
|
p.set_brush_shader(ColorShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_color", _color)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_mask(data: HTerrainData, position: Vector2):
|
||
|
var image := data.get_image(HTerrainData.CHANNEL_COLOR)
|
||
|
var texture := data.get_texture(HTerrainData.CHANNEL_COLOR, 0, true)
|
||
|
|
||
|
var mm := ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_COLOR
|
||
|
mm.map_index = 0
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
|
||
|
p.set_brush_shader(AlphaShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_value", 1.0 if _mask_flag else 0.0)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|
||
|
|
||
|
|
||
|
func _paint_detail(data: HTerrainData, position: Vector2):
|
||
|
var image := data.get_image(HTerrainData.CHANNEL_DETAIL, _detail_index)
|
||
|
var texture := data.get_texture(HTerrainData.CHANNEL_DETAIL, _detail_index, true)
|
||
|
|
||
|
var mm := ModifiedMap.new()
|
||
|
mm.map_type = HTerrainData.CHANNEL_DETAIL
|
||
|
mm.map_index = _detail_index
|
||
|
mm.painter_index = 0
|
||
|
_modified_maps = [mm]
|
||
|
|
||
|
var p : Painter = _painters[0]
|
||
|
var c := Color(_detail_density, _detail_density, _detail_density, 1.0)
|
||
|
|
||
|
# TODO Don't use this shader
|
||
|
p.set_brush_shader(ColorShader)
|
||
|
p.set_brush_shader_param("u_factor", _opacity)
|
||
|
p.set_brush_shader_param("u_color", c)
|
||
|
p.set_image(image, texture)
|
||
|
p.paint_input(position)
|