TerminalWidget.h 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. /*
  2. * Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/ByteString.h>
  9. #include <LibCore/ElapsedTimer.h>
  10. #include <LibCore/Notifier.h>
  11. #include <LibCore/Timer.h>
  12. #include <LibGUI/Clipboard.h>
  13. #include <LibGUI/Frame.h>
  14. #include <LibGfx/Bitmap.h>
  15. #include <LibGfx/Rect.h>
  16. #include <LibVT/Color.h>
  17. #include <LibVT/Range.h>
  18. #include <LibVT/Terminal.h>
  19. namespace VT {
  20. class TerminalWidget final
  21. : public GUI::Frame
  22. , public VT::TerminalClient
  23. , public GUI::Clipboard::ClipboardClient {
  24. C_OBJECT(TerminalWidget);
  25. public:
  26. virtual ~TerminalWidget() override = default;
  27. void set_pty_master_fd(int fd);
  28. void inject_string(StringView string)
  29. {
  30. m_terminal.inject_string(string);
  31. flush_dirty_lines();
  32. }
  33. void flush_dirty_lines();
  34. void apply_size_increments_to_window(GUI::Window&);
  35. void set_opacity(u8);
  36. float opacity() { return m_opacity; }
  37. void set_show_scrollbar(bool);
  38. enum class BellMode {
  39. Visible,
  40. AudibleBeep,
  41. Disabled
  42. };
  43. BellMode bell_mode() { return m_bell_mode; }
  44. void set_bell_mode(BellMode bm) { m_bell_mode = bm; }
  45. enum class AutoMarkMode {
  46. MarkNothing,
  47. MarkInteractiveShellPrompt,
  48. };
  49. AutoMarkMode auto_mark_mode() { return m_auto_mark_mode; }
  50. void set_auto_mark_mode(AutoMarkMode am) { m_auto_mark_mode = am; }
  51. bool has_selection() const;
  52. bool selection_contains(const VT::Position&) const;
  53. ByteString selected_text() const;
  54. VT::Range normalized_selection() const { return m_selection.normalized(); }
  55. void set_selection(const VT::Range& selection);
  56. VT::Position buffer_position_at(Gfx::IntPoint) const;
  57. VT::Range find_next(StringView, const VT::Position& start = {}, bool case_sensitivity = false, bool should_wrap = false);
  58. VT::Range find_previous(StringView, const VT::Position& start = {}, bool case_sensitivity = false, bool should_wrap = false);
  59. void scroll_to_bottom();
  60. void scroll_to_row(int);
  61. bool is_scrollable() const;
  62. int scroll_length() const;
  63. size_t max_history_size() const { return m_terminal.max_history_size(); }
  64. void set_max_history_size(size_t value) { m_terminal.set_max_history_size(value); }
  65. GUI::Action& copy_action() { return *m_copy_action; }
  66. GUI::Action& paste_action() { return *m_paste_action; }
  67. GUI::Action& clear_including_history_action() { return *m_clear_including_history_action; }
  68. GUI::Action& clear_to_previous_mark_action() { return *m_clear_to_previous_mark_action; }
  69. void copy();
  70. void paste();
  71. void clear_including_history();
  72. void clear_to_previous_mark();
  73. void set_startup_process_id(pid_t pid) { m_startup_process_id = pid; }
  74. const StringView color_scheme_name() const { return m_color_scheme_name; }
  75. Function<void(StringView)> on_title_change;
  76. Function<void(Gfx::IntSize)> on_terminal_size_change;
  77. Function<void()> on_command_exit;
  78. GUI::Menu& context_menu() { return *m_context_menu; }
  79. constexpr Gfx::Color terminal_color_to_rgb(VT::Color) const;
  80. void set_font_and_resize_to_fit(Gfx::Font const&);
  81. void update_color_scheme();
  82. void set_logical_focus(bool);
  83. VT::CursorShape cursor_shape() { return m_cursor_shape; }
  84. virtual void set_cursor_blinking(bool) override;
  85. virtual void set_cursor_shape(CursorShape) override;
  86. static Optional<VT::CursorShape> parse_cursor_shape(StringView);
  87. static ByteString stringify_cursor_shape(VT::CursorShape);
  88. private:
  89. TerminalWidget(int ptm_fd, bool automatic_size_policy);
  90. // ^GUI::Widget
  91. virtual void event(Core::Event&) override;
  92. virtual void paint_event(GUI::PaintEvent&) override;
  93. virtual void resize_event(GUI::ResizeEvent&) override;
  94. virtual void keydown_event(GUI::KeyEvent&) override;
  95. virtual void keyup_event(GUI::KeyEvent&) override;
  96. virtual void mousedown_event(GUI::MouseEvent&) override;
  97. virtual void mouseup_event(GUI::MouseEvent&) override;
  98. virtual void mousemove_event(GUI::MouseEvent&) override;
  99. virtual void mousewheel_event(GUI::MouseEvent&) override;
  100. virtual void doubleclick_event(GUI::MouseEvent&) override;
  101. virtual void focusin_event(GUI::FocusEvent&) override;
  102. virtual void focusout_event(GUI::FocusEvent&) override;
  103. virtual void context_menu_event(GUI::ContextMenuEvent&) override;
  104. virtual void drag_enter_event(GUI::DragEvent&) override;
  105. virtual void drop_event(GUI::DropEvent&) override;
  106. virtual void leave_event(Core::Event&) override;
  107. virtual void did_change_font() override;
  108. // ^TerminalClient
  109. virtual void beep() override;
  110. virtual void set_window_title(StringView) override;
  111. virtual void set_window_progress(int value, int max) override;
  112. virtual void terminal_did_resize(u16 columns, u16 rows) override;
  113. virtual void terminal_history_changed(int delta) override;
  114. virtual void terminal_did_perform_possibly_partial_clear() override;
  115. virtual void emit(u8 const*, size_t) override;
  116. // ^GUI::Clipboard::ClipboardClient
  117. virtual void clipboard_content_did_change(ByteString const&) override { update_paste_action(); }
  118. void send_non_user_input(ReadonlyBytes);
  119. Gfx::IntRect glyph_rect(u16 row, u16 column);
  120. Gfx::IntRect row_rect(u16 row);
  121. Gfx::IntSize widget_size_for_font(Gfx::Font const&) const;
  122. void update_cursor();
  123. void invalidate_cursor();
  124. void relayout(Gfx::IntSize);
  125. void update_copy_action();
  126. void update_paste_action();
  127. Gfx::IntSize compute_base_size() const;
  128. int first_selection_column_on_row(int row) const;
  129. int last_selection_column_on_row(int row) const;
  130. u32 code_point_at(const VT::Position&) const;
  131. VT::Position next_position_after(const VT::Position&, bool should_wrap) const;
  132. VT::Position previous_position_before(const VT::Position&, bool should_wrap) const;
  133. void update_cached_font_metrics();
  134. void handle_pty_owner_change(pid_t new_owner);
  135. VT::Terminal m_terminal;
  136. VT::Range m_selection;
  137. ByteString m_hovered_href;
  138. Optional<ByteString> m_hovered_href_id;
  139. ByteString m_active_href;
  140. Optional<ByteString> m_active_href_id;
  141. // Snapshot of m_hovered_href when opening a context menu for a hyperlink.
  142. ByteString m_context_menu_href;
  143. Gfx::Color m_colors[256];
  144. Gfx::Color m_default_foreground_color;
  145. Gfx::Color m_default_background_color;
  146. bool m_show_bold_text_as_bright { true };
  147. ByteString m_color_scheme_name;
  148. AutoMarkMode m_auto_mark_mode { AutoMarkMode::MarkInteractiveShellPrompt };
  149. BellMode m_bell_mode { BellMode::Visible };
  150. bool m_alt_key_held { false };
  151. bool m_rectangle_selection { false };
  152. int m_pixel_width { 0 };
  153. int m_pixel_height { 0 };
  154. int m_inset { 2 };
  155. int m_line_spacing { 4 };
  156. int m_line_height { 0 };
  157. int m_cell_height { 0 };
  158. int m_column_width { 0 };
  159. int m_ptm_fd { -1 };
  160. bool m_has_logical_focus { false };
  161. bool m_in_relayout { false };
  162. RefPtr<Core::Notifier> m_notifier;
  163. u8 m_opacity { 255 };
  164. bool m_cursor_blink_state { true };
  165. bool m_automatic_size_policy { false };
  166. VT::CursorShape m_cursor_shape { VT::CursorShape::Block };
  167. bool m_cursor_is_blinking_set { true };
  168. enum class AutoScrollDirection {
  169. None,
  170. Up,
  171. Down
  172. };
  173. void set_auto_scroll_direction(AutoScrollDirection);
  174. AutoScrollDirection m_auto_scroll_direction { AutoScrollDirection::None };
  175. RefPtr<Core::Timer> m_cursor_blink_timer;
  176. RefPtr<Core::Timer> m_visual_beep_timer;
  177. RefPtr<Core::Timer> m_auto_scroll_timer;
  178. RefPtr<GUI::Scrollbar> m_scrollbar;
  179. RefPtr<GUI::Action> m_copy_action;
  180. RefPtr<GUI::Action> m_paste_action;
  181. RefPtr<GUI::Action> m_clear_including_history_action;
  182. RefPtr<GUI::Action> m_clear_to_previous_mark_action;
  183. RefPtr<GUI::Menu> m_context_menu;
  184. RefPtr<GUI::Menu> m_context_menu_for_hyperlink;
  185. Core::ElapsedTimer m_triple_click_timer;
  186. Gfx::IntPoint m_left_mousedown_position;
  187. VT::Position m_left_mousedown_position_buffer;
  188. bool m_startup_process_owns_pty { false };
  189. pid_t m_startup_process_id { -1 };
  190. };
  191. }