Pixel-Composer/scripts/tiler_tool_fill/tiler_tool_fill.gml

141 lines
3.9 KiB
Text
Raw Permalink Normal View History

2025-01-02 08:54:31 +01:00
function tiler_tool_fill(_node, _brush, toolAttr) : tiler_tool(_node) constructor {
2024-10-16 12:34:26 +02:00
self.brush = _brush;
self.tool_attribute = toolAttr;
mouse_cur_x = -1;
mouse_cur_y = -1;
mouse_pre_x = -1;
mouse_pre_y = -1;
function step(hover, active, _x, _y, _s, _mx, _my, _snx, _sny) {
mouse_cur_x = floor(round((_mx - _x) / _s - 0.5) / tile_size[0]);
mouse_cur_y = floor(round((_my - _y) / _s - 0.5) / tile_size[1]);
surface_w = surface_get_width(drawing_surface);
surface_h = surface_get_height(drawing_surface);
var _auto = brush.autoterrain;
2024-10-24 12:40:14 +02:00
2024-10-16 12:34:26 +02:00
if(mouse_press(mb_left, active) && point_in_rectangle(mouse_cur_x, mouse_cur_y, 0, 0, surface_w - 1, surface_h - 1)) {
2025-01-02 08:54:31 +01:00
node.storeAction();
2024-10-16 12:34:26 +02:00
surface_set_target(drawing_surface);
tiler_flood_fill_scanline(drawing_surface, mouse_cur_x, mouse_cur_y, brush, tool_attribute.fillType);
surface_reset_target();
2024-10-24 12:40:14 +02:00
if(_auto != noone) {
2025-01-02 08:54:31 +01:00
_auto.drawing_start(drawing_surface);
2024-10-24 12:40:14 +02:00
tiler_flood_fill_scanline(drawing_surface, mouse_cur_x, mouse_cur_y, brush, tool_attribute.fillType);
_auto.drawing_end();
}
2024-10-16 12:34:26 +02:00
apply_draw_surface();
}
}
}
2024-10-24 12:40:14 +02:00
function _tiler_ff_getPixel(_x, _y) { return round(buffer_read_at(_ff_buff, (_y * _ff_w + _x) * 8, buffer_f16)); }
2024-10-16 12:34:26 +02:00
function tiler_flood_fill_scanline(_surf, _x, _y, brush, _corner = false) {
if(brush.brush_height * brush.brush_width == 0) return;
var _index = brush.brush_erase? -1 : brush.brush_indices[0][0];
var _c = surface_getpixel(_surf, _x, _y);
var baseC = _c[0];
var baseV = _c[1];
var indxC = _index[0];
var indxV = _index[1];
2024-10-16 12:34:26 +02:00
if(indxC == baseC && indxV == baseV) return;
2024-10-16 12:34:26 +02:00
_ff_w = surface_get_width(_surf);
_ff_h = surface_get_height(_surf);
2024-10-24 12:40:14 +02:00
_ff_buff = buffer_create(_ff_w * _ff_h * 8, buffer_fixed, 2);
2024-10-16 12:34:26 +02:00
buffer_get_surface(_ff_buff, _surf, 0);
var x1, y1, x_start;
var spanAbove, spanBelow;
2025-01-03 03:50:08 +01:00
var bw = brush.brush_width;
var bh = brush.brush_height;
2024-10-16 12:34:26 +02:00
var qx = ds_queue_create();
var qy = ds_queue_create();
ds_queue_enqueue(qx, _x);
ds_queue_enqueue(qy, _y);
shader_set(sh_draw_tile_brush);
BLEND_OVERRIDE
while(!ds_queue_empty(qx)) {
x1 = ds_queue_dequeue(qx);
y1 = ds_queue_dequeue(qy);
2025-01-03 03:50:08 +01:00
if(_tiler_ff_getPixel(x1, y1) == 1) continue; //Color in queue is already filled
2024-10-16 12:34:26 +02:00
while(x1 > 0 && baseC == _tiler_ff_getPixel(x1 - 1, y1)) //Move to the leftmost connected pixel in the same row.
2024-10-16 12:34:26 +02:00
x1--;
x_start = x1;
spanAbove = false;
spanBelow = false;
while(x1 < surface_w && baseC == _tiler_ff_getPixel(x1, y1)) {
2025-01-03 03:50:08 +01:00
var _b = brush.brush_indices[y1 % bh][x1 % bw];
shader_set_f("index", _b[0]);
shader_set_f("varient", _b[1]);
2024-10-16 12:34:26 +02:00
draw_point(x1, y1);
2025-01-03 03:50:08 +01:00
buffer_write_at(_ff_buff, (y1 * _ff_w + x1) * 8, buffer_f16, 1);
2024-10-16 12:34:26 +02:00
if(y1 > 0) {
if(_corner && x1 > 0 && baseC == _tiler_ff_getPixel(x1 - 1, y1 - 1)) { //Check top left pixel
2024-10-16 12:34:26 +02:00
ds_queue_enqueue(qx, x1 - 1);
ds_queue_enqueue(qy, y1 - 1);
}
if(baseC == _tiler_ff_getPixel(x1, y1 - 1)) { //Check top pixel
2024-10-16 12:34:26 +02:00
ds_queue_enqueue(qx, x1);
ds_queue_enqueue(qy, y1 - 1);
}
}
if(y1 < surface_h - 1) {
if(_corner && x1 > 0 && baseC == _tiler_ff_getPixel(x1 - 1, y1 + 1)) { //Check bottom left pixel
2024-10-16 12:34:26 +02:00
ds_queue_enqueue(qx, x1 - 1);
ds_queue_enqueue(qy, y1 + 1);
}
if(baseC == _tiler_ff_getPixel(x1, y1 + 1)) { //Check bottom pixel
2024-10-16 12:34:26 +02:00
ds_queue_enqueue(qx, x1);
ds_queue_enqueue(qy, y1 + 1);
}
}
if(_corner && x1 < surface_w - 1) {
if(y1 > 0 && baseC == _tiler_ff_getPixel(x1 + 1, y1 - 1)) { //Check top right pixel
2024-10-16 12:34:26 +02:00
ds_queue_enqueue(qx, x1 + 1);
ds_queue_enqueue(qy, y1 - 1);
}
if(y1 < surface_h - 1 && baseC == _tiler_ff_getPixel(x1 + 1, y1 + 1)) { //Check bottom right pixel
2024-10-16 12:34:26 +02:00
ds_queue_enqueue(qx, x1 + 1);
ds_queue_enqueue(qy, y1 + 1);
}
}
x1++;
}
}
BLEND_NORMAL
shader_reset();
ds_queue_destroy(qx);
ds_queue_destroy(qy);
draw_set_alpha(1);
buffer_delete(_ff_buff);
}