mirror of
https://github.com/RGBCube/serenity
synced 2025-07-25 13:07:46 +00:00
LibGUI: More work on GCheckBox.
- Make it track the mouse cursor just like GButton does so that changes only get committed if the mouseup event happens while inside the widget rect. - Draw a focus rect around the box when appropriate. - When focused, support toggling the checked state with the space bar.
This commit is contained in:
parent
90e898b771
commit
35c06f1520
3 changed files with 111 additions and 60 deletions
|
@ -1,10 +1,33 @@
|
||||||
#include "GCheckBox.h"
|
#include "GCheckBox.h"
|
||||||
#include <SharedGraphics/Painter.h>
|
#include <SharedGraphics/Painter.h>
|
||||||
#include <SharedGraphics/CharacterBitmap.h>
|
#include <SharedGraphics/CharacterBitmap.h>
|
||||||
|
#include <Kernel/KeyCode.h>
|
||||||
|
|
||||||
|
//#define GCHECKBOX_DEBUG
|
||||||
|
|
||||||
|
static const char* s_checked_bitmap_data = {
|
||||||
|
" "
|
||||||
|
" ## "
|
||||||
|
" ## "
|
||||||
|
" ## "
|
||||||
|
" ## "
|
||||||
|
" ## ## "
|
||||||
|
" #### "
|
||||||
|
" ## "
|
||||||
|
" "
|
||||||
|
};
|
||||||
|
|
||||||
|
static CharacterBitmap* s_checked_bitmap;
|
||||||
|
static const int s_checked_bitmap_width = 9;
|
||||||
|
static const int s_checked_bitmap_height = 9;
|
||||||
|
static const int s_box_width = 11;
|
||||||
|
static const int s_box_height = 11;
|
||||||
|
|
||||||
GCheckBox::GCheckBox(GWidget* parent)
|
GCheckBox::GCheckBox(GWidget* parent)
|
||||||
: GWidget(parent)
|
: GWidget(parent)
|
||||||
{
|
{
|
||||||
|
if (!s_checked_bitmap)
|
||||||
|
s_checked_bitmap = CharacterBitmap::create_from_ascii(s_checked_bitmap_data, s_checked_bitmap_width, s_checked_bitmap_height).leakRef();
|
||||||
}
|
}
|
||||||
|
|
||||||
GCheckBox::~GCheckBox()
|
GCheckBox::~GCheckBox()
|
||||||
|
@ -27,76 +50,95 @@ void GCheckBox::set_checked(bool b)
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char* uncheckedBitmap = {
|
|
||||||
"###########"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"# #"
|
|
||||||
"###########"
|
|
||||||
};
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
static const char* checkedBitmap = {
|
|
||||||
"############"
|
|
||||||
"# #"
|
|
||||||
"# ## #"
|
|
||||||
"# ## #"
|
|
||||||
"# ## #"
|
|
||||||
"# ## #"
|
|
||||||
"# ## #"
|
|
||||||
"# ## ## #"
|
|
||||||
"# ## ## #"
|
|
||||||
"# ### #"
|
|
||||||
"# #"
|
|
||||||
"############"
|
|
||||||
};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static const char* checkedBitmap = {
|
|
||||||
"###########"
|
|
||||||
"## ##"
|
|
||||||
"# # # #"
|
|
||||||
"# # # #"
|
|
||||||
"# # # #"
|
|
||||||
"# # #"
|
|
||||||
"# # # #"
|
|
||||||
"# # # #"
|
|
||||||
"# # # #"
|
|
||||||
"## ##"
|
|
||||||
"###########"
|
|
||||||
};
|
|
||||||
|
|
||||||
void GCheckBox::paint_event(GPaintEvent&)
|
void GCheckBox::paint_event(GPaintEvent&)
|
||||||
{
|
{
|
||||||
Painter painter(*this);
|
Painter painter(*this);
|
||||||
auto bitmap = CharacterBitmap::create_from_ascii(is_checked() ? checkedBitmap : uncheckedBitmap, 11, 11);
|
|
||||||
|
|
||||||
auto textRect = rect();
|
auto text_rect = rect();
|
||||||
textRect.set_left(bitmap->width() + 4);
|
text_rect.set_left(s_box_width + 4);
|
||||||
textRect.set_top(height() / 2 - font().glyph_height() / 2);
|
text_rect.set_top(height() / 2 - font().glyph_height() / 2);
|
||||||
|
|
||||||
Point bitmapPosition;
|
if (fill_with_background_color())
|
||||||
bitmapPosition.set_x(2);
|
painter.fill_rect(rect(), background_color());
|
||||||
bitmapPosition.set_y(height() / 2 - bitmap->height() / 2 - 1);
|
|
||||||
|
|
||||||
painter.fill_rect(rect(), background_color());
|
Rect box_rect {
|
||||||
painter.draw_bitmap(bitmapPosition, *bitmap, foreground_color());
|
2, height() / 2 - s_box_height / 2 - 1,
|
||||||
|
s_box_width, s_box_height
|
||||||
|
};
|
||||||
|
painter.draw_rect(box_rect, Color::Black);
|
||||||
|
|
||||||
if (!caption().is_empty()) {
|
if (m_being_modified) {
|
||||||
painter.draw_text(textRect, caption(), Painter::TextAlignment::TopLeft, foreground_color());
|
auto modification_rect = box_rect;
|
||||||
|
modification_rect.shrink(2, 2);
|
||||||
|
painter.draw_rect(modification_rect, Color::MidGray);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (m_checked) {
|
||||||
|
auto bitmap_rect = box_rect;
|
||||||
|
bitmap_rect.shrink(2, 2);
|
||||||
|
painter.draw_bitmap(bitmap_rect.location(), *s_checked_bitmap, foreground_color());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!caption().is_empty())
|
||||||
|
painter.draw_text(text_rect, caption(), Painter::TextAlignment::TopLeft, foreground_color());
|
||||||
|
|
||||||
|
if (is_focused()) {
|
||||||
|
// NOTE: Painter::draw_focus_rect() will shrink(2,2) the passed rect.
|
||||||
|
auto focus_rect = box_rect;
|
||||||
|
focus_rect.inflate(4, 4);
|
||||||
|
painter.draw_focus_rect(focus_rect);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GCheckBox::mousemove_event(GMouseEvent& event)
|
||||||
|
{
|
||||||
|
if (m_tracking_cursor) {
|
||||||
|
bool being_pressed = rect().contains(event.position());
|
||||||
|
if (being_pressed != m_being_modified) {
|
||||||
|
m_being_modified = being_pressed;
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
GWidget::mousemove_event(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
void GCheckBox::mousedown_event(GMouseEvent& event)
|
void GCheckBox::mousedown_event(GMouseEvent& event)
|
||||||
{
|
{
|
||||||
dbgprintf("GCheckBox::mouseDownEvent: x=%d, y=%d, button=%u\n", event.x(), event.y(), (unsigned)event.button());
|
#ifdef GCHECKBOX_DEBUG
|
||||||
|
dbgprintf("GCheckBox::mouse_down_event: x=%d, y=%d, button=%u\n", event.x(), event.y(), (unsigned)event.button());
|
||||||
set_checked(!is_checked());
|
#endif
|
||||||
|
if (event.button() == GMouseButton::Left) {
|
||||||
|
m_being_modified = true;
|
||||||
|
m_tracking_cursor = true;
|
||||||
|
set_global_cursor_tracking(true);
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
GWidget::mousedown_event(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GCheckBox::mouseup_event(GMouseEvent& event)
|
||||||
|
{
|
||||||
|
#ifdef GCHECKBOX_DEBUG
|
||||||
|
dbgprintf("GCheckBox::mouseup_event: x=%d, y=%d, button=%u\n", event.x(), event.y(), (unsigned)event.button());
|
||||||
|
#endif
|
||||||
|
if (event.button() == GMouseButton::Left) {
|
||||||
|
bool was_being_pressed = m_being_modified;
|
||||||
|
m_being_modified = false;
|
||||||
|
m_tracking_cursor = false;
|
||||||
|
set_global_cursor_tracking(false);
|
||||||
|
if (was_being_pressed) {
|
||||||
|
set_checked(!is_checked());
|
||||||
|
}
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
GWidget::mouseup_event(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GCheckBox::keydown_event(GKeyEvent& event)
|
||||||
|
{
|
||||||
|
if (!m_tracking_cursor && event.key() == KeyCode::Key_Space) {
|
||||||
|
set_checked(!is_checked());
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
GWidget::keydown_event(event);
|
||||||
|
}
|
||||||
|
|
|
@ -17,11 +17,15 @@ public:
|
||||||
private:
|
private:
|
||||||
virtual void paint_event(GPaintEvent&) override;
|
virtual void paint_event(GPaintEvent&) override;
|
||||||
virtual void mousedown_event(GMouseEvent&) override;
|
virtual void mousedown_event(GMouseEvent&) override;
|
||||||
|
virtual void mouseup_event(GMouseEvent&) override;
|
||||||
|
virtual void mousemove_event(GMouseEvent&) override;
|
||||||
|
virtual void keydown_event(GKeyEvent&) override;
|
||||||
virtual const char* class_name() const override { return "GCheckBox"; }
|
virtual const char* class_name() const override { return "GCheckBox"; }
|
||||||
virtual bool accepts_focus() const override { return true; }
|
virtual bool accepts_focus() const override { return true; }
|
||||||
|
|
||||||
String m_caption;
|
String m_caption;
|
||||||
bool m_checked { false };
|
bool m_checked { false };
|
||||||
|
bool m_being_modified { false };
|
||||||
|
bool m_tracking_cursor { false };
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
#include <LibGUI/GButton.h>
|
#include <LibGUI/GButton.h>
|
||||||
#include <LibGUI/GEventLoop.h>
|
#include <LibGUI/GEventLoop.h>
|
||||||
#include <LibGUI/GTextBox.h>
|
#include <LibGUI/GTextBox.h>
|
||||||
|
#include <LibGUI/GCheckBox.h>
|
||||||
|
|
||||||
static GWindow* make_font_test_window();
|
static GWindow* make_font_test_window();
|
||||||
static GWindow* make_launcher_window();
|
static GWindow* make_launcher_window();
|
||||||
|
@ -117,6 +118,10 @@ GWindow* make_launcher_window()
|
||||||
auto* other_textbox = new GTextBox(widget);
|
auto* other_textbox = new GTextBox(widget);
|
||||||
other_textbox->set_relative_rect({ 5, 140, 90, 20 });
|
other_textbox->set_relative_rect({ 5, 140, 90, 20 });
|
||||||
|
|
||||||
|
auto* checkbox = new GCheckBox(widget);
|
||||||
|
checkbox->set_relative_rect({ 5, 170, 90, 20 });
|
||||||
|
checkbox->set_caption("CheckBox");
|
||||||
|
|
||||||
window->set_focused_widget(textbox);
|
window->set_focused_widget(textbox);
|
||||||
|
|
||||||
return window;
|
return window;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue