diff options
Diffstat (limited to 'Userland/Libraries/LibGUI/Button.cpp')
-rw-r--r-- | Userland/Libraries/LibGUI/Button.cpp | 170 |
1 files changed, 170 insertions, 0 deletions
diff --git a/Userland/Libraries/LibGUI/Button.cpp b/Userland/Libraries/LibGUI/Button.cpp new file mode 100644 index 0000000000..94f7db79ae --- /dev/null +++ b/Userland/Libraries/LibGUI/Button.cpp @@ -0,0 +1,170 @@ +/* + * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org> + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * 1. Redistributions of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * 2. Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER + * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, + * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include <AK/StringBuilder.h> +#include <LibGUI/Action.h> +#include <LibGUI/ActionGroup.h> +#include <LibGUI/Button.h> +#include <LibGUI/Painter.h> +#include <LibGfx/Font.h> +#include <LibGfx/FontDatabase.h> +#include <LibGfx/Palette.h> +#include <LibGfx/StylePainter.h> + +REGISTER_WIDGET(GUI, Button) + +namespace GUI { + +Button::Button(String text) + : AbstractButton(move(text)) +{ + set_min_width(32); + set_fixed_height(22); + set_focus_policy(GUI::FocusPolicy::StrongFocus); + + REGISTER_ENUM_PROPERTY( + "button_style", button_style, set_button_style, Gfx::ButtonStyle, + { Gfx::ButtonStyle::Normal, "Normal" }, + { Gfx::ButtonStyle::CoolBar, "CoolBar" }); +} + +Button::~Button() +{ + if (m_action) + m_action->unregister_button({}, *this); +} + +void Button::paint_event(PaintEvent& event) +{ + Painter painter(*this); + painter.add_clip_rect(event.rect()); + + Gfx::StylePainter::paint_button(painter, rect(), palette(), m_button_style, is_being_pressed(), is_hovered(), is_checked(), is_enabled(), is_focused()); + + if (text().is_empty() && !m_icon) + return; + + auto content_rect = rect().shrunken(8, 2); + auto icon_location = m_icon ? content_rect.center().translated(-(m_icon->width() / 2), -(m_icon->height() / 2)) : Gfx::IntPoint(); + if (m_icon && !text().is_empty()) + icon_location.set_x(content_rect.x()); + + if (is_being_pressed() || is_checked()) + painter.translate(1, 1); + else if (m_icon && is_enabled() && is_hovered() && button_style() == Gfx::ButtonStyle::CoolBar) { + auto shadow_color = palette().button().darkened(0.7f); + painter.blit_filtered(icon_location.translated(1, 1), *m_icon, m_icon->rect(), [&shadow_color](auto) { + return shadow_color; + }); + icon_location.move_by(-1, -1); + } + + if (m_icon) { + if (is_enabled()) { + if (is_hovered()) + painter.blit_brightened(icon_location, *m_icon, m_icon->rect()); + else + painter.blit(icon_location, *m_icon, m_icon->rect()); + } else { + painter.blit_disabled(icon_location, *m_icon, m_icon->rect(), palette()); + } + } + auto& font = is_checked() ? Gfx::FontDatabase::default_bold_font() : this->font(); + if (m_icon && !text().is_empty()) { + content_rect.move_by(m_icon->width() + 4, 0); + content_rect.set_width(content_rect.width() - m_icon->width() - 4); + } + + Gfx::IntRect text_rect { 0, 0, font.width(text()), font.glyph_height() }; + if (text_rect.width() > content_rect.width()) + text_rect.set_width(content_rect.width()); + text_rect.align_within(content_rect, text_alignment()); + paint_text(painter, text_rect, font, text_alignment()); + + if (is_focused()) { + Gfx::IntRect focus_rect; + if (m_icon && !text().is_empty()) + focus_rect = text_rect.inflated(6, 6); + else + focus_rect = rect().shrunken(8, 8); + painter.draw_focus_rect(focus_rect, palette().focus_outline()); + } +} + +void Button::click(unsigned modifiers) +{ + if (!is_enabled()) + return; + + NonnullRefPtr protector = *this; + + if (is_checkable()) { + if (is_checked() && !is_uncheckable()) + return; + set_checked(!is_checked()); + } + if (on_click) + on_click(modifiers); + if (m_action) + m_action->activate(this); +} + +void Button::context_menu_event(ContextMenuEvent& context_menu_event) +{ + if (!is_enabled()) + return; + if (on_context_menu_request) + on_context_menu_request(context_menu_event); +} + +void Button::set_action(Action& action) +{ + m_action = action; + action.register_button({}, *this); + set_enabled(action.is_enabled()); + set_checkable(action.is_checkable()); + if (action.is_checkable()) + set_checked(action.is_checked()); +} + +void Button::set_icon(RefPtr<Gfx::Bitmap>&& icon) +{ + if (m_icon == icon) + return; + m_icon = move(icon); + update(); +} + +bool Button::is_uncheckable() const +{ + if (!m_action) + return true; + if (!m_action->group()) + return true; + return m_action->group()->is_unchecking_allowed(); +} + +} |