tazweb view src/main.c @ rev 87

Add an icon into the search entry
author Christophe Lincoln <pankso@slitaz.org>
date Wed Apr 27 02:04:13 2011 +0200 (2011-04-27)
parents bf2ff0574080
children 79aa29eeb0a1
line source
1 /*
2 * TazWeb is a radically simple web browser providing a single window
3 * with a single toolbar with buttons, an URL entry and search as well
4 * as a contextual menu, but no menu bar or tabs. Commented line code
5 * starts with // and comments are between * *
6 *
7 * Copyright (C) 2011 SliTaz GNU/Linux - BSD License
8 * See AUTHORS and LICENSE for detailed information
9 *
10 */
12 #include <gtk/gtk.h>
13 #include <webkit/webkit.h>
15 #define CONFIG g_strdup_printf("%s/.config/tazweb", g_get_home_dir())
16 #define START "file:///usr/share/webhome/index.html"
17 #define SEARCH "http://www.google.com/search?q=%s"
19 /* Needs AppleWebKit/531.2+ to handle all sites ? */
20 static gchar *useragent = "TazWeb (X11; SliTaz GNU/Linux; U; en_US)";
22 static GtkWidget* create_window(WebKitWebView** newwebview);
23 static GtkWidget *mainwindow, *vbox, *browser, *toolbar;
24 static WebKitWebView *webview;
25 static WebKitWebFrame *frame;
26 static gint count = 0;
27 const gchar* uri;
29 /* Create an icon */
30 static GdkPixbuf*
31 create_pixbuf(const gchar* image)
32 {
33 GdkPixbuf *pixbuf;
34 pixbuf = gdk_pixbuf_new_from_file(image, NULL);
35 return pixbuf;
36 }
38 /* Can be: http://hg.slitaz.org or hg.slitaz.org */
39 static void
40 check_requested_uri()
41 {
42 uri = g_strrstr(uri, "://") ? g_strdup(uri)
43 : g_strdup_printf("http://%s", uri);
44 }
46 /* Update title */
47 static void
48 update_title(GtkWindow* window, WebKitWebView* webview)
49 {
50 GString *string;
51 gdouble progress;
52 gchar *title;
54 string = g_string_new(webkit_web_view_get_title(webview));
55 progress = webkit_web_view_get_progress(webview) * 100;
56 if (progress < 100)
57 g_string_append_printf(string, " [ %f%% ] ", progress);
59 title = g_string_free(string, FALSE);
60 gtk_window_set_title(window, title);
61 g_free(title);
62 }
64 /* Get the page title */
65 static void
66 notify_title_cb(WebKitWebView* webview, GParamSpec* pspec, GtkWidget* window)
67 {
68 update_title(GTK_WINDOW(window), webview);
69 }
71 /* Request progress in window title */
72 static void
73 notify_progress_cb(WebKitWebView* webview, GParamSpec* pspec, GtkWidget* window)
74 {
75 update_title(GTK_WINDOW(window), webview);
76 }
78 /* Notify url entry */
79 static void
80 notify_load_status_cb(WebKitWebView* webview, GParamSpec* pspec, GtkWidget* urientry)
81 {
82 if (webkit_web_view_get_load_status(webview) == WEBKIT_LOAD_COMMITTED) {
83 frame = webkit_web_view_get_main_frame(webview);
84 uri = webkit_web_frame_get_uri(frame);
85 if (uri)
86 gtk_entry_set_text(GTK_ENTRY(urientry), uri);
87 }
88 }
90 /* Destroy the window */
91 static void
92 destroy_cb(GtkWidget* widget, GtkWindow* window)
93 {
94 if (g_atomic_int_dec_and_test(&count))
95 gtk_main_quit();
96 }
98 /* Show page source */
99 static void
100 view_source_cb(GtkWidget* widget, WebKitWebView* webview)
101 {
102 gboolean source;
104 frame = webkit_web_view_get_main_frame(webview);
105 uri = webkit_web_frame_get_uri(frame);
106 source = webkit_web_view_get_view_source_mode(webview);
108 webkit_web_view_set_view_source_mode(webview, !source);
109 webkit_web_view_reload(webview);
110 }
112 /* URL entry callback function */
113 static void
114 uri_entry_cb(GtkWidget* urientry, WebKitWebView* webview)
115 {
116 uri = gtk_entry_get_text(GTK_ENTRY(urientry));
117 g_assert(uri);
118 check_requested_uri();
119 webkit_web_view_load_uri(webview, uri);
120 }
122 /* Search entry callback function */
123 static void
124 search_entry_cb(GtkWidget* search, WebKitWebView* webview)
125 {
126 uri = g_strdup_printf(SEARCH, gtk_entry_get_text(GTK_ENTRY(search)));
127 g_assert(uri);
128 webkit_web_view_load_uri(webview, uri);
129 }
131 /* Navigation button function */
132 static void
133 go_home_cb(GtkWidget* widget, WebKitWebView* webview)
134 {
135 uri = g_strdup_printf("file://%s/home.html", CONFIG);
136 g_assert(uri);
137 webkit_web_view_load_uri(webview, uri);
138 }
140 static void
141 go_back_cb(GtkWidget* widget, WebKitWebView* webview)
142 {
143 webkit_web_view_go_back(webview);
144 }
146 static void
147 go_forward_cb(GtkWidget* widget, WebKitWebView* webview)
148 {
149 webkit_web_view_go_forward(webview);
150 }
152 /* Fullscreen and unfullscreen callback function */
153 static void
154 fullscreen_cb(GtkWindow* window, gpointer data)
155 {
156 GdkWindowState state;
157 state = gdk_window_get_state(gtk_widget_get_window(GTK_WIDGET(mainwindow)));
159 if(state & GDK_WINDOW_STATE_FULLSCREEN)
160 gtk_window_unfullscreen(GTK_WINDOW(mainwindow));
161 else
162 gtk_window_fullscreen(GTK_WINDOW(mainwindow));
163 }
165 /* TazWeb doc callback function */
166 static void
167 tazweb_doc_cb(GtkWidget* widget, WebKitWebView *webview)
168 {
169 uri = ("file:///usr/share/doc/tazweb/tazweb.html");
170 g_assert(uri);
171 webkit_web_view_load_uri(webview, uri);
172 }
174 /* Download function */
175 static gboolean
176 download_requested_cb(WebKitWebView *webview, WebKitDownload *download,
177 gpointer user_data)
178 {
179 uri = webkit_download_get_uri(download);
180 const gchar* buffer;
181 asprintf(&buffer,
182 "xterm -T 'Download' -geom 72x10+0-24 -e \
183 'cd $HOME/Downloads && wget -c %s; sleep 2' &", uri);
184 system(buffer);
185 }
187 /* Zoom out and in callback function */
188 static void
189 zoom_out_cb(GtkWidget *widget, WebKitWebView* webview)
190 {
191 webkit_web_view_zoom_out(webview);
192 }
194 static void
195 zoom_in_cb(GtkWidget *widget, WebKitWebView* webview)
196 {
197 webkit_web_view_zoom_in(webview);
198 }
200 /* New webview clallbacks */
201 static WebKitWebView*
202 create_web_view_cb(WebKitWebView* webview, WebKitWebFrame* web_frame,
203 GtkWidget* window)
204 {
205 WebKitWebView *newview;
206 create_window(&newview);
207 return newview;
208 }
210 static gboolean
211 webview_ready_cb(WebKitWebView* webview, GtkWidget* window)
212 {
213 gtk_widget_grab_focus(GTK_WIDGET(webview));
214 gtk_widget_show_all(window);
215 return FALSE;
216 }
218 static gboolean
219 close_webview_cb(WebKitWebView* webview, GtkWidget* window)
220 {
221 gtk_widget_destroy(window);
222 return TRUE;
223 }
225 /* Add items to WebKit contextual menu */
226 static void
227 populate_menu_cb(WebKitWebView *webview, GtkMenu *menu, gpointer data)
228 {
229 GtkWidget* item;
231 /* Separator */
232 item = gtk_separator_menu_item_new();
233 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
235 /* Zoom in */
236 item = gtk_image_menu_item_new_with_label("Zoom in");
237 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item),
238 gtk_image_new_from_stock(GTK_STOCK_ZOOM_IN, GTK_ICON_SIZE_MENU));
239 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
240 g_signal_connect(item, "activate", G_CALLBACK(zoom_in_cb), webview);
242 /* Zoom out */
243 item = gtk_image_menu_item_new_with_label("Zoom out");
244 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item),
245 gtk_image_new_from_stock(GTK_STOCK_ZOOM_OUT, GTK_ICON_SIZE_MENU));
246 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
247 g_signal_connect(item, "activate", G_CALLBACK(zoom_out_cb), webview);
249 /* Separator */
250 item = gtk_separator_menu_item_new();
251 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
253 /* TazWeb documentation */
254 item = gtk_image_menu_item_new_with_label("TazWeb manual");
255 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item),
256 gtk_image_new_from_stock(GTK_STOCK_INFO, GTK_ICON_SIZE_MENU));
257 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
258 g_signal_connect(item, "activate", G_CALLBACK(tazweb_doc_cb), webview);
260 /* View source mode */
261 item = gtk_image_menu_item_new_with_label("View source mode");
262 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item),
263 gtk_image_new_from_stock(GTK_STOCK_PROPERTIES, GTK_ICON_SIZE_MENU));
264 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
265 g_signal_connect(item, "activate", G_CALLBACK(view_source_cb), webview);
267 gtk_widget_show_all(GTK_WIDGET(menu));
268 }
270 /* Scrolled window for the webview */
271 static GtkWidget*
272 create_browser(GtkWidget* window, GtkWidget* urientry, GtkWidget* search,
273 WebKitWebView* webview)
274 {
275 WebKitWebSettings *settings;
277 browser = gtk_scrolled_window_new(NULL, NULL);
278 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(browser),
279 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
281 gtk_container_add(GTK_CONTAINER(browser), GTK_WIDGET(webview));
283 /* User agent */
284 settings = webkit_web_view_get_settings (webview);
285 g_object_set(G_OBJECT(settings), "user-agent", useragent, NULL);
287 /* Connect WebKit events */
288 g_signal_connect(webview, "notify::title",
289 G_CALLBACK(notify_title_cb), window);
290 g_signal_connect(webview, "notify::progress",
291 G_CALLBACK(notify_progress_cb), window);
292 g_signal_connect(webview, "notify::load-status",
293 G_CALLBACK(notify_load_status_cb), urientry);
294 g_signal_connect(webview, "download-requested",
295 G_CALLBACK(download_requested_cb), NULL);
296 g_signal_connect(webview, "create-web-view",
297 G_CALLBACK(create_web_view_cb), window);
298 g_signal_connect(webview, "web-view-ready",
299 G_CALLBACK(webview_ready_cb), window);
300 g_signal_connect(webview, "close-web-view",
301 G_CALLBACK(close_webview_cb), window);
303 /* Connect WebKit contextual menu items */
304 g_object_connect(G_OBJECT(webview), "signal::populate-popup",
305 G_CALLBACK(populate_menu_cb), webview, NULL);
307 return browser;
308 }
310 static GtkWidget*
311 create_toolbar(GtkWidget* urientry, GtkWidget* search, WebKitWebView* webview)
312 {
313 GtkToolItem* item;
315 toolbar = gtk_toolbar_new();
316 gtk_toolbar_set_orientation(GTK_TOOLBAR(toolbar),
317 GTK_ORIENTATION_HORIZONTAL);
318 gtk_toolbar_set_style(GTK_TOOLBAR(toolbar),
319 GTK_TOOLBAR_BOTH_HORIZ);
321 /* The back button */
322 item = gtk_tool_button_new_from_stock(GTK_STOCK_GO_BACK);
323 g_signal_connect(G_OBJECT(item), "clicked",
324 G_CALLBACK(go_back_cb), webview);
325 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
327 /* The forward button */
328 item = gtk_tool_button_new_from_stock(GTK_STOCK_GO_FORWARD);
329 g_signal_connect(G_OBJECT(item), "clicked",
330 G_CALLBACK(go_forward_cb), webview);
331 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
333 /* Home button */
334 item = gtk_tool_button_new_from_stock(GTK_STOCK_HOME);
335 g_signal_connect(G_OBJECT(item), "clicked",
336 G_CALLBACK(go_home_cb), webview);
337 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
339 /* URL entry */
340 item = gtk_tool_item_new();
341 gtk_tool_item_set_expand(item, TRUE);
342 gtk_widget_set_size_request(urientry, 0, 20);
343 gtk_container_add(GTK_CONTAINER(item), urientry);
344 g_signal_connect(G_OBJECT(urientry), "activate",
345 G_CALLBACK(uri_entry_cb), webview);
346 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
348 /* Separator */
349 item = gtk_separator_tool_item_new();
350 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
352 /* Search entry */
353 item = gtk_tool_item_new();
354 gtk_widget_set_size_request(search, 150, 20);
355 gtk_entry_set_icon_from_stock(GTK_ENTRY(search),
356 GTK_ENTRY_ICON_SECONDARY, GTK_STOCK_FIND);
357 /*
358 * FIXME: The icon can be activatable but must be connected to a callback to
359 * valid the search. With GTK_ENTRY_ICON_SECONDARY it do a segfault maybe
360 * with GTK_ENTRY_ICON_PRIMARY it dont crash but dont works.
361 */
362 gtk_entry_set_icon_activatable(GTK_ENTRY(search),
363 GTK_ENTRY_ICON_SECONDARY, FALSE);
364 //g_signal_connect(GTK_ENTRY(search), "icon-press",
365 // G_CALLBACK(search_entry_cb), webview);
366 //gtk_entry_set_icon_tooltip_text (GTK_ENTRY(search),
367 // GTK_ENTRY_ICON_SECONDARY, "Search the web");
368 gtk_container_add(GTK_CONTAINER(item), search);
369 g_signal_connect(G_OBJECT(search), "activate",
370 G_CALLBACK(search_entry_cb), webview);
371 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
373 /* The Fullscreen button */
374 item = gtk_tool_button_new_from_stock(GTK_STOCK_FULLSCREEN);
375 g_signal_connect(G_OBJECT(item), "clicked",
376 G_CALLBACK(fullscreen_cb), NULL);
377 gtk_toolbar_insert(GTK_TOOLBAR(toolbar), item, -1);
379 return toolbar;
380 }
382 /* Main window */
383 static GtkWidget*
384 create_window(WebKitWebView** newwebview)
385 {
386 GtkWidget *window;
387 GtkWidget *urientry;
388 GtkWidget *search;
390 g_atomic_int_inc(&count);
392 /* Default TazWeb window */
393 window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
394 gtk_window_set_default_size(GTK_WINDOW(window), 800, 600);
395 gtk_window_set_icon(GTK_WINDOW(window),
396 create_pixbuf("/usr/share/pixmaps/tazweb.png"));
397 gtk_widget_set_name(window, "TazWeb");
398 g_signal_connect(window, "destroy", G_CALLBACK(destroy_cb), NULL);
400 /* Webview and widgets */
401 webview = WEBKIT_WEB_VIEW(webkit_web_view_new());
402 urientry = gtk_entry_new();
403 search = gtk_entry_new();
404 vbox = gtk_vbox_new(FALSE, 0);
406 /* Pack box and container */
407 gtk_box_pack_start(GTK_BOX(vbox),
408 create_browser(window, urientry, search, webview), TRUE, TRUE, 0);
409 gtk_box_pack_start(GTK_BOX(vbox),
410 create_toolbar(urientry, search, webview), FALSE, FALSE, 0);
411 gtk_container_add(GTK_CONTAINER(window), vbox);
413 if (newwebview)
414 *newwebview = webview;
416 return window;
417 }
419 int
420 main(int argc, char* argv[])
421 {
422 gtk_init(NULL, NULL);
423 if (!g_thread_supported())
424 g_thread_init(NULL);
426 /* Get a default home.html if missing */
427 if (! g_file_test(CONFIG, G_FILE_TEST_EXISTS))
428 system("cp -r /usr/share/tazweb $HOME/.config/tazweb");
430 /* Load the start page file or the url in argument */
431 uri =(char*)(argc > 1 ? argv[1] : START);
432 if (argv[1])
433 check_requested_uri();
435 mainwindow = create_window(&webview);
437 gtk_widget_show_all(mainwindow);
438 webkit_web_view_load_uri(webview, uri);
439 gtk_widget_grab_focus(GTK_WIDGET(webview));
440 gtk_main();
442 return 0;
443 }