2020-05-29 00:16:44 +00:00
|
|
|
extends Node
|
|
|
|
|
|
|
|
|
2020-06-04 11:11:24 +00:00
|
|
|
var drawer := Drawer.new()
|
2020-05-29 00:16:44 +00:00
|
|
|
var mouse_press_pixels := [] # Cleared after mouse release
|
|
|
|
var mouse_press_pressure_values := [] # Cleared after mouse release
|
|
|
|
|
|
|
|
|
2020-06-04 11:11:24 +00:00
|
|
|
func reset() -> void:
|
|
|
|
drawer.reset()
|
|
|
|
mouse_press_pixels.clear()
|
|
|
|
mouse_press_pressure_values.clear()
|
|
|
|
|
|
|
|
|
|
|
|
func draw_pixel_blended(sprite : Image, pos : Vector2, color : Color, pen_pressure : float, current_mouse_button := -1, current_action := -1) -> void:
|
2020-06-05 13:06:59 +00:00
|
|
|
var x_min = Global.current_project.x_min
|
|
|
|
var x_max = Global.current_project.x_max
|
|
|
|
var y_min = Global.current_project.y_min
|
|
|
|
var y_max = Global.current_project.y_max
|
2020-06-10 01:17:39 +00:00
|
|
|
|
2020-06-09 18:13:34 +00:00
|
|
|
# #Check if Tiling is enabled and whether mouse is in TilingPreviews
|
|
|
|
if Global.tile_mode and point_in_rectangle(pos,Vector2( - Global.current_project.size.x - 1 , - Global.current_project.size.y -1 ), Vector2(2 * Global.current_project.size.x, 2 * Global.current_project.size.y)):
|
|
|
|
pos = pos.posmodv(Global.current_project.size)
|
2020-06-10 01:17:39 +00:00
|
|
|
|
2020-06-04 15:20:56 +00:00
|
|
|
if !point_in_rectangle(pos, Vector2(x_min - 1, y_min - 1), Vector2(x_max, y_max)):
|
2020-05-31 23:49:54 +00:00
|
|
|
return
|
|
|
|
|
2020-05-31 23:32:44 +00:00
|
|
|
var pos_floored := pos.floor()
|
|
|
|
var current_pixel_color = sprite.get_pixelv(pos)
|
2020-05-31 23:49:54 +00:00
|
|
|
var saved_pixel_index := mouse_press_pixels.find(pos_floored)
|
2020-05-31 23:32:44 +00:00
|
|
|
if current_action == Global.Tools.PENCIL && color.a < 1:
|
|
|
|
color = blend_colors(color, current_pixel_color)
|
|
|
|
|
2020-05-31 23:49:54 +00:00
|
|
|
if current_pixel_color != color && (saved_pixel_index == -1 || pen_pressure > mouse_press_pressure_values[saved_pixel_index]):
|
2020-05-31 23:32:44 +00:00
|
|
|
if current_action == Global.Tools.LIGHTENDARKEN:
|
|
|
|
var ld : int = Global.ld_modes[current_mouse_button]
|
|
|
|
var ld_amount : float = Global.ld_amounts[current_mouse_button]
|
|
|
|
if ld == Global.Lighten_Darken_Mode.LIGHTEN:
|
|
|
|
color = current_pixel_color.lightened(ld_amount)
|
|
|
|
else:
|
|
|
|
color = current_pixel_color.darkened(ld_amount)
|
|
|
|
|
|
|
|
if saved_pixel_index == -1:
|
|
|
|
mouse_press_pixels.append(pos_floored)
|
|
|
|
mouse_press_pressure_values.append(pen_pressure)
|
|
|
|
else:
|
|
|
|
mouse_press_pressure_values[saved_pixel_index] = pen_pressure
|
|
|
|
drawer.set_pixel(sprite, pos, color)
|
|
|
|
|
|
|
|
|
2020-05-31 15:46:47 +00:00
|
|
|
func draw_brush(sprite : Image, pos : Vector2, color : Color, current_mouse_button : int, pen_pressure : float, current_action := -1) -> void:
|
2020-05-29 21:46:58 +00:00
|
|
|
if Global.can_draw && Global.has_focus:
|
2020-06-05 13:06:59 +00:00
|
|
|
var x_min = Global.current_project.x_min
|
|
|
|
var x_max = Global.current_project.x_max
|
|
|
|
var y_min = Global.current_project.y_min
|
|
|
|
var y_max = Global.current_project.y_max
|
2020-05-29 21:46:58 +00:00
|
|
|
|
|
|
|
if Global.pressure_sensitivity_mode == Global.Pressure_Sensitivity.ALPHA:
|
2020-05-31 13:04:33 +00:00
|
|
|
if current_action == Global.Tools.PENCIL:
|
2020-05-29 21:46:58 +00:00
|
|
|
color.a *= pen_pressure
|
2020-05-31 13:04:33 +00:00
|
|
|
elif current_action == Global.Tools.ERASER: # This is not working
|
2020-05-29 21:46:58 +00:00
|
|
|
color.a *= (1.0 - pen_pressure)
|
|
|
|
|
2020-05-31 23:32:44 +00:00
|
|
|
var brush_size : int = Global.brush_sizes[current_mouse_button]
|
|
|
|
var brush_type : int = Global.current_brush_types[current_mouse_button]
|
2020-05-29 21:46:58 +00:00
|
|
|
|
2020-05-31 23:32:44 +00:00
|
|
|
var horizontal_mirror : bool = Global.horizontal_mirror[current_mouse_button]
|
|
|
|
var vertical_mirror : bool = Global.vertical_mirror[current_mouse_button]
|
2020-06-04 11:11:24 +00:00
|
|
|
var pixel_perfect : bool = Global.pixel_perfect[current_mouse_button]
|
2020-06-04 15:20:56 +00:00
|
|
|
|
2020-06-04 11:11:24 +00:00
|
|
|
drawer.pixel_perfect = pixel_perfect if brush_size == 1 else false
|
|
|
|
drawer.h_mirror = horizontal_mirror
|
|
|
|
drawer.v_mirror = vertical_mirror
|
2020-06-04 15:20:56 +00:00
|
|
|
|
2020-05-31 13:04:33 +00:00
|
|
|
if brush_type == Global.Brush_Types.PIXEL || current_action == Global.Tools.LIGHTENDARKEN:
|
2020-06-09 18:13:34 +00:00
|
|
|
var start_pos_x = floor(pos.x - (brush_size >> 1))
|
|
|
|
var start_pos_y = floor(pos.y - (brush_size >> 1))
|
|
|
|
var end_pos_x = floor(start_pos_x + brush_size)
|
|
|
|
var end_pos_y = floor(start_pos_y + brush_size)
|
2020-05-31 23:32:44 +00:00
|
|
|
|
2020-05-29 21:46:58 +00:00
|
|
|
for cur_pos_x in range(start_pos_x, end_pos_x):
|
2020-06-04 15:20:56 +00:00
|
|
|
for cur_pos_y in range(start_pos_y, end_pos_y):
|
2020-06-04 11:11:24 +00:00
|
|
|
draw_pixel_blended(sprite, Vector2(cur_pos_x, cur_pos_y), color, pen_pressure, current_mouse_button, current_action)
|
2020-05-31 23:49:54 +00:00
|
|
|
Global.canvas.sprite_changed_this_frame = true
|
2020-05-29 21:46:58 +00:00
|
|
|
|
|
|
|
elif brush_type == Global.Brush_Types.CIRCLE || brush_type == Global.Brush_Types.FILLED_CIRCLE:
|
2020-05-31 23:32:44 +00:00
|
|
|
plot_circle(sprite, pos.x, pos.y, brush_size, color, brush_type == Global.Brush_Types.FILLED_CIRCLE)
|
2020-05-29 21:46:58 +00:00
|
|
|
Global.canvas.sprite_changed_this_frame = true
|
|
|
|
|
|
|
|
else:
|
2020-06-01 00:11:00 +00:00
|
|
|
var brush_index : int = Global.custom_brush_indexes[current_mouse_button]
|
2020-05-31 23:32:44 +00:00
|
|
|
var custom_brush_image : Image
|
|
|
|
if brush_type != Global.Brush_Types.RANDOM_FILE:
|
2020-06-05 14:21:35 +00:00
|
|
|
custom_brush_image = Global.brush_images[current_mouse_button]
|
2020-05-31 23:32:44 +00:00
|
|
|
else: # Handle random brush
|
|
|
|
var brush_button = Global.file_brush_container.get_child(brush_index + 3)
|
|
|
|
var random_index = randi() % brush_button.random_brushes.size()
|
|
|
|
custom_brush_image = Image.new()
|
|
|
|
custom_brush_image.copy_from(brush_button.random_brushes[random_index])
|
|
|
|
var custom_brush_size = custom_brush_image.get_size()
|
|
|
|
custom_brush_image.resize(custom_brush_size.x * brush_size, custom_brush_size.y * brush_size, Image.INTERPOLATE_NEAREST)
|
|
|
|
custom_brush_image = Global.blend_image_with_color(custom_brush_image, color, Global.interpolate_spinboxes[current_mouse_button].value / 100)
|
|
|
|
custom_brush_image.lock()
|
|
|
|
|
2020-05-29 21:46:58 +00:00
|
|
|
var custom_brush_size := custom_brush_image.get_size() - Vector2.ONE
|
|
|
|
pos = pos.floor()
|
2020-06-09 18:13:34 +00:00
|
|
|
# #Check if Tiling is enabled and whether mouse is in TilingPreviews
|
|
|
|
if Global.tile_mode and point_in_rectangle(pos,Vector2( - Global.current_project.size.x - 1 , - Global.current_project.size.y -1 ), Vector2(2 * Global.current_project.size.x, 2 * Global.current_project.size.y)):
|
|
|
|
pos = pos.posmodv(Global.current_project.size)
|
2020-06-10 01:17:39 +00:00
|
|
|
|
2020-05-29 21:46:58 +00:00
|
|
|
var dst := rectangle_center(pos, custom_brush_size)
|
|
|
|
var src_rect := Rect2(Vector2.ZERO, custom_brush_size + Vector2.ONE)
|
|
|
|
# Rectangle with the same size as the brush, but at cursor's position
|
|
|
|
var pos_rect := Rect2(dst, custom_brush_size + Vector2.ONE)
|
|
|
|
|
|
|
|
# The selection rectangle
|
|
|
|
# If there's no rectangle, the whole canvas is considered a selection
|
|
|
|
var selection_rect := Rect2()
|
2020-06-04 15:20:56 +00:00
|
|
|
selection_rect.position = Vector2(x_min, y_min)
|
|
|
|
selection_rect.end = Vector2(x_max, y_max)
|
2020-05-29 21:46:58 +00:00
|
|
|
# Intersection of the position rectangle and selection
|
|
|
|
var pos_rect_clipped := pos_rect.clip(selection_rect)
|
|
|
|
# If the size is 0, that means that the brush wasn't positioned inside the selection
|
|
|
|
if pos_rect_clipped.size == Vector2.ZERO:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Re-position src_rect and dst based on the clipped position
|
|
|
|
var pos_difference := (pos_rect.position - pos_rect_clipped.position).abs()
|
|
|
|
# Obviously, if pos_rect and pos_rect_clipped are the same, pos_difference is Vector2.ZERO
|
|
|
|
src_rect.position = pos_difference
|
|
|
|
dst += pos_difference
|
|
|
|
src_rect.end -= pos_rect.end - pos_rect_clipped.end
|
|
|
|
# If the selection rectangle is smaller than the brush, ...
|
|
|
|
# ... make sure pixels aren't being drawn outside the selection by adjusting src_rect's size
|
|
|
|
src_rect.size.x = min(src_rect.size.x, selection_rect.size.x)
|
|
|
|
src_rect.size.y = min(src_rect.size.y, selection_rect.size.y)
|
|
|
|
|
|
|
|
# Handle mirroring
|
2020-06-04 15:20:56 +00:00
|
|
|
var mirror_x = x_max + x_min - pos.x - (pos.x - dst.x)
|
|
|
|
var mirror_y = y_max + y_min - pos.y - (pos.y - dst.y)
|
2020-05-29 21:46:58 +00:00
|
|
|
if int(pos_rect_clipped.size.x) % 2 != 0:
|
|
|
|
mirror_x -= 1
|
|
|
|
if int(pos_rect_clipped.size.y) % 2 != 0:
|
|
|
|
mirror_y -= 1
|
|
|
|
# Use custom blend function cause of godot's issue #31124
|
|
|
|
if color.a > 0: # If it's the pencil
|
2020-05-31 23:32:44 +00:00
|
|
|
blend_rect(sprite, custom_brush_image, src_rect, dst)
|
2020-05-29 21:46:58 +00:00
|
|
|
if horizontal_mirror:
|
2020-05-31 23:32:44 +00:00
|
|
|
blend_rect(sprite, custom_brush_image, src_rect, Vector2(mirror_x, dst.y))
|
2020-05-29 21:46:58 +00:00
|
|
|
if vertical_mirror:
|
2020-05-31 23:32:44 +00:00
|
|
|
blend_rect(sprite, custom_brush_image, src_rect, Vector2(dst.x, mirror_y))
|
2020-05-29 21:46:58 +00:00
|
|
|
if horizontal_mirror && vertical_mirror:
|
2020-05-31 23:32:44 +00:00
|
|
|
blend_rect(sprite, custom_brush_image, src_rect, Vector2(mirror_x, mirror_y))
|
2020-05-29 21:46:58 +00:00
|
|
|
|
|
|
|
else: # if it's transparent - if it's the eraser
|
|
|
|
var custom_brush := Image.new()
|
2020-06-05 14:21:35 +00:00
|
|
|
if brush_type == Global.Brush_Types.CUSTOM:
|
|
|
|
custom_brush.copy_from(Global.current_project.brushes[brush_index])
|
|
|
|
else:
|
|
|
|
custom_brush.copy_from(Global.file_brushes[brush_index])
|
2020-05-29 21:46:58 +00:00
|
|
|
custom_brush_size = custom_brush.get_size()
|
|
|
|
custom_brush.resize(custom_brush_size.x * brush_size, custom_brush_size.y * brush_size, Image.INTERPOLATE_NEAREST)
|
|
|
|
var custom_brush_blended = Global.blend_image_with_color(custom_brush, color, 1)
|
|
|
|
|
|
|
|
sprite.blit_rect_mask(custom_brush_blended, custom_brush, src_rect, dst)
|
|
|
|
if horizontal_mirror:
|
|
|
|
sprite.blit_rect_mask(custom_brush_blended, custom_brush, src_rect, Vector2(mirror_x, dst.y))
|
|
|
|
if vertical_mirror:
|
|
|
|
sprite.blit_rect_mask(custom_brush_blended, custom_brush, src_rect, Vector2(dst.x, mirror_y))
|
|
|
|
if horizontal_mirror && vertical_mirror:
|
|
|
|
sprite.blit_rect_mask(custom_brush_blended, custom_brush, src_rect, Vector2(mirror_x, mirror_y))
|
|
|
|
|
|
|
|
sprite.lock()
|
|
|
|
Global.canvas.sprite_changed_this_frame = true
|
|
|
|
|
|
|
|
Global.canvas.previous_mouse_pos_for_lines = pos.floor() + Vector2(0.5, 0.5)
|
2020-06-04 20:20:20 +00:00
|
|
|
Global.canvas.previous_mouse_pos_for_lines.x = clamp(Global.canvas.previous_mouse_pos_for_lines.x, Global.canvas.location.x, Global.canvas.location.x + Global.current_project.size.x)
|
|
|
|
Global.canvas.previous_mouse_pos_for_lines.y = clamp(Global.canvas.previous_mouse_pos_for_lines.y, Global.canvas.location.y, Global.canvas.location.y + Global.current_project.size.y)
|
2020-05-29 21:46:58 +00:00
|
|
|
if Global.canvas.is_making_line:
|
2020-06-10 11:19:20 +00:00
|
|
|
Global.canvas.line_pos[0] = Global.canvas.previous_mouse_pos_for_lines
|
2020-05-29 21:46:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Bresenham's Algorithm
|
|
|
|
# Thanks to https://godotengine.org/qa/35276/tile-based-line-drawing-algorithm-efficiency
|
2020-05-31 15:46:47 +00:00
|
|
|
func fill_gaps(sprite : Image, end_pos : Vector2, start_pos : Vector2, color : Color, current_mouse_button : int, pen_pressure : float, current_action := -1) -> void:
|
2020-05-29 21:46:58 +00:00
|
|
|
var previous_mouse_pos_floored = start_pos.floor()
|
|
|
|
var mouse_pos_floored = end_pos.floor()
|
|
|
|
var dx := int(abs(mouse_pos_floored.x - previous_mouse_pos_floored.x))
|
|
|
|
var dy := int(-abs(mouse_pos_floored.y - previous_mouse_pos_floored.y))
|
|
|
|
var err := dx + dy
|
|
|
|
var e2 := err << 1 # err * 2
|
|
|
|
var sx = 1 if previous_mouse_pos_floored.x < mouse_pos_floored.x else -1
|
|
|
|
var sy = 1 if previous_mouse_pos_floored.y < mouse_pos_floored.y else -1
|
|
|
|
var x = previous_mouse_pos_floored.x
|
|
|
|
var y = previous_mouse_pos_floored.y
|
|
|
|
while !(x == mouse_pos_floored.x && y == mouse_pos_floored.y):
|
|
|
|
draw_brush(sprite, Vector2(x, y), color, current_mouse_button, pen_pressure, current_action)
|
|
|
|
e2 = err << 1
|
|
|
|
if e2 >= dy:
|
|
|
|
err += dy
|
|
|
|
x += sx
|
|
|
|
if e2 <= dx:
|
|
|
|
err += dx
|
|
|
|
y += sy
|
|
|
|
|
|
|
|
|
2020-05-29 00:16:44 +00:00
|
|
|
# Algorithm based on http://members.chello.at/easyfilter/bresenham.html
|
|
|
|
func plot_circle(sprite : Image, xm : int, ym : int, r : int, color : Color, fill := false) -> void:
|
|
|
|
var radius := r # Used later for filling
|
|
|
|
var x := -r
|
|
|
|
var y := 0
|
|
|
|
var err := 2 - r * 2 # II. Quadrant
|
|
|
|
while x < 0:
|
|
|
|
var quadrant_1 := Vector2(xm - x, ym + y)
|
|
|
|
var quadrant_2 := Vector2(xm - y, ym - x)
|
|
|
|
var quadrant_3 := Vector2(xm + x, ym - y)
|
|
|
|
var quadrant_4 := Vector2(xm + y, ym + x)
|
2020-05-31 23:49:54 +00:00
|
|
|
draw_pixel_blended(sprite, quadrant_1, color, Global.canvas.pen_pressure)
|
|
|
|
draw_pixel_blended(sprite, quadrant_2, color, Global.canvas.pen_pressure)
|
|
|
|
draw_pixel_blended(sprite, quadrant_3, color, Global.canvas.pen_pressure)
|
|
|
|
draw_pixel_blended(sprite, quadrant_4, color, Global.canvas.pen_pressure)
|
2020-05-29 00:16:44 +00:00
|
|
|
|
|
|
|
r = err
|
|
|
|
if r <= y:
|
|
|
|
y += 1
|
|
|
|
err += y * 2 + 1
|
|
|
|
if r > x || err > y:
|
|
|
|
x += 1
|
|
|
|
err += x * 2 + 1
|
|
|
|
|
|
|
|
if fill:
|
|
|
|
for j in range (-radius, radius + 1):
|
|
|
|
for i in range (-radius, radius + 1):
|
|
|
|
if i * i + j * j <= radius * radius:
|
|
|
|
var draw_pos := Vector2(i + xm, j + ym)
|
2020-05-31 23:49:54 +00:00
|
|
|
draw_pixel_blended(sprite, draw_pos, color, Global.canvas.pen_pressure)
|
2020-05-29 00:16:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Thanks to https://en.wikipedia.org/wiki/Flood_fill
|
|
|
|
func flood_fill(sprite : Image, pos : Vector2, target_color : Color, replace_color : Color) -> void:
|
2020-06-05 13:06:59 +00:00
|
|
|
var x_min = Global.current_project.x_min
|
|
|
|
var x_max = Global.current_project.x_max
|
|
|
|
var y_min = Global.current_project.y_min
|
|
|
|
var y_max = Global.current_project.y_max
|
2020-05-29 00:16:44 +00:00
|
|
|
pos = pos.floor()
|
|
|
|
var pixel = sprite.get_pixelv(pos)
|
|
|
|
if target_color == replace_color:
|
|
|
|
return
|
|
|
|
elif pixel != target_color:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
|
2020-06-04 15:20:56 +00:00
|
|
|
if !point_in_rectangle(pos, Vector2(x_min - 1, y_min - 1), Vector2(x_max, y_max)):
|
2020-05-29 00:16:44 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
var q = [pos]
|
|
|
|
for n in q:
|
|
|
|
# If the difference in colors is very small, break the loop (thanks @azagaya on GitHub!)
|
|
|
|
if target_color == replace_color:
|
|
|
|
break
|
|
|
|
var west : Vector2 = n
|
|
|
|
var east : Vector2 = n
|
2020-06-04 15:20:56 +00:00
|
|
|
while west.x >= x_min && sprite.get_pixelv(west) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
west += Vector2.LEFT
|
2020-06-04 15:20:56 +00:00
|
|
|
while east.x < x_max && sprite.get_pixelv(east) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
east += Vector2.RIGHT
|
|
|
|
for px in range(west.x + 1, east.x):
|
|
|
|
var p := Vector2(px, n.y)
|
|
|
|
# Draw
|
|
|
|
sprite.set_pixelv(p, replace_color)
|
|
|
|
replace_color = sprite.get_pixelv(p)
|
|
|
|
var north := p + Vector2.UP
|
|
|
|
var south := p + Vector2.DOWN
|
2020-06-04 15:20:56 +00:00
|
|
|
if north.y >= y_min && sprite.get_pixelv(north) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
q.append(north)
|
2020-06-04 15:20:56 +00:00
|
|
|
if south.y < y_max && sprite.get_pixelv(south) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
q.append(south)
|
|
|
|
|
|
|
|
Global.canvas.sprite_changed_this_frame = true
|
|
|
|
|
|
|
|
|
|
|
|
func pattern_fill(sprite : Image, pos : Vector2, pattern : Image, target_color : Color, var offset : Vector2) -> void:
|
2020-06-05 13:06:59 +00:00
|
|
|
var x_min = Global.current_project.x_min
|
|
|
|
var x_max = Global.current_project.x_max
|
|
|
|
var y_min = Global.current_project.y_min
|
|
|
|
var y_max = Global.current_project.y_max
|
2020-05-29 00:16:44 +00:00
|
|
|
pos = pos.floor()
|
2020-06-04 15:20:56 +00:00
|
|
|
if !point_in_rectangle(pos, Vector2(x_min - 1, y_min - 1), Vector2(x_max, y_max)):
|
2020-05-29 00:16:44 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
pattern.lock()
|
|
|
|
var pattern_size := pattern.get_size()
|
|
|
|
var q = [pos]
|
|
|
|
|
|
|
|
for n in q:
|
|
|
|
var west : Vector2 = n
|
|
|
|
var east : Vector2 = n
|
2020-06-04 15:20:56 +00:00
|
|
|
while west.x >= x_min && sprite.get_pixelv(west) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
west += Vector2.LEFT
|
2020-06-04 15:20:56 +00:00
|
|
|
while east.x < x_max && sprite.get_pixelv(east) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
east += Vector2.RIGHT
|
|
|
|
|
|
|
|
for px in range(west.x + 1, east.x):
|
|
|
|
var p := Vector2(px, n.y)
|
|
|
|
var xx : int = int(px + offset.x) % int(pattern_size.x)
|
|
|
|
var yy : int = int(n.y + offset.y) % int(pattern_size.y)
|
|
|
|
var pattern_color : Color = pattern.get_pixel(xx, yy)
|
|
|
|
if pattern_color == target_color:
|
|
|
|
continue
|
|
|
|
sprite.set_pixelv(p, pattern_color)
|
|
|
|
|
|
|
|
var north := p + Vector2.UP
|
|
|
|
var south := p + Vector2.DOWN
|
2020-06-04 15:20:56 +00:00
|
|
|
if north.y >= y_min && sprite.get_pixelv(north) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
q.append(north)
|
2020-06-04 15:20:56 +00:00
|
|
|
if south.y < y_max && sprite.get_pixelv(south) == target_color:
|
2020-05-29 00:16:44 +00:00
|
|
|
q.append(south)
|
|
|
|
|
|
|
|
pattern.unlock()
|
|
|
|
Global.canvas.sprite_changed_this_frame = true
|
|
|
|
|
|
|
|
|
|
|
|
func blend_colors(color_1 : Color, color_2 : Color) -> Color:
|
|
|
|
var color := Color()
|
|
|
|
color.a = color_1.a + color_2.a * (1 - color_1.a) # Blend alpha
|
|
|
|
if color.a != 0:
|
|
|
|
# Blend colors
|
|
|
|
color.r = (color_1.r * color_1.a + color_2.r * color_2.a * (1-color_1.a)) / color.a
|
|
|
|
color.g = (color_1.g * color_1.a + color_2.g * color_2.a * (1-color_1.a)) / color.a
|
|
|
|
color.b = (color_1.b * color_1.a + color_2.b * color_2.a * (1-color_1.a)) / color.a
|
|
|
|
return color
|
|
|
|
|
|
|
|
|
|
|
|
# Custom blend rect function, needed because Godot's issue #31124
|
2020-06-13 12:30:58 +00:00
|
|
|
# The issue has been solved in Godot 3.2.2 and the method will be removed
|
|
|
|
# once 3.2.2 stable is released
|
2020-05-29 00:16:44 +00:00
|
|
|
func blend_rect(bg : Image, brush : Image, src_rect : Rect2, dst : Vector2) -> void:
|
2020-06-13 12:30:58 +00:00
|
|
|
var godot_version = Engine.get_version_info()
|
|
|
|
if godot_version.major >= 3 and godot_version.minor >= 2 and godot_version.patch >= 2:
|
|
|
|
bg.blend_rect(brush, src_rect, dst)
|
|
|
|
return
|
|
|
|
|
2020-05-29 00:16:44 +00:00
|
|
|
var brush_size := brush.get_size()
|
|
|
|
var clipped_src_rect := Rect2(Vector2.ZERO, brush_size).clip(src_rect)
|
|
|
|
if clipped_src_rect.size.x <= 0 || clipped_src_rect.size.y <= 0:
|
|
|
|
return
|
|
|
|
var src_underscan := Vector2(min(0, src_rect.position.x), min(0, src_rect.position.y))
|
|
|
|
var dest_rect := Rect2(0, 0, bg.get_width(), bg.get_height()).clip(Rect2(dst - src_underscan, clipped_src_rect.size))
|
|
|
|
|
|
|
|
for x in range(0, dest_rect.size.x):
|
|
|
|
for y in range(0, dest_rect.size.y):
|
|
|
|
var src_x := clipped_src_rect.position.x + x;
|
|
|
|
var src_y := clipped_src_rect.position.y + y;
|
|
|
|
|
|
|
|
var dst_x := dest_rect.position.x + x;
|
|
|
|
var dst_y := dest_rect.position.y + y;
|
|
|
|
|
|
|
|
brush.lock()
|
|
|
|
var brush_color := brush.get_pixel(src_x, src_y)
|
|
|
|
var bg_color := bg.get_pixel(dst_x, dst_y)
|
|
|
|
var out_color := blend_colors(brush_color, bg_color)
|
|
|
|
if out_color.a != 0:
|
|
|
|
bg.set_pixel(dst_x, dst_y, out_color)
|
|
|
|
brush.unlock()
|
|
|
|
|
|
|
|
|
|
|
|
func scale3X(sprite : Image, tol : float = 50) -> Image:
|
|
|
|
var scaled = Image.new()
|
|
|
|
scaled.create(sprite.get_width()*3, sprite.get_height()*3, false, Image.FORMAT_RGBA8)
|
|
|
|
scaled.lock()
|
|
|
|
sprite.lock()
|
|
|
|
var a : Color
|
|
|
|
var b : Color
|
|
|
|
var c : Color
|
|
|
|
var d : Color
|
|
|
|
var e : Color
|
|
|
|
var f : Color
|
|
|
|
var g : Color
|
|
|
|
var h : Color
|
|
|
|
var i : Color
|
|
|
|
|
|
|
|
for x in range(1,sprite.get_width()-1):
|
|
|
|
for y in range(1,sprite.get_height()-1):
|
|
|
|
var xs : float = 3*x
|
|
|
|
var ys : float = 3*y
|
|
|
|
|
|
|
|
a = sprite.get_pixel(x-1,y-1)
|
|
|
|
b = sprite.get_pixel(x,y-1)
|
|
|
|
c = sprite.get_pixel(x+1,y-1)
|
|
|
|
d = sprite.get_pixel(x-1,y)
|
|
|
|
e = sprite.get_pixel(x,y)
|
|
|
|
f = sprite.get_pixel(x+1,y)
|
|
|
|
g = sprite.get_pixel(x-1,y+1)
|
|
|
|
h = sprite.get_pixel(x,y+1)
|
|
|
|
i = sprite.get_pixel(x+1,y+1)
|
|
|
|
|
|
|
|
var db : bool = similarColors(d, b, tol)
|
|
|
|
var dh : bool = similarColors(d, h, tol)
|
|
|
|
var bf : bool = similarColors(f, b, tol)
|
|
|
|
var ec : bool = similarColors(e, c, tol)
|
|
|
|
var ea : bool = similarColors(e, a, tol)
|
|
|
|
var fh : bool = similarColors(f, h, tol)
|
|
|
|
var eg : bool = similarColors(e, g, tol)
|
|
|
|
var ei : bool = similarColors(e, i, tol)
|
|
|
|
|
|
|
|
scaled.set_pixel(xs-1, ys-1, d if (db and !dh and !bf) else e )
|
|
|
|
scaled.set_pixel(xs, ys-1, b if (db and !dh and !bf and !ec) or
|
|
|
|
(bf and !db and !fh and !ea) else e)
|
|
|
|
scaled.set_pixel(xs+1, ys-1, f if (bf and !db and !fh) else e)
|
|
|
|
scaled.set_pixel(xs-1, ys, d if (dh and !fh and !db and !ea) or
|
|
|
|
(db and !dh and !bf and !eg) else e)
|
|
|
|
scaled.set_pixel(xs, ys, e);
|
|
|
|
scaled.set_pixel(xs+1, ys, f if (bf and !db and !fh and !ei) or
|
|
|
|
(fh and !bf and !dh and !ec) else e)
|
|
|
|
scaled.set_pixel(xs-1, ys+1, d if (dh and !fh and !db) else e)
|
|
|
|
scaled.set_pixel(xs, ys+1, h if (fh and !bf and !dh and !eg) or
|
|
|
|
(dh and !fh and !db and !ei) else e)
|
|
|
|
scaled.set_pixel(xs+1, ys+1, f if (fh and !bf and !dh) else e)
|
|
|
|
|
|
|
|
scaled.unlock()
|
|
|
|
sprite.unlock()
|
|
|
|
return scaled
|
|
|
|
|
|
|
|
|
|
|
|
func rotxel(sprite : Image, angle : float) -> void:
|
|
|
|
# If angle is simple, then nn rotation is the best
|
|
|
|
|
|
|
|
if angle == 0 || angle == PI/2 || angle == PI || angle == 2*PI:
|
|
|
|
nn_rotate(sprite, angle)
|
|
|
|
return
|
|
|
|
|
|
|
|
var aux : Image = Image.new()
|
|
|
|
aux.copy_from(sprite)
|
2020-06-01 00:11:00 +00:00
|
|
|
# warning-ignore:integer_division
|
|
|
|
# warning-ignore:integer_division
|
|
|
|
var center : Vector2 = Vector2(sprite.get_width() / 2, sprite.get_height() / 2)
|
2020-05-29 00:16:44 +00:00
|
|
|
var ox : int
|
|
|
|
var oy : int
|
|
|
|
var p : Color
|
|
|
|
aux.lock()
|
|
|
|
sprite.lock()
|
|
|
|
for x in range(sprite.get_width()):
|
|
|
|
for y in range(sprite.get_height()):
|
|
|
|
var dx = 3*(x - center.x)
|
|
|
|
var dy = 3*(y - center.y)
|
|
|
|
var found_pixel : bool = false
|
|
|
|
for k in range(9):
|
|
|
|
var i = -1 + k % 3
|
2020-06-01 00:11:00 +00:00
|
|
|
# warning-ignore:integer_division
|
2020-05-29 00:16:44 +00:00
|
|
|
var j = -1 + int(k / 3)
|
|
|
|
var dir = atan2(dy + j, dx + i)
|
|
|
|
var mag = sqrt(pow(dx + i, 2) + pow(dy + j, 2))
|
|
|
|
dir -= angle
|
|
|
|
ox = round(center.x*3 + 1 + mag*cos(dir))
|
|
|
|
oy = round(center.y*3 + 1 + mag*sin(dir))
|
|
|
|
|
|
|
|
if (sprite.get_width() % 2 != 0):
|
|
|
|
ox += 1
|
|
|
|
oy += 1
|
|
|
|
|
|
|
|
if (ox >= 0 && ox < sprite.get_width()*3
|
|
|
|
&& oy >= 0 && oy < sprite.get_height()*3):
|
|
|
|
found_pixel = true
|
|
|
|
break
|
|
|
|
|
|
|
|
if !found_pixel:
|
|
|
|
sprite.set_pixel(x, y, Color(0,0,0,0))
|
|
|
|
continue
|
|
|
|
|
|
|
|
var fil : int = oy % 3
|
|
|
|
var col : int = ox % 3
|
|
|
|
var index : int = col + 3*fil
|
|
|
|
|
|
|
|
ox = round((ox - 1)/3.0);
|
|
|
|
oy = round((oy - 1)/3.0);
|
|
|
|
var a : Color
|
|
|
|
var b : Color
|
|
|
|
var c : Color
|
|
|
|
var d : Color
|
|
|
|
var e : Color
|
|
|
|
var f : Color
|
|
|
|
var g : Color
|
|
|
|
var h : Color
|
|
|
|
var i : Color
|
|
|
|
if (ox == 0 || ox == sprite.get_width() - 1 ||
|
|
|
|
oy == 0 || oy == sprite.get_height() - 1):
|
|
|
|
p = aux.get_pixel(ox, oy)
|
|
|
|
else:
|
|
|
|
a = aux.get_pixel(ox-1,oy-1);
|
|
|
|
b = aux.get_pixel(ox,oy-1);
|
|
|
|
c = aux.get_pixel(ox+1,oy-1);
|
|
|
|
d = aux.get_pixel(ox-1,oy);
|
|
|
|
e = aux.get_pixel(ox,oy);
|
|
|
|
f = aux.get_pixel(ox+1,oy);
|
|
|
|
g = aux.get_pixel(ox-1,oy+1);
|
|
|
|
h = aux.get_pixel(ox,oy+1);
|
|
|
|
i = aux.get_pixel(ox+1,oy+1);
|
|
|
|
|
|
|
|
match(index):
|
|
|
|
0:
|
|
|
|
p = d if (similarColors(d,b) && !similarColors(d,h)
|
|
|
|
&& !similarColors(b,f)) else e;
|
|
|
|
1:
|
|
|
|
p = b if ((similarColors(d,b) && !similarColors(d,h) &&
|
|
|
|
!similarColors(b,f) && !similarColors(e,c)) ||
|
|
|
|
(similarColors(b,f) && !similarColors(d,b) &&
|
|
|
|
!similarColors(f,h) && !similarColors(e,a))) else e;
|
|
|
|
2:
|
|
|
|
p = f if (similarColors(b,f) && !similarColors(d,b) &&
|
|
|
|
!similarColors(f,h)) else e;
|
|
|
|
3:
|
|
|
|
p = d if ((similarColors(d,h) && !similarColors(f,h) &&
|
|
|
|
!similarColors(d,b) && !similarColors(e,a)) ||
|
|
|
|
(similarColors(d,b) && !similarColors(d,h) &&
|
|
|
|
!similarColors(b,f) && !similarColors(e,g))) else e;
|
|
|
|
4:
|
|
|
|
p = e
|
|
|
|
5:
|
|
|
|
p = f if((similarColors(b,f) && !similarColors(d,b) &&
|
|
|
|
!similarColors(f,h) && !similarColors(e,i))
|
|
|
|
|| (similarColors(f,h) && !similarColors(b,f) &&
|
|
|
|
!similarColors(d,h) && !similarColors(e,c))) else e;
|
|
|
|
6:
|
|
|
|
p = d if (similarColors(d,h) && !similarColors(f,h) &&
|
|
|
|
!similarColors(d,b)) else e;
|
|
|
|
7:
|
|
|
|
p = h if ((similarColors(f,h) && !similarColors(f,b) &&
|
|
|
|
!similarColors(d,h) && !similarColors(e,g))
|
|
|
|
|| (similarColors(d,h) && !similarColors(f,h) &&
|
|
|
|
!similarColors(d,b) && !similarColors(e,i))) else e;
|
|
|
|
8:
|
|
|
|
p = f if (similarColors(f,h) && !similarColors(f,b) &&
|
|
|
|
!similarColors(d,h)) else e;
|
|
|
|
sprite.set_pixel(x, y, p)
|
|
|
|
sprite.unlock()
|
|
|
|
aux.unlock()
|
|
|
|
|
|
|
|
|
|
|
|
func fake_rotsprite(sprite : Image, angle : float) -> void:
|
|
|
|
sprite.copy_from(scale3X(sprite))
|
|
|
|
nn_rotate(sprite,angle)
|
2020-06-01 00:11:00 +00:00
|
|
|
# warning-ignore:integer_division
|
|
|
|
# warning-ignore:integer_division
|
|
|
|
sprite.resize(sprite.get_width() / 3, sprite.get_height() / 3, 0)
|
2020-05-29 00:16:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
func nn_rotate(sprite : Image, angle : float) -> void:
|
|
|
|
var aux : Image = Image.new()
|
|
|
|
aux.copy_from(sprite)
|
|
|
|
sprite.lock()
|
|
|
|
aux.lock()
|
|
|
|
var ox: int
|
|
|
|
var oy: int
|
2020-06-01 00:11:00 +00:00
|
|
|
# warning-ignore:integer_division
|
|
|
|
# warning-ignore:integer_division
|
|
|
|
var center : Vector2 = Vector2(sprite.get_width() / 2, sprite.get_height() / 2)
|
2020-05-29 00:16:44 +00:00
|
|
|
for x in range(sprite.get_width()):
|
|
|
|
for y in range(sprite.get_height()):
|
|
|
|
ox = (x - center.x)*cos(angle) + (y - center.y)*sin(angle) + center.x
|
|
|
|
oy = -(x - center.x)*sin(angle) + (y - center.y)*cos(angle) + center.y
|
|
|
|
if ox >= 0 && ox < sprite.get_width() && oy >= 0 && oy < sprite.get_height():
|
|
|
|
sprite.set_pixel(x, y, aux.get_pixel(ox, oy))
|
|
|
|
else:
|
|
|
|
sprite.set_pixel(x, y, Color(0,0,0,0))
|
|
|
|
sprite.unlock()
|
|
|
|
aux.unlock()
|
|
|
|
|
|
|
|
|
|
|
|
func similarColors(c1 : Color, c2 : Color, tol : float = 100) -> bool:
|
|
|
|
var dist = colorDistance(c1, c2)
|
|
|
|
return dist <= tol
|
|
|
|
|
|
|
|
|
|
|
|
func colorDistance(c1 : Color, c2 : Color) -> float:
|
|
|
|
return sqrt(pow((c1.r - c2.r)*255, 2) + pow((c1.g - c2.g)*255, 2)
|
|
|
|
+ pow((c1.b - c2.b)*255, 2) + pow((c1.a - c2.a)*255, 2))
|
|
|
|
|
2020-06-13 15:57:28 +00:00
|
|
|
# Image effects
|
|
|
|
|
|
|
|
func scale_image(width : int, height : int, interpolation : int) -> void:
|
|
|
|
Global.current_project.undos += 1
|
|
|
|
Global.current_project.undo_redo.create_action("Scale")
|
|
|
|
Global.current_project.undo_redo.add_do_property(Global.current_project, "size", Vector2(width, height).floor())
|
|
|
|
|
|
|
|
for f in Global.current_project.frames:
|
|
|
|
for i in range(f.cels.size() - 1, -1, -1):
|
|
|
|
var sprite := Image.new()
|
|
|
|
sprite.copy_from(f.cels[i].image)
|
|
|
|
sprite.resize(width, height, interpolation)
|
|
|
|
Global.current_project.undo_redo.add_do_property(f.cels[i].image, "data", sprite.data)
|
|
|
|
Global.current_project.undo_redo.add_undo_property(f.cels[i].image, "data", f.cels[i].image.data)
|
|
|
|
|
|
|
|
Global.current_project.undo_redo.add_undo_property(Global.current_project, "size", Global.current_project.size)
|
|
|
|
Global.current_project.undo_redo.add_undo_method(Global, "undo")
|
|
|
|
Global.current_project.undo_redo.add_do_method(Global, "redo")
|
|
|
|
Global.current_project.undo_redo.commit_action()
|
|
|
|
|
|
|
|
|
|
|
|
func crop_image(image : Image) -> void:
|
|
|
|
# Use first cel as a starting rectangle
|
|
|
|
var used_rect : Rect2 = image.get_used_rect()
|
|
|
|
|
|
|
|
for f in Global.current_project.frames:
|
|
|
|
# However, if first cel is empty, loop through all cels until we find one that isn't
|
|
|
|
for cel in f.cels:
|
|
|
|
if used_rect != Rect2(0, 0, 0, 0):
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
if cel.image.get_used_rect() != Rect2(0, 0, 0, 0):
|
|
|
|
used_rect = cel.image.get_used_rect()
|
|
|
|
|
|
|
|
# Merge all layers with content
|
|
|
|
for cel in f.cels:
|
|
|
|
if cel.image.get_used_rect() != Rect2(0, 0, 0, 0):
|
|
|
|
used_rect = used_rect.merge(cel.image.get_used_rect())
|
|
|
|
|
|
|
|
# If no layer has any content, just return
|
|
|
|
if used_rect == Rect2(0, 0, 0, 0):
|
|
|
|
return
|
|
|
|
|
|
|
|
var width := used_rect.size.x
|
|
|
|
var height := used_rect.size.y
|
|
|
|
Global.current_project.undos += 1
|
|
|
|
Global.current_project.undo_redo.create_action("Scale")
|
|
|
|
Global.current_project.undo_redo.add_do_property(Global.current_project, "size", Vector2(width, height).floor())
|
|
|
|
for f in Global.current_project.frames:
|
|
|
|
# Loop through all the layers to crop them
|
|
|
|
for j in range(Global.current_project.layers.size() - 1, -1, -1):
|
|
|
|
var sprite : Image = f.cels[j].image.get_rect(used_rect)
|
|
|
|
Global.current_project.undo_redo.add_do_property(f.cels[j].image, "data", sprite.data)
|
|
|
|
Global.current_project.undo_redo.add_undo_property(f.cels[j].image, "data", f.cels[j].image.data)
|
|
|
|
|
|
|
|
Global.current_project.undo_redo.add_undo_property(Global.current_project, "size", Global.current_project.size)
|
|
|
|
Global.current_project.undo_redo.add_undo_method(Global, "undo")
|
|
|
|
Global.current_project.undo_redo.add_do_method(Global, "redo")
|
|
|
|
Global.current_project.undo_redo.commit_action()
|
2020-06-13 17:22:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
func resize_canvas(width : int, height : int) -> void:
|
|
|
|
Global.current_project.undos += 1
|
|
|
|
Global.current_project.undo_redo.create_action("Scale")
|
|
|
|
Global.current_project.undo_redo.add_do_property(Global.current_project, "size", Vector2(width, height).floor())
|
|
|
|
for f in Global.current_project.frames:
|
|
|
|
for c in f.cels:
|
|
|
|
var sprite := Image.new()
|
|
|
|
sprite.copy_from(c.image)
|
|
|
|
sprite.crop(width, height)
|
|
|
|
Global.current_project.undo_redo.add_do_property(c.image, "data", sprite.data)
|
|
|
|
Global.current_project.undo_redo.add_undo_property(c.image, "data", c.image.data)
|
|
|
|
|
|
|
|
Global.current_project.undo_redo.add_undo_property(Global.current_project, "size", Global.current_project.size)
|
|
|
|
Global.current_project.undo_redo.add_undo_method(Global, "undo")
|
|
|
|
Global.current_project.undo_redo.add_do_method(Global, "redo")
|
|
|
|
Global.current_project.undo_redo.commit_action()
|
2020-06-13 15:57:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
func invert_image_colors(image : Image) -> void:
|
|
|
|
Global.canvas.handle_undo("Draw")
|
|
|
|
for xx in image.get_size().x:
|
|
|
|
for yy in image.get_size().y:
|
|
|
|
var px_color = image.get_pixel(xx, yy).inverted()
|
|
|
|
if px_color.a == 0:
|
|
|
|
continue
|
|
|
|
image.set_pixel(xx, yy, px_color)
|
|
|
|
Global.canvas.handle_redo("Draw")
|
|
|
|
|
|
|
|
|
|
|
|
func desaturate_image(image : Image) -> void:
|
|
|
|
Global.canvas.handle_undo("Draw")
|
|
|
|
for xx in image.get_size().x:
|
|
|
|
for yy in image.get_size().y:
|
|
|
|
var px_color = image.get_pixel(xx, yy)
|
|
|
|
if px_color.a == 0:
|
|
|
|
continue
|
|
|
|
var gray = image.get_pixel(xx, yy).v
|
|
|
|
px_color = Color(gray, gray, gray, px_color.a)
|
|
|
|
image.set_pixel(xx, yy, px_color)
|
|
|
|
Global.canvas.handle_redo("Draw")
|
|
|
|
|
|
|
|
|
|
|
|
func generate_outline(image : Image, outline_color : Color, thickness : int, diagonal : bool, inside_image : bool) -> void:
|
|
|
|
if image.is_invisible():
|
|
|
|
return
|
|
|
|
var new_image := Image.new()
|
|
|
|
new_image.copy_from(image)
|
|
|
|
new_image.lock()
|
|
|
|
|
|
|
|
Global.canvas.handle_undo("Draw")
|
|
|
|
for xx in image.get_size().x:
|
|
|
|
for yy in image.get_size().y:
|
|
|
|
var pos = Vector2(xx, yy)
|
|
|
|
var current_pixel := image.get_pixelv(pos)
|
|
|
|
if current_pixel.a == 0:
|
|
|
|
continue
|
|
|
|
|
|
|
|
for i in range(1, thickness + 1):
|
|
|
|
if inside_image:
|
|
|
|
var outline_pos : Vector2 = pos + Vector2.LEFT # Left
|
|
|
|
if outline_pos.x < 0 || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + Vector2.RIGHT * (i - 1)
|
|
|
|
if new_pos.x < Global.current_project.size.x:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
outline_pos = pos + Vector2.RIGHT # Right
|
|
|
|
if outline_pos.x >= Global.current_project.size.x || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + Vector2.LEFT * (i - 1)
|
|
|
|
if new_pos.x >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
outline_pos = pos + Vector2.UP # Up
|
|
|
|
if outline_pos.y < 0 || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + Vector2.DOWN * (i - 1)
|
|
|
|
if new_pos.y < Global.current_project.size.y:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
outline_pos = pos + Vector2.DOWN # Down
|
|
|
|
if outline_pos.y >= Global.current_project.size.y || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + Vector2.UP * (i - 1)
|
|
|
|
if new_pos.y >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
if diagonal:
|
|
|
|
outline_pos = pos + (Vector2.LEFT + Vector2.UP) # Top left
|
|
|
|
if (outline_pos.x < 0 && outline_pos.y < 0) || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + (Vector2.RIGHT + Vector2.DOWN) * (i - 1)
|
|
|
|
if new_pos.x < Global.current_project.size.x && new_pos.y < Global.current_project.size.y:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
outline_pos = pos + (Vector2.LEFT + Vector2.DOWN) # Bottom left
|
|
|
|
if (outline_pos.x < 0 && outline_pos.y >= Global.current_project.size.y) || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + (Vector2.RIGHT + Vector2.UP) * (i - 1)
|
|
|
|
if new_pos.x < Global.current_project.size.x && new_pos.y >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
outline_pos = pos + (Vector2.RIGHT + Vector2.UP) # Top right
|
|
|
|
if (outline_pos.x >= Global.current_project.size.x && outline_pos.y < 0) || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + (Vector2.LEFT + Vector2.DOWN) * (i - 1)
|
|
|
|
if new_pos.x >= 0 && new_pos.y < Global.current_project.size.y:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
outline_pos = pos + (Vector2.RIGHT + Vector2.DOWN) # Bottom right
|
|
|
|
if (outline_pos.x >= Global.current_project.size.x && outline_pos.y >= Global.current_project.size.y) || image.get_pixelv(outline_pos).a == 0:
|
|
|
|
var new_pos : Vector2 = pos + (Vector2.LEFT + Vector2.UP) * (i - 1)
|
|
|
|
if new_pos.x >= 0 && new_pos.y >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a > 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
else:
|
|
|
|
var new_pos : Vector2 = pos + Vector2.LEFT * i # Left
|
|
|
|
if new_pos.x >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
new_pos = pos + Vector2.RIGHT * i # Right
|
|
|
|
if new_pos.x < Global.current_project.size.x:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
new_pos = pos + Vector2.UP * i # Up
|
|
|
|
if new_pos.y >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
new_pos = pos + Vector2.DOWN * i # Down
|
|
|
|
if new_pos.y < Global.current_project.size.y:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
if diagonal:
|
|
|
|
new_pos = pos + (Vector2.LEFT + Vector2.UP) * i # Top left
|
|
|
|
if new_pos.x >= 0 && new_pos.y >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
new_pos = pos + (Vector2.LEFT + Vector2.DOWN) * i # Bottom left
|
|
|
|
if new_pos.x >= 0 && new_pos.y < Global.current_project.size.y:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
new_pos = pos + (Vector2.RIGHT + Vector2.UP) * i # Top right
|
|
|
|
if new_pos.x < Global.current_project.size.x && new_pos.y >= 0:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
new_pos = pos + (Vector2.RIGHT + Vector2.DOWN) * i # Bottom right
|
|
|
|
if new_pos.x < Global.current_project.size.x && new_pos.y < Global.current_project.size.y:
|
|
|
|
var new_pixel = image.get_pixelv(new_pos)
|
|
|
|
if new_pixel.a == 0:
|
|
|
|
new_image.set_pixelv(new_pos, outline_color)
|
|
|
|
|
|
|
|
image.copy_from(new_image)
|
|
|
|
Global.canvas.handle_redo("Draw")
|
|
|
|
|
2020-05-29 00:16:44 +00:00
|
|
|
|
|
|
|
func adjust_hsv(img: Image, id : int, delta : float) -> void:
|
2020-06-05 13:06:59 +00:00
|
|
|
var x_min = Global.current_project.x_min
|
|
|
|
var x_max = Global.current_project.x_max
|
|
|
|
var y_min = Global.current_project.y_min
|
|
|
|
var y_max = Global.current_project.y_max
|
2020-05-29 00:16:44 +00:00
|
|
|
img.lock()
|
|
|
|
|
|
|
|
match id:
|
|
|
|
0: # Hue
|
2020-06-04 15:20:56 +00:00
|
|
|
for i in range(x_min, x_max):
|
|
|
|
for j in range(y_min, y_max):
|
2020-05-29 00:16:44 +00:00
|
|
|
var c : Color = img.get_pixel(i,j)
|
|
|
|
var hue = range_lerp(c.h,0,1,-180,180)
|
|
|
|
hue = hue + delta
|
|
|
|
|
|
|
|
while(hue >= 180):
|
|
|
|
hue -= 360
|
|
|
|
while(hue < -180):
|
|
|
|
hue += 360
|
|
|
|
c.h = range_lerp(hue,-180,180,0,1)
|
|
|
|
img.set_pixel(i,j,c)
|
|
|
|
|
|
|
|
1: # Saturation
|
2020-06-04 15:20:56 +00:00
|
|
|
for i in range(x_min, x_max):
|
|
|
|
for j in range(y_min, y_max):
|
2020-05-29 00:16:44 +00:00
|
|
|
var c : Color = img.get_pixel(i,j)
|
|
|
|
var sat = c.s
|
|
|
|
if delta > 0:
|
|
|
|
sat = range_lerp(delta,0,100,c.s,1)
|
|
|
|
elif delta < 0:
|
|
|
|
sat = range_lerp(delta,-100,0,0,c.s)
|
|
|
|
c.s = sat
|
|
|
|
img.set_pixel(i,j,c)
|
|
|
|
|
|
|
|
2: # Value
|
2020-06-04 15:20:56 +00:00
|
|
|
for i in range(x_min, x_max):
|
|
|
|
for j in range(y_min, y_max):
|
2020-05-29 00:16:44 +00:00
|
|
|
var c : Color = img.get_pixel(i,j)
|
|
|
|
var val = c.v
|
|
|
|
if delta > 0:
|
|
|
|
val = range_lerp(delta,0,100,c.v,1)
|
|
|
|
elif delta < 0:
|
|
|
|
val = range_lerp(delta,-100,0,0,c.v)
|
|
|
|
|
|
|
|
c.v = val
|
|
|
|
img.set_pixel(i,j,c)
|
|
|
|
|
|
|
|
img.unlock()
|
|
|
|
|
|
|
|
|
|
|
|
# Checks if a point is inside a rectangle
|
|
|
|
func point_in_rectangle(p : Vector2, coord1 : Vector2, coord2 : Vector2) -> bool:
|
|
|
|
return p.x > coord1.x && p.y > coord1.y && p.x < coord2.x && p.y < coord2.y
|
2020-05-29 21:46:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Returns the position in the middle of a rectangle
|
|
|
|
func rectangle_center(rect_position : Vector2, rect_size : Vector2) -> Vector2:
|
|
|
|
return (rect_position - rect_size / 2).floor()
|