mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 17:28:11 +00:00
PixelPaint+LibGfx: Allow resizing images and layers
This PR adds resize ability to PixelPaint as per issue 11862. The previous behaviour was to always rescale the canvas when resizing an image. This adds a checkbox to toggle between rescaling, and resizing which blits the existing canvas to the top left of the new, resized canvas. As part of this, a new ScalingMode is added to LibGfx - None.
This commit is contained in:
parent
d6334dcab1
commit
4bf587811f
6 changed files with 35 additions and 3 deletions
|
@ -230,7 +230,11 @@ void Layer::resize(Gfx::IntSize const& new_size, Gfx::IntPoint const& new_locati
|
|||
auto dst = Gfx::Bitmap::try_create(Gfx::BitmapFormat::BGRA8888, new_size).release_value_but_fixme_should_propagate_errors();
|
||||
Gfx::Painter painter(dst);
|
||||
|
||||
painter.draw_scaled_bitmap(dst_rect, *m_content_bitmap, src_rect, 1.0f, scaling_mode);
|
||||
if (scaling_mode == Gfx::Painter::ScalingMode::None) {
|
||||
painter.blit(src_rect.top_left(), *m_content_bitmap, src_rect, 1.0f);
|
||||
} else {
|
||||
painter.draw_scaled_bitmap(dst_rect, *m_content_bitmap, src_rect, 1.0f, scaling_mode);
|
||||
}
|
||||
|
||||
m_content_bitmap = move(dst);
|
||||
}
|
||||
|
@ -239,7 +243,12 @@ void Layer::resize(Gfx::IntSize const& new_size, Gfx::IntPoint const& new_locati
|
|||
auto dst = Gfx::Bitmap::try_create(Gfx::BitmapFormat::BGRA8888, new_size).release_value_but_fixme_should_propagate_errors();
|
||||
Gfx::Painter painter(dst);
|
||||
|
||||
painter.draw_scaled_bitmap(dst_rect, *m_mask_bitmap, src_rect, 1.0f, scaling_mode);
|
||||
if (scaling_mode == Gfx::Painter::ScalingMode::None) {
|
||||
painter.blit(src_rect.top_left(), *m_content_bitmap, src_rect, 1.0f);
|
||||
} else {
|
||||
painter.draw_scaled_bitmap(dst_rect, *m_mask_bitmap, src_rect, 1.0f, scaling_mode);
|
||||
}
|
||||
|
||||
m_mask_bitmap = move(dst);
|
||||
}
|
||||
|
||||
|
|
|
@ -70,10 +70,12 @@ ResizeImageDialog::ResizeImageDialog(Gfx::IntSize const& suggested_size, GUI::Wi
|
|||
auto nearest_neighbor_radio = main_widget.find_descendant_of_type_named<GUI::RadioButton>("nearest_neighbor_radio");
|
||||
auto smooth_pixels_radio = main_widget.find_descendant_of_type_named<GUI::RadioButton>("smooth_pixels_radio");
|
||||
auto bilinear_radio = main_widget.find_descendant_of_type_named<GUI::RadioButton>("bilinear_radio");
|
||||
auto resize_canvas_radio = main_widget.find_descendant_of_type_named<GUI::RadioButton>("resize_canvas");
|
||||
|
||||
VERIFY(nearest_neighbor_radio);
|
||||
VERIFY(smooth_pixels_radio);
|
||||
VERIFY(bilinear_radio);
|
||||
VERIFY(resize_canvas_radio);
|
||||
|
||||
m_scaling_mode = Gfx::Painter::ScalingMode::NearestNeighbor;
|
||||
if (bilinear_radio->is_checked()) {
|
||||
|
@ -92,6 +94,10 @@ ResizeImageDialog::ResizeImageDialog(Gfx::IntSize const& suggested_size, GUI::Wi
|
|||
if (is_checked)
|
||||
m_scaling_mode = Gfx::Painter::ScalingMode::BilinearBlend;
|
||||
};
|
||||
resize_canvas_radio->on_checked = [this](bool is_checked) {
|
||||
if (is_checked)
|
||||
m_scaling_mode = Gfx::Painter::ScalingMode::None;
|
||||
};
|
||||
|
||||
auto ok_button = main_widget.find_descendant_of_type_named<GUI::Button>("ok_button");
|
||||
auto cancel_button = main_widget.find_descendant_of_type_named<GUI::Button>("cancel_button");
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
@GUI::Widget {
|
||||
fill_with_background_color: true
|
||||
min_width: 260
|
||||
min_height: 210
|
||||
min_height: 260
|
||||
layout: @GUI::VerticalBoxLayout {
|
||||
margins: [4]
|
||||
}
|
||||
|
@ -92,6 +92,12 @@
|
|||
text: "Bilinear"
|
||||
autosize: true
|
||||
}
|
||||
|
||||
@GUI::RadioButton {
|
||||
name: "resize_canvas"
|
||||
text: "Resize Canvas (None)"
|
||||
autosize: true
|
||||
}
|
||||
}
|
||||
|
||||
@GUI::Widget {
|
||||
|
|
|
@ -17,6 +17,7 @@ class ResizeImageDialog final : public GUI::Dialog {
|
|||
public:
|
||||
Gfx::IntSize const& desired_size() const { return m_desired_size; }
|
||||
Gfx::Painter::ScalingMode scaling_mode() const { return m_scaling_mode; }
|
||||
bool should_rescale() const { return m_rescale_image; }
|
||||
|
||||
private:
|
||||
ResizeImageDialog(Gfx::IntSize const& starting_size, GUI::Window* parent_window);
|
||||
|
@ -24,6 +25,7 @@ private:
|
|||
Gfx::IntSize m_desired_size;
|
||||
Gfx::Painter::ScalingMode m_scaling_mode;
|
||||
float m_starting_aspect_ratio;
|
||||
bool m_rescale_image;
|
||||
};
|
||||
|
||||
}
|
||||
|
|
|
@ -1248,6 +1248,9 @@ ALWAYS_INLINE static void do_draw_scaled_bitmap(Gfx::Bitmap& target, IntRect con
|
|||
case Painter::ScalingMode::BilinearBlend:
|
||||
do_draw_scaled_bitmap<has_alpha_channel, Painter::ScalingMode::BilinearBlend>(target, dst_rect, clipped_rect, source, src_rect, get_pixel, opacity);
|
||||
break;
|
||||
case Painter::ScalingMode::None:
|
||||
do_draw_scaled_bitmap<has_alpha_channel, Painter::ScalingMode::None>(target, dst_rect, clipped_rect, source, src_rect, get_pixel, opacity);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1262,6 +1265,11 @@ void Painter::draw_scaled_bitmap(IntRect const& a_dst_rect, Gfx::Bitmap const& s
|
|||
if (scale() == source.scale() && a_src_rect == int_src_rect && a_dst_rect.size() == int_src_rect.size())
|
||||
return blit(a_dst_rect.location(), source, int_src_rect, opacity);
|
||||
|
||||
if (scaling_mode == ScalingMode::None) {
|
||||
IntRect clipped_draw_rect { (int)a_src_rect.location().x(), (int)a_src_rect.location().y(), a_dst_rect.size().width(), a_dst_rect.size().height() };
|
||||
return blit(a_dst_rect.location(), source, clipped_draw_rect, opacity);
|
||||
}
|
||||
|
||||
auto dst_rect = to_physical(a_dst_rect);
|
||||
auto src_rect = a_src_rect * source.scale();
|
||||
auto clipped_rect = dst_rect.intersected(clip_rect() * scale());
|
||||
|
|
|
@ -38,6 +38,7 @@ public:
|
|||
NearestNeighbor,
|
||||
SmoothPixels,
|
||||
BilinearBlend,
|
||||
None,
|
||||
};
|
||||
|
||||
void clear_rect(IntRect const&, Color);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue