3 * Copyright (c) 2010 Marco Peereboom <marco@peereboom.us>
5 * Permission to use, copy, modify, and distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
20 * inverse color browsing
22 * download files status
23 * multi letter commands
24 * pre and post counts for commands
29 * autocompletion on various inputs
40 #include <sys/queue.h>
41 #include <sys/types.h>
45 #include <gdk/gdkkeysyms.h>
46 #include <webkit/webkit.h>
47 #include <libsoup/soup.h>
48 #include <JavaScriptCore/JavaScript.h>
50 static char *version
= "$xxxterm$";
53 /* #define XT_DEBUG */
55 #define DPRINTF(x...) do { if (swm_debug) fprintf(stderr, x); } while (0)
56 #define DNPRINTF(n,x...) do { if (swm_debug & n) fprintf(stderr, x); } while (0)
57 #define XT_D_MOVE 0x0001
58 #define XT_D_KEY 0x0002
59 #define XT_D_TAB 0x0004
60 #define XT_D_URL 0x0008
61 #define XT_D_CMD 0x0010
62 #define XT_D_NAV 0x0020
63 #define XT_D_DOWNLOAD 0x0040
64 #define XT_D_CONFIG 0x0080
65 u_int32_t swm_debug
= 0
77 #define DNPRINTF(n,x...)
80 #define LENGTH(x) (sizeof x / sizeof x[0])
81 #define CLEAN(mask) (mask & ~(GDK_MOD2_MASK) & \
82 ~(GDK_BUTTON1_MASK) & \
83 ~(GDK_BUTTON2_MASK) & \
84 ~(GDK_BUTTON3_MASK) & \
85 ~(GDK_BUTTON4_MASK) & \
89 TAILQ_ENTRY(tab
) entry
;
93 GtkWidget
*search_entry
;
95 GtkWidget
*browser_win
;
97 GtkToolItem
*backward
;
102 /* adjustments for browser */
105 GtkAdjustment
*adjust_h
;
106 GtkAdjustment
*adjust_v
;
117 WebKitWebSettings
*settings
;
119 TAILQ_HEAD(tab_list
, tab
);
127 #define XT_DIR (".xxxterm")
128 #define XT_CONF_FILE ("xxxterm.conf")
129 #define XT_CB_HANDLED (TRUE)
130 #define XT_CB_PASSTHROUGH (FALSE)
133 #define XT_MOVE_INVALID (0)
134 #define XT_MOVE_DOWN (1)
135 #define XT_MOVE_UP (2)
136 #define XT_MOVE_BOTTOM (3)
137 #define XT_MOVE_TOP (4)
138 #define XT_MOVE_PAGEDOWN (5)
139 #define XT_MOVE_PAGEUP (6)
140 #define XT_MOVE_LEFT (7)
141 #define XT_MOVE_FARLEFT (8)
142 #define XT_MOVE_RIGHT (9)
143 #define XT_MOVE_FARRIGHT (10)
145 #define XT_TAB_PREV (-2)
146 #define XT_TAB_NEXT (-1)
147 #define XT_TAB_INVALID (0)
148 #define XT_TAB_NEW (1)
149 #define XT_TAB_DELETE (2)
150 #define XT_TAB_DELQUIT (3)
151 #define XT_TAB_OPEN (4)
153 #define XT_NAV_INVALID (0)
154 #define XT_NAV_BACK (1)
155 #define XT_NAV_FORWARD (2)
156 #define XT_NAV_RELOAD (3)
158 #define XT_FOCUS_INVALID (0)
159 #define XT_FOCUS_URI (1)
161 #define XT_SEARCH_INVALID (0)
162 #define XT_SEARCH_NEXT (1)
163 #define XT_SEARCH_PREV (2)
166 extern char *__progname
;
168 GtkWidget
*main_window
;
169 GtkNotebook
*notebook
;
170 struct tab_list tabs
;
173 int showtabs
= 1; /* show tabs on notebook */
174 int showurl
= 1; /* show url toolbar on notebook */
175 int tabless
= 0; /* allow only 1 tab */
176 int ctrl_click_focus
= 0; /* ctrl click gets focus */
177 int cookies_enabled
= 1; /* enable cookies */
178 int read_only_cookies
= 0; /* enable to not write cookies */
179 int enable_scripts
= 1;
180 int enable_plugins
= 1;
181 int default_font_size
= 12;
182 int fancy_bar
= 1; /* fancy toolbar */
184 char *home
= "http://www.peereboom.us";
185 char *search_string
= NULL
;
186 char *http_proxy
= NULL
;
187 SoupURI
*proxy_uri
= NULL
;
188 char work_dir
[PATH_MAX
];
189 char cookie_file
[PATH_MAX
];
190 char download_dir
[PATH_MAX
];
191 SoupSession
*session
;
192 SoupCookieJar
*cookiejar
;
195 void create_new_tab(char *, int);
196 void delete_tab(struct tab
*);
198 struct valid_url_types
{
208 valid_url_type(char *url
)
212 for (i
= 0; i
< LENGTH(vut
); i
++)
213 if (!strncasecmp(vut
[i
].type
, url
, strlen(vut
[i
].type
)))
220 guess_url_type(char *url_in
)
223 char *url_out
= NULL
;
225 /* XXX not sure about this heuristic */
226 if (stat(url_in
, &sb
) == 0) {
227 if (asprintf(&url_out
, "file://%s", url_in
) == -1)
228 err(1, "aprintf file");
231 if (asprintf(&url_out
, "http://%s", url_in
) == -1)
232 err(1, "aprintf http");
236 err(1, "asprintf pointer");
238 DNPRINTF(XT_D_URL
, "guess_url_type: guessed %s\n", url_out
);
245 config_parse(char *filename
)
248 char *line
, *cp
, *var
, *val
;
249 size_t len
, lineno
= 0;
251 DNPRINTF(XT_D_CONFIG
, "config_parse: filename %s\n", filename
);
253 if (filename
== NULL
)
256 if ((config
= fopen(filename
, "r")) == NULL
) {
257 warn("config_parse: cannot open %s", filename
);
262 if ((line
= fparseln(config
, &len
, &lineno
, NULL
, 0)) == NULL
)
267 cp
+= (long)strspn(cp
, WS
);
274 if ((var
= strsep(&cp
, WS
)) == NULL
|| cp
== NULL
)
277 cp
+= (long)strspn(cp
, WS
);
279 if ((val
= strsep(&cp
, "\0")) == NULL
)
282 DNPRINTF(XT_D_CONFIG
, "config_parse: %s=%s\n",var
,val
);
285 if (!strcmp(var
, "home"))
287 else if (!strcmp(var
, "ctrl_click_focus"))
288 ctrl_click_focus
= atoi(val
);
289 else if (!strcmp(var
, "read_only_cookies"))
290 read_only_cookies
= atoi(val
);
291 else if (!strcmp(var
, "cookies_enabled"))
292 cookies_enabled
= atoi(val
);
293 else if (!strcmp(var
, "enable_scripts"))
294 enable_scripts
= atoi(val
);
295 else if (!strcmp(var
, "enable_plugins"))
296 enable_plugins
= atoi(val
);
297 else if (!strcmp(var
, "default_font_size"))
298 default_font_size
= atoi(val
);
299 else if (!strcmp(var
, "fancy_bar"))
300 fancy_bar
= atoi(val
);
301 else if (!strcmp(var
, "http_proxy")) {
302 http_proxy
= strdup(val
);
303 if (http_proxy
== NULL
)
304 err(1, "http_proxy");
305 } else if (!strcmp(var
, "search_string")) {
306 search_string
= strdup(val
);
307 if (search_string
== NULL
)
308 err(1, "search_string");
309 } else if (!strcmp(var
, "download_dir")) {
311 snprintf(download_dir
, sizeof download_dir
,
312 "%s/%s", pwd
->pw_dir
, &val
[1]);
314 strlcpy(download_dir
, val
, sizeof download_dir
);
315 fprintf(stderr
, "download dir: %s\n", download_dir
);
317 errx(1, "invalid conf file entry: %s=%s", var
, val
);
325 quit(struct tab
*t
, struct karg
*args
)
333 focus(struct tab
*t
, struct karg
*args
)
338 if (args
->i
== XT_FOCUS_URI
)
339 gtk_widget_grab_focus(GTK_WIDGET(t
->uri_entry
));
345 help(struct tab
*t
, struct karg
*args
)
350 webkit_web_view_load_string(t
->wv
,
351 "<html><body><h1>XXXTerm</h1></body></html>",
360 navaction(struct tab
*t
, struct karg
*args
)
362 DNPRINTF(XT_D_NAV
, "navaction: tab %d opcode %d\n",
367 webkit_web_view_go_back(t
->wv
);
370 webkit_web_view_go_forward(t
->wv
);
373 webkit_web_view_reload(t
->wv
);
376 return (XT_CB_PASSTHROUGH
);
380 move(struct tab
*t
, struct karg
*args
)
382 GtkAdjustment
*adjust
;
383 double pi
, si
, pos
, ps
, upper
, lower
, max
;
390 case XT_MOVE_PAGEDOWN
:
392 adjust
= t
->adjust_v
;
395 adjust
= t
->adjust_h
;
399 pos
= gtk_adjustment_get_value(adjust
);
400 ps
= gtk_adjustment_get_page_size(adjust
);
401 upper
= gtk_adjustment_get_upper(adjust
);
402 lower
= gtk_adjustment_get_lower(adjust
);
403 si
= gtk_adjustment_get_step_increment(adjust
);
404 pi
= gtk_adjustment_get_page_increment(adjust
);
407 DNPRINTF(XT_D_MOVE
, "move: opcode %d %s pos %f ps %f upper %f lower %f "
408 "max %f si %f pi %f\n",
409 args
->i
, adjust
== t
->adjust_h
? "horizontal" : "vertical",
410 pos
, ps
, upper
, lower
, max
, si
, pi
);
416 gtk_adjustment_set_value(adjust
, MIN(pos
, max
));
421 gtk_adjustment_set_value(adjust
, MAX(pos
, lower
));
424 case XT_MOVE_FARRIGHT
:
425 gtk_adjustment_set_value(adjust
, max
);
428 case XT_MOVE_FARLEFT
:
429 gtk_adjustment_set_value(adjust
, lower
);
431 case XT_MOVE_PAGEDOWN
:
433 gtk_adjustment_set_value(adjust
, MIN(pos
, max
));
437 gtk_adjustment_set_value(adjust
, MAX(pos
, lower
));
440 return (XT_CB_PASSTHROUGH
);
443 DNPRINTF(XT_D_MOVE
, "move: new pos %f %f\n", pos
, MIN(pos
, max
));
445 return (XT_CB_HANDLED
);
449 getparams(char *cmd
, char *cmp
)
454 if (!strncmp(cmd
, cmp
, strlen(cmp
))) {
455 rv
= cmd
+ strlen(cmp
);
467 tabaction(struct tab
*t
, struct karg
*args
)
469 int rv
= XT_CB_HANDLED
;
470 char *url
= NULL
, *newuri
= NULL
;
472 DNPRINTF(XT_D_TAB
, "tabaction: %p %d %d\n", t
, args
->i
, t
->focus_wv
);
475 return (XT_CB_PASSTHROUGH
);
479 if ((url
= getparams(args
->s
, "tabnew")))
480 create_new_tab(url
, 1);
482 create_new_tab(NULL
, 1);
488 if (gtk_notebook_get_n_pages(notebook
) > 1)
494 if ((url
= getparams(args
->s
, "open")) ||
495 ((url
= getparams(args
->s
, "op"))) ||
496 ((url
= getparams(args
->s
, "o"))))
499 rv
= XT_CB_PASSTHROUGH
;
503 if (valid_url_type(url
)) {
504 newuri
= guess_url_type(url
);
507 webkit_web_view_load_uri(t
->wv
, url
);
512 rv
= XT_CB_PASSTHROUGH
;
526 movetab(struct tab
*t
, struct karg
*args
)
531 DNPRINTF(XT_D_TAB
, "movetab: %p %d\n", t
, args
->i
);
534 return (XT_CB_PASSTHROUGH
);
536 if (args
->i
== XT_TAB_INVALID
)
537 return (XT_CB_PASSTHROUGH
);
539 if (args
->i
< XT_TAB_INVALID
) {
540 /* next or previous tab */
541 if (TAILQ_EMPTY(&tabs
))
542 return (XT_CB_PASSTHROUGH
);
544 if (args
->i
== XT_TAB_NEXT
)
545 gtk_notebook_next_page(notebook
);
547 gtk_notebook_prev_page(notebook
);
549 return (XT_CB_HANDLED
);
554 if (t
->tab_id
== x
) {
555 DNPRINTF(XT_D_TAB
, "movetab: do nothing\n");
556 return (XT_CB_HANDLED
);
559 TAILQ_FOREACH(tt
, &tabs
, entry
) {
560 if (tt
->tab_id
== x
) {
561 gtk_notebook_set_current_page(notebook
, x
);
562 DNPRINTF(XT_D_TAB
, "movetab: going to %d\n", x
);
564 gtk_widget_grab_focus(GTK_WIDGET(tt
->wv
));
568 return (XT_CB_HANDLED
);
572 command(struct tab
*t
, struct karg
*args
)
577 if (t
== NULL
|| args
== NULL
)
582 else if (args
->i
== ':')
585 warnx("invalid command %c\n", args
->i
);
586 return (XT_CB_PASSTHROUGH
);
589 DNPRINTF(XT_D_CMD
, "command: type %s\n", s
);
591 gtk_entry_set_text(GTK_ENTRY(t
->cmd
), s
);
592 gdk_color_parse("white", &color
);
593 gtk_widget_modify_base(t
->cmd
, GTK_STATE_NORMAL
, &color
);
594 gtk_widget_show(t
->cmd
);
595 gtk_widget_grab_focus(GTK_WIDGET(t
->cmd
));
596 gtk_editable_set_position(GTK_EDITABLE(t
->cmd
), -1);
598 return (XT_CB_HANDLED
);
602 search(struct tab
*t
, struct karg
*args
)
606 if (t
== NULL
|| args
== NULL
)
608 if (t
->search_text
== NULL
)
609 return (XT_CB_PASSTHROUGH
);
611 DNPRINTF(XT_D_CMD
, "search: tab %d opc %d text %s\n",
612 t
->tab_id
, args
->i
, t
->search_text
);
622 return (XT_CB_PASSTHROUGH
);
625 webkit_web_view_search_text(t
->wv
, t
->search_text
, FALSE
, d
, TRUE
);
627 return (XT_CB_HANDLED
);
630 /* inherent to GTK not all keys will be caught at all times */
635 int (*func
)(struct tab
*, struct karg
*);
638 { 0, 0, GDK_slash
, command
, {.i
= '/'} },
639 { GDK_SHIFT_MASK
, 0, GDK_colon
, command
, {.i
= ':'} },
640 { GDK_CONTROL_MASK
, 0, GDK_q
, quit
, {0} },
643 { 0, 0, GDK_n
, search
, {.i
= XT_SEARCH_NEXT
} },
644 { GDK_SHIFT_MASK
, 0, GDK_N
, search
, {.i
= XT_SEARCH_PREV
} },
647 { 0, 0, GDK_F6
, focus
, {.i
= XT_FOCUS_URI
} },
650 { 0, 0, GDK_BackSpace
, navaction
, {.i
= XT_NAV_BACK
} },
651 { GDK_MOD1_MASK
, 0, GDK_Left
, navaction
, {.i
= XT_NAV_BACK
} },
652 { GDK_SHIFT_MASK
, 0, GDK_BackSpace
, navaction
, {.i
= XT_NAV_FORWARD
} },
653 { GDK_MOD1_MASK
, 0, GDK_Right
, navaction
, {.i
= XT_NAV_FORWARD
} },
654 { 0, 0, GDK_F5
, navaction
, {.i
= XT_NAV_RELOAD
} },
656 /* vertical movement */
657 { 0, 0, GDK_j
, move
, {.i
= XT_MOVE_DOWN
} },
658 { 0, 0, GDK_Down
, move
, {.i
= XT_MOVE_DOWN
} },
659 { 0, 0, GDK_Up
, move
, {.i
= XT_MOVE_UP
} },
660 { 0, 0, GDK_k
, move
, {.i
= XT_MOVE_UP
} },
661 { GDK_SHIFT_MASK
, 0, GDK_G
, move
, {.i
= XT_MOVE_BOTTOM
} },
662 { 0, 0, GDK_End
, move
, {.i
= XT_MOVE_BOTTOM
} },
663 { 0, 0, GDK_Home
, move
, {.i
= XT_MOVE_TOP
} },
664 { 0, GDK_g
, GDK_g
, move
, {.i
= XT_MOVE_TOP
} }, /* XXX make this work */
665 { 0, 0, GDK_space
, move
, {.i
= XT_MOVE_PAGEDOWN
} },
666 { 0, 0, GDK_Page_Down
, move
, {.i
= XT_MOVE_PAGEDOWN
} },
667 { 0, 0, GDK_Page_Up
, move
, {.i
= XT_MOVE_PAGEUP
} },
668 /* horizontal movement */
669 { 0, 0, GDK_l
, move
, {.i
= XT_MOVE_RIGHT
} },
670 { 0, 0, GDK_Right
, move
, {.i
= XT_MOVE_RIGHT
} },
671 { 0, 0, GDK_Left
, move
, {.i
= XT_MOVE_LEFT
} },
672 { 0, 0, GDK_h
, move
, {.i
= XT_MOVE_LEFT
} },
673 { GDK_SHIFT_MASK
, 0, GDK_dollar
, move
, {.i
= XT_MOVE_FARRIGHT
} },
674 { 0, 0, GDK_0
, move
, {.i
= XT_MOVE_FARLEFT
} },
677 { GDK_CONTROL_MASK
, 0, GDK_t
, tabaction
, {.i
= XT_TAB_NEW
} },
678 { GDK_CONTROL_MASK
, 0, GDK_w
, tabaction
, {.i
= XT_TAB_DELETE
} },
679 { GDK_CONTROL_MASK
, 0, GDK_1
, movetab
, {.i
= 1} },
680 { GDK_CONTROL_MASK
, 0, GDK_2
, movetab
, {.i
= 2} },
681 { GDK_CONTROL_MASK
, 0, GDK_3
, movetab
, {.i
= 3} },
682 { GDK_CONTROL_MASK
, 0, GDK_4
, movetab
, {.i
= 4} },
683 { GDK_CONTROL_MASK
, 0, GDK_5
, movetab
, {.i
= 5} },
684 { GDK_CONTROL_MASK
, 0, GDK_6
, movetab
, {.i
= 6} },
685 { GDK_CONTROL_MASK
, 0, GDK_7
, movetab
, {.i
= 7} },
686 { GDK_CONTROL_MASK
, 0, GDK_8
, movetab
, {.i
= 8} },
687 { GDK_CONTROL_MASK
, 0, GDK_9
, movetab
, {.i
= 9} },
688 { GDK_CONTROL_MASK
, 0, GDK_0
, movetab
, {.i
= 10} },
694 int (*func
)(struct tab
*, struct karg
*);
697 { "q!", 0, quit
, {0} },
698 { "qa", 0, quit
, {0} },
699 { "qa!", 0, quit
, {0} },
700 { "help", 0, help
, {0} },
703 { "o", 1, tabaction
, {.i
= XT_TAB_OPEN
} },
704 { "op", 1, tabaction
, {.i
= XT_TAB_OPEN
} },
705 { "open", 1, tabaction
, {.i
= XT_TAB_OPEN
} },
706 { "tabnew", 1, tabaction
, {.i
= XT_TAB_NEW
} },
707 { "tabedit", 1, tabaction
, {.i
= XT_TAB_NEW
} },
708 { "tabe", 1, tabaction
, {.i
= XT_TAB_NEW
} },
709 { "tabclose", 0, tabaction
, {.i
= XT_TAB_DELETE
} },
710 { "tabc", 0, tabaction
, {.i
= XT_TAB_DELETE
} },
711 { "quit", 0, tabaction
, {.i
= XT_TAB_DELQUIT
} },
712 { "q", 0, tabaction
, {.i
= XT_TAB_DELQUIT
} },
713 /* XXX add count to these commands and add tabl and friends */
714 { "tabprevious", 0, movetab
, {.i
= XT_TAB_PREV
} },
715 { "tabp", 0, movetab
, {.i
= XT_TAB_PREV
} },
716 { "tabnext", 0, movetab
, {.i
= XT_TAB_NEXT
} },
717 { "tabn", 0, movetab
, {.i
= XT_TAB_NEXT
} },
721 focus_uri_entry_cb(GtkWidget
* w
, GtkDirectionType direction
, struct tab
*t
)
723 DNPRINTF(XT_D_URL
, "focus_uri_entry_cb: tab %d focus_wv %d\n",
724 t
->tab_id
, t
->focus_wv
);
727 errx(1, "focus_uri_entry_cb");
729 /* focus on wv instead */
731 gtk_widget_grab_focus(GTK_WIDGET(t
->wv
));
735 activate_uri_entry_cb(GtkWidget
* entry
, struct tab
*t
)
737 const gchar
*uri
= gtk_entry_get_text(GTK_ENTRY(entry
));
740 DNPRINTF(XT_D_URL
, "activate_uri_entry_cb: %s\n", uri
);
743 errx(1, "activate_uri_entry_cb");
748 if (valid_url_type((char *)uri
)) {
749 newuri
= guess_url_type((char *)uri
);
753 webkit_web_view_load_uri(t
->wv
, uri
);
754 gtk_widget_grab_focus(GTK_WIDGET(t
->wv
));
761 activate_search_entry_cb(GtkWidget
* entry
, struct tab
*t
)
763 const gchar
*search
= gtk_entry_get_text(GTK_ENTRY(entry
));
766 DNPRINTF(XT_D_URL
, "activate_search_entry_cb: %s\n", search
);
769 errx(1, "activate_search_entry_cb");
771 if (search_string
== NULL
) {
772 warnx("no search_string");
776 if (asprintf(&newuri
, search_string
, search
) == -1)
777 err(1, "activate_search_entry_cb");
779 webkit_web_view_load_uri(t
->wv
, newuri
);
780 gtk_widget_grab_focus(GTK_WIDGET(t
->wv
));
787 notify_load_status_cb(WebKitWebView
* wview
, GParamSpec
* pspec
, struct tab
*t
)
789 WebKitWebFrame
*frame
;
793 errx(1, "notify_load_status_cb");
795 switch (webkit_web_view_get_load_status(wview
)) {
796 case WEBKIT_LOAD_COMMITTED
:
797 frame
= webkit_web_view_get_main_frame(wview
);
798 uri
= webkit_web_frame_get_uri(frame
);
800 gtk_entry_set_text(GTK_ENTRY(t
->uri_entry
), uri
);
802 gtk_widget_set_sensitive(GTK_WIDGET(t
->stop
), TRUE
);
805 /* take focus if we are visible */
806 if (gtk_notebook_get_current_page(notebook
) == t
->tab_id
)
807 gtk_widget_grab_focus(GTK_WIDGET(t
->wv
));
809 case WEBKIT_LOAD_FIRST_VISUALLY_NON_EMPTY_LAYOUT
:
810 uri
= webkit_web_view_get_title(wview
);
812 frame
= webkit_web_view_get_main_frame(wview
);
813 uri
= webkit_web_frame_get_uri(frame
);
815 gtk_label_set_text(GTK_LABEL(t
->label
), uri
);
817 case WEBKIT_LOAD_PROVISIONAL
:
818 case WEBKIT_LOAD_FINISHED
:
819 case WEBKIT_LOAD_FAILED
:
820 gtk_widget_set_sensitive(GTK_WIDGET(t
->stop
), FALSE
);
825 gtk_widget_set_sensitive(GTK_WIDGET(t
->backward
),
826 webkit_web_view_can_go_back(wview
));
828 gtk_widget_set_sensitive(GTK_WIDGET(t
->forward
),
829 webkit_web_view_can_go_forward(wview
));
833 webview_npd_cb(WebKitWebView
*wv
, WebKitWebFrame
*wf
,
834 WebKitNetworkRequest
*request
, WebKitWebNavigationAction
*na
,
835 WebKitWebPolicyDecision
*pd
, struct tab
*t
)
840 errx(1, "webview_npd_cb");
842 DNPRINTF(XT_D_NAV
, "webview_npd_cb: %s\n",
843 webkit_network_request_get_uri(request
));
846 uri
= (char *)webkit_network_request_get_uri(request
);
847 create_new_tab(uri
, ctrl_click_focus
);
849 webkit_web_policy_decision_ignore(pd
);
851 return (TRUE
); /* we made the decission */
858 webview_event_cb(GtkWidget
*w
, GdkEventButton
*e
, struct tab
*t
)
860 /* we can not eat the event without throwing gtk off so defer it */
862 /* catch ctrl click */
863 if (e
->type
== GDK_BUTTON_RELEASE
&&
864 CLEAN(e
->state
) == GDK_CONTROL_MASK
)
869 return (XT_CB_PASSTHROUGH
);
873 webview_mimetype_cb(WebKitWebView
*wv
, WebKitWebFrame
*frame
,
874 WebKitNetworkRequest
*request
, char *mime_type
,
875 WebKitWebPolicyDecision
*decision
, struct tab
*t
)
878 errx(1, "webview_mimetype_cb");
880 DNPRINTF(XT_D_DOWNLOAD
, "webview_mimetype_cb: tab %d mime %s\n",
881 t
->tab_id
, mime_type
);
883 if (webkit_web_view_can_show_mime_type(wv
, mime_type
) == FALSE
) {
884 webkit_web_policy_decision_download(decision
);
892 webview_download_cb(WebKitWebView
*wv
, WebKitDownload
*download
, struct tab
*t
)
894 const gchar
*filename
;
897 if (download
== NULL
|| t
== NULL
)
898 errx(1, "webview_download_cb: invalid pointers");
900 filename
= webkit_download_get_suggested_filename(download
);
901 if (filename
== NULL
)
902 return (FALSE
); /* abort download */
904 if (asprintf(&uri
, "file://%s/%s", download_dir
, filename
) == -1)
905 err(1, "aprintf uri");
907 DNPRINTF(XT_D_DOWNLOAD
, "webview_download_cb: tab %d filename %s "
909 t
->tab_id
, filename
, uri
);
911 webkit_download_set_destination_uri(download
, uri
);
916 webkit_download_start(download
);
918 return (TRUE
); /* start download */
921 /* XXX currently unused */
923 webview_hover_cb(WebKitWebView
*wv
, gchar
*title
, gchar
*uri
, struct tab
*t
)
925 DNPRINTF(XT_D_KEY
, "webview_hover_cb: %s %s\n", title
, uri
);
928 errx(1, "webview_hover_cb");
935 t
->hover
= strdup(uri
);
936 } else if (t
->hover
) {
943 webview_keypress_cb(GtkWidget
*w
, GdkEventKey
*e
, struct tab
*t
)
947 /* don't use w directly; use t->whatever instead */
950 errx(1, "webview_keypress_cb");
952 DNPRINTF(XT_D_KEY
, "webview_keypress_cb: keyval 0x%x mask 0x%x t %p\n",
953 e
->keyval
, e
->state
, t
);
955 for (i
= 0; i
< LENGTH(keys
); i
++)
956 if (e
->keyval
== keys
[i
].key
&& CLEAN(e
->state
) ==
958 keys
[i
].func(t
, &keys
[i
].arg
);
959 return (XT_CB_HANDLED
);
962 return (XT_CB_PASSTHROUGH
);
966 cmd_keyrelease_cb(GtkEntry
*w
, GdkEventKey
*e
, struct tab
*t
)
968 const gchar
*c
= gtk_entry_get_text(w
);
971 DNPRINTF(XT_D_CMD
, "cmd_keyrelease_cb: keyval 0x%x mask 0x%x t %p\n",
972 e
->keyval
, e
->state
, t
);
975 errx(1, "cmd_keyrelease_cb");
977 DNPRINTF(XT_D_CMD
, "cmd_keyrelease_cb: keyval 0x%x mask 0x%x t %p\n",
978 e
->keyval
, e
->state
, t
);
986 if (webkit_web_view_search_text(t
->wv
, &c
[1], FALSE
, TRUE
, TRUE
) == FALSE
) {
987 /* not found, mark red */
988 gdk_color_parse("red", &color
);
989 gtk_widget_modify_base(t
->cmd
, GTK_STATE_NORMAL
, &color
);
990 /* unmark and remove selection */
991 webkit_web_view_unmark_text_matches(t
->wv
);
992 /* my kingdom for a way to unselect text in webview */
994 /* found, highlight all */
995 webkit_web_view_unmark_text_matches(t
->wv
);
996 webkit_web_view_mark_text_matches(t
->wv
, &c
[1], FALSE
, 0);
997 webkit_web_view_set_highlight_text_matches(t
->wv
, TRUE
);
998 gdk_color_parse("white", &color
);
999 gtk_widget_modify_base(t
->cmd
, GTK_STATE_NORMAL
, &color
);
1002 return (XT_CB_PASSTHROUGH
);
1006 cmd_keypress_cb(GtkEntry
*w
, GdkEventKey
*e
, struct tab
*t
)
1008 int rv
= XT_CB_HANDLED
;
1009 const gchar
*c
= gtk_entry_get_text(w
);
1012 errx(1, "cmd_keypress_cb");
1014 DNPRINTF(XT_D_CMD
, "cmd_keypress_cb: keyval 0x%x mask 0x%x t %p\n",
1015 e
->keyval
, e
->state
, t
);
1019 e
->keyval
= GDK_Escape
;
1020 else if (!(c
[0] == ':' || c
[0] == '/'))
1021 e
->keyval
= GDK_Escape
;
1023 switch (e
->keyval
) {
1025 if (!(!strcmp(c
, ":") || !strcmp(c
, "/")))
1029 gtk_widget_hide(t
->cmd
);
1030 gtk_widget_grab_focus(GTK_WIDGET(t
->wv
));
1034 rv
= XT_CB_PASSTHROUGH
;
1040 cmd_focusout_cb(GtkWidget
*w
, GdkEventFocus
*e
, struct tab
*t
)
1043 errx(1, "cmd_focusout_cb");
1045 DNPRINTF(XT_D_CMD
, "cmd_focusout_cb: tab %d focus_wv %d\n",
1046 t
->tab_id
, t
->focus_wv
);
1048 /* abort command when losing focus */
1049 gtk_widget_hide(t
->cmd
);
1051 gtk_widget_grab_focus(GTK_WIDGET(t
->wv
));
1053 gtk_widget_grab_focus(GTK_WIDGET(t
->uri_entry
));
1055 return (XT_CB_PASSTHROUGH
);
1059 cmd_activate_cb(GtkEntry
*entry
, struct tab
*t
)
1063 const gchar
*c
= gtk_entry_get_text(entry
);
1066 errx(1, "cmd_activate_cb");
1068 DNPRINTF(XT_D_CMD
, "cmd_activate_cb: tab %d %s\n", t
->tab_id
, c
);
1073 else if (!(c
[0] == ':' || c
[0] == '/'))
1080 if (t
->search_text
) {
1081 free(t
->search_text
);
1082 t
->search_text
= NULL
;
1085 t
->search_text
= strdup(s
);
1086 if (t
->search_text
== NULL
)
1087 err(1, "search_text");
1092 for (i
= 0; i
< LENGTH(cmds
); i
++)
1093 if (cmds
[i
].params
) {
1094 if (!strncmp(s
, cmds
[i
].cmd
, strlen(cmds
[i
].cmd
))) {
1095 cmds
[i
].arg
.s
= strdup(s
);
1096 cmds
[i
].func(t
, &cmds
[i
].arg
);
1099 if (!strcmp(s
, cmds
[i
].cmd
))
1100 cmds
[i
].func(t
, &cmds
[i
].arg
);
1104 gtk_widget_hide(t
->cmd
);
1108 backward_cb(GtkWidget
*w
, struct tab
*t
)
1111 errx(1, "backward_cb");
1113 DNPRINTF(XT_D_NAV
, "backward_cb: tab %d\n", t
->tab_id
);
1115 webkit_web_view_go_back(t
->wv
);
1119 forward_cb(GtkWidget
*w
, struct tab
*t
)
1122 errx(1, "forward_cb");
1124 DNPRINTF(XT_D_NAV
, "forward_cb: tab %d\n", t
->tab_id
);
1126 webkit_web_view_go_forward(t
->wv
);
1130 stop_cb(GtkWidget
*w
, struct tab
*t
)
1132 WebKitWebFrame
*frame
;
1137 DNPRINTF(XT_D_NAV
, "stop_cb: tab %d\n", t
->tab_id
);
1139 frame
= webkit_web_view_get_main_frame(t
->wv
);
1140 if (frame
== NULL
) {
1141 warnx("stop_cb: no frame");
1145 webkit_web_frame_stop_loading(frame
);
1149 create_browser(struct tab
*t
)
1154 errx(1, "create_browser");
1156 t
->sb_h
= GTK_SCROLLBAR(gtk_hscrollbar_new(NULL
));
1157 t
->sb_v
= GTK_SCROLLBAR(gtk_vscrollbar_new(NULL
));
1158 t
->adjust_h
= gtk_range_get_adjustment(GTK_RANGE(t
->sb_h
));
1159 t
->adjust_v
= gtk_range_get_adjustment(GTK_RANGE(t
->sb_v
));
1161 w
= gtk_scrolled_window_new(t
->adjust_h
, t
->adjust_v
);
1162 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(w
),
1163 GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
1165 t
->wv
= WEBKIT_WEB_VIEW(webkit_web_view_new());
1166 gtk_container_add(GTK_CONTAINER(w
), GTK_WIDGET(t
->wv
));
1168 g_signal_connect(t
->wv
, "notify::load-status",
1169 G_CALLBACK(notify_load_status_cb
), t
);
1179 w
= gtk_window_new(GTK_WINDOW_TOPLEVEL
);
1180 gtk_window_set_default_size(GTK_WINDOW(w
), 800, 600);
1181 gtk_widget_set_name(w
, "xxxterm");
1182 gtk_window_set_wmclass(GTK_WINDOW(w
), "xxxterm", "XXXTerm");
1188 create_toolbar(struct tab
*t
)
1190 GtkWidget
*toolbar
= gtk_toolbar_new();
1193 #if GTK_CHECK_VERSION(2,15,0)
1194 gtk_orientable_set_orientation(GTK_ORIENTABLE(toolbar
),
1195 GTK_ORIENTATION_HORIZONTAL
);
1197 gtk_toolbar_set_orientation(GTK_TOOLBAR(toolbar
),
1198 GTK_ORIENTATION_HORIZONTAL
);
1200 gtk_toolbar_set_style(GTK_TOOLBAR(toolbar
), GTK_TOOLBAR_BOTH_HORIZ
);
1203 /* backward button */
1204 t
->backward
= gtk_tool_button_new_from_stock(GTK_STOCK_GO_BACK
);
1205 gtk_widget_set_sensitive(GTK_WIDGET(t
->backward
), FALSE
);
1206 g_signal_connect(G_OBJECT(t
->backward
), "clicked",
1207 G_CALLBACK(backward_cb
), t
);
1208 gtk_toolbar_insert(GTK_TOOLBAR(toolbar
), t
->backward
, -1);
1210 /* forward button */
1212 gtk_tool_button_new_from_stock(GTK_STOCK_GO_FORWARD
);
1213 gtk_widget_set_sensitive(GTK_WIDGET(t
->forward
), FALSE
);
1214 g_signal_connect(G_OBJECT(t
->forward
), "clicked",
1215 G_CALLBACK(forward_cb
), t
);
1216 gtk_toolbar_insert(GTK_TOOLBAR(toolbar
), t
->forward
, -1);
1219 t
->stop
= gtk_tool_button_new_from_stock(GTK_STOCK_STOP
);
1220 gtk_widget_set_sensitive(GTK_WIDGET(t
->stop
), FALSE
);
1221 g_signal_connect(G_OBJECT(t
->stop
), "clicked",
1222 G_CALLBACK(stop_cb
), t
);
1223 gtk_toolbar_insert(GTK_TOOLBAR(toolbar
), t
->stop
, -1);
1227 i
= gtk_tool_item_new();
1228 gtk_tool_item_set_expand(i
, TRUE
);
1229 t
->uri_entry
= gtk_entry_new();
1230 gtk_container_add(GTK_CONTAINER(i
), t
->uri_entry
);
1231 g_signal_connect(G_OBJECT(t
->uri_entry
), "activate",
1232 G_CALLBACK(activate_uri_entry_cb
), t
);
1233 gtk_toolbar_insert(GTK_TOOLBAR(toolbar
), i
, -1);
1236 if (fancy_bar
&& search_string
) {
1237 i
= gtk_tool_item_new();
1238 t
->search_entry
= gtk_entry_new();
1239 gtk_entry_set_width_chars(GTK_ENTRY(t
->search_entry
), 30);
1240 gtk_container_add(GTK_CONTAINER(i
), t
->search_entry
);
1241 g_signal_connect(G_OBJECT(t
->search_entry
), "activate",
1242 G_CALLBACK(activate_search_entry_cb
), t
);
1243 gtk_toolbar_insert(GTK_TOOLBAR(toolbar
), i
, -1);
1250 delete_tab(struct tab
*t
)
1252 DNPRINTF(XT_D_TAB
, "delete_tab: %p\n", t
);
1257 TAILQ_REMOVE(&tabs
, t
, entry
);
1258 if (TAILQ_EMPTY(&tabs
))
1259 create_new_tab(NULL
, 1);
1261 gtk_widget_destroy(t
->vbox
);
1266 setup_webkit(struct tab
*t
)
1271 /* XXX this can't be called over and over; fix it */
1272 t
->settings
= webkit_web_settings_new();
1273 g_object_get((GObject
*)t
->settings
, "user-agent", &strval
, NULL
);
1274 if (strval
== NULL
) {
1275 warnx("setup_webkit: can't get user-agent property");
1279 if (asprintf(&ua
, "%s %s+", strval
, version
) == -1)
1280 err(1, "aprintf user-agent");
1282 g_object_set((GObject
*)t
->settings
,
1283 "user-agent", ua
, NULL
);
1284 g_object_set((GObject
*)t
->settings
,
1285 "enable-scripts", enable_scripts
, NULL
);
1286 g_object_set((GObject
*)t
->settings
,
1287 "enable-plugins", enable_plugins
, NULL
);
1288 g_object_set((GObject
*)t
->settings
,
1289 "default-font-size", default_font_size
, NULL
);
1291 webkit_web_view_set_settings(t
->wv
, t
->settings
);
1298 create_new_tab(char *title
, int focus
)
1302 char *newuri
= NULL
;
1304 DNPRINTF(XT_D_TAB
, "create_new_tab: title %s focus %d\n", title
, focus
);
1306 if (tabless
&& !TAILQ_EMPTY(&tabs
)) {
1307 DNPRINTF(XT_D_TAB
, "create_new_tab: new tab rejected\n");
1311 t
= g_malloc0(sizeof *t
);
1312 TAILQ_INSERT_TAIL(&tabs
, t
, entry
);
1314 if (title
== NULL
) {
1315 title
= "(untitled)";
1318 if (valid_url_type(title
)) {
1319 newuri
= guess_url_type(title
);
1324 t
->vbox
= gtk_vbox_new(FALSE
, 0);
1327 t
->label
= gtk_label_new(title
);
1328 gtk_widget_set_size_request(t
->label
, 100, -1);
1331 t
->toolbar
= create_toolbar(t
);
1332 gtk_box_pack_start(GTK_BOX(t
->vbox
), t
->toolbar
, FALSE
, FALSE
, 0);
1335 t
->browser_win
= create_browser(t
);
1336 gtk_box_pack_start(GTK_BOX(t
->vbox
), t
->browser_win
, TRUE
, TRUE
, 0);
1340 t
->cmd
= gtk_entry_new();
1341 gtk_entry_set_inner_border(GTK_ENTRY(t
->cmd
), NULL
);
1342 gtk_entry_set_has_frame(GTK_ENTRY(t
->cmd
), FALSE
);
1343 gtk_box_pack_end(GTK_BOX(t
->vbox
), t
->cmd
, FALSE
, FALSE
, 0);
1345 /* and show it all */
1346 gtk_widget_show_all(t
->vbox
);
1347 t
->tab_id
= gtk_notebook_append_page(notebook
, t
->vbox
,
1350 g_object_connect((GObject
*)t
->cmd
,
1351 "signal::key-press-event", (GCallback
)cmd_keypress_cb
, t
,
1352 "signal::key-release-event", (GCallback
)cmd_keyrelease_cb
, t
,
1353 "signal::focus-out-event", (GCallback
)cmd_focusout_cb
, t
,
1354 "signal::activate", (GCallback
)cmd_activate_cb
, t
,
1357 g_object_connect((GObject
*)t
->wv
,
1358 "signal-after::key-press-event", (GCallback
)webview_keypress_cb
, t
,
1359 /* "signal::hovering-over-link", (GCallback)webview_hover_cb, t, */
1360 "signal::download-requested", (GCallback
)webview_download_cb
, t
,
1361 "signal::mime-type-policy-decision-requested", (GCallback
)webview_mimetype_cb
, t
,
1362 "signal::navigation-policy-decision-requested", (GCallback
)webview_npd_cb
, t
,
1363 "signal::event", (GCallback
)webview_event_cb
, t
,
1366 /* hijack the unused keys as if we were the browser */
1367 g_object_connect((GObject
*)t
->toolbar
,
1368 "signal-after::key-press-event", (GCallback
)webview_keypress_cb
, t
,
1371 g_signal_connect(G_OBJECT(t
->uri_entry
), "focus",
1372 G_CALLBACK(focus_uri_entry_cb
), t
);
1375 gtk_widget_hide(t
->cmd
);
1377 gtk_widget_hide(t
->toolbar
);
1380 gtk_notebook_set_current_page(notebook
, t
->tab_id
);
1381 DNPRINTF(XT_D_TAB
, "create_new_tab: going to tab: %d\n",
1386 webkit_web_view_load_uri(t
->wv
, title
);
1388 gtk_widget_grab_focus(GTK_WIDGET(t
->uri_entry
));
1395 notebook_switchpage_cb(GtkNotebook
*nb
, GtkNotebookPage
*nbp
, guint pn
,
1400 DNPRINTF(XT_D_TAB
, "notebook_switchpage_cb: tab: %d\n", pn
);
1402 TAILQ_FOREACH(t
, &tabs
, entry
) {
1403 if (t
->tab_id
== pn
) {
1404 DNPRINTF(XT_D_TAB
, "notebook_switchpage_cb: going to "
1406 gtk_widget_hide(t
->cmd
);
1416 vbox
= gtk_vbox_new(FALSE
, 0);
1417 notebook
= GTK_NOTEBOOK(gtk_notebook_new());
1419 gtk_notebook_set_show_tabs(GTK_NOTEBOOK(notebook
), FALSE
);
1420 gtk_notebook_set_scrollable(notebook
, TRUE
);
1422 gtk_box_pack_start(GTK_BOX(vbox
), GTK_WIDGET(notebook
), TRUE
, TRUE
, 0);
1424 g_object_connect((GObject
*)notebook
,
1425 "signal::switch-page", (GCallback
)notebook_switchpage_cb
, NULL
,
1428 main_window
= create_window();
1429 gtk_container_add(GTK_CONTAINER(main_window
), vbox
);
1430 gtk_widget_show_all(main_window
);
1437 soup_session_remove_feature(session
,
1438 (SoupSessionFeature
*)cookiejar
);
1439 g_object_unref(cookiejar
);
1443 if (cookies_enabled
== 0)
1446 cookiejar
= soup_cookie_jar_text_new(cookie_file
, read_only_cookies
);
1447 soup_session_add_feature(session
, (SoupSessionFeature
*)cookiejar
);
1451 setup_proxy(char *uri
)
1454 g_object_set(session
, "proxy_uri", NULL
, NULL
);
1455 soup_uri_free(proxy_uri
);
1460 http_proxy
= strdup(uri
);
1461 if (http_proxy
== NULL
)
1462 err(1, "http_proxy");
1468 DNPRINTF(XT_D_CONFIG
, "setup_proxy: %s\n", uri
);
1469 proxy_uri
= soup_uri_new(uri
);
1471 g_object_set(session
, "proxy-uri", proxy_uri
, NULL
);
1478 "%s [-STVt][-f file] url ...\n", __progname
);
1483 main(int argc
, char *argv
[])
1487 char conf
[PATH_MAX
] = { '\0' };
1488 char *env_proxy
= NULL
;
1490 while ((c
= getopt(argc
, argv
, "STVf:t")) != -1) {
1499 errx(0 , "Version: %s", version
);
1502 strlcpy(conf
, optarg
, sizeof(conf
));
1518 gtk_init(&argc
, &argv
);
1519 if (!g_thread_supported())
1520 g_thread_init(NULL
);
1522 pwd
= getpwuid(getuid());
1524 errx(1, "invalid user %d", getuid());
1526 /* set download dir */
1527 strlcpy(download_dir
, pwd
->pw_dir
, sizeof download_dir
);
1529 /* read config file */
1530 if (strlen(conf
) == 0)
1531 snprintf(conf
, sizeof conf
, "%s/.%s",
1532 pwd
->pw_dir
, XT_CONF_FILE
);
1535 if (stat(download_dir
, &sb
))
1536 errx(1, "must specify a valid download_dir");
1538 /* working directory */
1539 snprintf(work_dir
, sizeof work_dir
, "%s/%s", pwd
->pw_dir
, XT_DIR
);
1540 if (stat(work_dir
, &sb
)) {
1541 if (mkdir(work_dir
, S_IRWXU
) == -1)
1543 } else if (((sb
.st_mode
& (S_IRWXU
| S_IRWXG
| S_IRWXO
))) != S_IRWXU
) {
1544 warnx("fixing invalid permissions on %s", work_dir
);
1545 if (chmod(work_dir
, S_IRWXU
) == -1)
1550 session
= webkit_get_default_session();
1551 snprintf(cookie_file
, sizeof cookie_file
, "%s/cookies.txt", work_dir
);
1555 env_proxy
= getenv("http_proxy");
1557 http_proxy
= strdup(env_proxy
);
1558 if (http_proxy
== NULL
)
1559 err(1, "http_proxy");
1561 setup_proxy(http_proxy
);
1566 create_new_tab(argv
[0], focus
);
1573 create_new_tab(home
, 1);