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);
|
|
|
|
|
2024-10-27 03:36:17 +01:00
|
|
|
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];
|
2025-01-02 09:02:39 +01:00
|
|
|
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
|
|
|
|
2025-01-02 09:02:39 +01: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;
|
|
|
|
|
|
|
|
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
|
2025-01-02 09:02:39 +01:00
|
|
|
shader_set_f("index", indxC);
|
|
|
|
shader_set_f("varient", indxV);
|
2024-10-16 12:34:26 +02:00
|
|
|
|
|
|
|
while(!ds_queue_empty(qx)) {
|
|
|
|
|
|
|
|
x1 = ds_queue_dequeue(qx);
|
|
|
|
y1 = ds_queue_dequeue(qy);
|
|
|
|
|
|
|
|
// print($"----Checking {x1}, {y1} - {_tiler_ff_getPixel(x1, y1)}")
|
|
|
|
|
2025-01-02 09:02:39 +01:00
|
|
|
if(_tiler_ff_getPixel(x1, y1) == indxC) continue; //Color in queue is already filled
|
2024-10-16 12:34:26 +02:00
|
|
|
|
2025-01-02 09:02:39 +01: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;
|
|
|
|
|
2025-01-02 09:02:39 +01:00
|
|
|
while(x1 < surface_w && baseC == _tiler_ff_getPixel(x1, y1)) {
|
2024-10-16 12:34:26 +02:00
|
|
|
draw_point(x1, y1);
|
2025-01-02 09:02:39 +01:00
|
|
|
buffer_write_at(_ff_buff, (y1 * _ff_w + x1) * 8, buffer_f16, indxC);
|
2024-10-16 12:34:26 +02:00
|
|
|
|
|
|
|
// print($"----Filling {x1}, {y1}")
|
|
|
|
|
|
|
|
if(y1 > 0) {
|
2025-01-02 09:02:39 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2025-01-02 09:02:39 +01:00
|
|
|
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) {
|
2025-01-02 09:02:39 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2025-01-02 09:02:39 +01:00
|
|
|
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) {
|
2025-01-02 09:02:39 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2025-01-02 09:02:39 +01:00
|
|
|
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);
|
|
|
|
}
|