mirror of
https://github.com/RGBCube/serenity
synced 2025-05-30 14:08:10 +00:00

The height of a line or column doesn't change unless the font changes, and we were already caching the line height. This patch extends it so we also cache the column width.
241 lines
7.4 KiB
C++
241 lines
7.4 KiB
C++
/*
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2022, the SerenityOS developers.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/DeprecatedString.h>
|
|
#include <LibCore/ElapsedTimer.h>
|
|
#include <LibCore/Notifier.h>
|
|
#include <LibCore/Timer.h>
|
|
#include <LibGUI/Clipboard.h>
|
|
#include <LibGUI/Frame.h>
|
|
#include <LibGfx/Bitmap.h>
|
|
#include <LibGfx/Rect.h>
|
|
#include <LibVT/Color.h>
|
|
#include <LibVT/Range.h>
|
|
#include <LibVT/Terminal.h>
|
|
|
|
namespace VT {
|
|
|
|
class TerminalWidget final
|
|
: public GUI::Frame
|
|
, public VT::TerminalClient
|
|
, public GUI::Clipboard::ClipboardClient {
|
|
C_OBJECT(TerminalWidget);
|
|
|
|
public:
|
|
virtual ~TerminalWidget() override = default;
|
|
|
|
void set_pty_master_fd(int fd);
|
|
void inject_string(StringView string)
|
|
{
|
|
m_terminal.inject_string(string);
|
|
flush_dirty_lines();
|
|
}
|
|
|
|
void flush_dirty_lines();
|
|
|
|
void apply_size_increments_to_window(GUI::Window&);
|
|
|
|
void set_opacity(u8);
|
|
float opacity() { return m_opacity; };
|
|
|
|
void set_show_scrollbar(bool);
|
|
|
|
enum class BellMode {
|
|
Visible,
|
|
AudibleBeep,
|
|
Disabled
|
|
};
|
|
|
|
BellMode bell_mode() { return m_bell_mode; }
|
|
void set_bell_mode(BellMode bm) { m_bell_mode = bm; };
|
|
|
|
bool has_selection() const;
|
|
bool selection_contains(const VT::Position&) const;
|
|
DeprecatedString selected_text() const;
|
|
VT::Range normalized_selection() const { return m_selection.normalized(); }
|
|
void set_selection(const VT::Range& selection);
|
|
VT::Position buffer_position_at(Gfx::IntPoint) const;
|
|
|
|
VT::Range find_next(StringView, const VT::Position& start = {}, bool case_sensitivity = false, bool should_wrap = false);
|
|
VT::Range find_previous(StringView, const VT::Position& start = {}, bool case_sensitivity = false, bool should_wrap = false);
|
|
|
|
void scroll_to_bottom();
|
|
void scroll_to_row(int);
|
|
|
|
bool is_scrollable() const;
|
|
int scroll_length() const;
|
|
|
|
size_t max_history_size() const { return m_terminal.max_history_size(); }
|
|
void set_max_history_size(size_t value) { m_terminal.set_max_history_size(value); }
|
|
|
|
GUI::Action& copy_action() { return *m_copy_action; }
|
|
GUI::Action& paste_action() { return *m_paste_action; }
|
|
GUI::Action& clear_including_history_action() { return *m_clear_including_history_action; }
|
|
|
|
void copy();
|
|
void paste();
|
|
void clear_including_history();
|
|
|
|
const StringView color_scheme_name() const { return m_color_scheme_name; }
|
|
|
|
Function<void(StringView)> on_title_change;
|
|
Function<void(Gfx::IntSize)> on_terminal_size_change;
|
|
Function<void()> on_command_exit;
|
|
|
|
GUI::Menu& context_menu() { return *m_context_menu; }
|
|
|
|
constexpr Gfx::Color terminal_color_to_rgb(VT::Color) const;
|
|
|
|
void set_font_and_resize_to_fit(Gfx::Font const&);
|
|
|
|
void update_color_scheme();
|
|
|
|
void set_logical_focus(bool);
|
|
|
|
VT::CursorShape cursor_shape() { return m_cursor_shape; }
|
|
virtual void set_cursor_blinking(bool) override;
|
|
virtual void set_cursor_shape(CursorShape) override;
|
|
|
|
static Optional<VT::CursorShape> parse_cursor_shape(StringView);
|
|
static DeprecatedString stringify_cursor_shape(VT::CursorShape);
|
|
|
|
private:
|
|
TerminalWidget(int ptm_fd, bool automatic_size_policy);
|
|
|
|
// ^GUI::Widget
|
|
virtual void event(Core::Event&) override;
|
|
virtual void paint_event(GUI::PaintEvent&) override;
|
|
virtual void resize_event(GUI::ResizeEvent&) override;
|
|
virtual void keydown_event(GUI::KeyEvent&) override;
|
|
virtual void keyup_event(GUI::KeyEvent&) override;
|
|
virtual void mousedown_event(GUI::MouseEvent&) override;
|
|
virtual void mouseup_event(GUI::MouseEvent&) override;
|
|
virtual void mousemove_event(GUI::MouseEvent&) override;
|
|
virtual void mousewheel_event(GUI::MouseEvent&) override;
|
|
virtual void doubleclick_event(GUI::MouseEvent&) override;
|
|
virtual void focusin_event(GUI::FocusEvent&) override;
|
|
virtual void focusout_event(GUI::FocusEvent&) override;
|
|
virtual void context_menu_event(GUI::ContextMenuEvent&) override;
|
|
virtual void drag_enter_event(GUI::DragEvent&) override;
|
|
virtual void drop_event(GUI::DropEvent&) override;
|
|
virtual void leave_event(Core::Event&) override;
|
|
virtual void did_change_font() override;
|
|
|
|
// ^TerminalClient
|
|
virtual void beep() override;
|
|
virtual void set_window_title(StringView) override;
|
|
virtual void set_window_progress(int value, int max) override;
|
|
virtual void terminal_did_resize(u16 columns, u16 rows) override;
|
|
virtual void terminal_history_changed(int delta) override;
|
|
virtual void emit(u8 const*, size_t) override;
|
|
|
|
// ^GUI::Clipboard::ClipboardClient
|
|
virtual void clipboard_content_did_change(DeprecatedString const&) override { update_paste_action(); }
|
|
|
|
void send_non_user_input(ReadonlyBytes);
|
|
|
|
Gfx::IntRect glyph_rect(u16 row, u16 column);
|
|
Gfx::IntRect row_rect(u16 row);
|
|
|
|
Gfx::IntSize widget_size_for_font(Gfx::Font const&) const;
|
|
|
|
void update_cursor();
|
|
void invalidate_cursor();
|
|
|
|
void relayout(Gfx::IntSize);
|
|
|
|
void update_copy_action();
|
|
void update_paste_action();
|
|
|
|
Gfx::IntSize compute_base_size() const;
|
|
int first_selection_column_on_row(int row) const;
|
|
int last_selection_column_on_row(int row) const;
|
|
|
|
u32 code_point_at(const VT::Position&) const;
|
|
VT::Position next_position_after(const VT::Position&, bool should_wrap) const;
|
|
VT::Position previous_position_before(const VT::Position&, bool should_wrap) const;
|
|
|
|
void update_cached_font_metrics();
|
|
|
|
VT::Terminal m_terminal;
|
|
|
|
VT::Range m_selection;
|
|
|
|
DeprecatedString m_hovered_href;
|
|
DeprecatedString m_hovered_href_id;
|
|
|
|
DeprecatedString m_active_href;
|
|
DeprecatedString m_active_href_id;
|
|
|
|
// Snapshot of m_hovered_href when opening a context menu for a hyperlink.
|
|
DeprecatedString m_context_menu_href;
|
|
|
|
Gfx::Color m_colors[256];
|
|
Gfx::Color m_default_foreground_color;
|
|
Gfx::Color m_default_background_color;
|
|
bool m_show_bold_text_as_bright { true };
|
|
|
|
DeprecatedString m_color_scheme_name;
|
|
|
|
BellMode m_bell_mode { BellMode::Visible };
|
|
bool m_alt_key_held { false };
|
|
bool m_rectangle_selection { false };
|
|
|
|
int m_pixel_width { 0 };
|
|
int m_pixel_height { 0 };
|
|
|
|
int m_inset { 2 };
|
|
int m_line_spacing { 4 };
|
|
int m_line_height { 0 };
|
|
int m_column_width { 0 };
|
|
|
|
int m_ptm_fd { -1 };
|
|
|
|
bool m_has_logical_focus { false };
|
|
bool m_in_relayout { false };
|
|
|
|
RefPtr<Core::Notifier> m_notifier;
|
|
|
|
u8 m_opacity { 255 };
|
|
bool m_cursor_blink_state { true };
|
|
bool m_automatic_size_policy { false };
|
|
|
|
VT::CursorShape m_cursor_shape { VT::CursorShape::Block };
|
|
bool m_cursor_is_blinking_set { true };
|
|
|
|
enum class AutoScrollDirection {
|
|
None,
|
|
Up,
|
|
Down
|
|
};
|
|
|
|
void set_auto_scroll_direction(AutoScrollDirection);
|
|
|
|
AutoScrollDirection m_auto_scroll_direction { AutoScrollDirection::None };
|
|
|
|
RefPtr<Core::Timer> m_cursor_blink_timer;
|
|
RefPtr<Core::Timer> m_visual_beep_timer;
|
|
RefPtr<Core::Timer> m_auto_scroll_timer;
|
|
|
|
RefPtr<GUI::Scrollbar> m_scrollbar;
|
|
|
|
RefPtr<GUI::Action> m_copy_action;
|
|
RefPtr<GUI::Action> m_paste_action;
|
|
RefPtr<GUI::Action> m_clear_including_history_action;
|
|
|
|
RefPtr<GUI::Menu> m_context_menu;
|
|
RefPtr<GUI::Menu> m_context_menu_for_hyperlink;
|
|
|
|
Core::ElapsedTimer m_triple_click_timer;
|
|
|
|
Gfx::IntPoint m_left_mousedown_position;
|
|
VT::Position m_left_mousedown_position_buffer;
|
|
};
|
|
|
|
}
|