Tab.cpp 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917
  1. /*
  2. * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021, Maciej Zygmanowski <sppmacd@pm.me>
  4. * Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
  5. * Copyright (c) 2022, the SerenityOS developers.
  6. * Copyright (c) 2022, networkException <networkexception@serenityos.org>
  7. * Copyright (c) 2023, Cameron Youell <cameronyouell@gmail.com>
  8. *
  9. * SPDX-License-Identifier: BSD-2-Clause
  10. */
  11. #include "Tab.h"
  12. #include "BookmarksBarWidget.h"
  13. #include "Browser.h"
  14. #include "BrowserWindow.h"
  15. #include "ConsoleWidget.h"
  16. #include "DownloadWidget.h"
  17. #include "History/HistoryWidget.h"
  18. #include "InspectorWidget.h"
  19. #include "StorageWidget.h"
  20. #include <AK/StringBuilder.h>
  21. #include <AK/URL.h>
  22. #include <Applications/Browser/TabGML.h>
  23. #include <Applications/BrowserSettings/Defaults.h>
  24. #include <LibConfig/Client.h>
  25. #include <LibGUI/Action.h>
  26. #include <LibGUI/Application.h>
  27. #include <LibGUI/BoxLayout.h>
  28. #include <LibGUI/Button.h>
  29. #include <LibGUI/Clipboard.h>
  30. #include <LibGUI/Menu.h>
  31. #include <LibGUI/MessageBox.h>
  32. #include <LibGUI/Statusbar.h>
  33. #include <LibGUI/TextBox.h>
  34. #include <LibGUI/Toolbar.h>
  35. #include <LibGUI/ToolbarContainer.h>
  36. #include <LibGUI/Window.h>
  37. #include <LibJS/Interpreter.h>
  38. #include <LibWeb/HTML/BrowsingContext.h>
  39. #include <LibWeb/HTML/SyntaxHighlighter/SyntaxHighlighter.h>
  40. #include <LibWeb/Layout/BlockContainer.h>
  41. #include <LibWeb/Layout/Viewport.h>
  42. #include <LibWeb/Loader/ResourceLoader.h>
  43. #include <LibWebView/OutOfProcessWebView.h>
  44. namespace Browser {
  45. Tab::~Tab()
  46. {
  47. close_sub_widgets();
  48. }
  49. URL url_from_user_input(DeprecatedString const& input)
  50. {
  51. if (input.starts_with('?') && !g_search_engine.is_empty())
  52. return URL(g_search_engine.replace("{}"sv, URL::percent_encode(input.substring_view(1)), ReplaceMode::FirstOnly));
  53. URL url_with_http_schema = URL(DeprecatedString::formatted("https://{}", input));
  54. if (url_with_http_schema.is_valid() && url_with_http_schema.port().has_value())
  55. return url_with_http_schema;
  56. URL url = URL(input);
  57. if (url.is_valid())
  58. return url;
  59. return url_with_http_schema;
  60. }
  61. void Tab::start_download(const URL& url)
  62. {
  63. auto window = GUI::Window::construct(&this->window());
  64. window->resize(300, 170);
  65. window->set_title(DeprecatedString::formatted("0% of {}", url.basename()));
  66. window->set_resizable(false);
  67. (void)window->set_main_widget<DownloadWidget>(url).release_value_but_fixme_should_propagate_errors();
  68. window->show();
  69. }
  70. void Tab::view_source(const URL& url, DeprecatedString const& source)
  71. {
  72. auto window = GUI::Window::construct(&this->window());
  73. auto editor = window->set_main_widget<GUI::TextEditor>().release_value_but_fixme_should_propagate_errors();
  74. editor->set_text(source);
  75. editor->set_mode(GUI::TextEditor::ReadOnly);
  76. editor->set_syntax_highlighter(make<Web::HTML::SyntaxHighlighter>());
  77. editor->set_ruler_visible(true);
  78. window->resize(640, 480);
  79. window->set_title(url.to_deprecated_string());
  80. window->set_icon(g_icon_bag.filetype_text);
  81. window->set_window_mode(GUI::WindowMode::Modeless);
  82. window->show();
  83. }
  84. void Tab::update_status(Optional<String> text_override, i32 count_waiting)
  85. {
  86. if (text_override.has_value()) {
  87. m_statusbar->set_text(*text_override);
  88. return;
  89. }
  90. if (m_loaded) {
  91. m_statusbar->set_text({});
  92. return;
  93. }
  94. VERIFY(m_navigating_url.has_value());
  95. if (count_waiting == 0) {
  96. // ex: "Loading google.com"
  97. m_statusbar->set_text(String::formatted("Loading {}", m_navigating_url->host()).release_value_but_fixme_should_propagate_errors());
  98. } else {
  99. // ex: "google.com is waiting on 5 resources"
  100. m_statusbar->set_text(String::formatted("{} is waiting on {} resource{}", m_navigating_url->host(), count_waiting, count_waiting == 1 ? ""sv : "s"sv).release_value_but_fixme_should_propagate_errors());
  101. }
  102. }
  103. Tab::Tab(BrowserWindow& window)
  104. {
  105. load_from_gml(tab_gml).release_value_but_fixme_should_propagate_errors();
  106. m_toolbar_container = *find_descendant_of_type_named<GUI::ToolbarContainer>("toolbar_container");
  107. auto& toolbar = *find_descendant_of_type_named<GUI::Toolbar>("toolbar");
  108. auto& webview_container = *find_descendant_of_type_named<GUI::Widget>("webview_container");
  109. m_web_content_view = webview_container.add<WebView::OutOfProcessWebView>();
  110. auto preferred_color_scheme = Web::CSS::preferred_color_scheme_from_string(Config::read_string("Browser"sv, "Preferences"sv, "ColorScheme"sv, Browser::default_color_scheme));
  111. m_web_content_view->set_preferred_color_scheme(preferred_color_scheme);
  112. content_filters_changed();
  113. autoplay_allowlist_changed();
  114. m_web_content_view->set_proxy_mappings(g_proxies, g_proxy_mappings);
  115. if (!g_webdriver_content_ipc_path.is_empty())
  116. enable_webdriver_mode();
  117. auto& go_back_button = toolbar.add_action(window.go_back_action());
  118. go_back_button.on_context_menu_request = [&](auto&) {
  119. if (!m_history.can_go_back())
  120. return;
  121. int i = 0;
  122. m_go_back_context_menu = GUI::Menu::construct();
  123. for (auto& url : m_history.get_back_title_history()) {
  124. i++;
  125. m_go_back_context_menu->add_action(GUI::Action::create(url.to_deprecated_string(), g_icon_bag.filetype_html, [this, i](auto&) { go_back(i); }));
  126. }
  127. m_go_back_context_menu->popup(go_back_button.screen_relative_rect().bottom_left().moved_up(1));
  128. };
  129. auto& go_forward_button = toolbar.add_action(window.go_forward_action());
  130. go_forward_button.on_context_menu_request = [&](auto&) {
  131. if (!m_history.can_go_forward())
  132. return;
  133. int i = 0;
  134. m_go_forward_context_menu = GUI::Menu::construct();
  135. for (auto& url : m_history.get_forward_title_history()) {
  136. i++;
  137. m_go_forward_context_menu->add_action(GUI::Action::create(url.to_deprecated_string(), g_icon_bag.filetype_html, [this, i](auto&) { go_forward(i); }));
  138. }
  139. m_go_forward_context_menu->popup(go_forward_button.screen_relative_rect().bottom_left().moved_up(1));
  140. };
  141. auto& go_home_button = toolbar.add_action(window.go_home_action());
  142. go_home_button.set_allowed_mouse_buttons_for_pressing(GUI::MouseButton::Primary | GUI::MouseButton::Middle);
  143. go_home_button.on_middle_mouse_click = [&](auto) {
  144. on_tab_open_request(Browser::url_from_user_input(g_home_url));
  145. };
  146. toolbar.add_action(window.reload_action());
  147. m_location_box = toolbar.add<GUI::UrlBox>();
  148. m_location_box->set_placeholder("Address"sv);
  149. m_location_box->on_return_pressed = [this] {
  150. auto url = url_from_location_bar();
  151. if (url.has_value())
  152. load(url.release_value());
  153. };
  154. m_location_box->on_ctrl_return_pressed = [this] {
  155. auto url = url_from_location_bar(MayAppendTLD::Yes);
  156. if (url.has_value())
  157. load(url.release_value());
  158. };
  159. m_location_box->add_custom_context_menu_action(GUI::Action::create("Paste && Go", [this](auto&) {
  160. auto [data, mime_type, _] = GUI::Clipboard::the().fetch_data_and_type();
  161. if (!mime_type.starts_with("text/"sv))
  162. return;
  163. auto const& paste_text = data;
  164. if (paste_text.is_empty())
  165. return;
  166. m_location_box->set_text(paste_text);
  167. m_location_box->on_return_pressed();
  168. }));
  169. auto bookmark_action = GUI::Action::create(
  170. "Bookmark current URL", { Mod_Ctrl, Key_D }, [this](auto&) {
  171. if (auto result = bookmark_current_url(); result.is_error())
  172. GUI::MessageBox::show_error(this->window().main_widget()->window(), MUST(String::formatted("Failed to bookmark URL: {}", result.error())));
  173. },
  174. this);
  175. m_reset_zoom_button = toolbar.add<GUI::Button>();
  176. m_reset_zoom_button->set_tooltip("Reset zoom level");
  177. m_reset_zoom_button->on_click = [&](auto) {
  178. view().reset_zoom();
  179. update_reset_zoom_button();
  180. };
  181. m_reset_zoom_button->set_button_style(Gfx::ButtonStyle::Coolbar);
  182. m_reset_zoom_button->set_visible(false);
  183. m_reset_zoom_button->set_preferred_width(GUI::SpecialDimension::Shrink);
  184. m_bookmark_button = toolbar.add<GUI::Button>();
  185. m_bookmark_button->set_action(bookmark_action);
  186. m_bookmark_button->set_button_style(Gfx::ButtonStyle::Coolbar);
  187. m_bookmark_button->set_focus_policy(GUI::FocusPolicy::TabFocus);
  188. m_bookmark_button->set_icon(g_icon_bag.bookmark_contour);
  189. m_bookmark_button->set_fixed_size(22, 22);
  190. view().on_load_start = [this](auto& url, bool is_redirect) {
  191. m_navigating_url = url;
  192. m_loaded = false;
  193. // If we are loading due to a redirect, we replace the current history entry
  194. // with the loaded URL
  195. if (is_redirect) {
  196. m_history.replace_current(url, title());
  197. }
  198. update_status();
  199. m_location_box->set_icon(nullptr);
  200. m_location_box->set_text(url.to_deprecated_string());
  201. // don't add to history if back or forward is pressed
  202. if (!m_is_history_navigation)
  203. m_history.push(url, title());
  204. m_is_history_navigation = false;
  205. update_actions();
  206. update_bookmark_button(url.to_deprecated_string());
  207. if (m_dom_inspector_widget)
  208. m_dom_inspector_widget->clear_dom_json();
  209. if (m_console_widget)
  210. m_console_widget->reset();
  211. };
  212. view().on_load_finish = [this](auto&) {
  213. m_navigating_url = {};
  214. m_loaded = true;
  215. update_status();
  216. if (m_dom_inspector_widget) {
  217. m_web_content_view->inspect_dom_tree();
  218. m_web_content_view->inspect_accessibility_tree();
  219. }
  220. };
  221. view().on_navigate_back = [this]() {
  222. go_back(1);
  223. };
  224. view().on_navigate_forward = [this]() {
  225. go_forward(1);
  226. };
  227. view().on_refresh = [this]() {
  228. reload();
  229. };
  230. view().on_link_click = [this](auto& url, auto& target, unsigned modifiers) {
  231. if (target == "_blank" || modifiers == Mod_Ctrl) {
  232. on_tab_open_request(url);
  233. } else {
  234. load(url);
  235. }
  236. };
  237. view().on_resource_status_change = [this](auto count_waiting) {
  238. update_status({}, count_waiting);
  239. };
  240. view().on_restore_window = [this]() {
  241. this->window().show();
  242. this->window().move_to_front();
  243. m_web_content_view->set_system_visibility_state(true);
  244. };
  245. view().on_reposition_window = [this](Gfx::IntPoint position) {
  246. this->window().move_to(position);
  247. return this->window().position();
  248. };
  249. view().on_resize_window = [this](Gfx::IntSize size) {
  250. this->window().resize(size);
  251. return this->window().size();
  252. };
  253. view().on_maximize_window = [this]() {
  254. this->window().set_maximized(true);
  255. return this->window().rect();
  256. };
  257. view().on_minimize_window = [this]() {
  258. this->window().set_minimized(true);
  259. m_web_content_view->set_system_visibility_state(false);
  260. return this->window().rect();
  261. };
  262. view().on_fullscreen_window = [this]() {
  263. this->window().set_fullscreen(true);
  264. return this->window().rect();
  265. };
  266. m_link_context_menu = GUI::Menu::construct();
  267. auto link_default_action = GUI::Action::create("&Open", g_icon_bag.go_to, [this](auto&) {
  268. view().on_link_click(m_link_context_menu_url, "", 0);
  269. });
  270. m_link_context_menu->add_action(link_default_action);
  271. m_link_context_menu_default_action = link_default_action;
  272. m_link_context_menu->add_action(GUI::Action::create("Open in New &Tab", g_icon_bag.new_tab, [this](auto&) {
  273. view().on_link_click(m_link_context_menu_url, "_blank", 0);
  274. }));
  275. m_link_context_menu->add_separator();
  276. m_link_context_menu->add_action(GUI::Action::create("&Copy URL", g_icon_bag.copy, [this](auto&) {
  277. GUI::Clipboard::the().set_plain_text(m_link_context_menu_url.to_deprecated_string());
  278. }));
  279. m_link_context_menu->add_separator();
  280. m_link_context_menu->add_action(GUI::Action::create("&Download", g_icon_bag.download, [this](auto&) {
  281. start_download(m_link_context_menu_url);
  282. }));
  283. m_link_context_menu->add_separator();
  284. m_link_context_menu->add_action(window.inspect_dom_node_action());
  285. view().on_link_context_menu_request = [this](auto& url, auto widget_position) {
  286. m_link_context_menu_url = url;
  287. auto screen_position = view().screen_relative_rect().location().translated(widget_position);
  288. m_link_context_menu->popup(screen_position, m_link_context_menu_default_action);
  289. };
  290. m_image_context_menu = GUI::Menu::construct();
  291. m_image_context_menu->add_action(GUI::Action::create("&Open Image", g_icon_bag.filetype_image, [this](auto&) {
  292. view().on_link_click(m_image_context_menu_url, "", 0);
  293. }));
  294. m_image_context_menu->add_action(GUI::Action::create("Open Image in New &Tab", g_icon_bag.new_tab, [this](auto&) {
  295. view().on_link_click(m_image_context_menu_url, "_blank", 0);
  296. }));
  297. m_image_context_menu->add_separator();
  298. m_image_context_menu->add_action(GUI::Action::create("&Copy Image", g_icon_bag.copy, [this](auto&) {
  299. if (m_image_context_menu_bitmap.is_valid())
  300. GUI::Clipboard::the().set_bitmap(*m_image_context_menu_bitmap.bitmap());
  301. }));
  302. m_image_context_menu->add_action(GUI::Action::create("Copy Image &URL", g_icon_bag.copy, [this](auto&) {
  303. GUI::Clipboard::the().set_plain_text(m_image_context_menu_url.to_deprecated_string());
  304. }));
  305. m_image_context_menu->add_separator();
  306. m_image_context_menu->add_action(GUI::Action::create("&Download", g_icon_bag.download, [this](auto&) {
  307. start_download(m_image_context_menu_url);
  308. }));
  309. m_image_context_menu->add_separator();
  310. m_image_context_menu->add_action(window.inspect_dom_node_action());
  311. view().on_image_context_menu_request = [this](auto& image_url, auto widget_position, Gfx::ShareableBitmap const& shareable_bitmap) {
  312. m_image_context_menu_url = image_url;
  313. m_image_context_menu_bitmap = shareable_bitmap;
  314. auto screen_position = view().screen_relative_rect().location().translated(widget_position);
  315. m_image_context_menu->popup(screen_position);
  316. };
  317. m_video_context_menu_play_pause_action = GUI::Action::create("&Play", g_icon_bag.play, [this](auto&) {
  318. view().toggle_video_play_state();
  319. });
  320. m_video_context_menu_controls_action = GUI::Action::create_checkable("Show &Controls", [this](auto&) {
  321. view().toggle_video_controls_state();
  322. });
  323. m_video_context_menu_loop_action = GUI::Action::create_checkable("&Loop Video", [this](auto&) {
  324. view().toggle_video_loop_state();
  325. });
  326. m_video_context_menu = GUI::Menu::construct();
  327. m_video_context_menu->add_action(*m_video_context_menu_play_pause_action);
  328. m_video_context_menu->add_action(*m_video_context_menu_controls_action);
  329. m_video_context_menu->add_action(*m_video_context_menu_loop_action);
  330. m_video_context_menu->add_separator();
  331. m_video_context_menu->add_action(GUI::Action::create("&Open Video", g_icon_bag.filetype_video, [this](auto&) {
  332. view().on_link_click(m_video_context_menu_url, "", 0);
  333. }));
  334. m_video_context_menu->add_action(GUI::Action::create("Open Video in New &Tab", g_icon_bag.new_tab, [this](auto&) {
  335. view().on_link_click(m_video_context_menu_url, "_blank", 0);
  336. }));
  337. m_video_context_menu->add_separator();
  338. m_video_context_menu->add_action(GUI::Action::create("Copy Video &URL", g_icon_bag.copy, [this](auto&) {
  339. GUI::Clipboard::the().set_plain_text(m_video_context_menu_url.to_deprecated_string());
  340. }));
  341. m_video_context_menu->add_separator();
  342. m_video_context_menu->add_action(GUI::Action::create("&Download", g_icon_bag.download, [this](auto&) {
  343. start_download(m_video_context_menu_url);
  344. }));
  345. m_video_context_menu->add_separator();
  346. m_video_context_menu->add_action(window.inspect_dom_node_action());
  347. view().on_video_context_menu_request = [this](auto& video_url, auto widget_position, bool is_playing, bool has_user_agent_controls, bool is_looping) {
  348. m_video_context_menu_url = video_url;
  349. if (is_playing) {
  350. m_video_context_menu_play_pause_action->set_icon(g_icon_bag.play);
  351. m_video_context_menu_play_pause_action->set_text("&Play"sv);
  352. } else {
  353. m_video_context_menu_play_pause_action->set_icon(g_icon_bag.pause);
  354. m_video_context_menu_play_pause_action->set_text("&Pause"sv);
  355. }
  356. m_video_context_menu_controls_action->set_checked(has_user_agent_controls);
  357. m_video_context_menu_loop_action->set_checked(is_looping);
  358. auto screen_position = view().screen_relative_rect().location().translated(widget_position);
  359. m_video_context_menu->popup(screen_position);
  360. };
  361. view().on_link_middle_click = [this](auto& href, auto&, auto) {
  362. view().on_link_click(href, "_blank", 0);
  363. };
  364. view().on_title_change = [this](auto const& title) {
  365. m_history.update_title(title);
  366. m_title = title;
  367. if (on_title_change)
  368. on_title_change(m_title);
  369. };
  370. view().on_favicon_change = [this](auto& icon) {
  371. m_icon = icon;
  372. m_location_box->set_icon(&icon);
  373. if (on_favicon_change)
  374. on_favicon_change(icon);
  375. };
  376. view().on_get_all_cookies = [this](auto& url) -> Vector<Web::Cookie::Cookie> {
  377. if (on_get_all_cookies)
  378. return on_get_all_cookies(url);
  379. return {};
  380. };
  381. view().on_get_named_cookie = [this](auto& url, auto& name) -> Optional<Web::Cookie::Cookie> {
  382. if (on_get_named_cookie)
  383. return on_get_named_cookie(url, name);
  384. return {};
  385. };
  386. view().on_get_cookie = [this](auto& url, auto source) -> DeprecatedString {
  387. if (on_get_cookie)
  388. return on_get_cookie(url, source);
  389. return {};
  390. };
  391. view().on_set_cookie = [this](auto& url, auto& cookie, auto source) {
  392. if (on_set_cookie)
  393. on_set_cookie(url, cookie, source);
  394. };
  395. view().on_update_cookie = [this](auto& cookie) {
  396. if (on_update_cookie)
  397. on_update_cookie(cookie);
  398. };
  399. view().on_get_source = [this](auto& url, auto& source) {
  400. view_source(url, source);
  401. };
  402. view().on_get_dom_tree = [this](auto& dom_tree) {
  403. if (m_dom_inspector_widget)
  404. m_dom_inspector_widget->set_dom_json(dom_tree);
  405. };
  406. view().on_get_dom_node_properties = [this](auto node_id, auto& specified, auto& computed, auto& custom_properties, auto& node_box_sizing) {
  407. m_dom_inspector_widget->set_dom_node_properties_json({ node_id }, specified, computed, custom_properties, node_box_sizing);
  408. };
  409. view().on_get_accessibility_tree = [this](auto& accessibility_tree) {
  410. if (m_dom_inspector_widget)
  411. m_dom_inspector_widget->set_accessibility_json(accessibility_tree);
  412. };
  413. view().on_js_console_new_message = [this](auto message_index) {
  414. if (m_console_widget)
  415. m_console_widget->notify_about_new_console_message(message_index);
  416. };
  417. view().on_get_js_console_messages = [this](auto start_index, auto& message_types, auto& messages) {
  418. if (m_console_widget)
  419. m_console_widget->handle_console_messages(start_index, message_types, messages);
  420. };
  421. auto focus_location_box_action = GUI::Action::create(
  422. "Focus location box", { Mod_Ctrl, Key_L }, Key_F6, [this](auto&) {
  423. m_location_box->set_focus(true);
  424. m_location_box->select_current_line();
  425. },
  426. this);
  427. m_statusbar = *find_descendant_of_type_named<GUI::Statusbar>("statusbar");
  428. view().on_link_hover = [this](auto& url) {
  429. update_status(String::from_deprecated_string(url.to_deprecated_string()).release_value_but_fixme_should_propagate_errors());
  430. };
  431. view().on_link_unhover = [this]() {
  432. view().set_override_cursor(Gfx::StandardCursor::None);
  433. update_status();
  434. };
  435. view().on_back_button = [this] {
  436. if (m_history.can_go_back())
  437. go_back();
  438. };
  439. view().on_forward_button = [this] {
  440. if (m_history.can_go_forward())
  441. go_forward();
  442. };
  443. view().on_new_tab = [this](auto activate_tab) {
  444. auto& tab = this->window().create_new_tab(URL("about:blank"), activate_tab);
  445. return tab.view().handle();
  446. };
  447. view().on_activate_tab = [this]() {
  448. on_activate_tab_request(*this);
  449. };
  450. view().on_close = [this] {
  451. on_tab_close_request(*this);
  452. };
  453. m_tab_context_menu = GUI::Menu::construct();
  454. m_tab_context_menu->add_action(GUI::CommonActions::make_reload_action([this](auto&) {
  455. reload();
  456. }));
  457. m_tab_context_menu->add_action(GUI::CommonActions::make_close_tab_action([this](auto&) {
  458. on_tab_close_request(*this);
  459. }));
  460. m_tab_context_menu->add_action(GUI::Action::create("&Duplicate Tab", g_icon_bag.duplicate_tab, [this](auto&) {
  461. on_tab_open_request(url());
  462. }));
  463. m_tab_context_menu->add_action(GUI::Action::create("Close &Other Tabs", g_icon_bag.close_other_tabs, [this](auto&) {
  464. on_tab_close_other_request(*this);
  465. }));
  466. auto take_visible_screenshot_action = GUI::Action::create(
  467. "Take &Visible Screenshot"sv, g_icon_bag.filetype_image, [this](auto&) {
  468. if (auto result = view().take_screenshot(WebView::ViewImplementation::ScreenshotType::Visible); result.is_error()) {
  469. auto error = String::formatted("{}", result.error()).release_value_but_fixme_should_propagate_errors();
  470. GUI::MessageBox::show_error(&this->window(), error);
  471. }
  472. },
  473. this);
  474. take_visible_screenshot_action->set_status_tip("Save a screenshot of the visible portion of the current tab to the Downloads directory"_string.release_value_but_fixme_should_propagate_errors());
  475. auto take_full_screenshot_action = GUI::Action::create(
  476. "Take &Full Screenshot"sv, g_icon_bag.filetype_image, [this](auto&) {
  477. if (auto result = view().take_screenshot(WebView::ViewImplementation::ScreenshotType::Full); result.is_error()) {
  478. auto error = String::formatted("{}", result.error()).release_value_but_fixme_should_propagate_errors();
  479. GUI::MessageBox::show_error(&this->window(), error);
  480. }
  481. },
  482. this);
  483. take_full_screenshot_action->set_status_tip("Save a screenshot of the entirety of the current tab to the Downloads directory"_string.release_value_but_fixme_should_propagate_errors());
  484. m_page_context_menu = GUI::Menu::construct();
  485. m_page_context_menu->add_action(window.go_back_action());
  486. m_page_context_menu->add_action(window.go_forward_action());
  487. m_page_context_menu->add_action(window.reload_action());
  488. m_page_context_menu->add_separator();
  489. m_page_context_menu->add_action(window.copy_selection_action());
  490. m_page_context_menu->add_action(window.select_all_action());
  491. m_page_context_menu->add_separator();
  492. m_page_context_menu->add_action(move(take_visible_screenshot_action));
  493. m_page_context_menu->add_action(move(take_full_screenshot_action));
  494. m_page_context_menu->add_separator();
  495. m_page_context_menu->add_action(window.view_source_action());
  496. m_page_context_menu->add_action(window.inspect_dom_tree_action());
  497. m_page_context_menu->add_action(window.inspect_dom_node_action());
  498. view().on_context_menu_request = [&](auto widget_position) {
  499. auto screen_position = view().screen_relative_rect().location().translated(widget_position);
  500. m_page_context_menu->popup(screen_position);
  501. };
  502. }
  503. void Tab::update_reset_zoom_button()
  504. {
  505. auto zoom_level = view().zoom_level();
  506. if (zoom_level != 1.0f) {
  507. m_reset_zoom_button->set_text(MUST(String::formatted("{}%", round_to<int>(zoom_level * 100))));
  508. m_reset_zoom_button->set_visible(true);
  509. } else {
  510. m_reset_zoom_button->set_visible(false);
  511. }
  512. }
  513. Optional<URL> Tab::url_from_location_bar(MayAppendTLD may_append_tld)
  514. {
  515. if (m_location_box->text().starts_with('?') && g_search_engine.is_empty()) {
  516. GUI::MessageBox::show(&this->window(), "Select a search engine in the Settings menu before searching."sv, "No search engine selected"sv, GUI::MessageBox::Type::Information);
  517. return {};
  518. }
  519. DeprecatedString text = m_location_box->text();
  520. StringBuilder builder;
  521. builder.append(text);
  522. if (may_append_tld == MayAppendTLD::Yes) {
  523. // FIXME: Expand the list of top level domains.
  524. if (!(text.ends_with(".com"sv) || text.ends_with(".net"sv) || text.ends_with(".org"sv))) {
  525. builder.append(".com"sv);
  526. }
  527. }
  528. auto final_text = builder.to_deprecated_string();
  529. auto url = url_from_user_input(final_text);
  530. return url;
  531. }
  532. void Tab::load(const URL& url, LoadType load_type)
  533. {
  534. m_is_history_navigation = (load_type == LoadType::HistoryNavigation);
  535. m_web_content_view->load(url);
  536. m_location_box->set_focus(false);
  537. }
  538. URL Tab::url() const
  539. {
  540. return m_web_content_view->url();
  541. }
  542. void Tab::reload()
  543. {
  544. load(url());
  545. }
  546. void Tab::go_back(int steps)
  547. {
  548. m_history.go_back(steps);
  549. update_actions();
  550. load(m_history.current().url, LoadType::HistoryNavigation);
  551. }
  552. void Tab::go_forward(int steps)
  553. {
  554. m_history.go_forward(steps);
  555. update_actions();
  556. load(m_history.current().url, LoadType::HistoryNavigation);
  557. }
  558. void Tab::update_actions()
  559. {
  560. auto& window = this->window();
  561. if (this != &window.active_tab())
  562. return;
  563. window.go_back_action().set_enabled(m_history.can_go_back());
  564. window.go_forward_action().set_enabled(m_history.can_go_forward());
  565. }
  566. ErrorOr<void> Tab::bookmark_current_url()
  567. {
  568. auto url = this->url().to_deprecated_string();
  569. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  570. TRY(BookmarksBarWidget::the().remove_bookmark(url));
  571. } else {
  572. TRY(BookmarksBarWidget::the().add_bookmark(url, m_title));
  573. }
  574. return {};
  575. }
  576. void Tab::update_bookmark_button(StringView url)
  577. {
  578. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  579. m_bookmark_button->set_icon(g_icon_bag.bookmark_filled);
  580. m_bookmark_button->set_tooltip("Remove Bookmark");
  581. } else {
  582. m_bookmark_button->set_icon(g_icon_bag.bookmark_contour);
  583. m_bookmark_button->set_tooltip("Add Bookmark");
  584. }
  585. }
  586. void Tab::did_become_active()
  587. {
  588. BookmarksBarWidget::the().on_bookmark_click = [this](auto& url, auto open) {
  589. if (open == BookmarksBarWidget::Open::InNewTab)
  590. on_tab_open_request(url);
  591. else if (open == BookmarksBarWidget::Open::InNewWindow)
  592. on_window_open_request(url);
  593. else
  594. load(url);
  595. };
  596. BookmarksBarWidget::the().on_bookmark_hover = [this](auto&, auto& url) {
  597. m_statusbar->set_text(String::from_deprecated_string(url).release_value_but_fixme_should_propagate_errors());
  598. };
  599. BookmarksBarWidget::the().on_bookmark_change = [this]() {
  600. update_bookmark_button(url().to_deprecated_string());
  601. };
  602. BookmarksBarWidget::the().remove_from_parent();
  603. m_toolbar_container->add_child(BookmarksBarWidget::the());
  604. auto is_fullscreen = window().is_fullscreen();
  605. m_toolbar_container->set_visible(!is_fullscreen);
  606. m_statusbar->set_visible(!is_fullscreen);
  607. update_actions();
  608. }
  609. void Tab::context_menu_requested(Gfx::IntPoint screen_position)
  610. {
  611. m_tab_context_menu->popup(screen_position);
  612. }
  613. void Tab::content_filters_changed()
  614. {
  615. if (g_content_filters_enabled)
  616. m_web_content_view->set_content_filters(g_content_filters);
  617. else
  618. m_web_content_view->set_content_filters({});
  619. }
  620. void Tab::autoplay_allowlist_changed()
  621. {
  622. if (g_autoplay_allowed_on_all_websites)
  623. m_web_content_view->set_autoplay_allowed_on_all_websites();
  624. else
  625. m_web_content_view->set_autoplay_allowlist(g_autoplay_allowlist);
  626. }
  627. void Tab::proxy_mappings_changed()
  628. {
  629. m_web_content_view->set_proxy_mappings(g_proxies, g_proxy_mappings);
  630. }
  631. void Tab::action_entered(GUI::Action& action)
  632. {
  633. m_statusbar->set_override_text(action.status_tip());
  634. }
  635. void Tab::action_left(GUI::Action&)
  636. {
  637. m_statusbar->set_override_text({});
  638. }
  639. void Tab::window_position_changed(Gfx::IntPoint position)
  640. {
  641. m_web_content_view->set_window_position(position);
  642. }
  643. void Tab::window_size_changed(Gfx::IntSize size)
  644. {
  645. m_web_content_view->set_window_size(size);
  646. }
  647. BrowserWindow const& Tab::window() const
  648. {
  649. return static_cast<BrowserWindow const&>(*Widget::window());
  650. }
  651. BrowserWindow& Tab::window()
  652. {
  653. return static_cast<BrowserWindow&>(*Widget::window());
  654. }
  655. void Tab::show_inspector_window(Browser::Tab::InspectorTarget inspector_target)
  656. {
  657. if (!m_dom_inspector_widget) {
  658. auto window = GUI::Window::construct(&this->window());
  659. window->set_window_mode(GUI::WindowMode::Modeless);
  660. window->resize(300, 500);
  661. window->set_title("Inspector");
  662. window->set_icon(g_icon_bag.inspector_object);
  663. window->on_close = [&]() {
  664. m_web_content_view->clear_inspected_dom_node();
  665. };
  666. m_dom_inspector_widget = window->set_main_widget<InspectorWidget>().release_value_but_fixme_should_propagate_errors();
  667. m_dom_inspector_widget->set_web_view(*m_web_content_view);
  668. m_web_content_view->inspect_dom_tree();
  669. m_web_content_view->inspect_accessibility_tree();
  670. }
  671. if (inspector_target == InspectorTarget::HoveredElement) {
  672. // FIXME: Handle pseudo-elements
  673. auto hovered_node = m_web_content_view->get_hovered_node_id();
  674. m_dom_inspector_widget->set_selection({ hovered_node });
  675. } else {
  676. VERIFY(inspector_target == InspectorTarget::Document);
  677. m_dom_inspector_widget->select_default_node();
  678. }
  679. auto* window = m_dom_inspector_widget->window();
  680. window->show();
  681. window->move_to_front();
  682. }
  683. void Tab::close_sub_widgets()
  684. {
  685. auto close_widget_window = [](auto& widget) {
  686. if (widget) {
  687. auto* window = widget->window();
  688. window->close();
  689. }
  690. };
  691. close_widget_window(m_dom_inspector_widget);
  692. close_widget_window(m_console_widget);
  693. close_widget_window(m_storage_widget);
  694. }
  695. void Tab::show_console_window()
  696. {
  697. if (!m_console_widget) {
  698. auto console_window = GUI::Window::construct(&window());
  699. console_window->resize(500, 300);
  700. console_window->set_title("JS Console");
  701. console_window->set_icon(g_icon_bag.filetype_javascript);
  702. m_console_widget = console_window->set_main_widget<ConsoleWidget>().release_value_but_fixme_should_propagate_errors();
  703. m_console_widget->on_js_input = [this](DeprecatedString const& js_source) {
  704. m_web_content_view->js_console_input(js_source);
  705. };
  706. m_console_widget->on_request_messages = [this](i32 start_index) {
  707. m_web_content_view->js_console_request_messages(start_index);
  708. };
  709. }
  710. auto* window = m_console_widget->window();
  711. window->show();
  712. window->move_to_front();
  713. }
  714. void Tab::show_storage_inspector()
  715. {
  716. if (!m_storage_widget) {
  717. auto storage_window = GUI::Window::construct(&window());
  718. storage_window->resize(500, 300);
  719. storage_window->set_title("Storage Inspector");
  720. storage_window->set_icon(g_icon_bag.cookie);
  721. m_storage_widget = storage_window->set_main_widget<StorageWidget>().release_value_but_fixme_should_propagate_errors();
  722. m_storage_widget->on_update_cookie = [this](Web::Cookie::Cookie cookie) {
  723. if (on_update_cookie)
  724. on_update_cookie(move(cookie));
  725. };
  726. }
  727. if (on_get_cookies_entries) {
  728. auto cookies = on_get_cookies_entries();
  729. m_storage_widget->clear_cookies();
  730. m_storage_widget->set_cookies_entries(cookies);
  731. }
  732. if (on_get_local_storage_entries) {
  733. auto local_storage_entries = on_get_local_storage_entries();
  734. m_storage_widget->clear_local_storage_entries();
  735. m_storage_widget->set_local_storage_entries(move(local_storage_entries));
  736. }
  737. if (on_get_session_storage_entries) {
  738. auto session_storage_entries = on_get_session_storage_entries();
  739. m_storage_widget->clear_session_storage_entries();
  740. m_storage_widget->set_session_storage_entries(move(session_storage_entries));
  741. }
  742. auto* window = m_storage_widget->window();
  743. window->show();
  744. window->move_to_front();
  745. }
  746. void Tab::show_history_inspector()
  747. {
  748. if (!m_history_widget) {
  749. auto history_window = GUI::Window::construct(&window());
  750. history_window->resize(500, 300);
  751. history_window->set_title("History");
  752. history_window->set_icon(g_icon_bag.history);
  753. m_history_widget = history_window->set_main_widget<HistoryWidget>().release_value_but_fixme_should_propagate_errors();
  754. }
  755. m_history_widget->clear_history_entries();
  756. m_history_widget->set_history_entries(m_history.get_all_history_entries());
  757. auto* window = m_history_widget->window();
  758. window->show();
  759. window->move_to_front();
  760. }
  761. void Tab::show_event(GUI::ShowEvent&)
  762. {
  763. m_web_content_view->set_visible(true);
  764. }
  765. void Tab::hide_event(GUI::HideEvent&)
  766. {
  767. m_web_content_view->set_visible(false);
  768. }
  769. void Tab::enable_webdriver_mode()
  770. {
  771. m_web_content_view->connect_to_webdriver(Browser::g_webdriver_content_ipc_path);
  772. auto& webdriver_banner = *find_descendant_of_type_named<GUI::Widget>("webdriver_banner");
  773. webdriver_banner.set_visible(true);
  774. }
  775. }