1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-24 23:32:06 +00:00
serenity/Userland/Libraries/LibGUI/CheckBox.cpp
Jelle Raaijmakers f391ccfe53 LibGfx+Everywhere: Change Gfx::Rect to be endpoint exclusive
Previously, calling `.right()` on a `Gfx::Rect` would return the last
column's coordinate still inside the rectangle, or `left + width - 1`.
This is called 'endpoint inclusive' and does not make a lot of sense for
`Gfx::Rect<float>` where a rectangle of width 5 at position (0, 0) would
return 4 as its right side. This same problem exists for `.bottom()`.

This changes `Gfx::Rect` to be endpoint exclusive, which gives us the
nice property that `width = right - left` and `height = bottom - top`.
It enables us to treat `Gfx::Rect<int>` and `Gfx::Rect<float>` exactly
the same.

All users of `Gfx::Rect` have been updated accordingly.
2023-05-23 12:35:42 +02:00

115 lines
3 KiB
C++

/*
* Copyright (c) 2018-2023, Andreas Kling <kling@serenityos.org>
* Copyright (c) 2022, the SerenityOS developers.
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibGUI/CheckBox.h>
#include <LibGUI/Painter.h>
#include <LibGfx/CharacterBitmap.h>
#include <LibGfx/Font/Font.h>
#include <LibGfx/Palette.h>
#include <LibGfx/StylePainter.h>
REGISTER_WIDGET(GUI, CheckBox)
namespace GUI {
CheckBox::CheckBox(String text)
: AbstractButton(move(text))
{
REGISTER_BOOL_PROPERTY("autosize", is_autosize, set_autosize);
REGISTER_ENUM_PROPERTY(
"checkbox_position", checkbox_position, set_checkbox_position, CheckBoxPosition,
{ CheckBoxPosition::Left, "Left" },
{ CheckBoxPosition::Right, "Right" });
set_min_size(SpecialDimension::Shrink, SpecialDimension::Shrink);
set_preferred_size(SpecialDimension::OpportunisticGrow, SpecialDimension::Shrink);
}
int CheckBox::gap_between_box_and_rect() const
{
return 6;
}
int CheckBox::horizontal_padding() const
{
return 2;
}
Gfx::IntRect CheckBox::box_rect() const
{
int box_size = max(13, height() - 10);
Gfx::IntRect box_rect {
0,
height() / 2 - box_size / 2 - 1,
box_size,
box_size,
};
if (m_checkbox_position == CheckBoxPosition::Right)
box_rect.set_right_without_resize(rect().right());
return box_rect;
}
void CheckBox::paint_event(PaintEvent& event)
{
Painter painter(*this);
painter.add_clip_rect(event.rect());
auto box_rect = this->box_rect();
auto text_rect = rect();
if (m_checkbox_position == CheckBoxPosition::Left)
text_rect.set_left(box_rect.right() + gap_between_box_and_rect());
text_rect.set_width(font().width_rounded_up(text()));
text_rect.set_top(height() / 2 - font().pixel_size_rounded_up() / 2);
text_rect.set_height(font().pixel_size_rounded_up());
if (fill_with_background_color())
painter.fill_rect(rect(), palette().window());
if (is_enabled() && is_hovered())
painter.fill_rect(rect(), palette().hover_highlight());
Gfx::StylePainter::paint_check_box(painter, box_rect, palette(), is_enabled(), is_checked(), is_being_pressed());
paint_text(painter, text_rect, font(), Gfx::TextAlignment::TopLeft);
if (is_focused())
painter.draw_focus_rect(text_rect.inflated(6, 6), palette().focus_outline());
}
void CheckBox::click(unsigned)
{
if (!is_enabled())
return;
set_checked(!is_checked());
}
void CheckBox::set_autosize(bool autosize)
{
if (m_autosize == autosize)
return;
m_autosize = autosize;
if (m_autosize)
size_to_fit();
}
void CheckBox::size_to_fit()
{
set_fixed_width(box_rect().width() + gap_between_box_and_rect() + font().width_rounded_up(text()) + horizontal_padding() * 2);
}
Optional<UISize> CheckBox::calculated_min_size() const
{
auto const& font = this->font();
int width = box_rect().width();
int height = max(22, max(static_cast<int>(ceilf(font.pixel_size())) + 8, box_rect().height()));
return UISize(width, height);
}
}