Tab.cpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678
  1. /*
  2. * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
  3. * All rights reserved.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are met:
  7. *
  8. * 1. Redistributions of source code must retain the above copyright notice, this
  9. * list of conditions and the following disclaimer.
  10. *
  11. * 2. Redistributions in binary form must reproduce the above copyright notice,
  12. * this list of conditions and the following disclaimer in the documentation
  13. * and/or other materials provided with the distribution.
  14. *
  15. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  16. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  17. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  18. * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
  19. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  20. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
  21. * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  22. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  23. * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  24. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  25. */
  26. #include "Tab.h"
  27. #include "BookmarksBarWidget.h"
  28. #include "Browser.h"
  29. #include "ConsoleWidget.h"
  30. #include "DownloadWidget.h"
  31. #include "InspectorWidget.h"
  32. #include "WindowActions.h"
  33. #include <AK/StringBuilder.h>
  34. #include <Applications/Browser/TabGML.h>
  35. #include <LibGUI/Action.h>
  36. #include <LibGUI/Application.h>
  37. #include <LibGUI/BoxLayout.h>
  38. #include <LibGUI/Button.h>
  39. #include <LibGUI/Clipboard.h>
  40. #include <LibGUI/InputBox.h>
  41. #include <LibGUI/Menu.h>
  42. #include <LibGUI/Menubar.h>
  43. #include <LibGUI/Statusbar.h>
  44. #include <LibGUI/TabWidget.h>
  45. #include <LibGUI/TextBox.h>
  46. #include <LibGUI/Toolbar.h>
  47. #include <LibGUI/ToolbarContainer.h>
  48. #include <LibGUI/Window.h>
  49. #include <LibJS/Interpreter.h>
  50. #include <LibWeb/Dump.h>
  51. #include <LibWeb/InProcessWebView.h>
  52. #include <LibWeb/Layout/BlockBox.h>
  53. #include <LibWeb/Layout/InitialContainingBlockBox.h>
  54. #include <LibWeb/Loader/ResourceLoader.h>
  55. #include <LibWeb/OutOfProcessWebView.h>
  56. #include <LibWeb/Page/Frame.h>
  57. namespace Browser {
  58. URL url_from_user_input(const String& input)
  59. {
  60. auto url = URL(input);
  61. if (url.is_valid())
  62. return url;
  63. StringBuilder builder;
  64. builder.append("http://");
  65. builder.append(input);
  66. return URL(builder.build());
  67. }
  68. void Tab::start_download(const URL& url)
  69. {
  70. auto window = GUI::Window::construct(this->window());
  71. window->resize(300, 150);
  72. window->set_title(String::formatted("0% of {}", url.basename()));
  73. window->set_resizable(false);
  74. window->set_main_widget<DownloadWidget>(url);
  75. window->show();
  76. [[maybe_unused]] auto& unused = window.leak_ref();
  77. }
  78. void Tab::view_source(const URL& url, const String& source)
  79. {
  80. auto window = GUI::Window::construct(this->window());
  81. auto& editor = window->set_main_widget<GUI::TextEditor>();
  82. editor.set_text(source);
  83. editor.set_mode(GUI::TextEditor::ReadOnly);
  84. editor.set_ruler_visible(true);
  85. window->resize(640, 480);
  86. window->set_title(url.to_string());
  87. window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/filetype-text.png"));
  88. window->show();
  89. [[maybe_unused]] auto& unused = window.leak_ref();
  90. }
  91. Tab::Tab(Type type)
  92. : m_type(type)
  93. {
  94. load_from_gml(tab_gml);
  95. m_toolbar_container = *find_descendant_of_type_named<GUI::ToolbarContainer>("toolbar_container");
  96. auto& toolbar = *find_descendant_of_type_named<GUI::Toolbar>("toolbar");
  97. auto& webview_container = *find_descendant_of_type_named<GUI::Widget>("webview_container");
  98. if (m_type == Type::InProcessWebView)
  99. m_page_view = webview_container.add<Web::InProcessWebView>();
  100. else
  101. m_web_content_view = webview_container.add<Web::OutOfProcessWebView>();
  102. m_go_back_action = GUI::CommonActions::make_go_back_action([this](auto&) { go_back(); }, this);
  103. m_go_forward_action = GUI::CommonActions::make_go_forward_action([this](auto&) { go_forward(); }, this);
  104. m_go_home_action = GUI::CommonActions::make_go_home_action([this](auto&) { load(g_home_url); }, this);
  105. m_go_home_action->set_status_tip("Go to home page");
  106. toolbar.add_action(*m_go_back_action);
  107. toolbar.add_action(*m_go_forward_action);
  108. toolbar.add_action(*m_go_home_action);
  109. m_reload_action = GUI::CommonActions::make_reload_action([this](auto&) { reload(); }, this);
  110. m_reload_action->set_status_tip("Reload current page");
  111. toolbar.add_action(*m_reload_action);
  112. m_location_box = toolbar.add<GUI::TextBox>();
  113. m_location_box->set_placeholder("Address");
  114. m_location_box->on_return_pressed = [this] {
  115. auto url = url_from_user_input(m_location_box->text());
  116. load(url);
  117. view().set_focus(true);
  118. };
  119. m_location_box->add_custom_context_menu_action(GUI::Action::create("Paste & Go", [this](auto&) {
  120. m_location_box->set_text(GUI::Clipboard::the().data());
  121. m_location_box->on_return_pressed();
  122. }));
  123. m_bookmark_button = toolbar.add<GUI::Button>();
  124. m_bookmark_button->set_button_style(Gfx::ButtonStyle::Coolbar);
  125. m_bookmark_button->set_focus_policy(GUI::FocusPolicy::TabFocus);
  126. m_bookmark_button->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/bookmark-contour.png"));
  127. m_bookmark_button->set_fixed_size(22, 22);
  128. m_bookmark_button->on_click = [this](auto) {
  129. auto url = this->url().to_string();
  130. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  131. BookmarksBarWidget::the().remove_bookmark(url);
  132. } else {
  133. BookmarksBarWidget::the().add_bookmark(url, m_title);
  134. }
  135. update_bookmark_button(url);
  136. };
  137. hooks().on_load_start = [this](auto& url) {
  138. m_location_box->set_icon(nullptr);
  139. m_location_box->set_text(url.to_string());
  140. // don't add to history if back or forward is pressed
  141. if (!m_is_history_navigation)
  142. m_history.push(url);
  143. m_is_history_navigation = false;
  144. update_actions();
  145. update_bookmark_button(url.to_string());
  146. };
  147. hooks().on_link_click = [this](auto& url, auto& target, unsigned modifiers) {
  148. if (target == "_blank" || modifiers == Mod_Ctrl) {
  149. on_tab_open_request(url);
  150. } else {
  151. load(url);
  152. }
  153. };
  154. m_link_context_menu = GUI::Menu::construct();
  155. auto link_default_action = GUI::Action::create("&Open", [this](auto&) {
  156. hooks().on_link_click(m_link_context_menu_url, "", 0);
  157. });
  158. m_link_context_menu->add_action(link_default_action);
  159. m_link_context_menu_default_action = link_default_action;
  160. m_link_context_menu->add_action(GUI::Action::create("Open in New &Tab", [this](auto&) {
  161. hooks().on_link_click(m_link_context_menu_url, "_blank", 0);
  162. }));
  163. m_link_context_menu->add_separator();
  164. m_link_context_menu->add_action(GUI::Action::create("&Copy URL", [this](auto&) {
  165. GUI::Clipboard::the().set_plain_text(m_link_context_menu_url.to_string());
  166. }));
  167. m_link_context_menu->add_separator();
  168. m_link_context_menu->add_action(GUI::Action::create("&Download", [this](auto&) {
  169. start_download(m_link_context_menu_url);
  170. }));
  171. hooks().on_link_context_menu_request = [this](auto& url, auto& screen_position) {
  172. m_link_context_menu_url = url;
  173. m_link_context_menu->popup(screen_position, m_link_context_menu_default_action);
  174. };
  175. m_image_context_menu = GUI::Menu::construct();
  176. m_image_context_menu->add_action(GUI::Action::create("&Open Image", [this](auto&) {
  177. hooks().on_link_click(m_image_context_menu_url, "", 0);
  178. }));
  179. m_image_context_menu->add_action(GUI::Action::create("Open Image in New &Tab", [this](auto&) {
  180. hooks().on_link_click(m_image_context_menu_url, "_blank", 0);
  181. }));
  182. m_image_context_menu->add_separator();
  183. m_image_context_menu->add_action(GUI::Action::create("&Copy Image", [this](auto&) {
  184. if (m_image_context_menu_bitmap.is_valid())
  185. GUI::Clipboard::the().set_bitmap(*m_image_context_menu_bitmap.bitmap());
  186. }));
  187. m_image_context_menu->add_action(GUI::Action::create("Copy Image &URL", [this](auto&) {
  188. GUI::Clipboard::the().set_plain_text(m_image_context_menu_url.to_string());
  189. }));
  190. m_image_context_menu->add_separator();
  191. m_image_context_menu->add_action(GUI::Action::create("&Download", [this](auto&) {
  192. start_download(m_image_context_menu_url);
  193. }));
  194. hooks().on_image_context_menu_request = [this](auto& image_url, auto& screen_position, const Gfx::ShareableBitmap& shareable_bitmap) {
  195. m_image_context_menu_url = image_url;
  196. m_image_context_menu_bitmap = shareable_bitmap;
  197. m_image_context_menu->popup(screen_position);
  198. };
  199. hooks().on_link_middle_click = [this](auto& href, auto&, auto) {
  200. hooks().on_link_click(href, "_blank", 0);
  201. };
  202. hooks().on_title_change = [this](auto& title) {
  203. if (title.is_null()) {
  204. m_title = url().to_string();
  205. } else {
  206. m_title = title;
  207. }
  208. if (on_title_change)
  209. on_title_change(m_title);
  210. };
  211. hooks().on_favicon_change = [this](auto& icon) {
  212. m_icon = icon;
  213. m_location_box->set_icon(&icon);
  214. if (on_favicon_change)
  215. on_favicon_change(icon);
  216. };
  217. hooks().on_get_cookie = [this](auto& url, auto source) -> String {
  218. if (on_get_cookie)
  219. return on_get_cookie(url, source);
  220. return {};
  221. };
  222. hooks().on_set_cookie = [this](auto& url, auto& cookie, auto source) {
  223. if (on_set_cookie)
  224. on_set_cookie(url, cookie, source);
  225. };
  226. hooks().on_get_source = [this](auto& url, auto& source) {
  227. view_source(url, source);
  228. };
  229. hooks().on_js_console_output = [this](auto& method, auto& line) {
  230. if (m_console_window) {
  231. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  232. console_widget->handle_js_console_output(method, line);
  233. }
  234. };
  235. if (m_type == Type::InProcessWebView) {
  236. hooks().on_set_document = [this](auto* document) {
  237. if (document && m_console_window) {
  238. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  239. console_widget->set_interpreter(document->interpreter().make_weak_ptr());
  240. }
  241. };
  242. }
  243. auto focus_location_box_action = GUI::Action::create(
  244. "Focus location box", { Mod_Ctrl, Key_L }, [this](auto&) {
  245. m_location_box->select_all();
  246. m_location_box->set_focus(true);
  247. },
  248. this);
  249. m_statusbar = *find_descendant_of_type_named<GUI::Statusbar>("statusbar");
  250. hooks().on_link_hover = [this](auto& url) {
  251. if (url.is_valid())
  252. m_statusbar->set_text(url.to_string());
  253. else
  254. m_statusbar->set_text("");
  255. };
  256. hooks().on_url_drop = [this](auto& url) {
  257. load(url);
  258. };
  259. m_menubar = GUI::Menubar::construct();
  260. auto& app_menu = m_menubar->add_menu("&File");
  261. app_menu.add_action(WindowActions::the().create_new_tab_action());
  262. auto close_tab_action = GUI::Action::create(
  263. "&Close Tab", { Mod_Ctrl, Key_W }, Gfx::Bitmap::load_from_file("/res/icons/16x16/close-tab.png"), [this](auto&) {
  264. on_tab_close_request(*this);
  265. },
  266. this);
  267. close_tab_action->set_status_tip("Close current tab");
  268. app_menu.add_action(close_tab_action);
  269. app_menu.add_separator();
  270. app_menu.add_action(GUI::CommonActions::make_quit_action([](auto&) {
  271. GUI::Application::the()->quit();
  272. }));
  273. auto& view_menu = m_menubar->add_menu("&View");
  274. view_menu.add_action(WindowActions::the().show_bookmarks_bar_action());
  275. view_menu.add_separator();
  276. view_menu.add_action(GUI::CommonActions::make_fullscreen_action(
  277. [this](auto&) {
  278. window()->set_fullscreen(!window()->is_fullscreen());
  279. auto is_fullscreen = window()->is_fullscreen();
  280. auto* tab_widget = static_cast<GUI::TabWidget*>(parent_widget());
  281. tab_widget->set_bar_visible(!is_fullscreen && tab_widget->children().size() > 1);
  282. m_toolbar_container->set_visible(!is_fullscreen);
  283. m_statusbar->set_visible(!is_fullscreen);
  284. if (is_fullscreen) {
  285. view().set_frame_thickness(0);
  286. } else {
  287. view().set_frame_thickness(2);
  288. }
  289. },
  290. this));
  291. auto& go_menu = m_menubar->add_menu("&Go");
  292. go_menu.add_action(*m_go_back_action);
  293. go_menu.add_action(*m_go_forward_action);
  294. go_menu.add_action(*m_go_home_action);
  295. go_menu.add_separator();
  296. go_menu.add_action(*m_reload_action);
  297. auto view_source_action = GUI::Action::create(
  298. "View &Source", { Mod_Ctrl, Key_U }, [this](auto&) {
  299. if (m_type == Type::InProcessWebView) {
  300. VERIFY(m_page_view->document());
  301. auto url = m_page_view->document()->url();
  302. auto source = m_page_view->document()->source();
  303. view_source(url, source);
  304. } else {
  305. m_web_content_view->get_source();
  306. }
  307. },
  308. this);
  309. view_source_action->set_status_tip("View source code of the current page");
  310. auto inspect_dom_tree_action = GUI::Action::create(
  311. "Inspect &DOM Tree", { Mod_None, Key_F12 }, [this](auto&) {
  312. if (m_type == Type::InProcessWebView) {
  313. if (!m_dom_inspector_window) {
  314. m_dom_inspector_window = GUI::Window::construct(window());
  315. m_dom_inspector_window->resize(300, 500);
  316. m_dom_inspector_window->set_title("DOM inspector");
  317. m_dom_inspector_window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/inspector-object.png"));
  318. m_dom_inspector_window->set_main_widget<InspectorWidget>();
  319. }
  320. auto* inspector_widget = static_cast<InspectorWidget*>(m_dom_inspector_window->main_widget());
  321. inspector_widget->set_document(m_page_view->document());
  322. m_dom_inspector_window->show();
  323. m_dom_inspector_window->move_to_front();
  324. } else {
  325. TODO();
  326. }
  327. },
  328. this);
  329. inspect_dom_tree_action->set_status_tip("Open DOM inspector window for this page");
  330. auto& inspect_menu = m_menubar->add_menu("&Inspect");
  331. inspect_menu.add_action(*view_source_action);
  332. inspect_menu.add_action(*inspect_dom_tree_action);
  333. auto js_console_action = GUI::Action::create(
  334. "Open &JS Console", { Mod_Ctrl, Key_I }, [this](auto&) {
  335. if (m_type == Type::InProcessWebView) {
  336. if (!m_console_window) {
  337. m_console_window = GUI::Window::construct(window());
  338. m_console_window->resize(500, 300);
  339. m_console_window->set_title("JS Console");
  340. m_console_window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/filetype-javascript.png"));
  341. m_console_window->set_main_widget<ConsoleWidget>();
  342. }
  343. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  344. console_widget->set_interpreter(m_page_view->document()->interpreter().make_weak_ptr());
  345. m_console_window->show();
  346. m_console_window->move_to_front();
  347. } else {
  348. if (!m_console_window) {
  349. m_console_window = GUI::Window::construct(window());
  350. m_console_window->resize(500, 300);
  351. m_console_window->set_title("JS Console");
  352. m_console_window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/filetype-javascript.png"));
  353. m_console_window->set_main_widget<ConsoleWidget>();
  354. }
  355. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  356. console_widget->on_js_input = [this](const String& js_source) {
  357. m_web_content_view->js_console_input(js_source);
  358. };
  359. console_widget->clear_output();
  360. m_web_content_view->js_console_initialize();
  361. m_console_window->show();
  362. m_console_window->move_to_front();
  363. }
  364. },
  365. this);
  366. js_console_action->set_status_tip("Open JavaScript console for this page");
  367. inspect_menu.add_action(js_console_action);
  368. auto& debug_menu = m_menubar->add_menu("&Debug");
  369. debug_menu.add_action(GUI::Action::create(
  370. "Dump &DOM Tree", [this](auto&) {
  371. if (m_type == Type::InProcessWebView) {
  372. Web::dump_tree(*m_page_view->document());
  373. } else {
  374. m_web_content_view->debug_request("dump-dom-tree");
  375. }
  376. },
  377. this));
  378. debug_menu.add_action(GUI::Action::create(
  379. "Dump &Layout Tree", [this](auto&) {
  380. if (m_type == Type::InProcessWebView) {
  381. Web::dump_tree(*m_page_view->document()->layout_node());
  382. } else {
  383. m_web_content_view->debug_request("dump-layout-tree");
  384. }
  385. },
  386. this));
  387. debug_menu.add_action(GUI::Action::create(
  388. "Dump &Style Sheets", [this](auto&) {
  389. if (m_type == Type::InProcessWebView) {
  390. for (auto& sheet : m_page_view->document()->style_sheets().sheets()) {
  391. Web::dump_sheet(sheet);
  392. }
  393. } else {
  394. m_web_content_view->debug_request("dump-style-sheets");
  395. }
  396. },
  397. this));
  398. debug_menu.add_action(GUI::Action::create("Dump &History", { Mod_Ctrl, Key_H }, [&](auto&) {
  399. m_history.dump();
  400. }));
  401. debug_menu.add_action(GUI::Action::create("Dump C&ookies", [&](auto&) {
  402. if (on_dump_cookies)
  403. on_dump_cookies();
  404. }));
  405. debug_menu.add_separator();
  406. auto line_box_borders_action = GUI::Action::create_checkable(
  407. "&Line Box Borders", [this](auto& action) {
  408. if (m_type == Type::InProcessWebView) {
  409. m_page_view->set_should_show_line_box_borders(action.is_checked());
  410. m_page_view->update();
  411. } else {
  412. m_web_content_view->debug_request("set-line-box-borders", action.is_checked() ? "on" : "off");
  413. }
  414. },
  415. this);
  416. line_box_borders_action->set_checked(false);
  417. debug_menu.add_action(line_box_borders_action);
  418. debug_menu.add_separator();
  419. debug_menu.add_action(GUI::Action::create("Collect &Garbage", { Mod_Ctrl | Mod_Shift, Key_G }, [this](auto&) {
  420. if (m_type == Type::InProcessWebView) {
  421. if (auto* document = m_page_view->document()) {
  422. document->interpreter().heap().collect_garbage(JS::Heap::CollectionType::CollectGarbage, true);
  423. }
  424. } else {
  425. m_web_content_view->debug_request("collect-garbage");
  426. }
  427. }));
  428. debug_menu.add_action(GUI::Action::create("Clear &Cache", { Mod_Ctrl | Mod_Shift, Key_C }, [this](auto&) {
  429. if (m_type == Type::InProcessWebView) {
  430. Web::ResourceLoader::the().clear_cache();
  431. } else {
  432. m_web_content_view->debug_request("clear-cache");
  433. }
  434. }));
  435. m_user_agent_spoof_actions.set_exclusive(true);
  436. auto& spoof_user_agent_menu = debug_menu.add_submenu("Spoof User Agent");
  437. m_disable_user_agent_spoofing = GUI::Action::create_checkable("Disabled", [&](auto&) {
  438. if (m_type == Type::InProcessWebView) {
  439. Web::ResourceLoader::the().set_user_agent(Web::default_user_agent);
  440. } else {
  441. m_web_content_view->debug_request("spoof-user-agent", Web::default_user_agent);
  442. }
  443. });
  444. m_disable_user_agent_spoofing->set_status_tip(Web::default_user_agent);
  445. spoof_user_agent_menu.add_action(*m_disable_user_agent_spoofing);
  446. m_user_agent_spoof_actions.add_action(*m_disable_user_agent_spoofing);
  447. m_disable_user_agent_spoofing->set_checked(true);
  448. auto add_user_agent = [&](auto& name, auto& user_agent) {
  449. auto action = GUI::Action::create_checkable(name, [&](auto&) {
  450. if (m_type == Type::InProcessWebView) {
  451. Web::ResourceLoader::the().set_user_agent(user_agent);
  452. } else {
  453. m_web_content_view->debug_request("spoof-user-agent", user_agent);
  454. }
  455. });
  456. action->set_status_tip(user_agent);
  457. spoof_user_agent_menu.add_action(action);
  458. m_user_agent_spoof_actions.add_action(action);
  459. };
  460. add_user_agent("Chrome Linux Desktop", "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36");
  461. add_user_agent("Firefox Linux Desktop", "Mozilla/5.0 (X11; Linux i686; rv:87.0) Gecko/20100101 Firefox/87.0");
  462. add_user_agent("Safari macOS Desktop", "Mozilla/5.0 (Macintosh; Intel Mac OS X 11_2_3) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/14.0.3 Safari/605.1.15");
  463. add_user_agent("Chrome Android Mobile", "Mozilla/5.0 (Linux; Android 10) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.66 Mobile Safari/537.36");
  464. add_user_agent("Firefox Android Mobile", "Mozilla/5.0 (Android 11; Mobile; rv:68.0) Gecko/68.0 Firefox/86.0");
  465. add_user_agent("Safari iOS Mobile", "Mozilla/5.0 (iPhone; CPU iPhone OS 14_4_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/14.0 Mobile/15E148 Safari/604.1");
  466. auto custom_user_agent = GUI::Action::create_checkable("Custom", [&](auto& action) {
  467. String user_agent;
  468. if (GUI::InputBox::show(window(), user_agent, "Enter User Agent:", "Custom User Agent") != GUI::InputBox::ExecOK || user_agent.is_empty() || user_agent.is_null()) {
  469. m_disable_user_agent_spoofing->activate();
  470. return;
  471. }
  472. if (m_type == Type::InProcessWebView) {
  473. Web::ResourceLoader::the().set_user_agent(user_agent);
  474. } else {
  475. m_web_content_view->debug_request("spoof-user-agent", user_agent);
  476. }
  477. action.set_status_tip(user_agent);
  478. });
  479. spoof_user_agent_menu.add_action(custom_user_agent);
  480. m_user_agent_spoof_actions.add_action(custom_user_agent);
  481. auto& help_menu = m_menubar->add_menu("&Help");
  482. help_menu.add_action(WindowActions::the().about_action());
  483. m_tab_context_menu = GUI::Menu::construct();
  484. m_tab_context_menu->add_action(GUI::Action::create("&Reload Tab", [this](auto&) {
  485. m_reload_action->activate();
  486. }));
  487. m_tab_context_menu->add_action(GUI::Action::create("&Close Tab", [this](auto&) {
  488. on_tab_close_request(*this);
  489. }));
  490. m_page_context_menu = GUI::Menu::construct();
  491. m_page_context_menu->add_action(*m_go_back_action);
  492. m_page_context_menu->add_action(*m_go_forward_action);
  493. m_page_context_menu->add_action(*m_reload_action);
  494. m_page_context_menu->add_separator();
  495. m_page_context_menu->add_action(*view_source_action);
  496. m_page_context_menu->add_action(*inspect_dom_tree_action);
  497. hooks().on_context_menu_request = [&](auto& screen_position) {
  498. m_page_context_menu->popup(screen_position);
  499. };
  500. }
  501. Tab::~Tab()
  502. {
  503. }
  504. void Tab::load(const URL& url, LoadType load_type)
  505. {
  506. m_is_history_navigation = (load_type == LoadType::HistoryNavigation);
  507. if (m_type == Type::InProcessWebView)
  508. m_page_view->load(url);
  509. else
  510. m_web_content_view->load(url);
  511. }
  512. URL Tab::url() const
  513. {
  514. if (m_type == Type::InProcessWebView)
  515. return m_page_view->url();
  516. return m_web_content_view->url();
  517. }
  518. void Tab::reload()
  519. {
  520. load(url());
  521. }
  522. void Tab::go_back()
  523. {
  524. m_history.go_back();
  525. update_actions();
  526. load(m_history.current(), LoadType::HistoryNavigation);
  527. }
  528. void Tab::go_forward()
  529. {
  530. m_history.go_forward();
  531. update_actions();
  532. load(m_history.current(), LoadType::HistoryNavigation);
  533. }
  534. void Tab::update_actions()
  535. {
  536. m_go_back_action->set_enabled(m_history.can_go_back());
  537. m_go_forward_action->set_enabled(m_history.can_go_forward());
  538. }
  539. void Tab::update_bookmark_button(const String& url)
  540. {
  541. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  542. m_bookmark_button->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/bookmark-filled.png"));
  543. m_bookmark_button->set_tooltip("Remove Bookmark");
  544. } else {
  545. m_bookmark_button->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/bookmark-contour.png"));
  546. m_bookmark_button->set_tooltip("Add Bookmark");
  547. }
  548. }
  549. void Tab::did_become_active()
  550. {
  551. Web::ResourceLoader::the().on_load_counter_change = [this] {
  552. if (Web::ResourceLoader::the().pending_loads() == 0) {
  553. m_statusbar->set_text("");
  554. return;
  555. }
  556. m_statusbar->set_text(String::formatted("Loading ({} pending resources...)", Web::ResourceLoader::the().pending_loads()));
  557. };
  558. BookmarksBarWidget::the().on_bookmark_click = [this](auto& url, unsigned modifiers) {
  559. if (modifiers & Mod_Ctrl)
  560. on_tab_open_request(url);
  561. else
  562. load(url);
  563. };
  564. BookmarksBarWidget::the().on_bookmark_hover = [this](auto&, auto& url) {
  565. m_statusbar->set_text(url);
  566. };
  567. BookmarksBarWidget::the().remove_from_parent();
  568. m_toolbar_container->add_child(BookmarksBarWidget::the());
  569. auto is_fullscreen = window()->is_fullscreen();
  570. m_toolbar_container->set_visible(!is_fullscreen);
  571. m_statusbar->set_visible(!is_fullscreen);
  572. window()->set_menubar(m_menubar);
  573. }
  574. void Tab::context_menu_requested(const Gfx::IntPoint& screen_position)
  575. {
  576. m_tab_context_menu->popup(screen_position);
  577. }
  578. GUI::ScrollableWidget& Tab::view()
  579. {
  580. if (m_type == Type::InProcessWebView)
  581. return *m_page_view;
  582. return *m_web_content_view;
  583. }
  584. Web::WebViewHooks& Tab::hooks()
  585. {
  586. if (m_type == Type::InProcessWebView)
  587. return *m_page_view;
  588. return *m_web_content_view;
  589. }
  590. void Tab::action_entered(GUI::Action& action)
  591. {
  592. m_statusbar->set_override_text(action.status_tip());
  593. }
  594. void Tab::action_left(GUI::Action&)
  595. {
  596. m_statusbar->set_override_text({});
  597. }
  598. }