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