0 #include <string.h>
1 #include <strings.h>
2 #include <unistd.h>
3 #define TB_IMPL
4 #include "wcwidth.h"
5 #undef wcwidth
6 #define wcwidth(x) mk_wcwidth(x)
7 #ifdef sun
8 #include <termios.h>
9 void cfmakeraw(struct termios *t);
10 #endif
11 #include <termbox.h>
12 #include "gemini.h"
13 #include "display.h"
14 #include "cert.h"
15 #include "str.h"
16 #include "url.h"
17 #include "sandbox.h"
18 #include "util.h"
19 #include <stdio.h>
20
21 int get_cursor_pos() {
22 int pos = 0;
23 for (int i = 0; i < client.input.cursor; i++) {
24 pos += tb_utf8_char_length(client.input.field[pos]);
25 if (!client.input.field[pos]) break;
26 }
27 return pos;
28 }
29
30 int vim_counter() {
31 int counter = atoi(client.vim.counter);
32 bzero(client.vim.counter, sizeof(client.vim.counter));
33 return counter?counter:1;
34 }
35
36 void fix_scroll(struct gmi_tab* tab) {
37 int h = tab->page.lines - tb_height() + 2 + (client.tabs_count > 1);
38 if (tab->scroll > h)
39 tab->scroll = h;
40 if (tab->scroll < 0)
41 tab->scroll = -1;
42 }
43
44 int command() {
45 struct gmi_tab* tab = client.tab;
46 struct gmi_page* page = &tab->page;
47
48 // Trim
49 for (int i = STRNLEN(client.input.field) - 1;
50 client.input.field[i] == ' ' ||
51 client.input.field[i] == '\t'; i--)
52 client.input.field[i] = '\0';
53
54 if (!STRCMP(client.input.field, ":q")) {
55 gmi_freetab(tab);
56 client.input.field[0] = '\0';
57 if (client.tabs_count < 1)
58 return 1;
59 if (client.tabs_count == 1)
60 fix_scroll(client.tab);
61 return 0;
62 }
63 if (!STRCMP(client.input.field, ":qa"))
64 return 1;
65 if ((client.input.field[1] == 'n' && client.input.field[2] == 't'
66 && client.input.field[3] == '\0')
67 || !STRCMP(client.input.field, ":tabnew")) {
68 client.tab = gmi_newtab();
69 client.input.field[0] = '\0';
70 return 0;
71 }
72 if (STARTWITH(client.input.field, ":nt ") ||
73 STARTWITH(client.input.field, ":tabnew ")) {
74 int i = client.input.field[1] == 'n' ? 4 : 8;
75 client.input.cursor = 0;
76 int id = atoi(&client.input.field[i]);
77 if (id != 0 ||
78 (client.input.field[i] == '0' &&
79 client.input.field[i + 1] == '\0')) {
80 gmi_goto_new(tab, id);
81 client.input.field[0] = '\0';
82 } else {
83 client.tab = gmi_newtab_url(&client.input.field[i]);
84 client.input.field[0] = '\0';
85 }
86 return 0;
87 }
88 if (STARTWITH(client.input.field, ":o ")) {
89 char urlbuf[MAX_URL];
90 if (STRLCPY(urlbuf, &client.input.field[3]) >= sizeof(urlbuf)) {
91 tab->show_error = 1;
92 snprintf(tab->error, sizeof(tab->error),
93 "Url too long");
94 return 0;
95 }
96 client.input.field[0] = '\0';
97 gmi_cleanforward(tab);
98 int bytes = gmi_request(tab, urlbuf, 1);
99 if (bytes > 0) {
100 tab->scroll = -1;
101 }
102 if (page->code == 11 || page->code == 10) {
103 client.input.mode = 1;
104 client.input.cursor = 0;
105 }
106 tab->selected = 0;
107 return 0;
108 }
109 if (STARTWITH(client.input.field, ":s")) {
110 char urlbuf[MAX_URL];
111 snprintf(urlbuf, sizeof(urlbuf),
112 "gemini://geminispace.info/search?%s",
113 &client.input.field[3]);
114 client.input.field[0] = '\0';
115 gmi_cleanforward(tab);
116 int bytes = gmi_request(tab, urlbuf, 1);
117 if (bytes > 0) {
118 tab->scroll = -1;
119 }
120 if (page->code == 11 || page->code == 10) {
121 client.input.mode = 1;
122 client.input.cursor = 0;
123 }
124 tab->selected = 0;
125 return 0;
126 }
127 if (STARTWITH(client.input.field, ":add")) {
128 char* title = client.input.field[4] == '\0'?
129 NULL:&client.input.field[5];
130 gmi_addbookmark(tab, tab->url, title);
131 client.input.field[0] = '\0';
132 tab->selected = 0;
133 if (!strcmp("about:home", tab->url)) {
134 gmi_freetab(tab);
135 gmi_gohome(tab, 1);
136 }
137 return 0;
138 }
139 int ignore = STARTWITH(client.input.field, ":ignore");
140 int forget = STARTWITH(client.input.field, ":forget");
141 if (forget || ignore) {
142 char* ptr = client.input.field + sizeof(":forget") - 1;
143 int space = 0;
144 for (; *ptr; ptr++) {
145 if (*ptr != ' ') break;
146 space++;
147 }
148 if (space == 0 || !*ptr) goto unknown;
149 if ((ignore && cert_ignore_expiration(ptr)) ||
150 (forget && cert_forget(ptr))) {
151 tab->show_error = 1;
152 if (forget) {
153 snprintf(tab->error, sizeof(tab->error),
154 "Unknown %s certificate", ptr);
155 } else {
156 strerror_r(errno, tab->error,
157 sizeof(tab->error));
158 }
159 return 0;
160 }
161 tab->show_info = 1;
162 snprintf(tab->info, sizeof(tab->info),
163 "%s %s", ptr, (forget ?
164 "certificate removed from known hosts" :
165 "certificate expiration will be ignored"));
166 client.input.field[0] = '\0';
167 return 0;
168 }
169 if (!STRCMP(client.input.field, ":gencert")) {
170 client.input.field[0] = '\0';
171 tab->selected = 0;
172 if (!STRCMP(tab->url, "about:home")) {
173 tab->show_error = 1;
174 snprintf(tab->error, sizeof(tab->error),
175 "Cannot create a certificate for this page");
176 return 0;
177 }
178 char host[256];
179 parse_url(tab->url, host, sizeof(host), NULL, 0, NULL);
180 if (cert_create(host, tab->error, sizeof(tab->error))) {
181 tab->show_error = 1;
182 return 0;
183 }
184 cert_getcert(host, 1);
185 tab->show_info = 1;
186 snprintf(tab->info, sizeof(tab->info),
187 "Certificate generated for %s", host);
188 return 0;
189 }
190 if (!STRCMP(client.input.field, ":exec")) {
191 #ifdef DISABLE_XDG
192 tab->show_error = 1;
193 snprintf(tab->error, sizeof(tab->error),
194 "xdg is disabled");
195 return 0;
196 #else
197 if (!*client.input.download) {
198 snprintf(tab->error, sizeof(tab->error),
199 "No file was downloaded");
200 tab->show_error = 1;
201 client.input.field[0] = '\0';
202 return 0;
203 }
204 xdg_open(client.input.download);
205 client.input.field[0] = '\0';
206 client.input.download[0] = '\0';
207 return 0;
208 #endif
209 }
210 if (STARTWITH(client.input.field, ":download")) {
211 char* ptr = client.input.field + sizeof(":download") - 1;
212 int space = 0;
213 for (; *ptr; ptr++) {
214 if (*ptr != ' ') break;
215 space++;
216 }
217 if (space == 0 && *ptr) goto unknown;
218 char urlbuf[1024];
219 char* url = strrchr(tab->history->url, '/');
220 if (url && (*(url+1) == '\0')) {
221 while (*url == '/' && url > tab->history->url)
222 url--;
223 char* ptr = url + 1;
224 while (*url != '/' && url > tab->history->url)
225 url--;
226 if (*url == '/') url++;
227 if (strlcpy(urlbuf, url, ptr - url + 1) >=
228 sizeof(urlbuf))
229 return fatalI();
230 url = urlbuf;
231 } else if (url) url++;
232 else url = tab->history->url;
233 char* download = *ptr ? ptr : url;
234 #ifdef SANDBOX_SUN
235 if (sandbox_download(tab, download))
236 return -1;
237 int fd = wr_pair[1];
238 #else
239 int fd = openat(getdownloadfd(), download,
240 O_CREAT|O_EXCL|O_WRONLY, 0600);
241 char buf[1024];
242 if (fd < 0 && errno == EEXIST) {
243 #ifdef __OpenBSD__
244 snprintf(buf, sizeof(buf), "%lld_%s",
245 #else
246 snprintf(buf, sizeof(buf), "%ld_%s",
247 #endif
248 time(NULL), download);
249 fd = openat(getdownloadfd(), buf,
250 O_CREAT|O_EXCL|O_WRONLY, 0600);
251 download = buf;
252 }
253 if (fd < 0) {
254 tab->show_error = -1;
255 snprintf(tab->error, sizeof(tab->error),
256 "Failed to write file : %s", strerror(errno));
257 return 0;
258 }
259 #endif
260 char* data = strnstr(tab->page.data, "\r\n",
261 tab->page.data_len);
262 uint64_t data_len = tab->page.data_len;
263 if (!data) data = tab->page.data;
264 else {
265 data += 2;
266 data_len -= (data - tab->page.data);
267 }
268 #ifdef SANDBOX_SUN
269 sandbox_dl_length(data_len);
270 #endif
271 write(fd, data, data_len);
272 #ifndef SANDBOX_SUN
273 close(fd);
274 #endif
275 tab->show_info = 1;
276 snprintf(tab->info, sizeof(tab->info),
277 "File downloaded : %s", download);
278 STRLCPY(client.input.download, download);
279 client.input.field[0] = '\0';
280 tab->selected = 0;
281 return 0;
282 }
283 if (client.input.field[0] == ':' && (atoi(&client.input.field[1]) ||
284 (client.input.field[1] == '0' && client.input.field[2] == '\0'))) {
285 client.tab->scroll = atoi(&client.input.field[1]) - 1 -
286 !!client.tabs_count;
287 if (client.tab->scroll < 0)
288 client.tab->scroll = -!!client.tabs_count;
289 fix_scroll(client.tab);
290 client.input.field[0] = '\0';
291 tab->selected = 0;
292 return 0;
293 }
294 if (client.input.field[0] == '/') {
295 STRLCPY(tab->search.entry, &client.input.field[1]);
296 client.input.field[0] = '\0';
297 } else if (client.input.field[1] == '\0') client.input.field[0] = '\0';
298 else {
299 unknown:
300 tab->show_error = -1;
301 snprintf(tab->error, sizeof(tab->error),
302 "Unknown input: %s", &client.input.field[1]);
303 }
304 return 0;
305 }
306
307 int input_page(struct tb_event ev) {
308 struct gmi_tab* tab = client.tab;
309 struct gmi_page* page = &tab->page;
310 int counter;
311 switch (ev.key) {
312 case TB_KEY_ESC:
313 tab->selected = 0;
314 bzero(client.vim.counter, sizeof(client.vim.counter));
315 client.vim.g = 0;
316 return 0;
317 case TB_KEY_DELETE:
318 if (strcmp(tab->url, "about:home")) return 0;
319 if (tab->selected && tab->selected > 0 &&
320 tab->selected <= page->links_count) {
321 gmi_removebookmark(tab->selected);
322 tab->selected = 0;
323 gmi_request(tab, tab->url, 0);
324 }
325 return 0;
326 case TB_KEY_BACK_TAB:
327 if (tab->selected && tab->selected > 0 &&
328 tab->selected <= page->links_count) {
329 int linkid = tab->selected;
330 tab->selected = 0;
331 gmi_goto_new(tab, linkid);
332 }
333 return 0;
334 case TB_KEY_ARROW_LEFT:
335 if (ev.mod == TB_MOD_SHIFT)
336 goto go_back;
337 goto tab_prev;
338 case TB_KEY_ARROW_RIGHT:
339 if (ev.mod == TB_MOD_SHIFT)
340 goto go_forward;
341 goto tab_next;
342 case TB_KEY_ARROW_UP:
343 goto move_up;
344 case TB_KEY_ARROW_DOWN:
345 goto move_down;
346 case TB_KEY_TAB:
347 client.vim.g = 0;
348 if (client.vim.counter[0] == '\0' ||
349 !atoi(client.vim.counter)) {
350 if (!tab->selected)
351 return 0;
352 gmi_goto(tab, tab->selected);
353 tab->selected = 0;
354 bzero(client.vim.counter, sizeof(client.vim.counter));
355 return 0;
356 }
357 tab->selected = atoi(client.vim.counter);
358 bzero(client.vim.counter, sizeof(client.vim.counter));
359 if (tab->selected > page->links_count) {
360 snprintf(tab->error, sizeof(tab->error),
361 "Invalid link number");
362 tab->selected = 0;
363 tab->show_error = 1;
364 return 0;
365 }
366 size_t len = STRLCPY(tab->selected_url,
367 page->links[tab->selected - 1]);
368 if (len >= sizeof(tab->selected_url)) {
369 snprintf(tab->error, sizeof(tab->error),
370 "Invalid link, above %lu characters",
371 sizeof(tab->selected_url));
372 tab->selected = 0;
373 tab->show_error = 1;
374 }
375 return 0;
376 case TB_KEY_ENTER:
377 if (client.vim.counter[0] != '\0' &&
378 atoi(client.vim.counter)) {
379 tab->scroll += atoi(client.vim.counter);
380 bzero(client.vim.counter, sizeof(client.vim.counter));
381 }
382 else tab->scroll++;
383 fix_scroll(tab);
384 client.vim.g = 0;
385 return 0;
386 case TB_KEY_PGUP:
387 tab->scroll -= vim_counter() * tb_height() -
388 2 - (client.tabs_count>1);
389 fix_scroll(tab);
390 client.vim.g = 0;
391 return 0;
392 case TB_KEY_PGDN:
393 tab->scroll += vim_counter() * tb_height() -
394 2 - (client.tabs_count>1);
395 fix_scroll(tab);
396 client.vim.g = 0;
397 return 0;
398 }
399 switch (ev.ch) {
400 case 'u':
401 tab->show_error = 0;
402 client.input.mode = 1;
403 snprintf(client.input.field,
404 sizeof(client.input.field),
405 ":o %s", tab->url);
406 client.input.cursor = utf8_len(client.input.field,
407 sizeof(client.input.field));
408 break;
409 case ':':
410 tab->show_error = 0;
411 client.input.mode = 1;
412 client.input.cursor = 1;
413 client.input.field[0] = ':';
414 client.input.field[1] = '\0';
415 break;
416 case '/':
417 tab->show_error = 0;
418 client.input.mode = 1;
419 client.input.cursor = 1;
420 client.input.field[0] = '/';
421 client.input.field[1] = '\0';
422 break;
423 case 'r': // Reload
424 if (!tab->history) break;
425 gmi_request(tab, tab->history->url, 0);
426 break;
427 case 'T': // Tab left
428 tab_prev:
429 if (!client.vim.g) break;
430 client.vim.g = 0;
431 counter = vim_counter();
432 if (!client.tab->next && !client.tab->prev) break;
433 for (int i = counter; i > 0; i--) {
434 if (client.tab->prev)
435 client.tab = client.tab->prev;
436 else
437 while (client.tab->next)
438 client.tab = client.tab->next;
439 fix_scroll(client.tab);
440 }
441 break;
442 case 't': // Tab right
443 tab_next:
444 if (!client.vim.g) break;
445 client.vim.g = 0;
446 counter = vim_counter() % client.tabs_count;
447 if (!client.tab->next && !client.tab->prev) break;
448 for (int i = counter; i > 0; i--) {
449 if (client.tab->next)
450 client.tab = client.tab->next;
451 else
452 while (client.tab->prev)
453 client.tab = client.tab->prev;
454 fix_scroll(client.tab);
455 }
456 break;
457 case 'H': // Back
458 go_back:
459 if (!tab->history || !tab->history->prev) break;
460 tab->selected = 0;
461 if (page->code == 20 || page->code == 10 || page->code == 11) {
462 tab->history->scroll = tab->scroll;
463 if (!tab->history->next) {
464 tab->history->page = tab->page;
465 tab->history->cached = 1;
466 }
467 tab->history = tab->history->prev;
468 tab->scroll = tab->history->scroll;
469 if (tab->history->cached)
470 tab->page = tab->history->page;
471 }
472 if (tab->history->cached) {
473 tab->page = tab->history->page;
474 STRLCPY(tab->url, tab->history->url);
475 } else if (gmi_request(tab, tab->history->url, 1) < 0) break;
476 fix_scroll(client.tab);
477 break;
478 case 'L': // Forward
479 go_forward:
480 if (!tab->history || !tab->history->next) break;
481 tab->selected = 0;
482 if (!tab->history->cached) {
483 tab->history->page = tab->page;
484 tab->history->cached = 1;
485 }
486 if (tab->history->next->cached)
487 tab->page = tab->history->next->page;
488 else if (gmi_request(tab, tab->history->next->url, 1) < 0)
489 break;
490 tab->history->scroll = tab->scroll;
491 tab->history = tab->history->next;
492 tab->scroll = tab->history->scroll;
493 STRLCPY(tab->url, tab->history->url);
494 fix_scroll(client.tab);
495 break;
496 case 'k': // UP
497 move_up:
498 tab->scroll -= vim_counter();
499 fix_scroll(tab);
500 client.vim.g = 0;
501 break;
502 case 'j': // DOWN
503 move_down:
504 tab->scroll += vim_counter();
505 fix_scroll(tab);
506 client.vim.g = 0;
507 break;
508 case 'f': // Show history
509 display_history();
510 break;
511 case 'g': // Start of file
512 if (client.vim.g) {
513 tab->scroll = -1;
514 client.vim.g = 0;
515 } else client.vim.g++;
516 break;
517 case 'G': // End of file
518 tab->scroll = page->lines-tb_height()+2;
519 if (client.tabs_count != 1)
520 tab->scroll++;
521 if (tb_height()-2-(client.tabs_count>1) > page->lines)
522 tab->scroll = -1;
523 break;
524 case 'n': // Next occurence
525 tab->search.cursor++;
526 tab->scroll = tab->search.pos[1] - tb_height()/2;
527 fix_scroll(tab);
528 break;
529 case 'N': // Previous occurence
530 tab->search.cursor--;
531 tab->scroll = tab->search.pos[0] - tb_height()/2;
532 fix_scroll(tab);
533 break;
534 default:
535 if (!(ev.ch >= '0' && ev.ch <= '9'))
536 break;
537 tab->show_error = 0;
538 tab->show_info = 0;
539 unsigned int len = STRNLEN(client.vim.counter);
540 if (len == 0 && ev.ch == '0') break;
541 if (len >= sizeof(client.vim.counter)) break;
542 client.vim.counter[len] = ev.ch;
543 client.vim.counter[len+1] = '\0';
544 }
545 return 0;
546 }
547
548 int input_field(struct tb_event ev) {
549 struct gmi_tab* tab = client.tab;
550 struct gmi_page* page = &tab->page;
551 int pos = 0;
552 switch (ev.key) {
553 case TB_KEY_ESC:
554 client.input.mode = 0;
555 client.input.cursor = 0;
556 if (page->code == 11 || page->code == 10)
557 page->code = 20;
558 client.input.field[0] = '\0';
559 tb_hide_cursor();
560 return 0;
561 case TB_KEY_BACKSPACE:
562 case TB_KEY_BACKSPACE2:
563 if (client.input.cursor >
564 ((page->code == 10 || page->code == 11)?0:1)) {
565 if (client.input.field[0] == '/' &&
566 page->code == 20)
567 tab->search.scroll = 1;
568 char* ptr = client.input.field;
569 int l = 1;
570 int pos = 0;
571 while (*ptr) {
572 l = tb_utf8_char_length(*ptr);
573 pos++;
574 if (pos == client.input.cursor)
575 break;
576 ptr += l;
577 }
578
579 strlcpy(ptr, ptr + l,
580 sizeof(client.input.field) -
581 (ptr - client.input.field));
582 client.input.cursor--;
583 }
584 return 0;
585 case TB_KEY_ENTER:
586 client.input.mode = 0;
587 tb_hide_cursor();
588 if (page->code == 10 || page->code == 11) {
589 char urlbuf[MAX_URL];
590 char* start = strstr(tab->url, "gemini://");
591 char* request = strrchr(tab->url, '?');
592 if (!(start?
593 strchr(&start[GMI], '/'):strchr(tab->url, '/')))
594 snprintf(urlbuf, sizeof(urlbuf),
595 "%s/?%s", tab->url,
596 client.input.field);
597 else if (request && request > strrchr(tab->url, '/')) {
598 *request = '\0';
599 snprintf(urlbuf, sizeof(urlbuf),
600 "%s?%s", tab->url,
601 client.input.field);
602 *request = '?';
603 } else
604 snprintf(urlbuf, sizeof(urlbuf),
605 "%s?%s", tab->url,
606 client.input.field);
607 int bytes = gmi_request(tab, urlbuf, 1);
608 if (bytes>0) {
609 tab = client.tab;
610 tab->scroll = -1;
611 }
612 client.input.field[0] = '\0';
613 return 0;
614 }
615 return command();
616 case TB_KEY_ARROW_LEFT:
617 {
618 int min = 1;
619 if (tab->page.code == 10 || tab->page.code == 11)
620 min = 0;
621 if (ev.mod != TB_MOD_CTRL) {
622 if (client.input.cursor > min)
623 client.input.cursor--;
624 return 0;
625 }
626 pos = get_cursor_pos();
627 while (client.input.cursor > min) {
628 pos -= tb_utf8_char_length(client.input.field[pos]);
629 client.input.cursor--;
630 if (client.input.field[pos] == ' ' ||
631 client.input.field[pos] == '.')
632 break;
633 }
634 return 0;
635 }
636 case TB_KEY_ARROW_RIGHT:
637 pos = get_cursor_pos();
638 if (ev.mod != TB_MOD_CTRL) {
639 if (client.input.field[pos])
640 client.input.cursor++;
641 return 0;
642 }
643 while (client.input.field[pos]) {
644 pos += tb_utf8_char_length(client.input.field[pos]);
645 client.input.cursor++;
646 if (!client.input.field[pos]) break;
647 if (client.input.field[pos] == ' ' ||
648 client.input.field[pos] == '.')
649 break;
650 }
651 return 0;
652 }
653 if (!ev.ch) {
654 tb_hide_cursor();
655 return 0;
656 }
657 char* end = client.input.field;
658 while (*end)
659 end += tb_utf8_char_length(*end);
660 if ((size_t)(end - client.input.field) >=
661 sizeof(client.input.field) - 1)
662 return 0;
663
664 char insert[16];
665 int insert_len = tb_utf8_unicode_to_char(insert, ev.ch);
666 char* start = client.input.field;
667 for (int i = 0; *start && i < client.input.cursor; i++)
668 start += tb_utf8_char_length(*start);
669 for (char* ptr = end; start <= ptr; ptr--)
670 ptr[insert_len] = *ptr;
671
672 memcpy(start, insert, insert_len);
673 client.input.cursor++;
674 end += insert_len;
675 *end = '\0';
676 if (client.input.field[0] != '/' || page->code != 20)
677 return 0;
678 int lines = 0;
679 int posx = 0;
680 int w = tb_width();
681 int found = 0;
682 for (int i = 0; i < tab->page.data_len - 1; i++) {
683 if (posx == 0 && tab->page.data[i] == '=' &&
684 tab->page.data[i + 1] == '>') {
685 int ignore = 0;
686 int firstspace = 0;
687 while (i + ignore < tab->page.data_len) {
688 if (tab->page.data[i + ignore] != '\n') {
689 i += 2 + firstspace;
690 break;
691 }
692 if (tab->page.data[i + ignore] == ' ') {
693 ignore++;
694 if (firstspace) continue;
695 i += ignore;
696 break;
697 }
698 ignore++;
699 firstspace = 1;
700 }
701 if (i > tab->page.data_len)
702 break;
703 }
704 if (lines &&
705 !strncasecmp(&client.input.field[1],
706 &tab->page.data[i],
707 strnlen(&client.input.field[1],
708 sizeof(client.input.field) - 1))) {
709 found = 1;
710 break;
711 }
712 if (posx == w || tab->page.data[i] == '\n') {
713 lines++;
714 posx = 0;
715 continue;
716 }
717 posx++;
718 }
719 if (found) {
720 tab->scroll = lines - tb_height()/2;
721 fix_scroll(tab);
722 }
723 return 0;
724 }
725
726 int input(struct tb_event ev) {
727 struct gmi_tab* tab = client.tab;
728 struct gmi_page* page = &tab->page;
729 if (page->code == 11 || page->code == 10) {
730 if (!client.input.mode) client.input.cursor = 0;
731 client.input.mode = 1;
732 }
733 if (ev.type == TB_EVENT_RESIZE) {
734 fix_scroll(tab);
735 return 0;
736 }
737 if (ev.type != TB_EVENT_KEY) return 0;
738 if (client.input.mode)
739 return input_field(ev);
740 return input_page(ev);
741 }
742
743 int tb_interupt() {
744 int sig = 0;
745 return write(global.resize_pipefd[1], &sig, sizeof(sig)) ==
746 sizeof(sig) ? 0 : -1;
747 }
748