1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-24 22:22:32 +00:00
serenity/Userland/Libraries/LibGUI/TextBox.cpp
faxe1008 86d5e5a90c LibGUI: Add button to PasswordBox to reveal the password
This patch adds an eye-like button to the PasswordBox
which allows the user to reveal the typed password.
Whether or not the button is shown is controllable via
an additional property within the PasswordBox.
2022-05-08 16:36:53 +02:00

161 lines
4.6 KiB
C++

/*
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
* Copyright (c) 2022, the SerenityOS developers.
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibGUI/Painter.h>
#include <LibGUI/TextBox.h>
#include <LibGfx/Palette.h>
REGISTER_WIDGET(GUI, TextBox)
REGISTER_WIDGET(GUI, PasswordBox)
REGISTER_WIDGET(GUI, UrlBox)
namespace GUI {
TextBox::TextBox()
: TextEditor(TextEditor::SingleLine)
{
set_min_width(32);
set_fixed_height(22);
}
void TextBox::keydown_event(GUI::KeyEvent& event)
{
TextEditor::keydown_event(event);
if (event.key() == Key_Up) {
if (on_up_pressed)
on_up_pressed();
if (has_no_history() || !can_go_backwards_in_history())
return;
if (m_history_index >= static_cast<int>(m_history.size()))
m_saved_input = text();
m_history_index--;
set_text(m_history[m_history_index]);
} else if (event.key() == Key_Down) {
if (on_down_pressed)
on_down_pressed();
if (has_no_history())
return;
if (can_go_forwards_in_history()) {
m_history_index++;
set_text(m_history[m_history_index]);
} else if (m_history_index < static_cast<int>(m_history.size())) {
m_history_index++;
set_text(m_saved_input);
}
}
}
void TextBox::add_current_text_to_history()
{
if (!m_history_enabled)
return;
auto input = text();
if (m_history.is_empty() || m_history.last() != input)
add_input_to_history(input);
m_history_index = static_cast<int>(m_history.size());
m_saved_input = {};
}
void TextBox::add_input_to_history(String input)
{
m_history.append(move(input));
m_history_index++;
}
constexpr u32 password_box_substitution_code_point = '*';
PasswordBox::PasswordBox()
: TextBox()
{
set_substitution_code_point(password_box_substitution_code_point);
set_text_is_secret(true);
REGISTER_BOOL_PROPERTY("show_reveal_button", is_showing_reveal_button, set_show_reveal_button);
}
Gfx::IntRect PasswordBox::reveal_password_button_rect() const
{
constexpr i32 button_box_margin = 3;
auto button_box_size = height() - button_box_margin - button_box_margin;
return { width() - button_box_size - button_box_margin, button_box_margin, button_box_size, button_box_size };
}
void PasswordBox::paint_event(PaintEvent& event)
{
TextBox::paint_event(event);
if (is_showing_reveal_button()) {
auto button_rect = reveal_password_button_rect();
Painter painter(*this);
painter.add_clip_rect(event.rect());
auto icon_color = palette().button_text();
if (substitution_code_point().has_value())
icon_color = palette().disabled_text_front();
i32 dot_indicator_padding = height() / 5;
Gfx::IntRect dot_indicator_rect = { button_rect.x() + dot_indicator_padding, button_rect.y() + dot_indicator_padding, button_rect.width() - dot_indicator_padding * 2, button_rect.height() - dot_indicator_padding * 2 };
painter.fill_ellipse(dot_indicator_rect, icon_color);
Gfx::IntPoint arc_start_point { dot_indicator_rect.x() - dot_indicator_padding / 2, dot_indicator_rect.y() + dot_indicator_padding / 2 };
Gfx::IntPoint arc_end_point = { dot_indicator_rect.top_right().x() + dot_indicator_padding / 2, dot_indicator_rect.top_right().y() + dot_indicator_padding / 2 };
Gfx::IntPoint arc_center_point = { dot_indicator_rect.center().x(), dot_indicator_rect.top() - dot_indicator_padding };
painter.draw_quadratic_bezier_curve(arc_center_point, arc_start_point, arc_end_point, icon_color, 1);
}
}
void PasswordBox::mousedown_event(GUI::MouseEvent& event)
{
if (is_showing_reveal_button() && reveal_password_button_rect().contains(event.position())) {
Optional<u32> next_substitution_code_point;
if (!substitution_code_point().has_value())
next_substitution_code_point = password_box_substitution_code_point;
set_substitution_code_point(next_substitution_code_point);
} else {
TextBox::mousedown_event(event);
}
}
UrlBox::UrlBox()
: TextBox()
{
set_auto_focusable(false);
}
void UrlBox::focusout_event(GUI::FocusEvent& event)
{
set_focus_transition(true);
TextBox::focusout_event(event);
}
void UrlBox::mousedown_event(GUI::MouseEvent& event)
{
if (is_displayonly())
return;
if (event.button() != MouseButton::Primary)
return;
if (is_focus_transition()) {
TextBox::select_current_line();
set_focus_transition(false);
} else {
TextBox::mousedown_event(event);
}
}
}