Optimized gradient parameters (passed as floats when possible)

This commit is contained in:
RodZill4 2020-03-26 08:56:55 +01:00
parent 8da80e163c
commit 9ab8820598
6 changed files with 89 additions and 46 deletions

View File

@ -114,15 +114,38 @@ func get_parameter(n : String):
else: else:
return get_parameter_def(n).default return get_parameter_def(n).default
class CustomGradientSorter:
static func compare(a, b) -> bool:
return a.pos < b.pos
func set_parameter(n : String, v) -> void: func set_parameter(n : String, v) -> void:
var old_value = parameters[n] if parameters.has(n) else null
parameters[n] = v parameters[n] = v
emit_signal("parameter_changed", n, v) emit_signal("parameter_changed", n, v)
if is_inside_tree(): if is_inside_tree():
var parameter_def : Dictionary = get_parameter_def(n) var parameter_def : Dictionary = get_parameter_def(n)
if parameter_def.has("type") and parameter_def.type == "float": if parameter_def.has("type"):
get_tree().call_group("preview", "on_float_parameter_changed", "p_o%s_%s" % [ str(get_instance_id()), n ], v) if parameter_def.type == "float":
else: var parameter_name = "p_o%s_%s" % [ str(get_instance_id()), n ]
source_changed(0) get_tree().call_group("preview", "on_float_parameter_changed", parameter_name, v)
return
elif parameter_def.type == "gradient":
if v.interpolation == old_value.interpolation && v.points.size() == old_value.points.size():
# convert from old format
for i in range(old_value.points.size()):
if old_value.points[i].has("v"):
var old = old_value.points[i]
old_value.points[i] = { pos=old.v, r=old.c.r, g=old.c.g, b=old.c.b, a=old.c.a }
old_value.points.sort_custom(CustomGradientSorter, "compare")
v.points.sort_custom(CustomGradientSorter, "compare")
for i in range(old_value.points.size()):
for f in [ "pos", "r", "g", "b", "a" ]:
if v.points[i][f] != old_value.points[i][f]:
var parameter_name = "p_o%s_%s_%d_%s" % [ str(get_instance_id()), n, i, f ]
get_tree().call_group("preview", "on_float_parameter_changed", parameter_name, v.points[i][f])
return
print("regenerating shader")
source_changed(0)
func notify_output_change(output_index : int) -> void: func notify_output_change(output_index : int) -> void:
var targets = get_targets(output_index) var targets = get_targets(output_index)

View File

@ -200,7 +200,7 @@ func subst(string : String, context : MMGenContext, uv : String = "") -> Diction
elif p.type == "color": elif p.type == "color":
value_string = "vec4(%.9f, %.9f, %.9f, %.9f)" % [ value.r, value.g, value.b, value.a ] value_string = "vec4(%.9f, %.9f, %.9f, %.9f)" % [ value.r, value.g, value.b, value.a ]
elif p.type == "gradient": elif p.type == "gradient":
value_string = genname+"_p_"+p.name+"_gradient_fct" value_string = genname+"_"+p.name+"_gradient_fct"
elif p.type == "boolean": elif p.type == "boolean":
value_string = "true" if value else "false" value_string = "true" if value else "false"
else: else:
@ -258,7 +258,10 @@ func _get_shader_code(uv : String, output_index : int, context : MMGenContext) -
if !(g is MMGradient): if !(g is MMGradient):
g = MMGradient.new() g = MMGradient.new()
g.deserialize(parameters[p.name]) g.deserialize(parameters[p.name])
rv.defs += g.get_shader(genname+"_p_"+p.name+"_gradient_fct") var params = g.get_shader_params(genname+"_"+p.name)
for sp in params.keys():
rv.defs += "uniform float "+sp+" = "+str(params[sp])+";\n"
rv.defs += g.get_shader(genname+"_"+p.name)
# Generate functions for inputs # Generate functions for inputs
if shader_model.has("inputs"): if shader_model.has("inputs"):
for i in range(shader_model.inputs.size()): for i in range(shader_model.inputs.size()):

View File

@ -33,6 +33,9 @@ func add_point(v, c) -> void:
func sort() -> void: func sort() -> void:
if !sorted: if !sorted:
points.sort_custom(CustomSorter, "compare") points.sort_custom(CustomSorter, "compare")
for i in range(points.size()-1):
if points[i].v+0.0000005 >= points[i+1].v:
points[i+1].v = points[i].v+0.000001
sorted = true sorted = true
func get_color(x) -> Color: func get_color(x) -> Color:
@ -52,69 +55,81 @@ func get_color(x) -> Color:
else: else:
return Color(0.0, 0.0, 0.0, 1.0) return Color(0.0, 0.0, 0.0, 1.0)
func get_shader_params(name) -> Dictionary:
sort()
var rv = {}
for i in range(points.size()):
rv["p_"+name+"_"+str(i)+"_pos"] = points[i].v
rv["p_"+name+"_"+str(i)+"_r"] = points[i].c.r
rv["p_"+name+"_"+str(i)+"_g"] = points[i].c.g
rv["p_"+name+"_"+str(i)+"_b"] = points[i].c.b
rv["p_"+name+"_"+str(i)+"_a"] = points[i].c.a
return rv
# get_color_in_shader # get_color_in_shader
func gcis(color) -> String: func gcis(color) -> String:
return "vec4(%.9f,%.9f,%.9f,%.9f)" % [color.r, color.g, color.b, color.a] return "vec4(%.9f,%.9f,%.9f,%.9f)" % [color.r, color.g, color.b, color.a]
func pv(name : String, i : int) -> String:
return "p_"+name+"_"+str(i)+"_pos"
func pc(name : String, i : int) -> String:
return "vec4(p_"+name+"_"+str(i)+"_r,p_"+name+"_"+str(i)+"_g,p_"+name+"_"+str(i)+"_b,p_"+name+"_"+str(i)+"_a)"
func get_shader(name) -> String: func get_shader(name) -> String:
sort() sort()
var shader var shader
shader = "vec4 "+name+"(float x) {\n" shader = "vec4 "+name+"_gradient_fct(float x) {\n"
match interpolation: match interpolation:
0: 0:
if points.size() > 0: if points.size() > 0:
shader += " if (x < %.9f) {\n" % (0.5*(points[0].v + points[1].v)) shader += " if (x < 0.5*(%s+%s)) {\n" % [ pv(name, 0), pv(name, 1) ]
shader += " return "+gcis(points[0].c)+";\n" shader += " return "+pc(name, 0)+";\n"
var s = points.size()-1 var s = points.size()-1
for i in range(s): for i in range(1, s):
if points[i+1].v-points[i].v > 0: shader += " } else if (x < 0.5*(%s+%s)) {\n" % [ pv(name, i), pv(name, i+1) ]
shader += " } else if (x < %.9f) {\n" % (0.5*(points[i].v + points[i+1].v)) shader += " return "+pc(name, i)+";\n"
shader += " return "+gcis(points[i].c)+";\n"
shader += " }\n" shader += " }\n"
shader += " return "+gcis(points[s].c)+";\n" shader += " return "+pc(name, s)+";\n"
else: else:
shader += " return vec4(0.0, 0.0, 0.0, 1.0);\n" shader += " return vec4(0.0, 0.0, 0.0, 1.0);\n"
1, 2: 1, 2:
if points.size() > 0: if points.size() > 0:
shader += " if (x < %.9f) {\n" % points[0].v shader += " if (x < %s) {\n" % pv(name, 0)
shader += " return "+gcis(points[0].c)+";\n" shader += " return "+pc(name, 0)+";\n"
var s = points.size()-1 var s = points.size()-1
for i in range(s): for i in range(s):
var p1mp0 = points[i+1].v-points[i].v shader += " } else if (x < %s) {\n" % pv(name, i+1)
if p1mp0 > 0: var function = "(" if interpolation == 1 else "0.5-0.5*cos(3.14159265359*"
shader += " } else if (x < %.9f) {\n" % points[i+1].v shader += " return mix(%s, %s, %s(x-%s)/(%s-%s)));\n" % [ pc(name, i), pc(name, i+1), function, pv(name, i), pv(name, i+1), pv(name, i) ]
var function = "(" if interpolation == 1 else "0.5-0.5*cos(3.14159265359*"
shader += " return mix(%s, %s, %s(x-%.9f)/%.9f));\n" % [ gcis(points[i].c), gcis(points[i+1].c), function, points[i].v, p1mp0 ]
shader += " }\n" shader += " }\n"
shader += " return "+gcis(points[s].c)+";\n" shader += " return "+pc(name, s)+";\n"
else: else:
shader += " return vec4(0.0, 0.0, 0.0, 1.0);\n" shader += " return vec4(0.0, 0.0, 0.0, 1.0);\n"
3: 3:
if points.size() > 0: if points.size() > 0:
shader += " if (x < %.9f) {\n" % points[0].v shader += " if (x < %s) {\n" % pv(name, 0)
shader += " return "+gcis(points[0].c)+";\n" shader += " return "+pc(name, 0)+";\n"
var s = points.size()-1 var s = points.size()-1
for i in range(s): for i in range(s):
var p1mp0 = points[i+1].v-points[i].v shader += " } else if (x < %s) {\n" % pv(name, i+1)
if p1mp0 > 0: var dx : String = "(x-%s)/(%s-%s)" % [ pv(name, i), pv(name, i+1), pv(name, i) ]
shader += " } else if (x < %.9f) {\n" % points[i+1].v var b : String = "mix(%s, %s, %s)" % [ pc(name, i), pc(name, i+1), dx ]
var dx : String = "(x-%.9f)/%.9f" % [ points[i].v, p1mp0 ] if i > 0:
var b : String = "mix(%s, %s, %s)" % [ gcis(points[i].c), gcis(points[i+1].c), dx ] var a : String = "mix(%s, %s, (x-%s)/(%s-%s))" % [ pc(name, i-1), pc(name, i), pv(name, i-1), pv(name, i), pv(name, i-1) ]
if i > 0 and points[i-1].v < points[i].v: if i < s-1:
var a : String = "mix(%s, %s, (x-%.9f)/%.9f)" % [ gcis(points[i-1].c), gcis(points[i].c), points[i-1].v, points[i].v-points[i-1].v ] var c : String = "mix(%s, %s, (x-%s)/(%s-%s))" % [ pc(name, i+1), pc(name, i+2), pv(name, i+1), pv(name, i+2), pv(name, i+1) ]
if i < s-1 and points[i+1].v < points[i+2].v: var ac : String = "mix("+a+", "+c+", 0.5-0.5*cos(3.14159265359*"+dx+"))"
var c : String = "mix(%s, %s, (x-%.9f)/%.9f)" % [ gcis(points[i+1].c), gcis(points[i+2].c), points[i+1].v, points[i+2].v-points[i+1].v ] shader += " return 0.5*("+b+" + "+ac+");\n"
var ac : String = "mix("+a+", "+c+", 0.5-0.5*cos(3.14159265359*"+dx+"))"
shader += " return 0.5*("+b+" + "+ac+");\n"
else:
shader += " return mix("+a+", "+b+", 0.5+0.5*"+dx+");\n"
elif i < s-1 and points[i+1].v < points[i+2].v:
var c : String = "mix(%s, %s, (x-%.9f)/%.9f)" % [ gcis(points[i+1].c), gcis(points[i+2].c), points[i+1].v, points[i+2].v-points[i+1].v ]
shader += " return mix("+c+", "+b+", 1.0-0.5*"+dx+");\n"
else: else:
shader += " return "+b+";\n" shader += " return mix("+a+", "+b+", 0.5+0.5*"+dx+");\n"
elif i < s-1:
var c : String = "mix(%s, %s, (x-%s)/(%s-%s))" % [ pc(name, i+1), pc(name, i+2), pv(name, i+1), pv(name, i+2), pv(name, i+1) ]
shader += " return mix("+c+", "+b+", 1.0-0.5*"+dx+");\n"
else:
shader += " return "+b+";\n"
shader += " }\n" shader += " }\n"
shader += " return "+gcis(points[s].c)+";\n" shader += " return "+pc(name, s)+";\n"
else: else:
shader += " return vec4(0.0, 0.0, 0.0, 1.0);\n" shader += " return vec4(0.0, 0.0, 0.0, 1.0);\n"
_: _:

View File

@ -109,7 +109,6 @@ func _on_gradient_changed(new_gradient, variable) -> void:
ignore_parameter_change = variable ignore_parameter_change = variable
generator.set_parameter(variable, MMType.serialize_value(new_gradient)) generator.set_parameter(variable, MMType.serialize_value(new_gradient))
ignore_parameter_change = "" ignore_parameter_change = ""
update_shaders()
func create_parameter_control(p : Dictionary) -> Control: func create_parameter_control(p : Dictionary) -> Control:
var control = null var control = null

View File

@ -46,7 +46,7 @@ func on_parameter_changed(n : String, v) -> void:
var p = generator.get_parameter_def(n) var p = generator.get_parameter_def(n)
if p.has("type"): if p.has("type"):
match p.type: match p.type:
"float": "float", "gradient":
pass pass
_: _:
set_generator(generator, output) set_generator(generator, output)

View File

@ -147,8 +147,11 @@ func get_gradient_color(x) -> Color:
func update_shader() -> void: func update_shader() -> void:
var shader var shader
shader = "shader_type canvas_item;\n" shader = "shader_type canvas_item;\n"
shader += value.get_shader("gradient") var params = value.get_shader_params("")
shader += "void fragment() { COLOR = gradient(UV.x); }" for sp in params.keys():
shader += "uniform float "+sp+" = "+str(params[sp])+";\n"
shader += value.get_shader("")
shader += "void fragment() { COLOR = _gradient_fct(UV.x); }"
$Gradient.material.shader.set_code(shader) $Gradient.material.shader.set_code(shader)
emit_signal("updated", value) emit_signal("updated", value)