1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-05-19 23:15:08 +00:00

LibGUI: Add a GProgressBar widget.

This commit is contained in:
Andreas Kling 2019-03-22 02:49:14 +01:00
parent 23fe630057
commit c3b0c1ba68
4 changed files with 123 additions and 2 deletions

65
LibGUI/GProgressBar.cpp Normal file
View file

@ -0,0 +1,65 @@
#include <LibGUI/GProgressBar.h>
#include <SharedGraphics/Painter.h>
GProgressBar::GProgressBar(GWidget* parent)
: GWidget(parent)
{
start_timer(10);
}
GProgressBar::~GProgressBar()
{
}
void GProgressBar::set_value(int value)
{
if (m_value == value)
return;
m_value = value;
update();
}
void GProgressBar::set_range(int min, int max)
{
ASSERT(min < max);
m_min = min;
m_max = max;
if (m_value > m_max)
m_value = m_max;
if (m_value < m_min)
m_value = m_min;
}
void GProgressBar::paint_event(GPaintEvent& event)
{
Painter painter(*this);
painter.set_clip_rect(event.rect());
// First we fill the entire widget with the gradient. This incurs a bit of
// overdraw but ensures a consistent look throughout the progression.
Color start_color(110, 34, 9);
Color end_color(244, 202, 158);
painter.fill_rect_with_gradient(rect(), start_color, end_color);
float range_size = m_max - m_min;
float progress = (m_value - m_min) / range_size;
// Then we draw the progress text over the gradient.
// We draw it twice, once offset (1, 1) for a drop shadow look.
auto progress_text = String::format("%d%%", (int)(progress * 100));
painter.draw_text(rect().translated(1, 1), progress_text, TextAlignment::Center, Color::Black);
painter.draw_text(rect(), progress_text, TextAlignment::Center, Color::White);
// Then we carve out a hole in the remaining part of the widget.
// We draw the text a third time, clipped and inverse, for sharp contrast.
painter.save();
float progress_width = progress * width();
Rect hole_rect { (int)progress_width, 0, (int)(width() - progress_width), height() };
painter.set_clip_rect(hole_rect);
painter.fill_rect(hole_rect, Color::White);
painter.draw_text(rect().translated(0, 0), progress_text, TextAlignment::Center, Color::Black);
painter.restore();
// Finally, draw a frame around the widget.
painter.draw_rect(rect(), Color::Black);
}

22
LibGUI/GProgressBar.h Normal file
View file

@ -0,0 +1,22 @@
#pragma once
#include <LibGUI/GWidget.h>
class GProgressBar : public GWidget {
public:
explicit GProgressBar(GWidget* parent);
virtual ~GProgressBar() override;
void set_range(int min, int max);
void set_value(int);
int value() const { return m_value; }
protected:
virtual void paint_event(GPaintEvent&) override;
private:
int m_min { 0 };
int m_max { 100 };
int m_value { 0 };
};

View file

@ -47,6 +47,7 @@ LIBGUI_OBJS = \
GInputBox.o \ GInputBox.o \
GDialog.o \ GDialog.o \
GDesktop.o \ GDesktop.o \
GProgressBar.o \
GWindow.o GWindow.o
OBJS = $(SHAREDGRAPHICS_OBJS) $(LIBGUI_OBJS) OBJS = $(SHAREDGRAPHICS_OBJS) $(LIBGUI_OBJS)

View file

@ -14,11 +14,14 @@
#include <LibGUI/GLabel.h> #include <LibGUI/GLabel.h>
#include <LibGUI/GButton.h> #include <LibGUI/GButton.h>
#include <LibGUI/GTextBox.h> #include <LibGUI/GTextBox.h>
#include <LibGUI/GBoxLayout.h>
#include <LibGUI/GCheckBox.h> #include <LibGUI/GCheckBox.h>
#include <LibGUI/GProgressBar.h>
#include <LibGUI/GApplication.h> #include <LibGUI/GApplication.h>
#include <signal.h> #include <signal.h>
static GWindow* make_launcher_window(); static GWindow* make_launcher_window();
static GWindow* make_progress_window();
void handle_sigchld(int) void handle_sigchld(int)
{ {
@ -38,18 +41,21 @@ int main(int argc, char** argv)
launcher_window->set_should_exit_event_loop_on_close(true); launcher_window->set_should_exit_event_loop_on_close(true);
launcher_window->show(); launcher_window->show();
auto* progress_window = make_progress_window();
progress_window->show();
return app.exec(); return app.exec();
} }
GWindow* make_launcher_window() GWindow* make_launcher_window()
{ {
auto* window = new GWindow; auto* window = new GWindow;
window->set_title("guitest2"); window->set_title("GUI Test II");
window->set_rect({ 100, 400, 100, 230 }); window->set_rect({ 100, 400, 100, 230 });
auto* widget = new GWidget; auto* widget = new GWidget;
widget->set_fill_with_background_color(true);
window->set_main_widget(widget); window->set_main_widget(widget);
widget->set_relative_rect({ 0, 0, 100, 230 });
auto* label = new GLabel(widget); auto* label = new GLabel(widget);
label->set_relative_rect({ 0, 0, 100, 20 }); label->set_relative_rect({ 0, 0, 100, 20 });
@ -116,3 +122,30 @@ GWindow* make_launcher_window()
return window; return window;
} }
static GWindow* make_progress_window()
{
auto* window = new GWindow;
window->set_title("Progress bar test");
window->set_rect({ 100, 400, 240, 80 });
auto* widget = new GWidget;
widget->set_fill_with_background_color(true);
window->set_main_widget(widget);
widget->set_layout(make<GBoxLayout>(Orientation::Vertical));
widget->layout()->set_margins({ 8, 8, 8, 8 });
auto* label = new GLabel("Hi /dpt/", widget);
label->set_size_policy(SizePolicy::Fill, SizePolicy::Fill);
auto* progress_bar = new GProgressBar(widget);
progress_bar->set_size_policy(SizePolicy::Fixed, SizePolicy::Fixed);
progress_bar->set_preferred_size({ 200, 20 });
progress_bar->set_range(0, 100);
progress_bar->set_value(25);
return window;
}