From 2221ee15bd08a925172a420389812dc24f69bd5f Mon Sep 17 00:00:00 2001 From: "Enno Boland (tox)" Date: Sun, 6 Sep 2009 13:40:41 +0200 Subject: [PATCH 1/2] adding config.h patch from quaker4lyf. Thanks :) --- surf.c | 250 +++++++++++++++++++++++++++++++++------------------------ 1 file changed, 145 insertions(+), 105 deletions(-) diff --git a/surf.c b/surf.c index 0d9988f..fb277cc 100644 --- a/surf.c +++ b/surf.c @@ -20,6 +20,15 @@ Display *dpy; Atom urlprop; +typedef union Arg Arg; +union Arg { + const gboolean b; + const int i; + const unsigned int ui; + const float f; + const void *v; +} ; + typedef struct Client { GtkWidget *win, *scroll, *vbox, *urlbar, *searchbar; WebKitWebView *view; @@ -28,6 +37,27 @@ typedef struct Client { gint progress; struct Client *next; } Client; + +typedef struct { + guint mod; + guint keyval; + void (*func)(Client *c, const Arg *arg); + const Arg arg; + gboolean stop; /* do not propagate keypress event/stop matching keys */ +} Key; + +typedef enum { + NONE, + SEARCHBAR, + URLBAR, +} Keypressmode; + +typedef struct { + Key *keys; + unsigned int numkeys; + Keypressmode mode; +} KeySet; + SoupCookieJar *cookiejar; Client *clients = NULL; gboolean embed = FALSE; @@ -37,32 +67,39 @@ extern char *optarg; extern int optind; static void cleanup(void); +static void clipboard(Client *c, const Arg *arg); static void destroyclient(Client *c); static void destroywin(GtkWidget* w, Client *c); static void die(char *str); static void download(WebKitDownload *o, GParamSpec *pspec, Client *c); static gboolean initdownload(WebKitWebView *view, WebKitDownload *o, Client *c); static gchar *geturi(Client *c); -static void hidesearch(Client *c); -static void hideurl(Client *c); +static void hidesearch(Client *c, const Arg *arg); +static void hideurl(Client *c, const Arg *arg); static gboolean keypress(GtkWidget* w, GdkEventKey *ev, Client *c); static void linkhover(WebKitWebView* page, const gchar* t, const gchar* l, Client *c); static void loadcommit(WebKitWebView *view, WebKitWebFrame *f, Client *c); static void loadstart(WebKitWebView *view, WebKitWebFrame *f, Client *c); static void loadfile(Client *c, const gchar *f); -static void loaduri(Client *c, const gchar *uri); +static void loaduri(Client *c, const Arg *arg); +static void navigate(Client *c, const Arg *arg); static Client *newclient(); static WebKitWebView *newwindow(WebKitWebView *v, WebKitWebFrame *f, Client *c); static void pasteurl(GtkClipboard *clipboard, const gchar *text, gpointer d); static GdkFilterReturn processx(GdkXEvent *xevent, GdkEvent *event, gpointer d); static void progresschange(WebKitWebView *view, gint p, Client *c); +static void reload(Client *c, const Arg *arg); static void setup(void); -static void showsearch(Client *c); -static void showurl(Client *c); -static void stop(Client *c); +static void searchtext(Client *c, const Arg *arg); +static void showsearch(Client *c, const Arg *arg); +static void showurl(Client *c, const Arg *arg); +static void stop(Client *c, const Arg *arg); static void titlechange(WebKitWebView* view, WebKitWebFrame* frame, const gchar* title, Client *c); static void usage(); static void updatetitle(Client *c, const gchar *title); +static void zoompage(Client *c, const Arg *arg); + +#include "config.h" void cleanup(void) { @@ -70,6 +107,15 @@ cleanup(void) { destroyclient(clients); } +void +clipboard(Client *c, const Arg *arg) { + gboolean paste = *(gboolean *)arg; + if(paste) + gtk_clipboard_request_text(gtk_clipboard_get(GDK_SELECTION_PRIMARY), pasteurl, c); + else + gtk_clipboard_set_text(gtk_clipboard_get(GDK_SELECTION_PRIMARY), webkit_web_view_get_uri(c->view), -1); +} + void destroyclient(Client *c) { Client *p; @@ -117,7 +163,7 @@ initdownload(WebKitWebView *view, WebKitDownload *o, Client *c) { const gchar *home, *filename; gchar *uri, *path, *html; - stop(c); + stop(c, NULL); c->download = o; home = g_get_home_dir(); filename = webkit_download_get_suggested_filename(o); @@ -148,106 +194,53 @@ geturi(Client *c) { } void -hidesearch(Client *c) { +hidesearch(Client *c, const Arg *arg) { gtk_widget_hide(c->searchbar); gtk_widget_grab_focus(GTK_WIDGET(c->view)); } void -hideurl(Client *c) { +hideurl(Client *c, const Arg *arg) { gtk_widget_hide(c->urlbar); gtk_widget_grab_focus(GTK_WIDGET(c->view)); } gboolean keypress(GtkWidget* w, GdkEventKey *ev, Client *c) { + unsigned int n, m; + if(ev->type != GDK_KEY_PRESS) return FALSE; - if(GTK_WIDGET_HAS_FOCUS(c->searchbar)) { - switch(ev->keyval) { - case GDK_Escape: - hidesearch(c); - return TRUE; - case GDK_Return: - webkit_web_view_search_text(c->view, - gtk_entry_get_text(GTK_ENTRY(c->searchbar)), - FALSE, - !(ev->state & GDK_SHIFT_MASK), - TRUE); - return TRUE; - case GDK_Left: - case GDK_Right: - return FALSE; + + for(n = 0; n < LENGTH(keysets); n++) + switch(keysets[n].mode) { + case SEARCHBAR: + if(GTK_WIDGET_HAS_FOCUS(c->searchbar)) + goto matchkeys; + break; + case URLBAR: + if(GTK_WIDGET_HAS_FOCUS(c->urlbar)) + goto matchkeys; + break; + case NONE: + goto matchkeys; + default: + fprintf(stderr, "keypress(): Unknown Keypressmode\n"); + break; } - } - else if(GTK_WIDGET_HAS_FOCUS(c->urlbar)) { - switch(ev->keyval) { - case GDK_Escape: - hideurl(c); - return TRUE; - case GDK_Return: - loaduri(c, gtk_entry_get_text(GTK_ENTRY(c->urlbar))); - hideurl(c); - return TRUE; - case GDK_Left: - case GDK_Right: - return FALSE; - } - } - if(ev->state & GDK_CONTROL_MASK) { - switch(ev->keyval) { - case GDK_p: - gtk_clipboard_request_text(gtk_clipboard_get(GDK_SELECTION_PRIMARY), pasteurl, c); - return TRUE; - case GDK_y: - gtk_clipboard_set_text(gtk_clipboard_get(GDK_SELECTION_PRIMARY), webkit_web_view_get_uri(c->view), -1); - return TRUE; - case GDK_r: - case GDK_R: - if((ev->state & GDK_SHIFT_MASK)) - webkit_web_view_reload_bypass_cache(c->view); - else - webkit_web_view_reload(c->view); - return TRUE; - case GDK_b: - return TRUE; - case GDK_g: - showurl(c); - return TRUE; - case GDK_slash: - showsearch(c); - return TRUE; - case GDK_plus: - case GDK_equal: - webkit_web_view_zoom_in(c->view); - return TRUE; - case GDK_minus: - webkit_web_view_zoom_out(c->view); - return TRUE; - case GDK_0: - webkit_web_view_set_zoom_level(c->view, 1.0); - return TRUE; - case GDK_n: - case GDK_N: - webkit_web_view_search_text(c->view, - gtk_entry_get_text(GTK_ENTRY(c->searchbar)), - FALSE, - !(ev->state & GDK_SHIFT_MASK), - TRUE); - return TRUE; - case GDK_Left: - webkit_web_view_go_back(c->view); - return TRUE; - case GDK_Right: - webkit_web_view_go_forward(c->view); - return TRUE; - } - } - else { - switch(ev->keyval) { - case GDK_Escape: - stop(c); - return TRUE; + + if(n < LENGTH(keysets)) { +matchkeys: + for(m = 0; m < keysets[n].numkeys; m++) { + Key *keys = keysets[n].keys; + if(ev->keyval == keys[m].keyval + && (ev->state == keys[m].mod + || (ev->state & keys[m].mod)) + && keys[m].func) { + keys[m].func(c, &(keys[m].arg)); + if(keys[m].stop) + return TRUE; + } } } return FALSE; @@ -284,6 +277,7 @@ loadfile(Client *c, const gchar *f) { GError *e = NULL; GString *code; gchar *line, *uri; + Arg arg; if(strcmp(f, "-") == 0) { chan = g_io_channel_unix_new(STDIN_FILENO); @@ -299,19 +293,22 @@ loadfile(Client *c, const gchar *f) { g_io_channel_shutdown(chan, FALSE, NULL); g_string_free(code, TRUE); } - uri = g_strdup("stdin"); + arg.v = uri = g_strdup("stdin"); } else { - uri = g_strdup_printf("file://%s", f); - loaduri(c, uri); + arg.v = uri = g_strdup_printf("file://%s", f); + loaduri(c, &arg); } updatetitle(c, uri); g_free(uri); } void -loaduri(Client *c, const gchar *uri) { +loaduri(Client *c, const Arg *arg) { gchar *u; + const gchar *uri = (gchar *)arg->v; + if(!uri) + uri = gtk_entry_get_text(GTK_ENTRY(c->urlbar)); u = g_strrstr(uri, ":") ? g_strdup(uri) : g_strdup_printf("http://%s", uri); webkit_web_view_load_uri(c->view, u); @@ -320,6 +317,15 @@ loaduri(Client *c, const gchar *uri) { g_free(u); } +void +navigate(Client *c, const Arg *arg) { + gboolean forward = *(gboolean *)arg; + if(forward) + webkit_web_view_go_forward(c->view); + else + webkit_web_view_go_back(c->view); +} + Client * newclient(void) { Client *c; @@ -404,8 +410,9 @@ newwindow(WebKitWebView *v, WebKitWebFrame *f, Client *c) { void pasteurl(GtkClipboard *clipboard, const gchar *text, gpointer d) { + Arg arg = {.v = text }; if(text != NULL) - loaduri((Client *) d, text); + loaduri((Client *) d, &arg); } GdkFilterReturn @@ -416,6 +423,7 @@ processx(GdkXEvent *e, GdkEvent *event, gpointer d) { int idummy; unsigned long ldummy; unsigned char *buf = NULL; + Arg arg; if(((XEvent *)e)->type == PropertyNotify) { ev = &((XEvent *)e)->xproperty; if(ev->atom == urlprop && ev->state == PropertyNewValue) { @@ -424,7 +432,8 @@ processx(GdkXEvent *e, GdkEvent *event, gpointer d) { else { XGetWindowProperty(dpy, ev->window, urlprop, 0L, BUFSIZ, False, XA_STRING, &adummy, &idummy, &ldummy, &ldummy, &buf); - loaduri(c, (gchar *)buf); + arg.v = buf; + loaduri(c, &arg); XFree(buf); } return GDK_FILTER_REMOVE; @@ -439,23 +448,42 @@ progresschange(WebKitWebView* view, gint p, Client *c) { updatetitle(c, NULL); } +void +reload(Client *c, const Arg *arg) { + gboolean nocache = *(gboolean *)arg; + if(nocache) + webkit_web_view_reload_bypass_cache(c->view); + else + webkit_web_view_reload(c->view); +} + void setup(void) { dpy = GDK_DISPLAY(); urlprop = XInternAtom(dpy, "_SURF_URL", False); } void -showsearch(Client *c) { - hideurl(c); +showsearch(Client *c, const Arg *arg) { + hideurl(c, NULL); gtk_widget_show(c->searchbar); gtk_widget_grab_focus(c->searchbar); } void -showurl(Client *c) { +searchtext(Client *c, const Arg *arg) { + gboolean forward = *(gboolean *)arg; + webkit_web_view_search_text(c->view, + gtk_entry_get_text(GTK_ENTRY(c->searchbar)), + FALSE, + forward, + TRUE); +} + +void +showurl(Client *c, const Arg *arg) { gchar *uri; - hidesearch(c); + hidesearch(c, NULL); uri = geturi(c); gtk_entry_set_text(GTK_ENTRY(c->urlbar), uri); gtk_widget_show(c->urlbar); @@ -463,7 +491,7 @@ showurl(Client *c) { } void -stop(Client *c) { +stop(Client *c, const Arg *arg) { if(c->download) webkit_download_cancel(c->download); else @@ -500,11 +528,22 @@ updatetitle(Client *c, const char *title) { } +void +zoompage(Client *c, const Arg *arg) { + if(*(float *)arg < 0) /* zoom out */ + webkit_web_view_zoom_out(c->view); + else if(*(float *)arg == 0) /* zoom in */ + webkit_web_view_zoom_in(c->view); + else /* absolute level */ + webkit_web_view_set_zoom_level(c->view, *(float *)arg); +} + int main(int argc, char *argv[]) { SoupSession *s; Client *c; int o; const gchar *home, *filename; + Arg arg; gtk_init(NULL, NULL); if (!g_thread_supported()) @@ -521,7 +560,8 @@ int main(int argc, char *argv[]) { break; case 'u': c = newclient(); - loaduri(c, optarg); + arg.v = optarg; + loaduri(c, &arg); break; case 'f': c = newclient(); From c20442a13864c870d7e5e24c31299284c6c45f97 Mon Sep 17 00:00:00 2001 From: "Enno Boland (tox)" Date: Sun, 6 Sep 2009 13:41:07 +0200 Subject: [PATCH 2/2] adding config.h --- config.h | 46 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 46 insertions(+) create mode 100644 config.h diff --git a/config.h b/config.h new file mode 100644 index 0000000..9406e48 --- /dev/null +++ b/config.h @@ -0,0 +1,46 @@ +/* modifier 0 means no modifier */ +static Key searchbar_keys[] = { + /* modifier keyval function arg stop event */ + { 0, GDK_Escape, hidesearch, {0}, TRUE }, + { 0, GDK_Return, searchtext, {.b = TRUE}, TRUE }, + { GDK_SHIFT_MASK, GDK_Return, searchtext, {.b = FALSE}, TRUE }, + { GDK_SHIFT_MASK, GDK_Left, NULL, {0}, FALSE }, + { GDK_SHIFT_MASK, GDK_Right, NULL, {0}, FALSE }, +}; + +static Key urlbar_keys[] = { + /* modifier keyval function arg stop event */ + { 0, GDK_Escape, hideurl, {0}, TRUE }, + /* able to "chain" commands; by setting stop event to FALSE */ + { 0, GDK_Return, loaduri, {.v = NULL}, FALSE }, + { 0, GDK_Return, hideurl, {0}, TRUE }, + { GDK_SHIFT_MASK, GDK_Left, NULL, {0}, FALSE }, + { GDK_SHIFT_MASK, GDK_Right, NULL, {0}, FALSE }, +}; + +static Key general_keys[] = { + /* modifier keyval function arg stop event */ + { GDK_CONTROL_MASK, GDK_p, clipboard, {.b = TRUE }, TRUE }, + { GDK_CONTROL_MASK, GDK_y, clipboard, {.b = FALSE}, TRUE }, + { GDK_CONTROL_MASK, GDK_R, reload, {.b = TRUE}, TRUE }, + { GDK_CONTROL_MASK, GDK_r, reload, {.b = FALSE}, TRUE }, + { GDK_CONTROL_MASK, GDK_b, NULL, {0}, TRUE }, + { GDK_CONTROL_MASK, GDK_g, showurl, {0}, TRUE }, + { GDK_CONTROL_MASK, GDK_slash, showsearch, {0}, TRUE }, + { GDK_CONTROL_MASK, GDK_plus, zoompage, {0}, TRUE }, + { GDK_CONTROL_MASK, GDK_minus, zoompage, {.f = -1.0 }, TRUE }, + { GDK_CONTROL_MASK, GDK_0, zoompage, {.f = +1.0 }, TRUE }, + { GDK_CONTROL_MASK, GDK_n, searchtext, {.b = TRUE}, TRUE }, + { GDK_CONTROL_MASK, GDK_N, searchtext, {.b = FALSE}, TRUE }, + { GDK_CONTROL_MASK, GDK_Right, navigate, {.b = TRUE}, TRUE }, + { GDK_CONTROL_MASK, GDK_Left, navigate, {.b = FALSE}, TRUE }, + { 0, GDK_Escape, stop, {0}, TRUE }, +}; + +/* Sequence of Keys to match against a keypress */ +static KeySet keysets[] = { + /* keyset (Key[]) numkeys focusedwidget/mode */ + { searchbar_keys, LENGTH(searchbar_keys), SEARCHBAR }, + { urlbar_keys, LENGTH(urlbar_keys), URLBAR }, + { general_keys, LENGTH(general_keys), NONE }, +};