panthera-www.cc
C++ source, ASCII text
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 14class WebPage : public Gtk::Box { 15 16}; 17 18class PantheraWww : public Gtk::Application { 19Gtk::Window *window = Gtk::make_managed<Gtk::Window>(); 20protected: 21std::shared_ptr<gPanthera::LayoutManager> layout_manager; 22std::shared_ptr<gPanthera::ContentManager> content_manager; 23Gtk::Entry *url_bar = nullptr; 24std::string cookie_file = "cookies.txt"; 25 26static void notify_callback(GObject *object, GParamSpec *pspec, gpointer data) { 27if(!gtk_widget_get_parent(GTK_WIDGET(object))) { 28return; 29} 30auto parent = gtk_widget_get_parent(gtk_widget_get_parent(GTK_WIDGET(object))); 31if(auto page = dynamic_cast<gPanthera::ContentPage*>(Glib::wrap(parent))) { 32if(g_strcmp0(pspec->name, "title") == 0) { 33if(auto label = dynamic_cast<Gtk::Label*>(page->tab_widget)) { 34label->set_label(webkit_web_view_get_title(WEBKIT_WEB_VIEW(object))); 35} 36} 37} 38} 39 40static void notify_focused_callback(GObject *object, GParamSpec *pspec, gpointer data) { 41if(!gtk_widget_get_parent(GTK_WIDGET(object))) { 42return; 43} 44auto this_ = static_cast<PantheraWww*>(data); 45auto parent = gtk_widget_get_parent(gtk_widget_get_parent(GTK_WIDGET(object))); 46if(auto page = dynamic_cast<gPanthera::ContentPage*>(Glib::wrap(parent))) { 47if(g_strcmp0(pspec->name, "uri") == 0) { 48this_->url_bar->set_text(webkit_web_view_get_uri(WEBKIT_WEB_VIEW(object))); 49} 50} 51} 52 53static void on_back_pressed(GtkGestureClick *gesture, int n_press, double x, double y, gpointer user_data) { 54WebKitWebView *webview = WEBKIT_WEB_VIEW(user_data); 55webkit_web_view_go_back(webview); 56} 57 58static void on_forward_pressed(GtkGestureClick *gesture, int n_press, double x, double y, gpointer user_data) { 59WebKitWebView *webview = WEBKIT_WEB_VIEW(user_data); 60webkit_web_view_go_forward(webview); 61} 62 63void on_new_tab(gPanthera::ContentStack *stack) { 64if(!stack) { 65// Find the current area 66stack = content_manager->get_last_operated_page()->get_stack(); 67} 68 69WebKitWebView *webview = WEBKIT_WEB_VIEW(webkit_web_view_new()); 70gtk_widget_set_hexpand(GTK_WIDGET(webview), true); 71gtk_widget_set_vexpand(GTK_WIDGET(webview), true); 72auto page_content = Gtk::make_managed<Gtk::Box>(); 73gtk_box_append(page_content->gobj(), GTK_WIDGET(webview)); 74auto page_tab = new Gtk::Label("Untitled"); 75auto page = Gtk::make_managed<gPanthera::ContentPage>(content_manager, stack, page_content, page_tab); 76g_signal_connect(webview, "notify", G_CALLBACK(notify_callback), page->gobj()); 77webkit_web_view_load_uri(webview, "about:blank"); 78auto cookie_manager = webkit_network_session_get_cookie_manager(webkit_web_view_get_network_session(webview)); 79webkit_cookie_manager_set_persistent_storage(cookie_manager, cookie_file.c_str(), WEBKIT_COOKIE_PERSISTENT_STORAGE_TEXT); 80webkit_cookie_manager_set_accept_policy(cookie_manager, WEBKIT_COOKIE_POLICY_ACCEPT_ALWAYS); 81GtkEventController *click_controller_back = GTK_EVENT_CONTROLLER(gtk_gesture_click_new()); 82gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(click_controller_back), 8); 83g_signal_connect(click_controller_back, "pressed", G_CALLBACK(on_back_pressed), webview); 84gtk_widget_add_controller(GTK_WIDGET(webview), click_controller_back); 85 86GtkEventController *click_controller_forward = GTK_EVENT_CONTROLLER(gtk_gesture_click_new()); 87gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(click_controller_forward), 9); 88g_signal_connect(click_controller_forward, "pressed", G_CALLBACK(on_forward_pressed), webview); 89gtk_widget_add_controller(GTK_WIDGET(webview), click_controller_forward); 90 91stack->add_page(*page); 92stack->set_visible_child(*page); 93content_manager->set_last_operated_page(page); 94} 95 96void on_startup() override { 97Gtk::Application::on_startup(); 98add_window(*window); 99window->set_default_size(600, 400); 100layout_manager = std::make_shared<gPanthera::LayoutManager>(); 101auto dock_stack_1 = Gtk::make_managed<gPanthera::DockStack>(layout_manager, "One", "one"); 102auto switcher_1 = Gtk::make_managed<gPanthera::DockStackSwitcher>(dock_stack_1, Gtk::Orientation::HORIZONTAL); 103auto dock_stack_2 = Gtk::make_managed<gPanthera::DockStack>(layout_manager, "Two", "two"); 104auto switcher_2 = Gtk::make_managed<gPanthera::DockStackSwitcher>(dock_stack_2, Gtk::Orientation::VERTICAL); 105auto pane_1_content = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 106auto debug_button = Gtk::make_managed<Gtk::Button>("Debug"); 107pane_1_content->append(*debug_button); 108auto pane_2_content = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 109auto test_notebook = Gtk::make_managed<Gtk::Notebook>(); 110test_notebook->append_page(*Gtk::make_managed<Gtk::Label>("Test 1"), *Gtk::make_managed<Gtk::Label>("Test 1")); 111pane_2_content->append(*test_notebook); 112auto pane_3_content = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 113pane_3_content->append(*Gtk::make_managed<Gtk::Label>("Pane 3 content")); 114auto pane_1_icon = Gtk::make_managed<Gtk::Image>(); 115pane_1_icon->set_from_icon_name("go-home-symbolic"); 116auto pane_2_icon = Gtk::make_managed<Gtk::Image>(); 117pane_2_icon->set_from_icon_name("folder-symbolic"); 118auto pane_3_icon = Gtk::make_managed<Gtk::Image>(); 119pane_3_icon->set_from_icon_name("network-transmit-receive-symbolic"); 120auto pane_1 = Gtk::make_managed<gPanthera::DockablePane>(layout_manager, *pane_1_content, "pane1", "Pane 1", pane_1_icon); 121auto pane_2 = Gtk::make_managed<gPanthera::DockablePane>(layout_manager, *pane_2_content, "pane2", "Pane 2", pane_2_icon); 122auto pane_3 = Gtk::make_managed<gPanthera::DockablePane>(layout_manager, *pane_3_content, "pane3", "Pane 3", pane_3_icon); 123 124dock_stack_1->set_transition_type(Gtk::StackTransitionType::SLIDE_LEFT_RIGHT); 125dock_stack_1->set_transition_duration(125); 126dock_stack_1->set_expand(true); 127dock_stack_2->set_transition_type(Gtk::StackTransitionType::SLIDE_UP_DOWN); 128dock_stack_2->set_transition_duration(125); 129dock_stack_2->set_expand(true); 130 131auto outer_grid = Gtk::make_managed<Gtk::Grid>(); 132outer_grid->attach(*switcher_2, 0, 1, 1, 1); 133outer_grid->attach(*switcher_1, 1, 2, 1, 1); 134auto outer_paned = Gtk::make_managed<Gtk::Paned>(Gtk::Orientation::HORIZONTAL); 135outer_paned->set_start_child(*dock_stack_2); 136auto inner_paned = Gtk::make_managed<Gtk::Paned>(Gtk::Orientation::VERTICAL); 137auto content = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::VERTICAL, 0); 138content_manager = std::make_shared<gPanthera::ContentManager>(); 139std::function<bool(gPanthera::ContentPage*)> detach_handler; 140detach_handler = [](gPanthera::ContentPage *widget) { 141auto new_stack = Gtk::make_managed<gPanthera::ContentStack>(widget->content_manager, widget->get_stack()->get_detach_handler()); 142auto 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()); 143auto new_notebook = Gtk::make_managed<gPanthera::ContentNotebook>(new_stack, new_switcher); 144auto window = new gPanthera::ContentWindow(new_notebook); 145widget->redock(new_stack); 146window->present(); 147new_stack->signal_leave_empty.connect([window]() { 148window->close(); 149delete window; 150}); 151return true; 152}; 153 154auto return_extra_child = [this](gPanthera::ContentTabBar *switcher) { 155auto new_tab_button = Gtk::make_managed<Gtk::Button>(); 156new_tab_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("list-add-symbolic"))); 157new_tab_button->set_tooltip_text("New tab"); 158new_tab_button->signal_clicked().connect([this, switcher]() { 159on_new_tab(switcher->get_stack()); 160}); 161return new_tab_button; 162}; 163auto content_stack = Gtk::make_managed<gPanthera::ContentStack>(content_manager, detach_handler); 164auto content_stack_switcher = Gtk::make_managed<gPanthera::ContentTabBar>(content_stack, Gtk::Orientation::HORIZONTAL, return_extra_child); 165content_manager->add_stack(content_stack); 166WebKitWebView *webview = WEBKIT_WEB_VIEW(webkit_web_view_new()); // for some reason, this has to be created 167content->set_name("content_box"); 168auto content_notebook = Gtk::make_managed<gPanthera::ContentNotebook>(content_stack, content_stack_switcher, Gtk::PositionType::TOP); 169content->append(*content_notebook); 170inner_paned->set_start_child(*content); 171inner_paned->set_end_child(*dock_stack_1); 172outer_paned->set_end_child(*inner_paned); 173outer_grid->attach(*outer_paned, 1, 1, 1, 1); 174auto main_toolbar = Gtk::make_managed<Gtk::Box>(Gtk::Orientation::HORIZONTAL, 8); 175url_bar = Gtk::make_managed<Gtk::Entry>(); 176url_bar->set_placeholder_text("Enter URL"); 177url_bar->set_hexpand(true); 178auto go_button = Gtk::make_managed<Gtk::Button>("Go"); 179auto back_button = Gtk::make_managed<Gtk::Button>(); 180back_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("go-previous-symbolic"))); 181back_button->set_tooltip_text("Back"); 182auto forward_button = Gtk::make_managed<Gtk::Button>(); 183forward_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("go-next-symbolic"))); 184forward_button->set_tooltip_text("Forward"); 185auto reload_button = Gtk::make_managed<Gtk::Button>(); 186reload_button->set_child(*Gtk::make_managed<Gtk::Image>(Gio::Icon::create("view-refresh-symbolic"))); 187reload_button->set_tooltip_text("Reload"); 188back_button->signal_clicked().connect([this]() { 189auto page = content_manager->get_last_operated_page(); 190if(page) { 191if(auto webview = WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj())) { 192webkit_web_view_go_back(webview); 193} 194} 195}); 196forward_button->signal_clicked().connect([this]() { 197auto page = content_manager->get_last_operated_page(); 198if(page) { 199if(auto webview = WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj())) { 200webkit_web_view_go_forward(webview); 201} 202} 203}); 204reload_button->signal_clicked().connect([this]() { 205auto page = content_manager->get_last_operated_page(); 206if(page) { 207if(auto webview = WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj())) { 208webkit_web_view_reload(webview); 209} 210} 211}); 212main_toolbar->append(*back_button); 213main_toolbar->append(*forward_button); 214main_toolbar->append(*reload_button); 215main_toolbar->append(*url_bar); 216main_toolbar->append(*go_button); 217outer_grid->attach(*main_toolbar, 0, 0, 2, 1); 218auto load_url_callback = [this]() { 219auto page = content_manager->get_last_operated_page(); 220if(page) { 221if(auto webview = WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj())) { 222webkit_web_view_load_uri(webview, url_bar->get_text().c_str()); 223} 224} 225}; 226go_button->signal_clicked().connect(load_url_callback); 227url_bar->signal_activate().connect(load_url_callback); 228content_manager->signal_page_operated.connect([this](gPanthera::ContentPage *page) { 229if(!page->get_child()) { 230return; 231} 232if(!page->get_child()->get_first_child()) { 233return; 234} 235url_bar->set_text(webkit_web_view_get_uri(WEBKIT_WEB_VIEW(page->get_child()->get_first_child()->gobj()))); 236guint url_update_handler = g_signal_connect(page->get_child()->get_first_child()->gobj(), "notify", G_CALLBACK(notify_focused_callback), this); 237std::shared_ptr<sigc::connection> control_signal_handler = std::make_shared<sigc::connection>(); 238*control_signal_handler = page->signal_control_status_changed.connect([this, page, control_signal_handler, url_update_handler](bool controlled) { 239if(!controlled) { 240control_signal_handler->disconnect(); 241g_signal_handler_disconnect(page->get_child()->get_first_child()->gobj(), url_update_handler); 242} 243}); 244}); 245window->set_child(*outer_grid); 246debug_button->signal_clicked().connect([this]() { 247if(content_manager->get_last_operated_page()) { 248std::cout << "Last operated page: " << content_manager->get_last_operated_page()->get_name() << std::endl; 249} else { 250std::cout << "No page operated!" << std::endl; 251} 252}); 253// TODO: Use the last operated page and allow opening tabs next to the last operated page using certain panes 254// Load the existing layout, if it exists 255std::ifstream layout_file_in("layout.json"); 256if(layout_file_in) { 257std::string layout_json((std::istreambuf_iterator<char>(layout_file_in)), std::istreambuf_iterator<char>()); 258layout_file_in.close(); 259layout_manager->restore_json_layout(layout_json); 260} else { 261// Create a new layout if the file doesn't exist 262layout_file_in.close(); 263 264dock_stack_1->add_pane(*pane_1); 265dock_stack_1->add_pane(*pane_3); 266dock_stack_2->add_pane(*pane_2); 267 268std::ofstream layout_file_out("layout.json"); 269layout_file_out << layout_manager->get_layout_as_json(); 270layout_file_out.close(); 271} 272// Save the layout when changed 273layout_manager->signal_pane_moved.connect([this](gPanthera::DockablePane *pane) { 274std::ofstream layout_file_out("layout.json"); 275layout_file_out << layout_manager->get_layout_as_json(); 276layout_file_out.close(); 277std::cout << "Layout changed: " << layout_manager->get_layout_as_json() << std::endl; 278}); 279 280auto new_tab_action = Gio::SimpleAction::create("new_tab"); 281new_tab_action->signal_activate().connect([this](const Glib::VariantBase&) { 282on_new_tab(nullptr); 283}); 284add_action(new_tab_action); 285set_accels_for_action("app.new_tab", {"<Primary>T"}); 286auto close_tab_action = Gio::SimpleAction::create("close_tab"); 287close_tab_action->signal_activate().connect([this](const Glib::VariantBase&) { 288auto page = content_manager->get_last_operated_page(); 289if(page) { 290page->close(); 291} 292}); 293add_action(close_tab_action); 294set_accels_for_action("app.close_tab", {"<Primary>W"}); 295} 296 297void on_activate() override { 298window->present(); 299} 300public: 301static Glib::RefPtr<PantheraWww> create() { 302return Glib::make_refptr_for_instance<PantheraWww>(new PantheraWww()); 303} 304}; 305 306int main(int argc, char *argv[]) { 307gPanthera::init(); 308auto app = PantheraWww::create(); 309return app->run(argc, argv); 310} 311