Tab.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. /*
  2. * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021, Maciej Zygmanowski <sppmacd@pm.me>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "Tab.h"
  8. #include "BookmarksBarWidget.h"
  9. #include "Browser.h"
  10. #include "BrowserWindow.h"
  11. #include "ConsoleWidget.h"
  12. #include "DownloadWidget.h"
  13. #include "InspectorWidget.h"
  14. #include <AK/StringBuilder.h>
  15. #include <AK/URL.h>
  16. #include <Applications/Browser/TabGML.h>
  17. #include <LibGUI/Action.h>
  18. #include <LibGUI/Application.h>
  19. #include <LibGUI/BoxLayout.h>
  20. #include <LibGUI/Button.h>
  21. #include <LibGUI/Clipboard.h>
  22. #include <LibGUI/Menu.h>
  23. #include <LibGUI/MessageBox.h>
  24. #include <LibGUI/Statusbar.h>
  25. #include <LibGUI/TextBox.h>
  26. #include <LibGUI/Toolbar.h>
  27. #include <LibGUI/ToolbarContainer.h>
  28. #include <LibGUI/Window.h>
  29. #include <LibJS/Interpreter.h>
  30. #include <LibWeb/HTML/SyntaxHighlighter/SyntaxHighlighter.h>
  31. #include <LibWeb/InProcessWebView.h>
  32. #include <LibWeb/Layout/BlockBox.h>
  33. #include <LibWeb/Layout/InitialContainingBlockBox.h>
  34. #include <LibWeb/Loader/ResourceLoader.h>
  35. #include <LibWeb/OutOfProcessWebView.h>
  36. #include <LibWeb/Page/BrowsingContext.h>
  37. namespace Browser {
  38. URL url_from_user_input(const String& input)
  39. {
  40. if (input.starts_with("?") && !g_search_engine.is_null()) {
  41. auto url = g_search_engine;
  42. url.replace("{}", URL::percent_encode(input.substring_view(1)));
  43. return URL(url);
  44. }
  45. auto url = URL(input);
  46. if (url.is_valid())
  47. return url;
  48. StringBuilder builder;
  49. builder.append("http://");
  50. builder.append(input);
  51. return URL(builder.build());
  52. }
  53. void Tab::start_download(const URL& url)
  54. {
  55. auto window = GUI::Window::construct(&this->window());
  56. window->resize(300, 170);
  57. window->set_title(String::formatted("0% of {}", url.basename()));
  58. window->set_resizable(false);
  59. window->set_main_widget<DownloadWidget>(url);
  60. window->show();
  61. }
  62. void Tab::view_source(const URL& url, const String& source)
  63. {
  64. auto window = GUI::Window::construct(&this->window());
  65. auto& editor = window->set_main_widget<GUI::TextEditor>();
  66. editor.set_text(source);
  67. editor.set_mode(GUI::TextEditor::ReadOnly);
  68. editor.set_syntax_highlighter(make<Web::HTML::SyntaxHighlighter>());
  69. editor.set_ruler_visible(true);
  70. window->resize(640, 480);
  71. window->set_title(url.to_string());
  72. window->set_icon(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/filetype-text.png"));
  73. window->show();
  74. }
  75. void Tab::view_dom_tree(const String& dom_tree)
  76. {
  77. auto window = GUI::Window::construct(&this->window());
  78. window->resize(300, 500);
  79. window->set_title("DOM inspector");
  80. window->set_icon(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/inspector-object.png"));
  81. window->set_main_widget<InspectorWidget>();
  82. auto* inspector_widget = static_cast<InspectorWidget*>(window->main_widget());
  83. inspector_widget->set_dom_json(dom_tree);
  84. window->show();
  85. window->move_to_front();
  86. }
  87. Tab::Tab(BrowserWindow& window, Type type)
  88. : m_type(type)
  89. {
  90. load_from_gml(tab_gml);
  91. m_toolbar_container = *find_descendant_of_type_named<GUI::ToolbarContainer>("toolbar_container");
  92. auto& toolbar = *find_descendant_of_type_named<GUI::Toolbar>("toolbar");
  93. auto& webview_container = *find_descendant_of_type_named<GUI::Widget>("webview_container");
  94. if (m_type == Type::InProcessWebView)
  95. m_page_view = webview_container.add<Web::InProcessWebView>();
  96. else
  97. m_web_content_view = webview_container.add<Web::OutOfProcessWebView>();
  98. auto& go_back_button = toolbar.add_action(window.go_back_action());
  99. go_back_button.on_context_menu_request = [this](auto& context_menu_event) {
  100. if (!m_history.can_go_back())
  101. return;
  102. int i = 0;
  103. m_go_back_context_menu = GUI::Menu::construct();
  104. for (auto& url : m_history.get_back_title_history()) {
  105. i++;
  106. m_go_back_context_menu->add_action(GUI::Action::create(url.to_string(),
  107. Gfx::Bitmap::try_load_from_file("/res/icons/16x16/filetype-html.png"),
  108. [this, i](auto&) { go_back(i); }));
  109. }
  110. m_go_back_context_menu->popup(context_menu_event.screen_position());
  111. };
  112. auto& go_forward_button = toolbar.add_action(window.go_forward_action());
  113. go_forward_button.on_context_menu_request = [this](auto& context_menu_event) {
  114. if (!m_history.can_go_forward())
  115. return;
  116. int i = 0;
  117. m_go_forward_context_menu = GUI::Menu::construct();
  118. for (auto& url : m_history.get_forward_title_history()) {
  119. i++;
  120. m_go_forward_context_menu->add_action(GUI::Action::create(url.to_string(),
  121. Gfx::Bitmap::try_load_from_file("/res/icons/16x16/filetype-html.png"),
  122. [this, i](auto&) { go_forward(i); }));
  123. }
  124. m_go_forward_context_menu->popup(context_menu_event.screen_position());
  125. };
  126. toolbar.add_action(window.go_home_action());
  127. toolbar.add_action(window.reload_action());
  128. m_location_box = toolbar.add<GUI::UrlBox>();
  129. m_location_box->set_placeholder("Address");
  130. m_location_box->on_return_pressed = [this] {
  131. if (m_location_box->text().starts_with('?') && g_search_engine.is_empty()) {
  132. GUI::MessageBox::show(&this->window(), "Select a search engine in the Settings menu before searching.", "No search engine selected", GUI::MessageBox::Type::Information);
  133. return;
  134. }
  135. auto url = url_from_user_input(m_location_box->text());
  136. load(url);
  137. view();
  138. };
  139. m_location_box->add_custom_context_menu_action(GUI::Action::create("Paste && Go", [this](auto&) {
  140. if (!GUI::Clipboard::the().mime_type().starts_with("text/"))
  141. return;
  142. auto const& paste_text = GUI::Clipboard::the().data();
  143. if (paste_text.is_empty())
  144. return;
  145. m_location_box->set_text(paste_text);
  146. m_location_box->on_return_pressed();
  147. }));
  148. m_bookmark_button = toolbar.add<GUI::Button>();
  149. m_bookmark_button->set_button_style(Gfx::ButtonStyle::Coolbar);
  150. m_bookmark_button->set_focus_policy(GUI::FocusPolicy::TabFocus);
  151. m_bookmark_button->set_icon(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/bookmark-contour.png"));
  152. m_bookmark_button->set_fixed_size(22, 22);
  153. m_bookmark_button->on_click = [this](auto) {
  154. bookmark_current_url();
  155. };
  156. auto bookmark_action = GUI::Action::create(
  157. "Bookmark current URL", { Mod_Ctrl, Key_D }, [this](auto&) {
  158. bookmark_current_url();
  159. },
  160. this);
  161. hooks().on_load_start = [this](auto& url) {
  162. m_location_box->set_icon(nullptr);
  163. m_location_box->set_text(url.to_string());
  164. // don't add to history if back or forward is pressed
  165. if (!m_is_history_navigation)
  166. m_history.push(url, title());
  167. m_is_history_navigation = false;
  168. update_actions();
  169. update_bookmark_button(url.to_string());
  170. };
  171. hooks().on_link_click = [this](auto& url, auto& target, unsigned modifiers) {
  172. if (target == "_blank" || modifiers == Mod_Ctrl) {
  173. on_tab_open_request(url);
  174. } else {
  175. load(url);
  176. }
  177. };
  178. m_link_context_menu = GUI::Menu::construct();
  179. auto link_default_action = GUI::Action::create("&Open", [this](auto&) {
  180. hooks().on_link_click(m_link_context_menu_url, "", 0);
  181. });
  182. m_link_context_menu->add_action(link_default_action);
  183. m_link_context_menu_default_action = link_default_action;
  184. m_link_context_menu->add_action(GUI::Action::create("Open in New &Tab", [this](auto&) {
  185. hooks().on_link_click(m_link_context_menu_url, "_blank", 0);
  186. }));
  187. m_link_context_menu->add_separator();
  188. m_link_context_menu->add_action(GUI::Action::create("&Copy URL", [this](auto&) {
  189. GUI::Clipboard::the().set_plain_text(m_link_context_menu_url.to_string());
  190. }));
  191. m_link_context_menu->add_separator();
  192. m_link_context_menu->add_action(GUI::Action::create("&Download", [this](auto&) {
  193. start_download(m_link_context_menu_url);
  194. }));
  195. m_link_context_menu->add_separator();
  196. m_link_context_menu->add_action(window.inspect_dom_node_action());
  197. hooks().on_link_context_menu_request = [this](auto& url, auto& screen_position) {
  198. m_link_context_menu_url = url;
  199. m_link_context_menu->popup(screen_position, m_link_context_menu_default_action);
  200. };
  201. m_image_context_menu = GUI::Menu::construct();
  202. m_image_context_menu->add_action(GUI::Action::create("&Open Image", [this](auto&) {
  203. hooks().on_link_click(m_image_context_menu_url, "", 0);
  204. }));
  205. m_image_context_menu->add_action(GUI::Action::create("Open Image in New &Tab", [this](auto&) {
  206. hooks().on_link_click(m_image_context_menu_url, "_blank", 0);
  207. }));
  208. m_image_context_menu->add_separator();
  209. m_image_context_menu->add_action(GUI::Action::create("&Copy Image", [this](auto&) {
  210. if (m_image_context_menu_bitmap.is_valid())
  211. GUI::Clipboard::the().set_bitmap(*m_image_context_menu_bitmap.bitmap());
  212. }));
  213. m_image_context_menu->add_action(GUI::Action::create("Copy Image &URL", [this](auto&) {
  214. GUI::Clipboard::the().set_plain_text(m_image_context_menu_url.to_string());
  215. }));
  216. m_image_context_menu->add_separator();
  217. m_image_context_menu->add_action(GUI::Action::create("&Download", [this](auto&) {
  218. start_download(m_image_context_menu_url);
  219. }));
  220. m_image_context_menu->add_separator();
  221. m_image_context_menu->add_action(window.inspect_dom_node_action());
  222. hooks().on_image_context_menu_request = [this](auto& image_url, auto& screen_position, const Gfx::ShareableBitmap& shareable_bitmap) {
  223. m_image_context_menu_url = image_url;
  224. m_image_context_menu_bitmap = shareable_bitmap;
  225. m_image_context_menu->popup(screen_position);
  226. };
  227. hooks().on_link_middle_click = [this](auto& href, auto&, auto) {
  228. hooks().on_link_click(href, "_blank", 0);
  229. };
  230. hooks().on_title_change = [this](auto& title) {
  231. if (title.is_null()) {
  232. m_history.update_title(url().to_string());
  233. m_title = url().to_string();
  234. } else {
  235. m_history.update_title(title);
  236. m_title = title;
  237. }
  238. if (on_title_change)
  239. on_title_change(m_title);
  240. };
  241. hooks().on_favicon_change = [this](auto& icon) {
  242. m_icon = icon;
  243. m_location_box->set_icon(&icon);
  244. if (on_favicon_change)
  245. on_favicon_change(icon);
  246. };
  247. hooks().on_get_cookie = [this](auto& url, auto source) -> String {
  248. if (on_get_cookie)
  249. return on_get_cookie(url, source);
  250. return {};
  251. };
  252. hooks().on_set_cookie = [this](auto& url, auto& cookie, auto source) {
  253. if (on_set_cookie)
  254. on_set_cookie(url, cookie, source);
  255. };
  256. hooks().on_get_source = [this](auto& url, auto& source) {
  257. view_source(url, source);
  258. };
  259. hooks().on_get_dom_tree = [this](auto& dom_tree) {
  260. view_dom_tree(dom_tree);
  261. };
  262. hooks().on_js_console_output = [this](auto& method, auto& line) {
  263. if (m_console_window) {
  264. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  265. console_widget->handle_js_console_output(method, line);
  266. }
  267. };
  268. if (m_type == Type::InProcessWebView) {
  269. hooks().on_set_document = [this](auto* document) {
  270. if (document && m_console_window) {
  271. auto* console_widget = static_cast<ConsoleWidget*>(m_console_window->main_widget());
  272. console_widget->set_interpreter(document->interpreter().make_weak_ptr());
  273. }
  274. };
  275. }
  276. auto focus_location_box_action = GUI::Action::create(
  277. "Focus location box", { Mod_Ctrl, Key_L }, Key_F6, [this](auto&) {
  278. m_location_box->set_focus(true);
  279. m_location_box->select_current_line();
  280. },
  281. this);
  282. m_statusbar = *find_descendant_of_type_named<GUI::Statusbar>("statusbar");
  283. hooks().on_link_hover = [this](auto& url) {
  284. if (url.is_valid())
  285. m_statusbar->set_text(url.to_string());
  286. else
  287. m_statusbar->set_text("");
  288. };
  289. hooks().on_url_drop = [this](auto& url) {
  290. load(url);
  291. };
  292. m_tab_context_menu = GUI::Menu::construct();
  293. m_tab_context_menu->add_action(GUI::Action::create("&Reload Tab", [this](auto&) {
  294. this->window().reload_action().activate();
  295. }));
  296. m_tab_context_menu->add_action(GUI::Action::create("&Close Tab", [this](auto&) {
  297. on_tab_close_request(*this);
  298. }));
  299. m_tab_context_menu->add_action(GUI::Action::create("&Duplicate Tab", [this](auto&) {
  300. on_tab_open_request(url());
  301. }));
  302. m_tab_context_menu->add_action(GUI::Action::create("Close &Other Tabs", [this](auto&) {
  303. on_tab_close_other_request(*this);
  304. }));
  305. m_page_context_menu = GUI::Menu::construct();
  306. m_page_context_menu->add_action(window.go_back_action());
  307. m_page_context_menu->add_action(window.go_forward_action());
  308. m_page_context_menu->add_action(window.reload_action());
  309. m_page_context_menu->add_separator();
  310. m_page_context_menu->add_action(window.copy_selection_action());
  311. m_page_context_menu->add_action(window.select_all_action());
  312. m_page_context_menu->add_separator();
  313. m_page_context_menu->add_action(window.view_source_action());
  314. m_page_context_menu->add_action(window.inspect_dom_tree_action());
  315. m_page_context_menu->add_action(window.inspect_dom_node_action());
  316. hooks().on_context_menu_request = [&](auto& screen_position) {
  317. m_page_context_menu->popup(screen_position);
  318. };
  319. // FIXME: This is temporary, until the OOPWV properly supports the DOM Inspector
  320. window.inspect_dom_node_action().set_enabled(type == Type::InProcessWebView);
  321. }
  322. Tab::~Tab()
  323. {
  324. }
  325. void Tab::load(const URL& url, LoadType load_type)
  326. {
  327. m_is_history_navigation = (load_type == LoadType::HistoryNavigation);
  328. if (m_type == Type::InProcessWebView)
  329. m_page_view->load(url);
  330. else
  331. m_web_content_view->load(url);
  332. m_location_box->set_focus(false);
  333. }
  334. URL Tab::url() const
  335. {
  336. if (m_type == Type::InProcessWebView)
  337. return m_page_view->url();
  338. return m_web_content_view->url();
  339. }
  340. void Tab::reload()
  341. {
  342. load(url());
  343. }
  344. void Tab::go_back(int steps)
  345. {
  346. m_history.go_back(steps);
  347. update_actions();
  348. load(m_history.current().url, LoadType::HistoryNavigation);
  349. }
  350. void Tab::go_forward(int steps)
  351. {
  352. m_history.go_forward(steps);
  353. update_actions();
  354. load(m_history.current().url, LoadType::HistoryNavigation);
  355. }
  356. void Tab::update_actions()
  357. {
  358. auto& window = this->window();
  359. if (this != &window.active_tab())
  360. return;
  361. window.go_back_action().set_enabled(m_history.can_go_back());
  362. window.go_forward_action().set_enabled(m_history.can_go_forward());
  363. }
  364. void Tab::bookmark_current_url()
  365. {
  366. auto url = this->url().to_string();
  367. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  368. BookmarksBarWidget::the().remove_bookmark(url);
  369. } else {
  370. BookmarksBarWidget::the().add_bookmark(url, m_title);
  371. }
  372. update_bookmark_button(url);
  373. }
  374. void Tab::update_bookmark_button(const String& url)
  375. {
  376. if (BookmarksBarWidget::the().contains_bookmark(url)) {
  377. m_bookmark_button->set_icon(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/bookmark-filled.png"));
  378. m_bookmark_button->set_tooltip("Remove Bookmark");
  379. } else {
  380. m_bookmark_button->set_icon(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/bookmark-contour.png"));
  381. m_bookmark_button->set_tooltip("Add Bookmark");
  382. }
  383. }
  384. void Tab::did_become_active()
  385. {
  386. if (m_type == Type::InProcessWebView) {
  387. Web::ResourceLoader::the().on_load_counter_change = [this] {
  388. if (Web::ResourceLoader::the().pending_loads() == 0) {
  389. m_statusbar->set_text("");
  390. return;
  391. }
  392. m_statusbar->set_text(String::formatted("Loading ({} pending resources...)", Web::ResourceLoader::the().pending_loads()));
  393. };
  394. }
  395. BookmarksBarWidget::the().on_bookmark_click = [this](auto& url, unsigned modifiers) {
  396. if (modifiers & Mod_Ctrl)
  397. on_tab_open_request(url);
  398. else
  399. load(url);
  400. };
  401. BookmarksBarWidget::the().on_bookmark_hover = [this](auto&, auto& url) {
  402. m_statusbar->set_text(url);
  403. };
  404. BookmarksBarWidget::the().remove_from_parent();
  405. m_toolbar_container->add_child(BookmarksBarWidget::the());
  406. auto is_fullscreen = window().is_fullscreen();
  407. m_toolbar_container->set_visible(!is_fullscreen);
  408. m_statusbar->set_visible(!is_fullscreen);
  409. update_actions();
  410. }
  411. void Tab::context_menu_requested(const Gfx::IntPoint& screen_position)
  412. {
  413. m_tab_context_menu->popup(screen_position);
  414. }
  415. GUI::AbstractScrollableWidget& Tab::view()
  416. {
  417. if (m_type == Type::InProcessWebView)
  418. return *m_page_view;
  419. return *m_web_content_view;
  420. }
  421. Web::WebViewHooks& Tab::hooks()
  422. {
  423. if (m_type == Type::InProcessWebView)
  424. return *m_page_view;
  425. return *m_web_content_view;
  426. }
  427. void Tab::action_entered(GUI::Action& action)
  428. {
  429. m_statusbar->set_override_text(action.status_tip());
  430. }
  431. void Tab::action_left(GUI::Action&)
  432. {
  433. m_statusbar->set_override_text({});
  434. }
  435. BrowserWindow const& Tab::window() const
  436. {
  437. return static_cast<BrowserWindow const&>(*Widget::window());
  438. }
  439. BrowserWindow& Tab::window()
  440. {
  441. return static_cast<BrowserWindow&>(*Widget::window());
  442. }
  443. void Tab::show_inspector_window(Browser::Tab::InspectorTarget target)
  444. {
  445. if (m_type == Tab::Type::InProcessWebView) {
  446. if (!m_dom_inspector_window) {
  447. m_dom_inspector_window = GUI::Window::construct(this);
  448. m_dom_inspector_window->resize(300, 500);
  449. m_dom_inspector_window->set_title("DOM inspector");
  450. m_dom_inspector_window->set_icon(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/inspector-object.png"));
  451. m_dom_inspector_window->set_main_widget<InspectorWidget>();
  452. m_dom_inspector_window->on_close = [&]() {
  453. m_page_view->document()->set_inspected_node(nullptr);
  454. };
  455. }
  456. auto* inspector_widget = static_cast<InspectorWidget*>(m_dom_inspector_window->main_widget());
  457. inspector_widget->set_document(m_page_view->document());
  458. switch (target) {
  459. case InspectorTarget::Document:
  460. inspector_widget->set_inspected_node(nullptr);
  461. break;
  462. case InspectorTarget::HoveredElement:
  463. inspector_widget->set_inspected_node(m_page_view->document()->hovered_node());
  464. break;
  465. }
  466. m_dom_inspector_window->show();
  467. m_dom_inspector_window->move_to_front();
  468. } else {
  469. m_web_content_view->inspect_dom_tree();
  470. }
  471. }
  472. }