Page.h 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. /*
  2. * Copyright (c) 2020-2023, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021-2022, Linus Groh <linusg@serenityos.org>
  4. * Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
  5. * Copyright (c) 2023, Shannon Booth <shannon@serenityos.org>
  6. *
  7. * SPDX-License-Identifier: BSD-2-Clause
  8. */
  9. #pragma once
  10. #include <AK/Noncopyable.h>
  11. #include <AK/OwnPtr.h>
  12. #include <AK/RefPtr.h>
  13. #include <AK/WeakPtr.h>
  14. #include <AK/Weakable.h>
  15. #include <LibGfx/Forward.h>
  16. #include <LibGfx/Palette.h>
  17. #include <LibGfx/Point.h>
  18. #include <LibGfx/Rect.h>
  19. #include <LibGfx/ShareableBitmap.h>
  20. #include <LibGfx/Size.h>
  21. #include <LibGfx/StandardCursor.h>
  22. #include <LibIPC/Forward.h>
  23. #include <LibJS/Heap/Handle.h>
  24. #include <LibJS/Heap/Heap.h>
  25. #include <LibURL/URL.h>
  26. #include <LibWeb/CSS/PreferredColorScheme.h>
  27. #include <LibWeb/CSS/PreferredContrast.h>
  28. #include <LibWeb/CSS/PreferredMotion.h>
  29. #include <LibWeb/CSS/Selector.h>
  30. #include <LibWeb/Cookie/Cookie.h>
  31. #include <LibWeb/Forward.h>
  32. #include <LibWeb/HTML/ActivateTab.h>
  33. #include <LibWeb/HTML/AudioPlayState.h>
  34. #include <LibWeb/HTML/ColorPickerUpdateState.h>
  35. #include <LibWeb/HTML/FileFilter.h>
  36. #include <LibWeb/HTML/SelectItem.h>
  37. #include <LibWeb/HTML/TokenizedFeatures.h>
  38. #include <LibWeb/HTML/WebViewHints.h>
  39. #include <LibWeb/Loader/FileRequest.h>
  40. #include <LibWeb/PixelUnits.h>
  41. #include <LibWeb/UIEvents/KeyCode.h>
  42. #ifdef HAS_ACCELERATED_GRAPHICS
  43. # include <LibAccelGfx/Context.h>
  44. #endif
  45. namespace Web {
  46. class PageClient;
  47. class Page final : public JS::Cell {
  48. JS_CELL(Page, JS::Cell);
  49. JS_DECLARE_ALLOCATOR(Page);
  50. public:
  51. static JS::NonnullGCPtr<Page> create(JS::VM&, JS::NonnullGCPtr<PageClient>);
  52. ~Page();
  53. PageClient& client() { return m_client; }
  54. PageClient const& client() const { return m_client; }
  55. void set_top_level_traversable(JS::NonnullGCPtr<HTML::TraversableNavigable>);
  56. // FIXME: This is a hack.
  57. bool top_level_traversable_is_initialized() const;
  58. HTML::BrowsingContext& top_level_browsing_context();
  59. HTML::BrowsingContext const& top_level_browsing_context() const;
  60. JS::NonnullGCPtr<HTML::TraversableNavigable> top_level_traversable() const;
  61. HTML::Navigable& focused_navigable();
  62. HTML::Navigable const& focused_navigable() const { return const_cast<Page*>(this)->focused_navigable(); }
  63. void set_focused_navigable(Badge<EventHandler>, HTML::Navigable&);
  64. void load(URL::URL const&);
  65. void load_html(StringView);
  66. void reload();
  67. void traverse_the_history_by_delta(int delta);
  68. CSSPixelPoint device_to_css_point(DevicePixelPoint) const;
  69. DevicePixelPoint css_to_device_point(CSSPixelPoint) const;
  70. DevicePixelRect css_to_device_rect(CSSPixelRect) const;
  71. CSSPixelRect device_to_css_rect(DevicePixelRect) const;
  72. CSSPixelSize device_to_css_size(DevicePixelSize) const;
  73. DevicePixelRect enclosing_device_rect(CSSPixelRect) const;
  74. DevicePixelRect rounded_device_rect(CSSPixelRect) const;
  75. bool handle_mouseup(DevicePixelPoint, DevicePixelPoint screen_position, unsigned button, unsigned buttons, unsigned modifiers);
  76. bool handle_mousedown(DevicePixelPoint, DevicePixelPoint screen_position, unsigned button, unsigned buttons, unsigned modifiers);
  77. bool handle_mousemove(DevicePixelPoint, DevicePixelPoint screen_position, unsigned buttons, unsigned modifiers);
  78. bool handle_mousewheel(DevicePixelPoint, DevicePixelPoint screen_position, unsigned button, unsigned buttons, unsigned modifiers, DevicePixels wheel_delta_x, DevicePixels wheel_delta_y);
  79. bool handle_doubleclick(DevicePixelPoint, DevicePixelPoint screen_position, unsigned button, unsigned buttons, unsigned modifiers);
  80. bool handle_keydown(UIEvents::KeyCode, unsigned modifiers, u32 code_point);
  81. bool handle_keyup(UIEvents::KeyCode, unsigned modifiers, u32 code_point);
  82. Gfx::Palette palette() const;
  83. CSSPixelRect web_exposed_screen_area() const;
  84. CSS::PreferredColorScheme preferred_color_scheme() const;
  85. CSS::PreferredContrast preferred_contrast() const;
  86. CSS::PreferredMotion preferred_motion() const;
  87. bool is_same_origin_policy_enabled() const { return m_same_origin_policy_enabled; }
  88. void set_same_origin_policy_enabled(bool b) { m_same_origin_policy_enabled = b; }
  89. bool is_scripting_enabled() const { return m_is_scripting_enabled; }
  90. void set_is_scripting_enabled(bool b) { m_is_scripting_enabled = b; }
  91. bool should_block_pop_ups() const { return m_should_block_pop_ups; }
  92. void set_should_block_pop_ups(bool b) { m_should_block_pop_ups = b; }
  93. bool is_webdriver_active() const { return m_is_webdriver_active; }
  94. void set_is_webdriver_active(bool b) { m_is_webdriver_active = b; }
  95. DevicePixelPoint window_position() const { return m_window_position; }
  96. void set_window_position(DevicePixelPoint position) { m_window_position = position; }
  97. DevicePixelSize window_size() const { return m_window_size; }
  98. void set_window_size(DevicePixelSize size) { m_window_size = size; }
  99. void did_request_alert(String const& message);
  100. void alert_closed();
  101. bool did_request_confirm(String const& message);
  102. void confirm_closed(bool accepted);
  103. Optional<String> did_request_prompt(String const& message, String const& default_);
  104. void prompt_closed(Optional<String> response);
  105. enum class PendingDialog {
  106. None,
  107. Alert,
  108. Confirm,
  109. Prompt,
  110. };
  111. bool has_pending_dialog() const { return m_pending_dialog != PendingDialog::None; }
  112. PendingDialog pending_dialog() const { return m_pending_dialog; }
  113. Optional<String> const& pending_dialog_text() const { return m_pending_dialog_text; }
  114. void dismiss_dialog();
  115. void accept_dialog();
  116. void did_request_color_picker(WeakPtr<HTML::HTMLInputElement> target, Color current_color);
  117. void color_picker_update(Optional<Color> picked_color, HTML::ColorPickerUpdateState state);
  118. void did_request_file_picker(WeakPtr<HTML::HTMLInputElement> target, HTML::FileFilter accepted_file_types, HTML::AllowMultipleFiles);
  119. void file_picker_closed(Span<HTML::SelectedFile> selected_files);
  120. void did_request_select_dropdown(WeakPtr<HTML::HTMLSelectElement> target, Web::CSSPixelPoint content_position, Web::CSSPixels minimum_width, Vector<Web::HTML::SelectItem> items);
  121. void select_dropdown_closed(Optional<u32> const& selected_item_id);
  122. enum class PendingNonBlockingDialog {
  123. None,
  124. ColorPicker,
  125. FilePicker,
  126. Select,
  127. };
  128. void register_media_element(Badge<HTML::HTMLMediaElement>, int media_id);
  129. void unregister_media_element(Badge<HTML::HTMLMediaElement>, int media_id);
  130. struct MediaContextMenu {
  131. URL::URL media_url;
  132. bool is_video { false };
  133. bool is_playing { false };
  134. bool is_muted { false };
  135. bool has_user_agent_controls { false };
  136. bool is_looping { false };
  137. };
  138. void did_request_media_context_menu(i32 media_id, CSSPixelPoint, ByteString const& target, unsigned modifiers, MediaContextMenu);
  139. WebIDL::ExceptionOr<void> toggle_media_play_state();
  140. void toggle_media_mute_state();
  141. WebIDL::ExceptionOr<void> toggle_media_loop_state();
  142. WebIDL::ExceptionOr<void> toggle_media_controls_state();
  143. HTML::MuteState page_mute_state() const { return m_mute_state; }
  144. void toggle_page_mute_state();
  145. Optional<String> const& user_style() const { return m_user_style_sheet_source; }
  146. void set_user_style(String source);
  147. bool pdf_viewer_supported() const { return m_pdf_viewer_supported; }
  148. void clear_selection();
  149. struct FindInPageQuery {
  150. String string {};
  151. CaseSensitivity case_sensitivity { CaseSensitivity::CaseInsensitive };
  152. };
  153. struct FindInPageResult {
  154. size_t current_match_index { 0 };
  155. Optional<size_t> total_match_count {};
  156. };
  157. FindInPageResult find_in_page(FindInPageQuery const&);
  158. FindInPageResult find_in_page_next_match();
  159. FindInPageResult find_in_page_previous_match();
  160. private:
  161. explicit Page(JS::NonnullGCPtr<PageClient>);
  162. virtual void visit_edges(Visitor&) override;
  163. JS::GCPtr<HTML::HTMLMediaElement> media_context_menu_element();
  164. Vector<JS::Handle<DOM::Document>> documents_in_active_window() const;
  165. enum class SearchDirection {
  166. Forward,
  167. Backward,
  168. };
  169. FindInPageResult perform_find_in_page_query(FindInPageQuery const&, Optional<SearchDirection> = {});
  170. void update_find_in_page_selection(Vector<JS::Handle<DOM::Range>> matches);
  171. JS::NonnullGCPtr<PageClient> m_client;
  172. WeakPtr<HTML::Navigable> m_focused_navigable;
  173. JS::GCPtr<HTML::TraversableNavigable> m_top_level_traversable;
  174. // FIXME: Enable this by default once CORS preflight checks are supported.
  175. bool m_same_origin_policy_enabled { false };
  176. bool m_is_scripting_enabled { true };
  177. bool m_should_block_pop_ups { true };
  178. // https://w3c.github.io/webdriver/#dfn-webdriver-active-flag
  179. // The webdriver-active flag is set to true when the user agent is under remote control. It is initially false.
  180. bool m_is_webdriver_active { false };
  181. DevicePixelPoint m_window_position {};
  182. DevicePixelSize m_window_size {};
  183. PendingDialog m_pending_dialog { PendingDialog::None };
  184. Optional<String> m_pending_dialog_text;
  185. Optional<Empty> m_pending_alert_response;
  186. Optional<bool> m_pending_confirm_response;
  187. Optional<Optional<String>> m_pending_prompt_response;
  188. PendingNonBlockingDialog m_pending_non_blocking_dialog { PendingNonBlockingDialog::None };
  189. WeakPtr<HTML::HTMLElement> m_pending_non_blocking_dialog_target;
  190. Vector<int> m_media_elements;
  191. Optional<int> m_media_context_menu_element_id;
  192. Web::HTML::MuteState m_mute_state { Web::HTML::MuteState::Unmuted };
  193. Optional<String> m_user_style_sheet_source;
  194. // https://html.spec.whatwg.org/multipage/system-state.html#pdf-viewer-supported
  195. // Each user agent has a PDF viewer supported boolean, whose value is implementation-defined (and might vary according to user preferences).
  196. // Spec Note: This value also impacts the navigation processing model.
  197. // FIXME: Actually support pdf viewing
  198. bool m_pdf_viewer_supported { false };
  199. size_t m_find_in_page_match_index { 0 };
  200. Optional<FindInPageQuery> m_last_find_in_page_query;
  201. URL::URL m_last_find_in_page_url;
  202. };
  203. struct PaintOptions {
  204. enum class PaintOverlay {
  205. No,
  206. Yes,
  207. };
  208. PaintOverlay paint_overlay { PaintOverlay::Yes };
  209. bool should_show_line_box_borders { false };
  210. bool has_focus { false };
  211. #ifdef HAS_ACCELERATED_GRAPHICS
  212. AccelGfx::Context* accelerated_graphics_context { nullptr };
  213. #endif
  214. };
  215. enum class PaintingCommandExecutorType {
  216. CPU,
  217. GPU,
  218. Skia
  219. };
  220. class PageClient : public JS::Cell {
  221. JS_CELL(PageClient, JS::Cell);
  222. public:
  223. virtual Page& page() = 0;
  224. virtual Page const& page() const = 0;
  225. virtual bool is_connection_open() const = 0;
  226. virtual Gfx::Palette palette() const = 0;
  227. virtual DevicePixelRect screen_rect() const = 0;
  228. virtual double device_pixels_per_css_pixel() const = 0;
  229. virtual CSS::PreferredColorScheme preferred_color_scheme() const = 0;
  230. virtual CSS::PreferredContrast preferred_contrast() const = 0;
  231. virtual CSS::PreferredMotion preferred_motion() const = 0;
  232. virtual void paint_next_frame() = 0;
  233. virtual void paint(DevicePixelRect const&, Gfx::Bitmap&, PaintOptions = {}) = 0;
  234. virtual void page_did_change_title(ByteString const&) { }
  235. virtual void page_did_change_url(URL::URL const&) { }
  236. virtual void page_did_request_navigate_back() { }
  237. virtual void page_did_request_navigate_forward() { }
  238. virtual void page_did_request_refresh() { }
  239. virtual Gfx::IntSize page_did_request_resize_window(Gfx::IntSize) { return {}; }
  240. virtual Gfx::IntPoint page_did_request_reposition_window(Gfx::IntPoint) { return {}; }
  241. virtual void page_did_request_restore_window() { }
  242. virtual Gfx::IntRect page_did_request_maximize_window() { return {}; }
  243. virtual Gfx::IntRect page_did_request_minimize_window() { return {}; }
  244. virtual Gfx::IntRect page_did_request_fullscreen_window() { return {}; }
  245. virtual void page_did_start_loading(URL::URL const&, bool is_redirect) { (void)is_redirect; }
  246. virtual void page_did_create_new_document(Web::DOM::Document&) { }
  247. virtual void page_did_change_active_document_in_top_level_browsing_context(Web::DOM::Document&) { }
  248. virtual void page_did_destroy_document(Web::DOM::Document&) { }
  249. virtual void page_did_finish_loading(URL::URL const&) { }
  250. virtual void page_did_request_cursor_change(Gfx::StandardCursor) { }
  251. virtual void page_did_request_context_menu(CSSPixelPoint) { }
  252. virtual void page_did_request_link_context_menu(CSSPixelPoint, URL::URL const&, [[maybe_unused]] ByteString const& target, [[maybe_unused]] unsigned modifiers) { }
  253. virtual void page_did_request_image_context_menu(CSSPixelPoint, URL::URL const&, [[maybe_unused]] ByteString const& target, [[maybe_unused]] unsigned modifiers, Gfx::Bitmap const*) { }
  254. virtual void page_did_request_media_context_menu(CSSPixelPoint, [[maybe_unused]] ByteString const& target, [[maybe_unused]] unsigned modifiers, Page::MediaContextMenu) { }
  255. virtual void page_did_middle_click_link(URL::URL const&, [[maybe_unused]] ByteString const& target, [[maybe_unused]] unsigned modifiers) { }
  256. virtual void page_did_enter_tooltip_area(CSSPixelPoint, ByteString const&) { }
  257. virtual void page_did_leave_tooltip_area() { }
  258. virtual void page_did_hover_link(URL::URL const&) { }
  259. virtual void page_did_unhover_link() { }
  260. virtual void page_did_change_favicon(Gfx::Bitmap const&) { }
  261. virtual void page_did_layout() { }
  262. virtual void page_did_request_alert(String const&) { }
  263. virtual void page_did_request_confirm(String const&) { }
  264. virtual void page_did_request_prompt(String const&, String const&) { }
  265. virtual void page_did_request_set_prompt_text(String const&) { }
  266. virtual void page_did_request_accept_dialog() { }
  267. virtual void page_did_request_dismiss_dialog() { }
  268. virtual Vector<Web::Cookie::Cookie> page_did_request_all_cookies(URL::URL const&) { return {}; }
  269. virtual Optional<Web::Cookie::Cookie> page_did_request_named_cookie(URL::URL const&, String const&) { return {}; }
  270. virtual String page_did_request_cookie(URL::URL const&, Cookie::Source) { return {}; }
  271. virtual void page_did_set_cookie(URL::URL const&, Cookie::ParsedCookie const&, Cookie::Source) { }
  272. virtual void page_did_update_cookie(Web::Cookie::Cookie) { }
  273. virtual void page_did_update_resource_count(i32) { }
  274. struct NewWebViewResult {
  275. JS::GCPtr<Page> page;
  276. String window_handle;
  277. };
  278. virtual NewWebViewResult page_did_request_new_web_view(HTML::ActivateTab, HTML::WebViewHints, HTML::TokenizedFeature::NoOpener) { return {}; }
  279. virtual void page_did_request_activate_tab() { }
  280. virtual void page_did_close_top_level_traversable() { }
  281. virtual void page_did_update_navigation_buttons_state([[maybe_unused]] bool back_enabled, [[maybe_unused]] bool forward_enabled) { }
  282. virtual void page_did_allocate_backing_stores([[maybe_unused]] i32 front_bitmap_id, [[maybe_unused]] Gfx::ShareableBitmap front_bitmap, [[maybe_unused]] i32 back_bitmap_id, [[maybe_unused]] Gfx::ShareableBitmap back_bitmap) { }
  283. virtual void request_file(FileRequest) = 0;
  284. // https://html.spec.whatwg.org/multipage/input.html#show-the-picker,-if-applicable
  285. virtual void page_did_request_color_picker([[maybe_unused]] Color current_color) { }
  286. virtual void page_did_request_file_picker([[maybe_unused]] HTML::FileFilter accepted_file_types, Web::HTML::AllowMultipleFiles) { }
  287. virtual void page_did_request_select_dropdown([[maybe_unused]] Web::CSSPixelPoint content_position, [[maybe_unused]] Web::CSSPixels minimum_width, [[maybe_unused]] Vector<Web::HTML::SelectItem> items) { }
  288. virtual void page_did_finish_text_test() { }
  289. virtual void page_did_change_theme_color(Gfx::Color) { }
  290. virtual void page_did_insert_clipboard_entry([[maybe_unused]] String data, [[maybe_unused]] String presentation_style, [[maybe_unused]] String mime_type) { }
  291. virtual void page_did_change_audio_play_state(HTML::AudioPlayState) { }
  292. virtual IPC::File request_worker_agent() { return IPC::File {}; }
  293. virtual void inspector_did_load() { }
  294. virtual void inspector_did_select_dom_node([[maybe_unused]] i32 node_id, [[maybe_unused]] Optional<CSS::Selector::PseudoElement::Type> const& pseudo_element) { }
  295. virtual void inspector_did_set_dom_node_text([[maybe_unused]] i32 node_id, [[maybe_unused]] String const& text) { }
  296. virtual void inspector_did_set_dom_node_tag([[maybe_unused]] i32 node_id, [[maybe_unused]] String const& tag) { }
  297. virtual void inspector_did_add_dom_node_attributes([[maybe_unused]] i32 node_id, [[maybe_unused]] JS::NonnullGCPtr<DOM::NamedNodeMap> attributes) { }
  298. virtual void inspector_did_replace_dom_node_attribute([[maybe_unused]] i32 node_id, [[maybe_unused]] size_t attribute_index, [[maybe_unused]] JS::NonnullGCPtr<DOM::NamedNodeMap> replacement_attributes) { }
  299. virtual void inspector_did_request_dom_tree_context_menu([[maybe_unused]] i32 node_id, [[maybe_unused]] CSSPixelPoint position, [[maybe_unused]] String const& type, [[maybe_unused]] Optional<String> const& tag, [[maybe_unused]] Optional<size_t> const& attribute_index) { }
  300. virtual void inspector_did_execute_console_script([[maybe_unused]] String const& script) { }
  301. virtual void schedule_repaint() = 0;
  302. virtual bool is_ready_to_paint() const = 0;
  303. virtual PaintingCommandExecutorType painting_command_executor_type() const = 0;
  304. protected:
  305. virtual ~PageClient() = default;
  306. };
  307. }
  308. namespace IPC {
  309. template<>
  310. ErrorOr<void> encode(Encoder&, Web::Page::MediaContextMenu const&);
  311. template<>
  312. ErrorOr<Web::Page::MediaContextMenu> decode(Decoder&);
  313. }