Tab.cpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656
  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. toolbar.add_action(*m_go_back_action);
  106. toolbar.add_action(*m_go_forward_action);
  107. toolbar.add_action(*m_go_home_action);
  108. m_reload_action = GUI::CommonActions::make_reload_action([this](auto&) { reload(); }, this);
  109. toolbar.add_action(*m_reload_action);
  110. m_location_box = toolbar.add<GUI::TextBox>();
  111. m_location_box->set_placeholder("Address");
  112. m_location_box->on_return_pressed = [this] {
  113. auto url = url_from_user_input(m_location_box->text());
  114. load(url);
  115. view().set_focus(true);
  116. };
  117. m_location_box->add_custom_context_menu_action(GUI::Action::create("Paste & Go", [this](auto&) {
  118. m_location_box->set_text(GUI::Clipboard::the().data());
  119. m_location_box->on_return_pressed();
  120. }));
  121. m_bookmark_button = toolbar.add<GUI::Button>();
  122. m_bookmark_button->set_button_style(Gfx::ButtonStyle::Coolbar);
  123. m_bookmark_button->set_focus_policy(GUI::FocusPolicy::TabFocus);
  124. m_bookmark_button->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/bookmark-contour.png"));
  125. m_bookmark_button->set_fixed_size(22, 22);
  126. m_bookmark_button->on_click = [this](auto) {
  127. auto url = this->url().to_string();
  128. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  129. BookmarksBarWidget::the().remove_bookmark(url);
  130. } else {
  131. BookmarksBarWidget::the().add_bookmark(url, m_title);
  132. }
  133. update_bookmark_button(url);
  134. };
  135. hooks().on_load_start = [this](auto& url) {
  136. m_location_box->set_icon(nullptr);
  137. m_location_box->set_text(url.to_string());
  138. // don't add to history if back or forward is pressed
  139. if (!m_is_history_navigation)
  140. m_history.push(url);
  141. m_is_history_navigation = false;
  142. update_actions();
  143. update_bookmark_button(url.to_string());
  144. };
  145. hooks().on_link_click = [this](auto& url, auto& target, unsigned modifiers) {
  146. if (target == "_blank" || modifiers == Mod_Ctrl) {
  147. on_tab_open_request(url);
  148. } else {
  149. load(url);
  150. }
  151. };
  152. m_link_context_menu = GUI::Menu::construct();
  153. auto link_default_action = GUI::Action::create("&Open", [this](auto&) {
  154. hooks().on_link_click(m_link_context_menu_url, "", 0);
  155. });
  156. m_link_context_menu->add_action(link_default_action);
  157. m_link_context_menu_default_action = link_default_action;
  158. m_link_context_menu->add_action(GUI::Action::create("Open in New &Tab", [this](auto&) {
  159. hooks().on_link_click(m_link_context_menu_url, "_blank", 0);
  160. }));
  161. m_link_context_menu->add_separator();
  162. m_link_context_menu->add_action(GUI::Action::create("&Copy URL", [this](auto&) {
  163. GUI::Clipboard::the().set_plain_text(m_link_context_menu_url.to_string());
  164. }));
  165. m_link_context_menu->add_separator();
  166. m_link_context_menu->add_action(GUI::Action::create("&Download", [this](auto&) {
  167. start_download(m_link_context_menu_url);
  168. }));
  169. hooks().on_link_context_menu_request = [this](auto& url, auto& screen_position) {
  170. m_link_context_menu_url = url;
  171. m_link_context_menu->popup(screen_position, m_link_context_menu_default_action);
  172. };
  173. m_image_context_menu = GUI::Menu::construct();
  174. m_image_context_menu->add_action(GUI::Action::create("&Open Image", [this](auto&) {
  175. hooks().on_link_click(m_image_context_menu_url, "", 0);
  176. }));
  177. m_image_context_menu->add_action(GUI::Action::create("Open Image in New &Tab", [this](auto&) {
  178. hooks().on_link_click(m_image_context_menu_url, "_blank", 0);
  179. }));
  180. m_image_context_menu->add_separator();
  181. m_image_context_menu->add_action(GUI::Action::create("&Copy Image", [this](auto&) {
  182. if (m_image_context_menu_bitmap.is_valid())
  183. GUI::Clipboard::the().set_bitmap(*m_image_context_menu_bitmap.bitmap());
  184. }));
  185. m_image_context_menu->add_action(GUI::Action::create("Copy Image &URL", [this](auto&) {
  186. GUI::Clipboard::the().set_plain_text(m_image_context_menu_url.to_string());
  187. }));
  188. m_image_context_menu->add_separator();
  189. m_image_context_menu->add_action(GUI::Action::create("&Download", [this](auto&) {
  190. start_download(m_image_context_menu_url);
  191. }));
  192. hooks().on_image_context_menu_request = [this](auto& image_url, auto& screen_position, const Gfx::ShareableBitmap& shareable_bitmap) {
  193. m_image_context_menu_url = image_url;
  194. m_image_context_menu_bitmap = shareable_bitmap;
  195. m_image_context_menu->popup(screen_position);
  196. };
  197. hooks().on_link_middle_click = [this](auto& href, auto&, auto) {
  198. hooks().on_link_click(href, "_blank", 0);
  199. };
  200. hooks().on_title_change = [this](auto& title) {
  201. if (title.is_null()) {
  202. m_title = url().to_string();
  203. } else {
  204. m_title = title;
  205. }
  206. if (on_title_change)
  207. on_title_change(m_title);
  208. };
  209. hooks().on_favicon_change = [this](auto& icon) {
  210. m_icon = icon;
  211. m_location_box->set_icon(&icon);
  212. if (on_favicon_change)
  213. on_favicon_change(icon);
  214. };
  215. hooks().on_get_cookie = [this](auto& url, auto source) -> String {
  216. if (on_get_cookie)
  217. return on_get_cookie(url, source);
  218. return {};
  219. };
  220. hooks().on_set_cookie = [this](auto& url, auto& cookie, auto source) {
  221. if (on_set_cookie)
  222. on_set_cookie(url, cookie, source);
  223. };
  224. hooks().on_get_source = [this](auto& url, auto& source) {
  225. view_source(url, source);
  226. };
  227. hooks().on_js_console_output = [this](auto& method, auto& line) {
  228. if (m_console_window) {
  229. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  230. console_widget->handle_js_console_output(method, line);
  231. }
  232. };
  233. if (m_type == Type::InProcessWebView) {
  234. hooks().on_set_document = [this](auto* document) {
  235. if (document && m_console_window) {
  236. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  237. console_widget->set_interpreter(document->interpreter().make_weak_ptr());
  238. }
  239. };
  240. }
  241. auto focus_location_box_action = GUI::Action::create(
  242. "Focus location box", { Mod_Ctrl, Key_L }, [this](auto&) {
  243. m_location_box->select_all();
  244. m_location_box->set_focus(true);
  245. },
  246. this);
  247. m_statusbar = *find_descendant_of_type_named<GUI::Statusbar>("statusbar");
  248. hooks().on_link_hover = [this](auto& url) {
  249. if (url.is_valid())
  250. m_statusbar->set_text(url.to_string());
  251. else
  252. m_statusbar->set_text("");
  253. };
  254. hooks().on_url_drop = [this](auto& url) {
  255. load(url);
  256. };
  257. m_menubar = GUI::Menubar::construct();
  258. auto& app_menu = m_menubar->add_menu("&File");
  259. app_menu.add_action(WindowActions::the().create_new_tab_action());
  260. app_menu.add_action(GUI::Action::create(
  261. "&Close Tab", { Mod_Ctrl, Key_W }, Gfx::Bitmap::load_from_file("/res/icons/16x16/close-tab.png"), [this](auto&) {
  262. on_tab_close_request(*this);
  263. },
  264. this));
  265. app_menu.add_separator();
  266. app_menu.add_action(GUI::CommonActions::make_quit_action([](auto&) {
  267. GUI::Application::the()->quit();
  268. }));
  269. auto& view_menu = m_menubar->add_menu("&View");
  270. view_menu.add_action(WindowActions::the().show_bookmarks_bar_action());
  271. view_menu.add_separator();
  272. view_menu.add_action(GUI::CommonActions::make_fullscreen_action(
  273. [this](auto&) {
  274. window()->set_fullscreen(!window()->is_fullscreen());
  275. auto is_fullscreen = window()->is_fullscreen();
  276. auto* tab_widget = static_cast<GUI::TabWidget*>(parent_widget());
  277. tab_widget->set_bar_visible(!is_fullscreen && tab_widget->children().size() > 1);
  278. m_toolbar_container->set_visible(!is_fullscreen);
  279. m_statusbar->set_visible(!is_fullscreen);
  280. if (is_fullscreen) {
  281. view().set_frame_thickness(0);
  282. } else {
  283. view().set_frame_thickness(2);
  284. }
  285. },
  286. this));
  287. auto& go_menu = m_menubar->add_menu("&Go");
  288. go_menu.add_action(*m_go_back_action);
  289. go_menu.add_action(*m_go_forward_action);
  290. go_menu.add_action(*m_go_home_action);
  291. go_menu.add_separator();
  292. go_menu.add_action(*m_reload_action);
  293. auto view_source_action = GUI::Action::create(
  294. "View &Source", { Mod_Ctrl, Key_U }, [this](auto&) {
  295. if (m_type == Type::InProcessWebView) {
  296. VERIFY(m_page_view->document());
  297. auto url = m_page_view->document()->url();
  298. auto source = m_page_view->document()->source();
  299. view_source(url, source);
  300. } else {
  301. m_web_content_view->get_source();
  302. }
  303. },
  304. this);
  305. auto inspect_dom_tree_action = GUI::Action::create(
  306. "Inspect &DOM Tree", { Mod_None, Key_F12 }, [this](auto&) {
  307. if (m_type == Type::InProcessWebView) {
  308. if (!m_dom_inspector_window) {
  309. m_dom_inspector_window = GUI::Window::construct(window());
  310. m_dom_inspector_window->resize(300, 500);
  311. m_dom_inspector_window->set_title("DOM inspector");
  312. m_dom_inspector_window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/inspector-object.png"));
  313. m_dom_inspector_window->set_main_widget<InspectorWidget>();
  314. }
  315. auto* inspector_widget = static_cast<InspectorWidget*>(m_dom_inspector_window->main_widget());
  316. inspector_widget->set_document(m_page_view->document());
  317. m_dom_inspector_window->show();
  318. m_dom_inspector_window->move_to_front();
  319. } else {
  320. TODO();
  321. }
  322. },
  323. this);
  324. auto& inspect_menu = m_menubar->add_menu("&Inspect");
  325. inspect_menu.add_action(*view_source_action);
  326. inspect_menu.add_action(*inspect_dom_tree_action);
  327. inspect_menu.add_action(GUI::Action::create(
  328. "Open &JS Console", { Mod_Ctrl, Key_I }, [this](auto&) {
  329. if (m_type == Type::InProcessWebView) {
  330. if (!m_console_window) {
  331. m_console_window = GUI::Window::construct(window());
  332. m_console_window->resize(500, 300);
  333. m_console_window->set_title("JS Console");
  334. m_console_window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/filetype-javascript.png"));
  335. m_console_window->set_main_widget<ConsoleWidget>();
  336. }
  337. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  338. console_widget->set_interpreter(m_page_view->document()->interpreter().make_weak_ptr());
  339. m_console_window->show();
  340. m_console_window->move_to_front();
  341. } else {
  342. if (!m_console_window) {
  343. m_console_window = GUI::Window::construct(window());
  344. m_console_window->resize(500, 300);
  345. m_console_window->set_title("JS Console");
  346. m_console_window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/filetype-javascript.png"));
  347. m_console_window->set_main_widget<ConsoleWidget>();
  348. }
  349. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  350. console_widget->on_js_input = [this](const String& js_source) {
  351. m_web_content_view->js_console_input(js_source);
  352. };
  353. console_widget->clear_output();
  354. m_web_content_view->js_console_initialize();
  355. m_console_window->show();
  356. m_console_window->move_to_front();
  357. }
  358. },
  359. this));
  360. auto& debug_menu = m_menubar->add_menu("&Debug");
  361. debug_menu.add_action(GUI::Action::create(
  362. "Dump &DOM Tree", [this](auto&) {
  363. if (m_type == Type::InProcessWebView) {
  364. Web::dump_tree(*m_page_view->document());
  365. } else {
  366. m_web_content_view->debug_request("dump-dom-tree");
  367. }
  368. },
  369. this));
  370. debug_menu.add_action(GUI::Action::create(
  371. "Dump &Layout Tree", [this](auto&) {
  372. if (m_type == Type::InProcessWebView) {
  373. Web::dump_tree(*m_page_view->document()->layout_node());
  374. } else {
  375. m_web_content_view->debug_request("dump-layout-tree");
  376. }
  377. },
  378. this));
  379. debug_menu.add_action(GUI::Action::create(
  380. "Dump &Style Sheets", [this](auto&) {
  381. if (m_type == Type::InProcessWebView) {
  382. for (auto& sheet : m_page_view->document()->style_sheets().sheets()) {
  383. Web::dump_sheet(sheet);
  384. }
  385. } else {
  386. m_web_content_view->debug_request("dump-style-sheets");
  387. }
  388. },
  389. this));
  390. debug_menu.add_action(GUI::Action::create("Dump &History", { Mod_Ctrl, Key_H }, [&](auto&) {
  391. m_history.dump();
  392. }));
  393. debug_menu.add_action(GUI::Action::create("Dump C&ookies", [&](auto&) {
  394. if (on_dump_cookies)
  395. on_dump_cookies();
  396. }));
  397. debug_menu.add_separator();
  398. auto line_box_borders_action = GUI::Action::create_checkable(
  399. "&Line Box Borders", [this](auto& action) {
  400. if (m_type == Type::InProcessWebView) {
  401. m_page_view->set_should_show_line_box_borders(action.is_checked());
  402. m_page_view->update();
  403. } else {
  404. m_web_content_view->debug_request("set-line-box-borders", action.is_checked() ? "on" : "off");
  405. }
  406. },
  407. this);
  408. line_box_borders_action->set_checked(false);
  409. debug_menu.add_action(line_box_borders_action);
  410. debug_menu.add_separator();
  411. debug_menu.add_action(GUI::Action::create("Collect &Garbage", { Mod_Ctrl | Mod_Shift, Key_G }, [this](auto&) {
  412. if (m_type == Type::InProcessWebView) {
  413. if (auto* document = m_page_view->document()) {
  414. document->interpreter().heap().collect_garbage(JS::Heap::CollectionType::CollectGarbage, true);
  415. }
  416. } else {
  417. m_web_content_view->debug_request("collect-garbage");
  418. }
  419. }));
  420. debug_menu.add_action(GUI::Action::create("Clear &Cache", { Mod_Ctrl | Mod_Shift, Key_C }, [this](auto&) {
  421. if (m_type == Type::InProcessWebView) {
  422. Web::ResourceLoader::the().clear_cache();
  423. } else {
  424. m_web_content_view->debug_request("clear-cache");
  425. }
  426. }));
  427. m_user_agent_spoof_actions.set_exclusive(true);
  428. auto& spoof_user_agent_menu = debug_menu.add_submenu("Spoof User Agent");
  429. m_disable_user_agent_spoofing = GUI::Action::create_checkable("Disabled", [&](auto&) {
  430. if (m_type == Type::InProcessWebView) {
  431. Web::ResourceLoader::the().set_user_agent(Web::default_user_agent);
  432. } else {
  433. m_web_content_view->debug_request("spoof-user-agent", Web::default_user_agent);
  434. }
  435. });
  436. m_disable_user_agent_spoofing->set_checked(true);
  437. spoof_user_agent_menu.add_action(*m_disable_user_agent_spoofing);
  438. m_user_agent_spoof_actions.add_action(*m_disable_user_agent_spoofing);
  439. auto add_user_agent = [&](auto& name, auto& user_agent) {
  440. auto action = GUI::Action::create_checkable(name, [&](auto&) {
  441. if (m_type == Type::InProcessWebView) {
  442. Web::ResourceLoader::the().set_user_agent(user_agent);
  443. } else {
  444. m_web_content_view->debug_request("spoof-user-agent", user_agent);
  445. }
  446. });
  447. spoof_user_agent_menu.add_action(action);
  448. m_user_agent_spoof_actions.add_action(action);
  449. };
  450. 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");
  451. add_user_agent("Firefox Linux Desktop", "Mozilla/5.0 (X11; Linux i686; rv:87.0) Gecko/20100101 Firefox/87.0");
  452. 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");
  453. 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");
  454. add_user_agent("Firefox Android Mobile", "Mozilla/5.0 (Android 11; Mobile; rv:68.0) Gecko/68.0 Firefox/86.0");
  455. 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");
  456. auto custom_user_agent = GUI::Action::create_checkable("Custom", [&](auto&) {
  457. String user_agent;
  458. if (GUI::InputBox::show(window(), user_agent, "Enter User Agent:", "Custom User Agent") != GUI::InputBox::ExecOK || user_agent.is_empty() || user_agent.is_null()) {
  459. m_disable_user_agent_spoofing->activate();
  460. return;
  461. }
  462. if (m_type == Type::InProcessWebView) {
  463. Web::ResourceLoader::the().set_user_agent(user_agent);
  464. } else {
  465. m_web_content_view->debug_request("spoof-user-agent", user_agent);
  466. }
  467. });
  468. spoof_user_agent_menu.add_action(custom_user_agent);
  469. m_user_agent_spoof_actions.add_action(custom_user_agent);
  470. auto& help_menu = m_menubar->add_menu("&Help");
  471. help_menu.add_action(WindowActions::the().about_action());
  472. m_tab_context_menu = GUI::Menu::construct();
  473. m_tab_context_menu->add_action(GUI::Action::create("&Reload Tab", [this](auto&) {
  474. m_reload_action->activate();
  475. }));
  476. m_tab_context_menu->add_action(GUI::Action::create("&Close Tab", [this](auto&) {
  477. on_tab_close_request(*this);
  478. }));
  479. m_page_context_menu = GUI::Menu::construct();
  480. m_page_context_menu->add_action(*m_go_back_action);
  481. m_page_context_menu->add_action(*m_go_forward_action);
  482. m_page_context_menu->add_action(*m_reload_action);
  483. m_page_context_menu->add_separator();
  484. m_page_context_menu->add_action(*view_source_action);
  485. m_page_context_menu->add_action(*inspect_dom_tree_action);
  486. hooks().on_context_menu_request = [&](auto& screen_position) {
  487. m_page_context_menu->popup(screen_position);
  488. };
  489. }
  490. Tab::~Tab()
  491. {
  492. }
  493. void Tab::load(const URL& url, LoadType load_type)
  494. {
  495. m_is_history_navigation = (load_type == LoadType::HistoryNavigation);
  496. if (m_type == Type::InProcessWebView)
  497. m_page_view->load(url);
  498. else
  499. m_web_content_view->load(url);
  500. }
  501. URL Tab::url() const
  502. {
  503. if (m_type == Type::InProcessWebView)
  504. return m_page_view->url();
  505. return m_web_content_view->url();
  506. }
  507. void Tab::reload()
  508. {
  509. load(url());
  510. }
  511. void Tab::go_back()
  512. {
  513. m_history.go_back();
  514. update_actions();
  515. load(m_history.current(), LoadType::HistoryNavigation);
  516. }
  517. void Tab::go_forward()
  518. {
  519. m_history.go_forward();
  520. update_actions();
  521. load(m_history.current(), LoadType::HistoryNavigation);
  522. }
  523. void Tab::update_actions()
  524. {
  525. m_go_back_action->set_enabled(m_history.can_go_back());
  526. m_go_forward_action->set_enabled(m_history.can_go_forward());
  527. }
  528. void Tab::update_bookmark_button(const String& url)
  529. {
  530. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  531. m_bookmark_button->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/bookmark-filled.png"));
  532. m_bookmark_button->set_tooltip("Remove Bookmark");
  533. } else {
  534. m_bookmark_button->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/bookmark-contour.png"));
  535. m_bookmark_button->set_tooltip("Add Bookmark");
  536. }
  537. }
  538. void Tab::did_become_active()
  539. {
  540. Web::ResourceLoader::the().on_load_counter_change = [this] {
  541. if (Web::ResourceLoader::the().pending_loads() == 0) {
  542. m_statusbar->set_text("");
  543. return;
  544. }
  545. m_statusbar->set_text(String::formatted("Loading ({} pending resources...)", Web::ResourceLoader::the().pending_loads()));
  546. };
  547. BookmarksBarWidget::the().on_bookmark_click = [this](auto& url, unsigned modifiers) {
  548. if (modifiers & Mod_Ctrl)
  549. on_tab_open_request(url);
  550. else
  551. load(url);
  552. };
  553. BookmarksBarWidget::the().on_bookmark_hover = [this](auto&, auto& url) {
  554. m_statusbar->set_text(url);
  555. };
  556. BookmarksBarWidget::the().remove_from_parent();
  557. m_toolbar_container->add_child(BookmarksBarWidget::the());
  558. auto is_fullscreen = window()->is_fullscreen();
  559. m_toolbar_container->set_visible(!is_fullscreen);
  560. m_statusbar->set_visible(!is_fullscreen);
  561. window()->set_menubar(m_menubar);
  562. }
  563. void Tab::context_menu_requested(const Gfx::IntPoint& screen_position)
  564. {
  565. m_tab_context_menu->popup(screen_position);
  566. }
  567. GUI::ScrollableWidget& Tab::view()
  568. {
  569. if (m_type == Type::InProcessWebView)
  570. return *m_page_view;
  571. return *m_web_content_view;
  572. }
  573. Web::WebViewHooks& Tab::hooks()
  574. {
  575. if (m_type == Type::InProcessWebView)
  576. return *m_page_view;
  577. return *m_web_content_view;
  578. }
  579. }