Tab.cpp 18 KB

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