Blob


1 /*
2 * Copyright (c) 2021 Omar Polo <op@omarpolo.com>
3 *
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
17 #ifndef TELESCOPE_H
18 #define TELESCOPE_H
20 #include "compat.h"
22 #include <event.h>
24 #include "url.h"
26 #define MIN(a, b) ((a) < (b) ? (a) : (b))
27 #define MAX(a, b) ((a) > (b) ? (a) : (b))
29 #define GEMINI_URL_LEN 1024
31 enum imsg_type {
32 /* ui <-> client/fs */
33 IMSG_GET, /* data is URL, peerid the tab id */
34 IMSG_ERR,
35 IMSG_CHECK_CERT,
36 IMSG_CERT_STATUS,
37 IMSG_GOT_CODE,
38 IMSG_GOT_META,
39 IMSG_PROCEED,
40 IMSG_STOP,
41 IMSG_BUF,
42 IMSG_EOF,
43 IMSG_QUIT,
45 /* ui <-> fs */
46 IMSG_BOOKMARK_PAGE,
47 IMSG_BOOKMARK_OK,
48 };
50 enum line_type {
51 LINE_TEXT,
52 LINE_LINK,
53 LINE_TITLE_1,
54 LINE_TITLE_2,
55 LINE_TITLE_3,
56 LINE_ITEM,
57 LINE_QUOTE,
58 LINE_PRE_START,
59 LINE_PRE_CONTENT,
60 LINE_PRE_END,
61 };
63 struct line {
64 enum line_type type;
65 char *line;
66 char *alt;
67 int flags;
68 TAILQ_ENTRY(line) lines;
69 };
71 struct vline {
72 const struct line *parent;
73 char *line;
74 int flags;
75 TAILQ_ENTRY(vline) vlines;
76 };
78 struct parser;
79 struct page;
81 /* typedef void (*initparserfn)(struct parser*); */
83 typedef int (*parsechunkfn)(struct parser*, const char*, size_t);
84 typedef int (*parserfreefn)(struct parser*);
86 typedef void (imsg_handlerfn)(struct imsg*, size_t);
88 struct parser {
89 const char *name;
90 char title[32+1];
91 char *buf;
92 size_t len;
93 size_t cap;
94 int flags;
95 parsechunkfn parse;
96 parserfreefn free;
98 TAILQ_HEAD(, line) head;
99 };
101 struct histhead {
102 TAILQ_HEAD(mhisthead, hist) head;
103 size_t len;
104 };
105 struct hist {
106 char h[1025];
107 TAILQ_ENTRY(hist) entries;
108 };
110 struct ui_state {
111 int curs_x;
112 int curs_y;
113 size_t line_off;
114 size_t line_max;
115 struct vline *current_line;
116 size_t line_x;
118 short loading_anim;
119 short loading_anim_step;
120 struct event loadingev;
122 TAILQ_HEAD(vhead, vline) head;
123 };
125 /*
126 * differnt types of trust for a certificate. Following
127 * gemini://thfr.info/gemini/modified-trust-verify.gmi
128 */
129 enum trust_state {
130 TS_UNKNOWN,
131 TS_UNTRUSTED,
132 TS_TRUSTED,
133 TS_VERIFIED,
134 };
136 extern TAILQ_HEAD(tabshead, tab) tabshead;
137 struct tab {
138 struct parser page;
139 TAILQ_ENTRY(tab) tabs;
140 uint32_t id;
141 uint32_t flags;
143 enum trust_state trust;
144 struct url url;
145 struct histhead hist;
146 struct hist *hist_cur;
147 size_t hist_off;
149 int code;
150 char meta[GEMINI_URL_LEN];
151 int redirect_count;
153 struct ui_state s;
154 };
156 struct proto {
157 const char *schema;
159 /* should load the given url in the tab. Optionally, it may
160 * consider the given url as relative to the one already
161 * present in tab. It must set tab->urlstr to a serialized
162 * human-friendly URL. */
163 void (*loadfn)(struct tab*, const char*);
164 };
166 struct kmap {
167 TAILQ_HEAD(map, keymap) m;
168 void (*unhandled_input)(void);
169 };
171 struct keymap {
172 int meta;
173 int key;
174 struct kmap map;
175 void (*fn)(struct tab*);
177 TAILQ_ENTRY(keymap) keymaps;
178 };
180 /* fs.c */
181 int fs_main(struct imsgbuf*);
183 /* gemini.c */
184 int client_main(struct imsgbuf*);
186 /* gemtext.c */
187 void gemtext_initparser(struct parser*);
189 /* hist.c */
190 void hist_clear_forward(struct histhead*, struct hist*);
191 void hist_push(struct histhead*, struct hist*);
193 /* keymap.c */
194 int kbd(const char*);
195 const char *unkbd(int);
196 int kmap_define_key(struct kmap*, const char*, void(*)(struct tab*));
198 /* mime.c */
199 int setup_parser_for(struct tab*);
201 /* pages.c */
202 extern const char *about_new;
204 #define CANNOT_FETCH 0
205 #define TOO_MUCH_REDIRECTS 1
206 #define MALFORMED_RESPONSE 2
207 #define UNKNOWN_TYPE_OR_CSET 3
208 extern const char *err_pages[70];
210 /* parser.c */
211 int parser_append(struct parser*, const char*, size_t);
212 int parser_set_buf(struct parser*, const char*, size_t);
214 /* sandbox.c */
215 void sandbox_network_process(void);
216 void sandbox_ui_process(void);
217 void sandbox_fs_process(void);
219 /* telescope.c */
220 void load_about_url(struct tab*, const char*);
221 void load_gemini_url(struct tab*, const char*);
222 void load_url(struct tab*, const char*);
223 int load_previous_page(struct tab*);
224 int load_next_page(struct tab*);
225 void stop_tab(struct tab*);
226 void add_to_bookmarks(const char*);
228 /* textplain.c */
229 void textplain_initparser(struct parser*);
231 /* ui.c */
232 int ui_init(void);
233 void ui_on_tab_loaded(struct tab*);
234 void ui_on_tab_refresh(struct tab*);
235 void ui_require_input(struct tab*, int);
236 void ui_notify(const char*, ...) __attribute__((format(printf, 1, 2)));
237 void ui_end(void);
239 /* util.c */
240 int mark_nonblock(int);
241 int has_prefix(const char*, const char*);
242 void dispatch_imsg(struct imsgbuf*, imsg_handlerfn**, size_t);
244 /* wrap.c */
245 void wrap_text(struct tab*, const char*, struct line*, size_t);
246 int hardwrap_text(struct tab*, struct line*, size_t);
248 #endif /* TELESCOPE_H */