0 /*
1 * ISC License
2 * Copyright (c) 2023 RMF <rawmonk@firemail.cc>
3 */
4 #include <stdio.h>
5 #include <stdlib.h>
6 #include <string.h>
7 #include <stdint.h>
8 #include <time.h>
9 #include <errno.h>
10 #include "macro.h"
11 #include "error.h"
12 #include "config.h"
13 #include "gemini.h"
14 #include "page.h"
15 #include "request.h"
16 #include "tab.h"
17 #include "client.h"
18 #include "termbox.h"
19 #include "utf8.h"
20 #include "input.h"
21 #include "strlcpy.h"
22 #include "known_hosts.h"
23 #include "storage.h"
24 #include "sandbox.h"
25 #include "parser.h"
26 #include "bookmarks.h"
27 #include "image.h"
28 #include "history.h"
29 #include "xdg.h"
30 #include "url.h"
31
32 int client_destroy(struct client *client) {
33 struct tab *tab;
34 if (!client) return ERROR_NULL_ARGUMENT;
35 for (tab = client->tab; tab && tab->prev; tab = tab->prev) ;
36 while (tab) {
37 struct tab *next = tab->next;
38 tab_free(tab);
39 tab = next;
40 }
41 known_hosts_free();
42 history_save();
43 history_free();
44 config_save();
45 free(bookmarks);
46 if (tb_shutdown()) return ERROR_TERMBOX_FAILURE;
47 return 0;
48 }
49
50 int client_closetab(struct client *client) {
51 struct tab *tab, *next, *prev;
52 if (!client->tab) return 1;
53 tab = client->tab;
54 next = tab->next;
55 prev = tab->prev;
56 if (next) next->prev = prev;
57 if (prev) prev->next = next;
58 if (next) {
59 client->tab = next;
60 } else if (prev) {
61 client->tab = prev;
62 } else client->tab = NULL;
63 tab_free(tab);
64 return 0;
65 }
66
67 int client_newtab(struct client *client, const char *url) {
68 struct tab *tab;
69 int proto;
70 if (!url) url = "about:newtab";
71 proto = protocol_from_url(url);
72 if (proto != PROTOCOL_GEMINI && proto != PROTOCOL_NONE) {
73 return tab_request(client->tab, url);
74 }
75 tab = tab_new();
76 if (!tab) return ERROR_MEMORY_FAILURE;
77 if (client->tab) {
78 tab->next = client->tab->next;
79 if (tab->next) tab->next->prev = tab;
80 tab->prev = client->tab;
81 client->tab->next = tab;
82 }
83 client->tab = tab;
84 return tab_request(client->tab, url);
85 }
86
87 int client_input(struct client *client) {
88
89 struct tb_event ev;
90 struct request *req = NULL;
91 int ret = 0;
92
93 if (!client->tab || !client->tab->request ||
94 client->tab->request->state != STATE_ONGOING) {
95 #ifdef FUZZING_MODE
96 ret = tb_peek_event(&ev, 50);
97 if (ret == TB_ERR_NO_EVENT) {
98 ret = TB_OK;
99 ev.ch = 'r';
100 }
101 #else
102 ret = tb_poll_event(&ev);
103 #endif
104 } else {
105 ret = tb_peek_event(&ev, 100);
106 }
107
108 if (ret == TB_ERR_NO_EVENT) return 0;
109 if (ret == TB_ERR_POLL) ret = TB_OK;
110 if (ret != TB_OK) return ERROR_TERMBOX_FAILURE;
111
112 if (client->tab)
113 req = tab_input(client->tab);
114
115 if (req && gemini_isinput(req->status) && client->mode == MODE_NORMAL) {
116 client_enter_mode_cmdline(client);
117 client->cursor = snprintf(V(client->cmd), "%s: ", req->meta);
118 }
119
120 switch (client->mode) {
121 case MODE_NORMAL:
122 return client_input_normal(client, ev);
123 case MODE_CMDLINE:
124 {
125 int ret = client_input_cmdline(client, ev);
126 if (client->mode == MODE_NORMAL)
127 tb_hide_cursor();
128 return ret;
129 }
130 }
131
132 return 0;
133 }
134
135 int client_init(struct client* client) {
136
137 int ret;
138
139 memset(client, 0, sizeof(*client));
140 if ((ret = storage_init())) return ret;
141 config_load();
142 if ((ret = parser_request_create())) return ret;
143 if ((ret = parser_page_create())) return ret;
144 #ifdef ENABLE_IMAGE
145 if ((ret = image_init())) return ret;
146 #endif
147 #ifndef DISABLE_XDG
148 if (xdg_available()) if ((ret = xdg_init())) return ret;
149 #endif
150 if ((ret = known_hosts_load())) return ret;
151 if ((ret = bookmark_load())) return ret;
152 if (tb_init()) return ERROR_TERMBOX_FAILURE;
153 #ifdef ENABLE_IMAGE
154 if (tb_set_output_mode(TB_OUTPUT_256)) return ERROR_TERMBOX_FAILURE;
155 #endif
156 if ((ret = sandbox_init())) {
157 tb_shutdown();
158 return ret;
159 }
160 history_init();
161
162 return 0;
163 }
164