/* * Copyright (c) 2018-2020, Andreas Kling * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "BookmarksBarWidget.h" #include "History.h" #include "InspectorWidget.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static const char* home_url = "file:///home/anon/www/welcome.html"; static const char* bookmarks_filename = "/home/anon/bookmarks.json"; static String s_title = ""; int main(int argc, char** argv) { if (getuid() == 0) { fprintf(stderr, "Refusing to run as root\n"); return 1; } if (pledge("stdio shared_buffer accept unix cpath rpath wpath fattr", nullptr) < 0) { perror("pledge"); return 1; } GUI::Application app(argc, argv); // Connect to the ProtocolServer immediately so we can drop the "unix" pledge. Web::ResourceLoader::the(); if (pledge("stdio shared_buffer accept cpath rpath wpath", nullptr) < 0) { perror("pledge"); return 1; } if (unveil("/home", "rwc") < 0) { perror("unveil"); return 1; } if (unveil("/res", "r") < 0) { perror("unveil"); return 1; } unveil(nullptr, nullptr); auto window = GUI::Window::construct(); window->set_rect(100, 100, 640, 480); auto& widget = window->set_main_widget(); widget.set_fill_with_background_color(true); widget.set_layout(); widget.layout()->set_spacing(0); bool bookmarksbar_enabled = true; auto& toolbar = widget.add(); auto& bookmarksbar = widget.add(bookmarks_filename, bookmarksbar_enabled); auto& html_widget = widget.add(); bookmarksbar.on_bookmark_click = [&](auto&, auto& url) { html_widget.load(url); }; History history; RefPtr go_back_action; RefPtr go_forward_action; auto update_actions = [&]() { go_back_action->set_enabled(history.can_go_back()); go_forward_action->set_enabled(history.can_go_forward()); }; bool should_push_loads_to_history = true; go_back_action = GUI::CommonActions::make_go_back_action([&](auto&) { history.go_back(); update_actions(); TemporaryChange change(should_push_loads_to_history, false); html_widget.load(history.current()); }); go_forward_action = GUI::CommonActions::make_go_forward_action([&](auto&) { history.go_forward(); update_actions(); TemporaryChange change(should_push_loads_to_history, false); html_widget.load(history.current()); }); toolbar.add_action(*go_back_action); toolbar.add_action(*go_forward_action); toolbar.add_action(GUI::CommonActions::make_go_home_action([&](auto&) { html_widget.load(home_url); })); toolbar.add_action(GUI::CommonActions::make_reload_action([&](auto&) { TemporaryChange change(should_push_loads_to_history, false); html_widget.reload(); })); auto& location_box = toolbar.add(); location_box.on_return_pressed = [&] { html_widget.load(location_box.text()); }; auto& bookmark_button = toolbar.add(); bookmark_button.set_button_style(Gfx::ButtonStyle::CoolBar); bookmark_button.set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/star-black.png")); bookmark_button.set_size_policy(GUI::SizePolicy::Fixed, GUI::SizePolicy::Fixed); bookmark_button.set_preferred_size(22, 22); auto update_bookmark_button = [&](const String& url) { if (bookmarksbar.contains_bookmark(url)) { bookmark_button.set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/star-yellow.png")); } else { bookmark_button.set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/star-contour.png")); } }; bookmark_button.on_click = [&] { auto url = html_widget.main_frame().document()->url().to_string(); if (bookmarksbar.contains_bookmark(url)) { bookmarksbar.remove_bookmark(url); } else { bookmarksbar.add_bookmark(url, s_title); } update_bookmark_button(url); }; html_widget.on_load_start = [&](auto& url) { location_box.set_text(url.to_string()); if (should_push_loads_to_history) history.push(url); update_actions(); update_bookmark_button(url.to_string()); }; html_widget.on_link_click = [&](auto& url) { if (url.starts_with("#")) { html_widget.scroll_to_anchor(url.substring_view(1, url.length() - 1)); } else { html_widget.load(html_widget.document()->complete_url(url)); } }; html_widget.on_title_change = [&](auto& title) { s_title = title; window->set_title(String::format("%s - Browser", title.characters())); }; auto focus_location_box_action = GUI::Action::create("Focus location box", { Mod_Ctrl, Key_L }, [&](auto&) { location_box.select_all(); location_box.set_focus(true); }); auto& statusbar = widget.add(); html_widget.on_link_hover = [&](auto& href) { statusbar.set_text(href); }; bookmarksbar.on_bookmark_hover = [&](auto&, auto& url) { statusbar.set_text(url); }; Web::ResourceLoader::the().on_load_counter_change = [&] { if (Web::ResourceLoader::the().pending_loads() == 0) { statusbar.set_text(""); return; } statusbar.set_text(String::format("Loading (%d pending resources...)", Web::ResourceLoader::the().pending_loads())); }; auto menubar = make(); auto& app_menu = menubar->add_menu("Browser"); app_menu.add_action(GUI::Action::create("Reload", { Mod_None, Key_F5 }, Gfx::Bitmap::load_from_file("/res/icons/16x16/reload.png"), [&](auto&) { TemporaryChange change(should_push_loads_to_history, false); html_widget.reload(); })); app_menu.add_separator(); app_menu.add_action(GUI::CommonActions::make_quit_action([&](auto&) { app.quit(); })); RefPtr dom_inspector_window; auto& inspect_menu = menubar->add_menu("Inspect"); inspect_menu.add_action(GUI::Action::create("View source", { Mod_Ctrl, Key_U }, [&](auto&) { String filename_to_open; char tmp_filename[] = "/tmp/view-source.XXXXXX"; ASSERT(html_widget.document()); if (html_widget.document()->url().protocol() == "file") { filename_to_open = html_widget.document()->url().path(); } else { int fd = mkstemp(tmp_filename); ASSERT(fd >= 0); auto source = html_widget.document()->source(); write(fd, source.characters(), source.length()); close(fd); filename_to_open = tmp_filename; } if (fork() == 0) { execl("/bin/TextEditor", "TextEditor", filename_to_open.characters(), nullptr); ASSERT_NOT_REACHED(); } })); inspect_menu.add_action(GUI::Action::create("Inspect DOM tree", { Mod_None, Key_F12 }, [&](auto&) { if (!dom_inspector_window) { dom_inspector_window = GUI::Window::construct(); dom_inspector_window->set_rect(100, 100, 300, 500); dom_inspector_window->set_title("DOM inspector"); dom_inspector_window->set_main_widget(); } auto* inspector_widget = static_cast(dom_inspector_window->main_widget()); inspector_widget->set_document(html_widget.document()); dom_inspector_window->show(); dom_inspector_window->move_to_front(); })); auto& debug_menu = menubar->add_menu("Debug"); debug_menu.add_action(GUI::Action::create("Dump DOM tree", [&](auto&) { dump_tree(*html_widget.document()); })); debug_menu.add_action(GUI::Action::create("Dump Layout tree", [&](auto&) { dump_tree(*html_widget.document()->layout_node()); })); debug_menu.add_action(GUI::Action::create("Dump Style sheets", [&](auto&) { for (auto& sheet : html_widget.document()->stylesheets()) { dump_sheet(sheet); } })); debug_menu.add_separator(); auto line_box_borders_action = GUI::Action::create("Line box borders", [&](auto& action) { action.set_checked(!action.is_checked()); html_widget.set_should_show_line_box_borders(action.is_checked()); html_widget.update(); }); line_box_borders_action->set_checkable(true); line_box_borders_action->set_checked(false); debug_menu.add_action(line_box_borders_action); auto& bookmarks_menu = menubar->add_menu("Bookmarks"); auto show_bookmarksbar_action = GUI::Action::create("Show bookmarks bar", [&](auto& action) { action.set_checked(!action.is_checked()); bookmarksbar.set_visible(action.is_checked()); bookmarksbar.update(); }); show_bookmarksbar_action->set_checkable(true); show_bookmarksbar_action->set_checked(bookmarksbar_enabled); bookmarks_menu.add_action(show_bookmarksbar_action); auto& help_menu = menubar->add_menu("Help"); help_menu.add_action(GUI::Action::create("About", [&](const GUI::Action&) { GUI::AboutDialog::show("Browser", Gfx::Bitmap::load_from_file("/res/icons/32x32/filetype-html.png"), window); })); app.set_menubar(move(menubar)); window->set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/filetype-html.png")); window->set_title("Browser"); window->show(); URL url_to_load = home_url; if (app.args().size() >= 1) { url_to_load = URL(); url_to_load.set_protocol("file"); url_to_load.set_path(app.args()[0]); } html_widget.load(url_to_load); return app.exec(); }