mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-05-12 14:12:52 +00:00
LibGUI: Add a spinbox widget.
This is essentially a combo widget containing a single-line GTextEditor and two buttons for increment and decrement. The GTextEditor::on_change callback is hooked to prevent non-numeric input but it's not entirely perfect since that callback is asynchronous. This will work until we have some more sophisticated input validation mechanism though.
This commit is contained in:
parent
4ba5e472be
commit
4c0f586f2b
Notes:
sideshowbarker
2024-07-19 14:47:11 +09:00
Author: https://github.com/awesomekling
Commit: 4c0f586f2b
4 changed files with 115 additions and 14 deletions
|
@ -6,6 +6,7 @@
|
||||||
#include <LibGUI/GLabel.h>
|
#include <LibGUI/GLabel.h>
|
||||||
#include <LibGUI/GTextBox.h>
|
#include <LibGUI/GTextBox.h>
|
||||||
#include <LibGUI/GCheckBox.h>
|
#include <LibGUI/GCheckBox.h>
|
||||||
|
#include <LibGUI/GSpinBox.h>
|
||||||
|
|
||||||
FontEditorWidget::FontEditorWidget(const String& path, RetainPtr<Font>&& edited_font, GWidget* parent)
|
FontEditorWidget::FontEditorWidget(const String& path, RetainPtr<Font>&& edited_font, GWidget* parent)
|
||||||
: GWidget(parent)
|
: GWidget(parent)
|
||||||
|
@ -62,8 +63,9 @@ FontEditorWidget::FontEditorWidget(const String& path, RetainPtr<Font>&& edited_
|
||||||
info_label->set_text_alignment(TextAlignment::CenterLeft);
|
info_label->set_text_alignment(TextAlignment::CenterLeft);
|
||||||
info_label->set_relative_rect({ 5, 110, 100, 20 });
|
info_label->set_relative_rect({ 5, 110, 100, 20 });
|
||||||
|
|
||||||
auto* width_textbox = new GTextBox(this);
|
auto* width_spinbox = new GSpinBox(this);
|
||||||
width_textbox->set_relative_rect({ 5, 135, 100, 20 });
|
width_spinbox->set_range(0, 32);
|
||||||
|
width_spinbox->set_relative_rect({ 5, 135, 100, 20 });
|
||||||
|
|
||||||
auto* demo_label_1 = new GLabel(this);
|
auto* demo_label_1 = new GLabel(this);
|
||||||
demo_label_1->set_font(m_edited_font.copy_ref());
|
demo_label_1->set_font(m_edited_font.copy_ref());
|
||||||
|
@ -85,27 +87,23 @@ FontEditorWidget::FontEditorWidget(const String& path, RetainPtr<Font>&& edited_
|
||||||
update_demo();
|
update_demo();
|
||||||
};
|
};
|
||||||
|
|
||||||
m_glyph_map_widget->on_glyph_selected = [this, info_label, width_textbox] (byte glyph) {
|
m_glyph_map_widget->on_glyph_selected = [this, info_label, width_spinbox] (byte glyph) {
|
||||||
m_glyph_editor_widget->set_glyph(glyph);
|
m_glyph_editor_widget->set_glyph(glyph);
|
||||||
width_textbox->set_text(String::format("%u", m_edited_font->glyph_width(m_glyph_map_widget->selected_glyph())));
|
width_spinbox->set_value(m_edited_font->glyph_width(m_glyph_map_widget->selected_glyph()));
|
||||||
info_label->set_text(String::format("0x%b (%c)", glyph, glyph));
|
info_label->set_text(String::format("0x%b (%c)", glyph, glyph));
|
||||||
};
|
};
|
||||||
|
|
||||||
fixed_width_checkbox->on_change = [this, width_textbox, update_demo] (GCheckBox&, bool is_checked) {
|
fixed_width_checkbox->on_change = [this, width_spinbox, update_demo] (GCheckBox&, bool is_checked) {
|
||||||
m_edited_font->set_fixed_width(is_checked);
|
m_edited_font->set_fixed_width(is_checked);
|
||||||
width_textbox->set_text(String::format("%u", m_edited_font->glyph_width(m_glyph_map_widget->selected_glyph())));
|
width_spinbox->set_value(m_edited_font->glyph_width(m_glyph_map_widget->selected_glyph()));
|
||||||
m_glyph_editor_widget->update();
|
m_glyph_editor_widget->update();
|
||||||
update_demo();
|
update_demo();
|
||||||
};
|
};
|
||||||
|
|
||||||
width_textbox->on_change = [this, update_demo] (GTextBox& textbox) {
|
width_spinbox->on_change = [this, update_demo] (int value) {
|
||||||
bool ok;
|
m_edited_font->set_glyph_width(m_glyph_map_widget->selected_glyph(), value);
|
||||||
unsigned width = textbox.text().to_uint(ok);
|
m_glyph_editor_widget->update();
|
||||||
if (ok) {
|
update_demo();
|
||||||
m_edited_font->set_glyph_width(m_glyph_map_widget->selected_glyph(), width);
|
|
||||||
m_glyph_editor_widget->update();
|
|
||||||
update_demo();
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
m_glyph_map_widget->set_selected_glyph('A');
|
m_glyph_map_widget->set_selected_glyph('A');
|
||||||
|
|
71
LibGUI/GSpinBox.cpp
Normal file
71
LibGUI/GSpinBox.cpp
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
#include <LibGUI/GSpinBox.h>
|
||||||
|
#include <LibGUI/GButton.h>
|
||||||
|
#include <LibGUI/GTextEditor.h>
|
||||||
|
|
||||||
|
GSpinBox::GSpinBox(GWidget* parent)
|
||||||
|
: GWidget(parent)
|
||||||
|
{
|
||||||
|
m_editor = new GTextEditor(GTextEditor::Type::SingleLine, this);
|
||||||
|
m_editor->on_change = [this] {
|
||||||
|
bool ok;
|
||||||
|
int value = m_editor->text().to_uint(ok);
|
||||||
|
if (ok)
|
||||||
|
set_value(value);
|
||||||
|
else
|
||||||
|
m_editor->set_text(String::format("%d", m_value));
|
||||||
|
};
|
||||||
|
m_increment_button = new GButton(this);
|
||||||
|
m_increment_button->set_caption("+");
|
||||||
|
m_increment_button->on_click = [this] (GButton&) { set_value(m_value + 1); };
|
||||||
|
m_decrement_button = new GButton(this);
|
||||||
|
m_decrement_button->set_caption("-");
|
||||||
|
m_decrement_button->on_click = [this] (GButton&) { set_value(m_value - 1); };
|
||||||
|
}
|
||||||
|
|
||||||
|
GSpinBox::~GSpinBox()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void GSpinBox::set_value(int value)
|
||||||
|
{
|
||||||
|
if (value < m_min)
|
||||||
|
value = m_min;
|
||||||
|
if (value > m_max)
|
||||||
|
value = m_max;
|
||||||
|
if (m_value == value)
|
||||||
|
return;
|
||||||
|
m_value = value;
|
||||||
|
m_editor->set_text(String::format("%d", value));
|
||||||
|
update();
|
||||||
|
if (on_change)
|
||||||
|
on_change(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GSpinBox::set_range(int min, int max)
|
||||||
|
{
|
||||||
|
ASSERT(min <= max);
|
||||||
|
if (m_min == min && m_max == max)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_min = min;
|
||||||
|
m_max = max;
|
||||||
|
|
||||||
|
int old_value = m_value;
|
||||||
|
if (m_value < m_min)
|
||||||
|
m_value = m_min;
|
||||||
|
if (m_value > m_max)
|
||||||
|
m_value = m_max;
|
||||||
|
if (on_change && m_value != old_value)
|
||||||
|
on_change(m_value);
|
||||||
|
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void GSpinBox::resize_event(GResizeEvent& event)
|
||||||
|
{
|
||||||
|
int button_height = event.size().height() / 2;
|
||||||
|
int button_width = 16;
|
||||||
|
m_increment_button->set_relative_rect(width() - button_width, 0, button_width, button_height);
|
||||||
|
m_decrement_button->set_relative_rect(width() - button_width, button_height, button_width, button_height);
|
||||||
|
m_editor->set_relative_rect(0, 0, width() - button_width, height());
|
||||||
|
}
|
31
LibGUI/GSpinBox.h
Normal file
31
LibGUI/GSpinBox.h
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <LibGUI/GWidget.h>
|
||||||
|
|
||||||
|
class GButton;
|
||||||
|
class GTextEditor;
|
||||||
|
|
||||||
|
class GSpinBox : public GWidget {
|
||||||
|
public:
|
||||||
|
GSpinBox(GWidget* parent = nullptr);
|
||||||
|
virtual ~GSpinBox() override;
|
||||||
|
|
||||||
|
int value() const { return m_value; }
|
||||||
|
void set_value(int);
|
||||||
|
|
||||||
|
void set_range(int min, int max);
|
||||||
|
|
||||||
|
Function<void(int value)> on_change;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual void resize_event(GResizeEvent&) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
GTextEditor* m_editor { nullptr };
|
||||||
|
GButton* m_increment_button { nullptr };
|
||||||
|
GButton* m_decrement_button { nullptr };
|
||||||
|
|
||||||
|
int m_min { 0 };
|
||||||
|
int m_max { 100 };
|
||||||
|
int m_value { 0 };
|
||||||
|
};
|
|
@ -63,6 +63,7 @@ LIBGUI_OBJS = \
|
||||||
GHttpRequest.o \
|
GHttpRequest.o \
|
||||||
GHttpResponse.o \
|
GHttpResponse.o \
|
||||||
GHttpJob.o \
|
GHttpJob.o \
|
||||||
|
GSpinBox.o \
|
||||||
GWindow.o
|
GWindow.o
|
||||||
|
|
||||||
OBJS = $(SHAREDGRAPHICS_OBJS) $(LIBGUI_OBJS)
|
OBJS = $(SHAREDGRAPHICS_OBJS) $(LIBGUI_OBJS)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue