TerminalWidget.h 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  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 Optional<BellMode> parse_bell(StringView);
  88. static Optional<AutoMarkMode> parse_automark_mode(StringView);
  89. static ByteString stringify_cursor_shape(VT::CursorShape);
  90. static ByteString stringify_bell(BellMode);
  91. static ByteString stringify_automark_mode(AutoMarkMode);
  92. private:
  93. TerminalWidget(int ptm_fd, bool automatic_size_policy);
  94. // ^GUI::Widget
  95. virtual void event(Core::Event&) override;
  96. virtual void paint_event(GUI::PaintEvent&) override;
  97. virtual void resize_event(GUI::ResizeEvent&) override;
  98. virtual void keydown_event(GUI::KeyEvent&) override;
  99. virtual void keyup_event(GUI::KeyEvent&) override;
  100. virtual void mousedown_event(GUI::MouseEvent&) override;
  101. virtual void mouseup_event(GUI::MouseEvent&) override;
  102. virtual void mousemove_event(GUI::MouseEvent&) override;
  103. virtual void mousewheel_event(GUI::MouseEvent&) override;
  104. virtual void doubleclick_event(GUI::MouseEvent&) override;
  105. virtual void focusin_event(GUI::FocusEvent&) override;
  106. virtual void focusout_event(GUI::FocusEvent&) override;
  107. virtual void context_menu_event(GUI::ContextMenuEvent&) override;
  108. virtual void drag_enter_event(GUI::DragEvent&) override;
  109. virtual void drop_event(GUI::DropEvent&) override;
  110. virtual void leave_event(Core::Event&) override;
  111. virtual void did_change_font() override;
  112. // ^TerminalClient
  113. virtual void beep() override;
  114. virtual void set_window_title(StringView) override;
  115. virtual void set_window_progress(int value, int max) override;
  116. virtual void terminal_did_resize(u16 columns, u16 rows) override;
  117. virtual void terminal_history_changed(int delta) override;
  118. virtual void terminal_did_perform_possibly_partial_clear() override;
  119. virtual void emit(u8 const*, size_t) override;
  120. // ^GUI::Clipboard::ClipboardClient
  121. virtual void clipboard_content_did_change(ByteString const&) override { update_paste_action(); }
  122. void send_non_user_input(ReadonlyBytes);
  123. Gfx::IntRect glyph_rect(u16 row, u16 column);
  124. Gfx::IntRect row_rect(u16 row);
  125. Gfx::IntSize widget_size_for_font(Gfx::Font const&) const;
  126. void update_cursor();
  127. void invalidate_cursor();
  128. void relayout(Gfx::IntSize);
  129. void update_copy_action();
  130. void update_paste_action();
  131. Gfx::IntSize compute_base_size() const;
  132. int first_selection_column_on_row(int row) const;
  133. int last_selection_column_on_row(int row) const;
  134. u32 code_point_at(const VT::Position&) const;
  135. VT::Position next_position_after(const VT::Position&, bool should_wrap) const;
  136. VT::Position previous_position_before(const VT::Position&, bool should_wrap) const;
  137. void update_cached_font_metrics();
  138. void handle_pty_owner_change(pid_t new_owner);
  139. VT::Terminal m_terminal;
  140. VT::Range m_selection;
  141. ByteString m_hovered_href;
  142. Optional<ByteString> m_hovered_href_id;
  143. ByteString m_active_href;
  144. Optional<ByteString> m_active_href_id;
  145. // Snapshot of m_hovered_href when opening a context menu for a hyperlink.
  146. ByteString m_context_menu_href;
  147. Gfx::Color m_colors[256];
  148. Gfx::Color m_default_foreground_color;
  149. Gfx::Color m_default_background_color;
  150. bool m_show_bold_text_as_bright { true };
  151. ByteString m_color_scheme_name;
  152. AutoMarkMode m_auto_mark_mode { AutoMarkMode::MarkInteractiveShellPrompt };
  153. BellMode m_bell_mode { BellMode::Visible };
  154. bool m_alt_key_held { false };
  155. bool m_rectangle_selection { false };
  156. int m_pixel_width { 0 };
  157. int m_pixel_height { 0 };
  158. int m_inset { 2 };
  159. int m_line_spacing { 4 };
  160. int m_line_height { 0 };
  161. int m_cell_height { 0 };
  162. int m_column_width { 0 };
  163. int m_ptm_fd { -1 };
  164. bool m_has_logical_focus { false };
  165. bool m_in_relayout { false };
  166. RefPtr<Core::Notifier> m_notifier;
  167. u8 m_opacity { 255 };
  168. bool m_cursor_blink_state { true };
  169. bool m_automatic_size_policy { false };
  170. VT::CursorShape m_cursor_shape { VT::CursorShape::Block };
  171. bool m_cursor_is_blinking_set { true };
  172. enum class AutoScrollDirection {
  173. None,
  174. Up,
  175. Down
  176. };
  177. void set_auto_scroll_direction(AutoScrollDirection);
  178. AutoScrollDirection m_auto_scroll_direction { AutoScrollDirection::None };
  179. RefPtr<Core::Timer> m_cursor_blink_timer;
  180. RefPtr<Core::Timer> m_visual_beep_timer;
  181. RefPtr<Core::Timer> m_auto_scroll_timer;
  182. RefPtr<GUI::Scrollbar> m_scrollbar;
  183. RefPtr<GUI::Action> m_copy_action;
  184. RefPtr<GUI::Action> m_paste_action;
  185. RefPtr<GUI::Action> m_clear_including_history_action;
  186. RefPtr<GUI::Action> m_clear_to_previous_mark_action;
  187. RefPtr<GUI::Menu> m_context_menu;
  188. RefPtr<GUI::Menu> m_context_menu_for_hyperlink;
  189. Core::ElapsedTimer m_triple_click_timer;
  190. Gfx::IntPoint m_left_mousedown_position;
  191. VT::Position m_left_mousedown_position_buffer;
  192. bool m_startup_process_owns_pty { false };
  193. pid_t m_startup_process_id { -1 };
  194. };
  195. }