/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include "BucketTool.h" #include "ImageEditor.h" #include "Layer.h" #include #include #include #include #include #include #include namespace PixelPaint { BucketTool::BucketTool() { } BucketTool::~BucketTool() { } static float color_distance_squared(Gfx::Color const& lhs, Gfx::Color const& rhs) { int a = rhs.red() - lhs.red(); int b = rhs.green() - lhs.green(); int c = rhs.blue() - lhs.blue(); return (a * a + b * b + c * c) / (255.0f * 255.0f); } static void flood_fill(Gfx::Bitmap& bitmap, Gfx::IntPoint const& start_position, Color target_color, Color fill_color, int threshold) { VERIFY(bitmap.bpp() == 32); if (target_color == fill_color) return; if (!bitmap.rect().contains(start_position)) return; float threshold_normalized_squared = (threshold / 100.0f) * (threshold / 100.0f); Queue queue; queue.enqueue(start_position); while (!queue.is_empty()) { auto position = queue.dequeue(); auto pixel_color = bitmap.get_pixel(position.x(), position.y()); if (color_distance_squared(pixel_color, target_color) > threshold_normalized_squared) continue; bitmap.set_pixel(position.x(), position.y(), fill_color); if (position.x() != 0) queue.enqueue(position.translated(-1, 0)); if (position.x() != bitmap.width() - 1) queue.enqueue(position.translated(1, 0)); if (position.y() != 0) queue.enqueue(position.translated(0, -1)); if (position.y() != bitmap.height() - 1) queue.enqueue(position.translated(0, 1)); } } void BucketTool::on_mousedown(Layer& layer, GUI::MouseEvent& event, GUI::MouseEvent&) { if (!layer.rect().contains(event.position())) return; GUI::Painter painter(layer.bitmap()); auto target_color = layer.bitmap().get_pixel(event.x(), event.y()); flood_fill(layer.bitmap(), event.position(), target_color, m_editor->color_for(event), m_threshold); layer.did_modify_bitmap(); m_editor->did_complete_action(); } GUI::Widget* BucketTool::get_properties_widget() { if (!m_properties_widget) { m_properties_widget = GUI::Widget::construct(); m_properties_widget->set_layout(); auto& threshold_container = m_properties_widget->add(); threshold_container.set_fixed_height(20); threshold_container.set_layout(); auto& threshold_label = threshold_container.add("Threshold:"); threshold_label.set_text_alignment(Gfx::TextAlignment::CenterLeft); threshold_label.set_fixed_size(80, 20); auto& threshold_slider = threshold_container.add(); threshold_slider.set_fixed_height(20); threshold_slider.set_range(0, 100); threshold_slider.set_value(m_threshold); threshold_slider.on_change = [this](int value) { m_threshold = value; }; } return m_properties_widget.ptr(); } }