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