panthera-www.cc
C++ source, ASCII text, with very long lines (382)
1#include "gpanthera.hh" 2#include <gtkmm.h> 3#include <glibmm.h> 4#include <glibmm/ustring.h> 5#include <iostream> 6#include <memory> 7#include <libintl.h> 8#include <locale.h> 9#include <gtk/gtk.h> 10#include <gdk/gdk.h> 11#include <webkit/webkit.h> 12#include <fstream> 13#include <nlohmann/json.hpp> 14#include <iomanip> 15#include <sstream> 16#include <dlfcn.h> 17#include "external/sqlite_orm/sqlite_orm.h" 18//#include "plugin_api.h" 19#define _(STRING) gettext(STRING) 20 21using plugin_entrypoint_type = void(*)(void); 22 23std::string url_encode(const std::string &value) { 24// Thanks https://stackoverflow.com/a/17708801 25std::ostringstream escaped; 26escaped.fill('0'); 27escaped << std::hex; 28 29for(char c : value) { 30// Keep alphanumeric and other accepted characters intact 31if (isalnum(c) || c == '-' || c == '_' || c == '.' || c == '~') { 32escaped << c; 33continue; 34} 35 36// Any other characters are percent-encoded 37escaped << std::uppercase; 38escaped << '%' << std::setw(2) << int((unsigned char) c); 39escaped << std::nouppercase; 40} 41 42return escaped.str(); 43} 44 45bool starts_with(const Glib::ustring &str, const Glib::ustring &prefix) { 46return str.size() >= prefix.size() && str.compare(0, prefix.size(), prefix) == 0; 47} 48 49bool ends_with(const Glib::ustring &str, const Glib::ustring &suffix) { 50return str.size() >= suffix.size() && str.compare(str.size() - suffix.size(), suffix.size(), suffix) == 0; 51} 52 53struct SearchEngine { 54Glib::ustring name; 55Glib::ustring url; 56Glib::ustring get_search_url(const std::string &query) { 57auto pos = url.find("%s"); 58if(pos == Glib::ustring::npos) { 59throw std::runtime_error("Invalid search engine URL: missing '%s' placeholder"); 60} 61auto new_url = url; 62new_url.replace(pos, 2, url_encode(query)); 63return new_url; 64} 65}; 66 67struct HistoryEntry { 68int id; 69std::string url, title; 70int64_t timestamp; 71}; 72 73using StorageType = decltype(sqlite_orm::make_storage("", 74sqlite_orm::make_table("history", 75sqlite_orm::make_column("id", &HistoryEntry::id, sqlite_orm::primary_key()), 76sqlite_orm::make_column("url", &HistoryEntry::url), 77sqlite_orm::make_column("title", &HistoryEntry::title), 78sqlite_orm::make_column("timestamp", &HistoryEntry::timestamp) 79) 80)); 81 82class HistoryManager { 83public: 84static auto make_storage(const std::string& path) { 85return sqlite_orm::make_storage(path, 86sqlite_orm::make_table("history", 87sqlite_orm::make_column("id", &HistoryEntry::id, sqlite_orm::primary_key()), 88sqlite_orm::make_column("url", &HistoryEntry::url), 89sqlite_orm::make_column("title", &HistoryEntry::title), 90sqlite_orm::make_column("timestamp", &HistoryEntry::timestamp) 91) 92); 93} 94StorageType storage; 95explicit HistoryManager(const std::string &db_path) : storage(make_storage(db_path)) { 96storage.sync_schema(); 97} 98 99void log_url(const std::string &url, const std::string title, const int64_t timestamp = Glib::DateTime::create_now_local().to_unix()) { 100storage.insert(HistoryEntry{-1, url, title, timestamp}); 101} 102 103std::vector<HistoryEntry> get_history() { 104return storage.get_all<HistoryEntry>(); 105} 106 107std::vector<HistoryEntry> get_history_between(int64_t start_time, int64_t end_time) { 108using namespace sqlite_orm; 109return storage.get_all<HistoryEntry>( 110where(c(&HistoryEntry::timestamp) >= start_time and c(&HistoryEntry::timestamp) < end_time), 111order_by(&HistoryEntry::timestamp).desc() 112); 113} 114}; 115 116class HistoryViewer : public gPanthera::DockablePane { 117protected: 118std::shared_ptr<HistoryManager> history_manager; 119Gtk::Calendar *calendar; 120Gtk::Label *date_label; 121Gtk::Box *box_place; 122public: 123sigc::signal<void(const std::string &url)> signal_open_url; 124HistoryViewer(std::shared_ptr<gPanthera::LayoutManager> layout_manager, std::shared_ptr<HistoryManager> history_manager) : gPanthera::DockablePane(layout_manager, *Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0), "history", _("History"), Gtk::make_managed<Gtk::Image>(Gio::Icon::create("document-open-recent-symbolic"))), history_manager(std::move(history_manager)) { 125auto history_toolbar = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::HORIZONTAL, 0); 126auto history_button_previous = Gtk::make_managed<Gtk::Button>(); 127history_button_previous->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("go-previous-symbolic"))); 128history_button_previous->set_tooltip_text(_("Previous day")); 129auto history_button_next = Gtk::make_managed<Gtk::Button>(); 130history_button_next->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("go-next-symbolic"))); 131history_button_next->set_tooltip_text(_("Next day")); 132 133date_label = Gtk::make_managed<Gtk::Label>(""); 134auto date_button = Gtk::make_managed<Gtk::MenuButton>(); 135date_button->set_child(*date_label); 136date_button->set_tooltip_text(_("Select date")); 137date_button->set_direction(Gtk::ArrowType::NONE); 138auto date_popover = Gtk::make_managed<Gtk::Popover>(); 139calendar = Gtk::make_managed<Gtk::Calendar>(); 140date_popover->set_child(*calendar); 141date_button->set_popover(*date_popover); 142date_button->set_has_frame(false); 143date_button->set_hexpand(true); 144 145history_toolbar->append(*history_button_previous); 146history_toolbar->append(*date_button); 147history_toolbar->append(*history_button_next); 148auto box = dynamic_cast<Gtk::Box*>(this->child); 149box->append(*history_toolbar); 150 151history_button_next->signal_clicked().connect([this]() { 152Glib::DateTime calendar_day = calendar->get_date(); 153Glib::DateTime datetime = Glib::DateTime::create_local( 154calendar_day.get_year(), 155calendar_day.get_month(), 156calendar_day.get_day_of_month(), 1570, 0, 0 158); 159auto next_entries = this->history_manager->storage.get_all<HistoryEntry>( 160sqlite_orm::where(sqlite_orm::c(&HistoryEntry::timestamp) >= datetime.to_unix() + 86400), 161sqlite_orm::order_by(&HistoryEntry::timestamp).asc() 162); 163if(next_entries.empty()) { 164return; 165} 166auto last_entry = next_entries.front(); 167Glib::DateTime next_day = Glib::DateTime::create_now_local(last_entry.timestamp); 168Glib::DateTime new_date = Glib::DateTime::create_local( 169next_day.get_year(), 170next_day.get_month(), 171next_day.get_day_of_month(), 1720, 0, 0 173); 174calendar->select_day(new_date); 175reload_history(); 176}); 177 178history_button_previous->signal_clicked().connect([this]() { 179Glib::DateTime calendar_day = calendar->get_date(); 180Glib::DateTime datetime = Glib::DateTime::create_local( 181calendar_day.get_year(), 182calendar_day.get_month(), 183calendar_day.get_day_of_month(), 1840, 0, 0 185); 186auto previous_entries = this->history_manager->storage.get_all<HistoryEntry>( 187sqlite_orm::where(sqlite_orm::c(&HistoryEntry::timestamp) < datetime.to_unix()), 188sqlite_orm::order_by(&HistoryEntry::timestamp).desc() 189); 190if(previous_entries.empty()) { 191return; 192} 193auto last_entry = previous_entries.front(); 194Glib::DateTime previous_day = Glib::DateTime::create_now_local(last_entry.timestamp); 195Glib::DateTime new_date = Glib::DateTime::create_local( 196previous_day.get_year(), 197previous_day.get_month(), 198previous_day.get_day_of_month(), 1990, 0, 0 200); 201calendar->select_day(new_date); 202reload_history(); 203}); 204 205calendar->signal_day_selected().connect(sigc::mem_fun(*this, &HistoryViewer::reload_history)); 206calendar->signal_prev_month().connect(sigc::mem_fun(*this, &HistoryViewer::reload_history)); 207calendar->signal_next_month().connect(sigc::mem_fun(*this, &HistoryViewer::reload_history)); 208calendar->signal_prev_year().connect(sigc::mem_fun(*this, &HistoryViewer::reload_history)); 209calendar->signal_next_year().connect(sigc::mem_fun(*this, &HistoryViewer::reload_history)); 210Glib::DateTime datetime = Glib::DateTime::create_now_local(); 211calendar->select_day(datetime); 212 213this->signal_pane_shown.connect([this]() { 214// Load the current date 215reload_history(); 216}); 217 218auto scrolled_window = Gtk::make_managed<Gtk::ScrolledWindow>(); 219box_place = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 220auto viewport = Gtk::make_managed<Gtk::Viewport>(nullptr, nullptr); 221viewport->set_child(*box_place); 222scrolled_window->set_child(*viewport); 223scrolled_window->set_policy(Gtk::PolicyType::AUTOMATIC, Gtk::PolicyType::AUTOMATIC); 224scrolled_window->set_vexpand(true); 225scrolled_window->set_hexpand(true); 226box->append(*scrolled_window); 227 228// TODO: allow deleting entries 229} 230 231void reload_history() { 232Glib::DateTime calendar_day = calendar->get_date(); 233Glib::DateTime datetime = Glib::DateTime::create_local( 234calendar_day.get_year(), 235calendar_day.get_month(), 236calendar_day.get_day_of_month(), 2370, 0, 0 238); 239date_label->set_text(datetime.format("%x")); 240auto box = Gtk::make_managed<Gtk::ListBox>(); 241auto history_entries = history_manager->get_history_between(datetime.to_unix(), datetime.to_unix() + 86400); 242for(const auto &entry : history_entries) { 243auto row = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 4); 244auto title = Gtk::make_managed<Gtk::Label>(entry.title.empty() ? "Untitled" : entry.title); 245title->set_halign(Gtk::Align::START); 246title->set_ellipsize(Pango::EllipsizeMode::MIDDLE); 247auto label = Gtk::make_managed<Gtk::Label>(entry.url); 248label->set_halign(Gtk::Align::START); 249label->set_ellipsize(Pango::EllipsizeMode::MIDDLE); 250row->append(*title); 251row->append(*label); 252if(auto time = Glib::DateTime::create_now_local(entry.timestamp)) { 253label->set_tooltip_text(time.format("%c")); 254box->append(*row); 255} 256} 257if(box_place->get_first_child()) { 258box_place->remove(*box_place->get_first_child()); 259} 260box->signal_row_activated().connect([this](Gtk::ListBoxRow *row) { 261if(auto box_row = dynamic_cast<Gtk::Box*>(row->get_child())) { 262if(auto label = dynamic_cast<Gtk::Label*>(box_row->get_last_child())) { 263auto url = label->get_text(); 264if(!url.empty()) { 265signal_open_url.emit(url); 266} 267} 268} 269}); 270box_place->append(*box); 271} 272}; 273 274class PantheraWindow; 275 276class PantheraWww : public Gtk::Application { 277private: 278std::shared_ptr<gPanthera::ContentManager> content_manager; 279std::string cookie_file; 280std::vector<SearchEngine> search_engines; 281SearchEngine *default_search_engine; 282std::shared_ptr<HistoryManager> history_manager; 283Glib::RefPtr<Gio::Menu> main_menu; 284std::string new_tab_page; 285gPanthera::ContentStack *controlled_stack = nullptr; 286 287static void load_change_callback(WebKitWebView* object, WebKitLoadEvent load_event, gpointer data); 288static void notify_callback(GObject* object, GParamSpec* pspec, gpointer data); 289static void notify_focused_callback(GObject* object, GParamSpec* pspec, gpointer data); 290static void on_back_pressed(GtkGestureClick* gesture, int n_press, double x, double y, gpointer user_data); 291static void on_forward_pressed(GtkGestureClick* gesture, int n_press, double x, double y, gpointer user_data); 292static gboolean on_decide_policy(WebKitWebView* source, WebKitPolicyDecision* decision, WebKitPolicyDecisionType type, gpointer user_data); 293 294protected: 295void on_startup() override; 296void on_activate() override; 297void on_shutdown() override; 298PantheraWindow *make_window(); 299 300public: 301PantheraWww(); 302~PantheraWww(); 303friend class PantheraWindow; 304void on_new_tab(gPanthera::ContentStack* stack, const Glib::ustring& url = "", bool focus = true, bool new_window = false); 305void set_search_engines_from_json(const std::string& json_string); 306static Glib::RefPtr<PantheraWww> create(); 307std::shared_ptr<gPanthera::ContentManager> get_content_manager(); 308void load_plugin(const std::string& plugin_path); 309}; 310 311class PantheraWindow : public Gtk::ApplicationWindow { 312private: 313std::shared_ptr<gPanthera::LayoutManager> layout_manager; 314Gtk::Entry *url_bar; 315HistoryViewer *history_viewer; 316gPanthera::ContentPage *controlled_page = nullptr; 317gPanthera::ContentStack *controlled_stack = nullptr; 318public: 319friend class PantheraWww; 320explicit PantheraWindow(Gtk::Application *application) : Gtk::ApplicationWindow() { 321// There is a constructor with Glib::RefPtr<Gtk::Application>, but it is not appropriate 322// because the window will own the application, creating a cycle 323auto panthera = dynamic_cast<PantheraWww*>(application); 324if(!panthera) { 325throw std::runtime_error("Application is not a PantheraWww instance"); 326} 327panthera->add_window(*this); 328this->set_default_size(800, 600); 329layout_manager = std::make_shared<gPanthera::LayoutManager>(); 330 331auto dock_stack_1 = Gtk::make_managed<gPanthera::DockStack>(layout_manager, _("One"), "one"); 332auto switcher_1 = Gtk::make_managed<gPanthera::DockStackSwitcher>(dock_stack_1, Gtk::Orientation::HORIZONTAL); 333auto dock_stack_2 = Gtk::make_managed<gPanthera::DockStack>(layout_manager, _("Two"), "two"); 334auto switcher_2 = Gtk::make_managed<gPanthera::DockStackSwitcher>(dock_stack_2, Gtk::Orientation::VERTICAL); 335auto pane_1_content = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 336auto debug_button = Gtk::make_managed<Gtk::Button>("Debug"); 337auto print_history_button = Gtk::make_managed<Gtk::Button>("Print history"); 338print_history_button->signal_clicked().connect([this, panthera]() { 339auto history = panthera->history_manager->get_history(); 340for(const auto &entry : history) { 341std::cout << "entry " << entry.id << ", url " << entry.url << ", unix time " << entry.timestamp << std::endl; 342// TODO: exclude redirecting URLs 343} 344}); 345pane_1_content->append(*debug_button); 346pane_1_content->append(*print_history_button); 347auto pane_1_icon = Gtk::make_managed<Gtk::Image>(Gio::Icon::create("help-about-symbolic")); 348auto pane_1 = Gtk::make_managed<gPanthera::DockablePane>(layout_manager, *pane_1_content, "debug", _("Debugging options"), pane_1_icon); 349 350dock_stack_1->set_transition_type(Gtk::StackTransitionType::SLIDE_LEFT_RIGHT); 351dock_stack_1->set_transition_duration(125); 352dock_stack_1->set_expand(true); 353dock_stack_2->set_transition_type(Gtk::StackTransitionType::SLIDE_UP_DOWN); 354dock_stack_2->set_transition_duration(125); 355dock_stack_2->set_expand(true); 356 357auto outer_grid = Gtk::make_managed<Gtk::Grid>(); 358outer_grid->attach(*switcher_2, 0, 2, 1, 1); 359outer_grid->attach(*switcher_1, 1, 3, 1, 1); 360auto outer_paned = Gtk::make_managed<Gtk::Paned>(Gtk::Orientation::HORIZONTAL); 361outer_paned->set_start_child(*dock_stack_2); 362auto inner_paned = Gtk::make_managed<Gtk::Paned>(Gtk::Orientation::VERTICAL); 363auto content = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 364panthera->content_manager = std::make_shared<gPanthera::ContentManager>(); 365std::function<bool(gPanthera::ContentPage*)> detach_handler; 366detach_handler = [](gPanthera::ContentPage *widget) { 367auto new_stack = Gtk::make_managed<gPanthera::ContentStack>(widget->content_manager, widget->get_stack()->get_detach_handler()); 368auto new_switcher = Gtk::make_managed<gPanthera::ContentTabBar>(new_stack, Gtk::Orientation::HORIZONTAL, dynamic_cast<gPanthera::ContentTabBar*>(widget->get_stack()->get_parent()->get_first_child())->get_extra_child_function()); 369auto new_notebook = Gtk::make_managed<gPanthera::ContentNotebook>(new_stack, new_switcher); 370auto window = new gPanthera::ContentWindow(new_notebook); 371widget->redock(new_stack); 372window->present(); 373new_stack->signal_leave_empty.connect([window]() { 374window->close(); 375delete window; 376}); 377return true; 378}; 379 380auto return_extra_child = [this, panthera](gPanthera::ContentTabBar *switcher) { 381auto new_tab_button = Gtk::make_managed<Gtk::Button>(); 382new_tab_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("list-add-symbolic"))); 383new_tab_button->set_tooltip_text(_("New Tab")); 384new_tab_button->signal_clicked().connect([this, switcher, panthera]() { 385panthera->on_new_tab(switcher->get_stack()); 386}); 387return new_tab_button; 388}; 389controlled_stack = Gtk::make_managed<gPanthera::ContentStack>(panthera->content_manager, detach_handler); 390panthera->controlled_stack = controlled_stack; 391auto content_stack_switcher = Gtk::make_managed<gPanthera::ContentTabBar>(controlled_stack, Gtk::Orientation::HORIZONTAL, return_extra_child); 392panthera->content_manager->add_stack(controlled_stack); 393WebKitWebView *webview = WEBKIT_WEB_VIEW(webkit_web_view_new()); // for some reason, this has to be created 394content->set_name("content_box"); 395auto content_notebook = Gtk::make_managed<gPanthera::ContentNotebook>(controlled_stack, content_stack_switcher, Gtk::PositionType::TOP); 396content->append(*content_notebook); 397inner_paned->set_start_child(*content); 398inner_paned->set_end_child(*dock_stack_1); 399outer_paned->set_end_child(*inner_paned); 400outer_grid->attach(*outer_paned, 1, 2, 1, 1); 401// Create the toolbar 402auto main_toolbar = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::HORIZONTAL, 8); 403// URL bar 404url_bar = Gtk::make_managed<Gtk::Entry>(); 405url_bar->set_placeholder_text(_("Enter URL")); 406url_bar->set_hexpand(true); 407auto load_url_callback = [this, panthera]() { 408bool has_protocol = url_bar->get_text().find("://") != std::string::npos; 409if(!has_protocol) { 410url_bar->set_text("http://" + url_bar->get_text()); 411} 412if(controlled_page) { 413if(auto webview = WEBKIT_WEB_VIEW(controlled_page->get_child()->get_first_child()->gobj())) { 414webkit_web_view_load_uri(webview, url_bar->get_text().c_str()); 415} 416} 417}; 418// Go 419auto go_button = Gtk::make_managed<Gtk::Button>(_("Go")); 420go_button->signal_clicked().connect(load_url_callback); 421url_bar->signal_activate().connect(load_url_callback); 422panthera->content_manager->signal_page_operated.connect([this, panthera](gPanthera::ContentPage *page) { 423if(!page->get_child()) { 424return; 425} 426if(!page->get_child()->get_first_child()) { 427return; 428} 429if(page->get_root() != this) { 430// The page is in some other window 431return; 432} 433controlled_page = page; 434controlled_stack = page->get_stack(); 435panthera->controlled_stack = page->get_stack(); 436url_bar->set_text(webkit_web_view_get_uri(WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj()))); 437guint url_update_handler = g_signal_connect(page->get_child()->get_first_child()->gobj(), "notify", G_CALLBACK(panthera->notify_focused_callback), this); 438std::shared_ptr<sigc::connection> control_signal_handler = std::make_shared<sigc::connection>(); 439*control_signal_handler = page->signal_control_status_changed.connect([this, page, control_signal_handler, url_update_handler](bool controlled) { 440if(!controlled) { 441control_signal_handler->disconnect(); 442if(page->get_child() && page->get_child()->get_first_child() && WEBKIT_IS_WEB_VIEW(page->get_child()->get_first_child()->gobj())) { 443g_signal_handler_disconnect(page->get_child()->get_first_child()->gobj(), url_update_handler); 444} 445} 446}); 447}); 448history_viewer = Gtk::make_managed<HistoryViewer>(layout_manager, panthera->history_manager); 449history_viewer->signal_open_url.connect([this, panthera](const std::string &url) { 450panthera->on_new_tab(nullptr, url, true); 451}); 452// Back, forward, reload 453auto back_button = Gtk::make_managed<Gtk::Button>(); 454back_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("go-previous-symbolic"))); 455back_button->set_tooltip_text(_("Back")); 456auto forward_button = Gtk::make_managed<Gtk::Button>(); 457forward_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("go-next-symbolic"))); 458forward_button->set_tooltip_text(_("Forward")); 459auto reload_button = Gtk::make_managed<Gtk::Button>(); 460reload_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("view-refresh-symbolic"))); 461reload_button->set_tooltip_text(_("Reload")); 462 463auto back_action = Gio::SimpleAction::create("go_back"); 464add_action(back_action); 465back_action->signal_activate().connect([this, panthera](const Glib::VariantBase&) { 466if(controlled_page) { 467if(auto webview = WEBKIT_WEB_VIEW(controlled_page->get_child()->get_first_child()->gobj())) { 468webkit_web_view_go_back(webview); 469} 470} 471}); 472back_button->set_action_name("win.go_back"); 473 474auto forward_action = Gio::SimpleAction::create("go_forward"); 475add_action(forward_action); 476forward_action->signal_activate().connect([this, panthera](const Glib::VariantBase&) { 477if(controlled_page) { 478if(auto webview = WEBKIT_WEB_VIEW(controlled_page->get_child()->get_first_child()->gobj())) { 479webkit_web_view_go_forward(webview); 480} 481} 482}); 483forward_button->set_action_name("win.go_forward"); 484 485auto reload_action = Gio::SimpleAction::create("reload"); 486add_action(reload_action); 487reload_action->signal_activate().connect([this, panthera](const Glib::VariantBase&) { 488if(controlled_page) { 489if(auto webview = WEBKIT_WEB_VIEW(controlled_page->get_child()->get_first_child()->gobj())) { 490webkit_web_view_reload(webview); 491} 492} 493}); 494reload_button->set_action_name("win.reload"); 495 496// Search bar 497// TODO: provide history, provide a menu button with the other engines 498auto search_bar = Gtk::make_managed<Gtk::Entry>(); 499search_bar->set_placeholder_text(_("Search")); 500search_bar->set_hexpand(true); 501if(panthera->search_engines.empty()) { 502search_bar->set_sensitive(false); 503search_bar->set_placeholder_text(_("No search")); 504} 505auto search_callback = [this, search_bar, panthera]() { 506// Create a new tab with the search results 507panthera->on_new_tab(controlled_stack); 508auto page = panthera->content_manager->get_last_operated_page(); 509if(page) { 510if(auto webview = WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj())) { 511auto search_url = panthera->default_search_engine->get_search_url(search_bar->get_text()); 512webkit_web_view_load_uri(webview, search_url.c_str()); 513} 514} 515}; 516if(panthera->default_search_engine) { 517search_bar->signal_activate().connect(search_callback); 518} 519 520// Assemble the toolbar 521main_toolbar->append(*back_button); 522main_toolbar->append(*forward_button); 523main_toolbar->append(*reload_button); 524main_toolbar->append(*url_bar); 525main_toolbar->append(*go_button); 526main_toolbar->append(*search_bar); 527if(panthera->default_search_engine) { 528auto search_button = Gtk::make_managed<Gtk::Button>(panthera->default_search_engine->name); 529search_button->signal_clicked().connect(search_callback); 530main_toolbar->append(*search_button); 531} 532outer_grid->attach(*main_toolbar, 0, 1, 2, 1); 533this->set_child(*outer_grid); 534debug_button->signal_clicked().connect([this, panthera]() { 535if(panthera->content_manager->get_last_operated_page()) { 536std::cout << "Last operated page: " << panthera->content_manager->get_last_operated_page()->get_name() << std::endl; 537} else { 538std::cout << "No page operated!" << std::endl; 539} 540}); 541// TODO: Use the last operated page and allow opening tabs next to the last operated page using certain panes 542// Load the existing layout, if it exists 543std::ifstream layout_file_in("layout.json"); 544if(layout_file_in) { 545std::string layout_json((std::istreambuf_iterator<char>(layout_file_in)), std::istreambuf_iterator<char>()); 546layout_file_in.close(); 547layout_manager->restore_json_layout(layout_json); 548} else { 549// Create a new layout if the file doesn't exist 550layout_file_in.close(); 551 552dock_stack_1->add_pane(*pane_1); 553dock_stack_1->add_pane(*history_viewer); 554 555std::ofstream layout_file_out("layout.json"); 556layout_file_out << layout_manager->get_layout_as_json(); 557layout_file_out.close(); 558} 559// Save the layout when changed 560layout_manager->signal_pane_moved.connect([this, panthera](gPanthera::DockablePane *pane) { 561std::ofstream layout_file_out("layout.json"); 562layout_file_out << layout_manager->get_layout_as_json(); 563layout_file_out.close(); 564std::cout << "Layout changed: " << layout_manager->get_layout_as_json() << std::endl; 565}); 566set_show_menubar(true); 567} 568~PantheraWindow() override = default; 569}; 570 571void PantheraWww::load_change_callback(WebKitWebView *object, WebKitLoadEvent load_event, gpointer data) { 572if(auto self = static_cast<PantheraWww*>(data)) { 573if(load_event == WEBKIT_LOAD_COMMITTED) { 574if(webkit_web_view_get_title(WEBKIT_WEB_VIEW(object)) == nullptr) { 575self->history_manager->log_url(webkit_web_view_get_uri(WEBKIT_WEB_VIEW(object)), ""); 576} else { 577self->history_manager->log_url(webkit_web_view_get_uri(WEBKIT_WEB_VIEW(object)), 578webkit_web_view_get_title(WEBKIT_WEB_VIEW(object))); 579} 580// TODO: reload visible history viewers 581} 582} 583} 584 585void PantheraWww::notify_callback(GObject *object, GParamSpec *pspec, gpointer data) { 586if(!gtk_widget_get_parent(GTK_WIDGET(object))) { 587return; 588} 589if(auto self = static_cast<PantheraWww*>(data)) { 590auto parent = gtk_widget_get_parent(gtk_widget_get_parent(GTK_WIDGET(object))); 591if(auto page = dynamic_cast<gPanthera::ContentPage*>(Glib::wrap(parent))) { 592if(g_strcmp0(pspec->name, "title") == 0) { 593if(auto label = dynamic_cast<Gtk::Label*>(page->tab_widget->get_last_child())) { 594if(strlen(webkit_web_view_get_title(WEBKIT_WEB_VIEW(object))) == 0) { 595label->set_label(_("Untitled")); 596} else { 597label->set_label(webkit_web_view_get_title(WEBKIT_WEB_VIEW(object))); 598} 599} 600} else if(g_strcmp0(pspec->name, "favicon") == 0) { 601// Update favicons 602if(auto image = dynamic_cast<Gtk::Image*>(page->tab_widget->get_first_child())) { 603image->set_from_icon_name("image-loading-symbolic"); 604if(auto favicon = webkit_web_view_get_favicon(WEBKIT_WEB_VIEW(object))) { 605gtk_image_set_from_paintable(image->gobj(), GDK_PAINTABLE(favicon)); 606} 607} 608} 609} 610} 611} 612 613void PantheraWww::notify_focused_callback(GObject *object, GParamSpec *pspec, gpointer data) { 614if(!gtk_widget_get_parent(GTK_WIDGET(object))) { 615return; 616} 617auto this_ = static_cast<PantheraWww*>(data); 618auto parent = gtk_widget_get_parent(gtk_widget_get_parent(GTK_WIDGET(object))); 619if(auto page = dynamic_cast<gPanthera::ContentPage*>(Glib::wrap(parent))) { 620if(g_strcmp0(pspec->name, "uri") == 0) { 621if(auto main_window = dynamic_cast<PantheraWindow*>(page->get_root())) { 622main_window->url_bar->set_text(webkit_web_view_get_uri(WEBKIT_WEB_VIEW(object))); 623} 624} 625} 626} 627 628void PantheraWww::on_back_pressed(GtkGestureClick *gesture, int n_press, double x, double y, gpointer user_data) { 629WebKitWebView *webview = WEBKIT_WEB_VIEW(user_data); 630webkit_web_view_go_back(webview); 631} 632 633void PantheraWww::on_forward_pressed(GtkGestureClick *gesture, int n_press, double x, double y, gpointer user_data) { 634WebKitWebView *webview = WEBKIT_WEB_VIEW(user_data); 635webkit_web_view_go_forward(webview); 636} 637 638gboolean PantheraWww::on_decide_policy(WebKitWebView *source, WebKitPolicyDecision *decision, WebKitPolicyDecisionType type, gpointer user_data) { 639if(auto self = static_cast<PantheraWww*>(user_data)) { 640if(type == WEBKIT_POLICY_DECISION_TYPE_NAVIGATION_ACTION) { 641auto action = webkit_navigation_policy_decision_get_navigation_action(WEBKIT_NAVIGATION_POLICY_DECISION(decision)); 642Glib::ustring url = Glib::ustring(webkit_uri_request_get_uri(webkit_navigation_action_get_request(action))); 643if(!starts_with(url, "about:") && url.find("://") == Glib::ustring::npos) { 644// It is a special scheme (mailto:, tel: etc.) 645try { 646Gio::AppInfo::launch_default_for_uri(url); 647} catch(const Glib::Error &exception) { 648std::cerr << "Failed to launch special URI: " << exception.what() << std::endl; 649} 650webkit_policy_decision_ignore(decision); 651return true; 652} 653if(webkit_navigation_action_get_mouse_button(action) == 2) { 654// Middle-click opens in a new window 655self->on_new_tab(nullptr, url, false); 656webkit_policy_decision_ignore(decision); 657return true; 658} 659} else if(type == WEBKIT_POLICY_DECISION_TYPE_NEW_WINDOW_ACTION) { 660auto action = webkit_navigation_policy_decision_get_navigation_action(WEBKIT_NAVIGATION_POLICY_DECISION(decision)); 661self->on_new_tab(nullptr, webkit_uri_request_get_uri(webkit_navigation_action_get_request(action)), false, true); 662return true; 663} 664} 665return false; 666} 667 668void PantheraWww::on_new_tab(gPanthera::ContentStack *stack, const Glib::ustring &url, bool focus, bool new_window) { 669if(!stack) { 670// Find the current area 671stack = controlled_stack; 672} 673Glib::ustring url_ = url; 674if(url.empty()) { 675url_ = new_tab_page; 676} 677 678WebKitWebView *webview = WEBKIT_WEB_VIEW(webkit_web_view_new()); 679gtk_widget_set_hexpand(GTK_WIDGET(webview), true); 680gtk_widget_set_vexpand(GTK_WIDGET(webview), true); 681auto page_content = Gtk::make_managed<Gtk::Box>(); 682gtk_box_append(page_content->gobj(), GTK_WIDGET(webview)); 683auto page_tab = new Gtk::Box(); 684page_tab->set_orientation(Gtk::Orientation::HORIZONTAL); 685auto initial_icon = Gtk::make_managed<Gtk::Image>(Gio::Icon::create("image-loading-symbolic")); 686page_tab->append(*initial_icon); 687page_tab->append(*Gtk::make_managed<Gtk::Label>(_("Untitled"))); 688page_tab->set_spacing(4); 689auto page = Gtk::make_managed<gPanthera::ContentPage>(content_manager, stack, page_content, page_tab); 690g_signal_connect(webview, "notify", G_CALLBACK(notify_callback), this); 691g_signal_connect(webview, "load-changed", G_CALLBACK(load_change_callback), this); 692g_signal_connect(webview, "decide-policy", G_CALLBACK(on_decide_policy), this); 693webkit_web_view_load_uri(webview, url_.data()); 694auto cookie_manager = webkit_network_session_get_cookie_manager(webkit_web_view_get_network_session(webview)); 695webkit_cookie_manager_set_persistent_storage(cookie_manager, cookie_file.c_str(), WEBKIT_COOKIE_PERSISTENT_STORAGE_TEXT); 696webkit_cookie_manager_set_accept_policy(cookie_manager, WEBKIT_COOKIE_POLICY_ACCEPT_ALWAYS); 697auto website_data_manager = webkit_network_session_get_website_data_manager(webkit_web_view_get_network_session(webview)); 698webkit_website_data_manager_set_favicons_enabled(website_data_manager, true); 699GtkEventController *click_controller_back = GTK_EVENT_CONTROLLER(gtk_gesture_click_new()); 700gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(click_controller_back), 8); 701g_signal_connect(click_controller_back, "pressed", G_CALLBACK(on_back_pressed), webview); 702gtk_widget_add_controller(GTK_WIDGET(webview), click_controller_back); 703 704GtkEventController *click_controller_forward = GTK_EVENT_CONTROLLER(gtk_gesture_click_new()); 705gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(click_controller_forward), 9); 706g_signal_connect(click_controller_forward, "pressed", G_CALLBACK(on_forward_pressed), webview); 707gtk_widget_add_controller(GTK_WIDGET(webview), click_controller_forward); 708 709stack->add_page(*page); 710if(focus) { 711stack->set_visible_child(*page); 712content_manager->set_last_operated_page(page); 713} 714if(new_window) { 715bool result = stack->signal_detach.emit(page); 716} 717} 718 719PantheraWindow *PantheraWww::make_window() { 720auto window = Gtk::make_managed<PantheraWindow>(this); 721return window; 722} 723 724void PantheraWww::on_startup() { 725bindtextdomain("panthera-www", "./locales"); 726textdomain("panthera-www"); 727Gtk::Application::on_startup(); 728// Get search engines 729std::ifstream search_engines_file_in("search_engines.json"); 730if(search_engines_file_in) { 731std::string search_engines_json((std::istreambuf_iterator<char>(search_engines_file_in)), std::istreambuf_iterator<char>()); 732search_engines_file_in.close(); 733set_search_engines_from_json(search_engines_json); 734} else { 735search_engines_file_in.close(); 736auto empty_json = nlohmann::json::array(); 737std::ofstream search_engines_file_out("search_engines.json"); 738search_engines_file_out << empty_json.dump(4); 739} 740 741// Load settings 742new_tab_page = "about:blank"; 743std::ifstream settings_file_in("settings.json"); 744if(settings_file_in) { 745std::string settings_json((std::istreambuf_iterator<char>(settings_file_in)), std::istreambuf_iterator<char>()); 746settings_file_in.close(); 747auto json = nlohmann::json::parse(settings_json); 748if(json.contains("ntp")) { 749new_tab_page = json["ntp"].get<std::string>(); 750} 751} else { 752settings_file_in.close(); 753auto empty_json = nlohmann::json::object(); 754empty_json["ntp"] = "about:blank"; 755std::ofstream settings_file_out("settings.json"); 756settings_file_out << empty_json.dump(4); 757settings_file_out.close(); 758} 759 760// Load plugins 761std::ifstream plugin_file_in("plugins.json"); 762if(plugin_file_in) { 763std::string plugin_json((std::istreambuf_iterator<char>(plugin_file_in)), std::istreambuf_iterator<char>()); 764plugin_file_in.close(); 765auto json = nlohmann::json::parse(plugin_json); 766for(auto &plugin : json) { 767auto plugin_path = plugin["path"].get<std::string>(); 768//load_plugin(plugin_path); 769} 770} else { 771plugin_file_in.close(); 772auto empty_json = nlohmann::json::array(); 773std::ofstream plugin_file_out("plugins.json"); 774plugin_file_out << empty_json.dump(4); 775plugin_file_out.close(); 776} 777history_manager = std::make_shared<HistoryManager>("history.db"); 778 779// Known bug <https://gitlab.gnome.org/GNOME/epiphany/-/issues/2714>: 780// JS can't veto the application's accelerators using `preventDefault()`. This is 781// not possible in WebKitGTK, or at least I can't find a way to do it. 782 783main_menu = Gio::Menu::create(); 784// File 785auto file_menu = Gio::Menu::create(); 786// New tab 787auto new_tab_action = Gio::SimpleAction::create("new_tab"); 788new_tab_action->signal_activate().connect([this](const Glib::VariantBase&) { 789on_new_tab(nullptr); 790}); 791add_action(new_tab_action); 792set_accels_for_action("app.new_tab", {"<Primary>T"}); 793file_menu->append(_("New _Tab"), "app.new_tab"); 794// Close tab 795auto close_tab_action = Gio::SimpleAction::create("close_tab"); 796close_tab_action->signal_activate().connect([this](const Glib::VariantBase&) { 797auto page = content_manager->get_last_operated_page(); 798if(page) { 799page->close(); 800} 801}); 802add_action(close_tab_action); 803set_accels_for_action("app.close_tab", {"<Primary>W"}); 804file_menu->append(_("_Close Tab"), "app.close_tab"); 805// New window 806auto new_window_action = Gio::SimpleAction::create("new_window"); 807new_window_action->signal_activate().connect([this](const Glib::VariantBase&) { 808on_activate(); 809}); 810add_action(new_window_action); 811set_accels_for_action("app.new_window", {"<Primary>N"}); 812file_menu->append(_("_New Window"), "app.new_window"); 813// Quit 814auto quit_action = Gio::SimpleAction::create("quit"); 815quit_action->signal_activate().connect([this](const Glib::VariantBase&) { 816quit(); 817}); 818add_action(quit_action); 819set_accels_for_action("app.quit", {"<Primary>Q"}); 820file_menu->append(_("_Quit"), "app.quit"); 821 822main_menu->append_submenu(_("_File"), file_menu); 823 824// View 825auto view_menu = Gio::Menu::create(); 826// Reload 827set_accels_for_action("win.reload", {"F5", "<Primary>R"}); 828view_menu->append(_("_Reload"), "win.reload"); 829 830main_menu->append_submenu(_("_View"), view_menu); 831 832// Go 833auto go_menu = Gio::Menu::create(); 834// Back 835set_accels_for_action("win.go_back", {"<Alt>Left"}); 836go_menu->append(_("_Back"), "win.go_back"); 837// Forward 838set_accels_for_action("win.go_forward", {"<Alt>Right"}); 839go_menu->append(_("_Forward"), "win.go_forward"); 840 841main_menu->append_submenu(_("_Go"), go_menu); 842 843set_menubar(main_menu); 844} 845 846void PantheraWww::on_activate() { 847auto window = make_window(); 848window->present(); 849} 850 851void PantheraWww::on_shutdown() { 852Gtk::Application::on_shutdown(); 853} 854 855void PantheraWww::set_search_engines_from_json(const std::string &json_string) { 856auto json = nlohmann::json::parse(json_string); 857Glib::ustring default_search_engine_name; 858for(auto &engine : json) { 859SearchEngine search_engine; 860search_engine.name = engine["name"].get<std::string>(); 861search_engine.url = engine["url"].get<std::string>(); 862search_engines.push_back(search_engine); 863if(engine.contains("default") && engine["default"].get<bool>()) { 864default_search_engine_name = engine["name"].get<std::string>(); 865} 866} 867for(auto &search_engine : search_engines) { 868if(search_engine.name == default_search_engine_name) { 869default_search_engine = &search_engine; 870break; 871} 872} 873} 874 875PantheraWww::PantheraWww() : Gtk::Application("com.roundabout_host.roundabout.PantheraWww", Gio::Application::Flags::NONE) { 876} 877 878Glib::RefPtr<PantheraWww> PantheraWww::create() { 879return Glib::make_refptr_for_instance<PantheraWww>(new PantheraWww()); 880} 881 882void PantheraWww::load_plugin(const std::string &plugin_path) { 883// TODO: needs to be reworked, to work with multi-window 884void *handle = dlopen(plugin_path.c_str(), RTLD_LAZY | RTLD_GLOBAL); 885if(!handle) { 886std::cerr << "Failed to load plugin: " << dlerror() << std::endl; 887return; 888} 889auto entrypoint = (plugin_entrypoint_type)dlsym(handle, "plugin_init"); 890if(!entrypoint) { 891std::cerr << "Failed to find entrypoint in plugin: " << dlerror() << std::endl; 892dlclose(handle); 893return; 894} 895entrypoint(); 896} 897 898PantheraWww::~PantheraWww() = default; 899 900/* 901extern "C" { 902void panthera_log(const char *message) { 903std::cerr << message << std::endl; 904} 905 906void panthera_add_pane(const char *id, const char *label, GtkImage *icon, GtkWidget *pane) { 907auto pane_child = Glib::wrap(pane); 908auto icon_cc = Glib::wrap(icon); 909auto label_cc = Glib::ustring(label); 910auto id_cc = Glib::ustring(id); 911//auto new_pane = Gtk::make_managed<gPanthera::DockablePane>(app->get_layout_manager(), *pane_child, id_cc, label_cc, icon_cc); 912// TODO: Port to multi-window 913//app->get_layout_manager()->add_pane(new_pane); 914} 915} 916*/ 917 918int main(int argc, char *argv[]) { 919gPanthera::init(); 920 921auto app = PantheraWww::create(); 922 923return app->run(argc, argv); 924} 925