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