GTK docking interfaces

By using this site, you agree to have cookies stored on your device, strictly for functional purposes, such as storing your session and preferences.

Dismiss

 panthera-www.cc

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