mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 15:02:46 +00:00 
			
		
		
		
	 b9b4ca064f
			
		
	
	
		b9b4ca064f
		
	
	
	
	
		
			
			This patch optimizes how the Brush-Tool modifies the pixels. The new logic generates a "reference brush" with the required size, falloff and color only once and uses that for the rawing operations. If no editing mask is used the reference brush is writen via a blit operation to the content or mask image. This increases the drawing speed and therefore also allows bigger brush sizes.
		
			
				
	
	
		
			123 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			123 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  * Copyright (c) 2021, Mustafa Quraish <mustafa@serenityos.org>
 | |
|  * Copyright (c) 2022, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include "Tool.h"
 | |
| #include "../ImageEditor.h"
 | |
| #include "../Layer.h"
 | |
| #include <LibGUI/Action.h>
 | |
| 
 | |
| namespace PixelPaint {
 | |
| 
 | |
| void Tool::setup(ImageEditor& editor)
 | |
| {
 | |
|     m_editor = editor;
 | |
| }
 | |
| 
 | |
| void Tool::set_action(GUI::Action* action)
 | |
| {
 | |
|     m_action = action;
 | |
| }
 | |
| 
 | |
| bool Tool::on_keydown(GUI::KeyEvent& event)
 | |
| {
 | |
|     switch (event.key()) {
 | |
|     case KeyCode::Key_LeftBracket:
 | |
|         if (m_primary_slider) {
 | |
|             m_primary_slider->decrease_slider_by(1);
 | |
|             return true;
 | |
|         }
 | |
|         break;
 | |
|     case KeyCode::Key_RightBracket:
 | |
|         if (m_primary_slider) {
 | |
|             m_primary_slider->increase_slider_by(1);
 | |
|             return true;
 | |
|         }
 | |
|         break;
 | |
|     case KeyCode::Key_LeftBrace:
 | |
|         if (m_secondary_slider) {
 | |
|             m_secondary_slider->decrease_slider_by(1);
 | |
|             return true;
 | |
|         }
 | |
|         break;
 | |
|     case KeyCode::Key_RightBrace:
 | |
|         if (m_secondary_slider) {
 | |
|             m_secondary_slider->increase_slider_by(1);
 | |
|             return true;
 | |
|         }
 | |
|         break;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| Gfx::IntPoint Tool::editor_layer_location(Layer const& layer) const
 | |
| {
 | |
|     return (Gfx::FloatPoint { layer.location() } * m_editor->scale()).to_rounded<int>();
 | |
| }
 | |
| 
 | |
| Gfx::IntPoint Tool::editor_stroke_position(Gfx::IntPoint pixel_coords, int stroke_thickness) const
 | |
| {
 | |
|     auto position = m_editor->content_to_frame_position(pixel_coords);
 | |
|     auto offset = (stroke_thickness % 2 == 0) ? 0 : m_editor->scale() / 2;
 | |
|     position = position.translated(offset, offset);
 | |
|     return position.to_type<int>();
 | |
| }
 | |
| 
 | |
| Gfx::IntPoint Tool::constrain_line_angle(Gfx::IntPoint start_pos, Gfx::IntPoint end_pos, float angle_increment)
 | |
| {
 | |
|     float current_angle = AK::atan2<float>(end_pos.y() - start_pos.y(), end_pos.x() - start_pos.x()) + float { M_PI * 2 };
 | |
| 
 | |
|     float constrained_angle = ((int)((current_angle + angle_increment / 2) / angle_increment)) * angle_increment;
 | |
| 
 | |
|     auto diff = end_pos - start_pos;
 | |
|     float line_length = AK::hypot<float>(diff.x(), diff.y());
 | |
| 
 | |
|     return { start_pos.x() + (int)(AK::cos(constrained_angle) * line_length),
 | |
|         start_pos.y() + (int)(AK::sin(constrained_angle) * line_length) };
 | |
| }
 | |
| 
 | |
| template<>
 | |
| void Tool::set_pixel_with_possible_mask<Gfx::StorageFormat::BGRA8888>(int x, int y, Gfx::Color color, Gfx::Bitmap& bitmap)
 | |
| {
 | |
|     if (!m_editor || !m_editor->active_layer())
 | |
|         return;
 | |
| 
 | |
|     switch (m_editor->active_layer()->edit_mode()) {
 | |
|     case Layer::EditMode::Content:
 | |
|         if (m_editor->active_layer()->mask_type() == Layer::MaskType::EditingMask)
 | |
|             bitmap.set_pixel<Gfx::StorageFormat::BGRA8888>(x, y, m_editor->active_layer()->modify_pixel_with_editing_mask(x, y, color, bitmap.get_pixel(x, y)));
 | |
|         else
 | |
|             bitmap.set_pixel(x, y, color);
 | |
|         break;
 | |
|     case Layer::EditMode::Mask:
 | |
|         bitmap.set_pixel<Gfx::StorageFormat::BGRA8888>(x, y, color);
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Tool::set_pixel_with_possible_mask(int x, int y, Gfx::Color color, Gfx::Bitmap& bitmap)
 | |
| {
 | |
|     if (!m_editor || !m_editor->active_layer())
 | |
|         return;
 | |
| 
 | |
|     switch (m_editor->active_layer()->edit_mode()) {
 | |
|     case Layer::EditMode::Content:
 | |
|         if (m_editor->active_layer()->mask_type() == Layer::MaskType::EditingMask)
 | |
|             bitmap.set_pixel(x, y, m_editor->active_layer()->modify_pixel_with_editing_mask(x, y, color, bitmap.get_pixel(x, y)));
 | |
|         else
 | |
|             bitmap.set_pixel(x, y, color);
 | |
|         break;
 | |
|     case Layer::EditMode::Mask:
 | |
|         bitmap.set_pixel(x, y, color);
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| }
 |