Blame


1 5e11c00c 2021-03-02 op /*
2 5e11c00c 2021-03-02 op * Copyright (c) 2021 Omar Polo <op@omarpolo.com>
3 5e11c00c 2021-03-02 op *
4 5e11c00c 2021-03-02 op * Permission to use, copy, modify, and distribute this software for any
5 5e11c00c 2021-03-02 op * purpose with or without fee is hereby granted, provided that the above
6 5e11c00c 2021-03-02 op * copyright notice and this permission notice appear in all copies.
7 5e11c00c 2021-03-02 op *
8 5e11c00c 2021-03-02 op * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 5e11c00c 2021-03-02 op * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 5e11c00c 2021-03-02 op * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 5e11c00c 2021-03-02 op * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 5e11c00c 2021-03-02 op * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 5e11c00c 2021-03-02 op * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 5e11c00c 2021-03-02 op * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 5e11c00c 2021-03-02 op */
16 5e11c00c 2021-03-02 op
17 1d08c280 2021-03-06 op /*
18 1d08c280 2021-03-06 op * Ncurses UI for telescope.
19 1d08c280 2021-03-06 op *
20 1d08c280 2021-03-06 op * Text scrolling
21 1d08c280 2021-03-06 op * ==============
22 1d08c280 2021-03-06 op *
23 1d08c280 2021-03-06 op * ncurses allows you to scroll a window, but when a line goes out of
24 1d08c280 2021-03-06 op * the visible area it's forgotten. We keep a list of formatted lines
25 1d08c280 2021-03-06 op * (``visual lines'') that we know fits in the window, and draw them.
26 1d08c280 2021-03-06 op * This way is easy to scroll: just call wscrl and then render the
27 1d08c280 2021-03-06 op * first/last line!
28 1d08c280 2021-03-06 op *
29 1d08c280 2021-03-06 op * This means that on every resize we have to clear our list of lines
30 1d08c280 2021-03-06 op * and re-render everything. A clever approach would be to do this
31 970deec6 2021-04-24 op * ``on-demand'', but it's still missing.
32 1d08c280 2021-03-06 op *
33 1d08c280 2021-03-06 op */
34 1d08c280 2021-03-06 op
35 e2226342 2021-05-12 op #include "telescope.h"
36 09312eb3 2021-05-14 op #include "cmd.gen.h"
37 5e11c00c 2021-03-02 op
38 bddc7bbd 2021-04-01 op #include <assert.h>
39 5e11c00c 2021-03-02 op #include <curses.h>
40 5e11c00c 2021-03-02 op #include <event.h>
41 5e11c00c 2021-03-02 op #include <locale.h>
42 5e11c00c 2021-03-02 op #include <signal.h>
43 7953dd72 2021-03-07 op #include <stdarg.h>
44 eb259e66 2021-03-02 op #include <stdlib.h>
45 eb259e66 2021-03-02 op #include <string.h>
46 f832146f 2021-03-09 op #include <unistd.h>
47 5e11c00c 2021-03-02 op
48 5e11c00c 2021-03-02 op #define TAB_CURRENT 0x1
49 e8a76665 2021-05-12 op #define TAB_URGENT 0x2
50 5e11c00c 2021-03-02 op
51 1b81bc33 2021-03-15 op #define NEW_TAB_URL "about:new"
52 1b81bc33 2021-03-15 op
53 5e11c00c 2021-03-02 op static struct event stdioev, winchev;
54 5e11c00c 2021-03-02 op
55 f832146f 2021-03-09 op static void load_default_keys(void);
56 2ba66cea 2021-03-22 op static void restore_cursor(struct window*);
57 870210fb 2021-03-26 op
58 870210fb 2021-03-26 op static void global_key_unbound(void);
59 22268e11 2021-03-11 op static void minibuffer_hist_save_entry(void);
60 22268e11 2021-03-11 op static void minibuffer_taint_hist(void);
61 5cd2ebb1 2021-03-11 op static void minibuffer_self_insert(void);
62 9ca15951 2021-03-09 op static void eecmd_self_insert(void);
63 b360ebb3 2021-03-10 op static void eecmd_select(void);
64 5cd2ebb1 2021-03-11 op static void ir_self_insert(void);
65 5cd2ebb1 2021-03-11 op static void ir_select(void);
66 5cd2ebb1 2021-03-11 op static void lu_self_insert(void);
67 5cd2ebb1 2021-03-11 op static void lu_select(void);
68 740f578b 2021-03-15 op static void bp_select(void);
69 5d1bac73 2021-03-25 op static void yornp_self_insert(void);
70 5d1bac73 2021-03-25 op static void yornp_abort(void);
71 9ca15951 2021-03-09 op
72 2ba66cea 2021-03-22 op static struct vline *nth_line(struct window*, size_t);
73 5e11c00c 2021-03-02 op static struct tab *current_tab(void);
74 2ba66cea 2021-03-22 op static struct window *current_window(void);
75 8947c1f2 2021-03-21 op static int readkey(void);
76 5e11c00c 2021-03-02 op static void dispatch_stdio(int, short, void*);
77 a6d450c1 2021-03-06 op static void handle_clear_minibuf(int, short, void*);
78 5e11c00c 2021-03-02 op static void handle_resize(int, short, void*);
79 831deb20 2021-05-12 op static void handle_resize_timeout(int, short, void*);
80 bddc7bbd 2021-04-01 op static int wrap_page(struct window*, int);
81 bddc7bbd 2021-04-01 op static void print_vline(WINDOW*, struct vline*);
82 8af5e5ed 2021-03-08 op static void redraw_tabline(void);
83 bddc7bbd 2021-04-01 op static void redraw_window(WINDOW*, int, struct window*);
84 bddc7bbd 2021-04-01 op static void redraw_help(void);
85 e19f9a04 2021-03-11 op static void redraw_body(struct tab*);
86 8af5e5ed 2021-03-08 op static void redraw_modeline(struct tab*);
87 9ca15951 2021-03-09 op static void redraw_minibuffer(void);
88 5e11c00c 2021-03-02 op static void redraw_tab(struct tab*);
89 bddc7bbd 2021-04-01 op static void emit_help_item(char*, void*);
90 bddc7bbd 2021-04-01 op static void rec_compute_help(struct kmap*, char*, size_t);
91 bddc7bbd 2021-04-01 op static void recompute_help(void);
92 740f578b 2021-03-15 op static void vmessage(const char*, va_list);
93 7953dd72 2021-03-07 op static void message(const char*, ...) __attribute__((format(printf, 1, 2)));
94 8af5e5ed 2021-03-08 op static void start_loading_anim(struct tab*);
95 8af5e5ed 2021-03-08 op static void update_loading_anim(int, short, void*);
96 8af5e5ed 2021-03-08 op static void stop_loading_anim(struct tab*);
97 43a1b8d0 2021-03-09 op static void load_url_in_tab(struct tab*, const char*);
98 3148eeac 2021-03-13 op static void enter_minibuffer(void(*)(void), void(*)(void), void(*)(void), struct histhead*);
99 b360ebb3 2021-03-10 op static void exit_minibuffer(void);
100 5cd2ebb1 2021-03-11 op static void switch_to_tab(struct tab*);
101 1b81bc33 2021-03-15 op static struct tab *new_tab(const char*);
102 c7107cec 2021-04-01 op static void session_new_tab_cb(const char*);
103 941b3761 2021-03-18 op static void usage(void);
104 5e11c00c 2021-03-02 op
105 8947c1f2 2021-03-21 op static struct { short meta; int key; uint32_t cp; } thiskey;
106 9ca15951 2021-03-09 op
107 831deb20 2021-05-12 op static struct event resizeev;
108 831deb20 2021-05-12 op static struct timeval resize_timer = { 0, 250000 };
109 831deb20 2021-05-12 op
110 48e9d457 2021-03-06 op static WINDOW *tabline, *body, *modeline, *minibuf;
111 48e9d457 2021-03-06 op static int body_lines, body_cols;
112 48e9d457 2021-03-06 op
113 bddc7bbd 2021-04-01 op static WINDOW *help;
114 bddc7bbd 2021-04-01 op static struct window helpwin;
115 bddc7bbd 2021-04-01 op static int help_lines, help_cols;
116 bddc7bbd 2021-04-01 op
117 bddc7bbd 2021-04-01 op static int side_window;
118 bddc7bbd 2021-04-01 op
119 48e9d457 2021-03-06 op static struct event clminibufev;
120 a6d450c1 2021-03-06 op static struct timeval clminibufev_timer = { 5, 0 };
121 8af5e5ed 2021-03-08 op static struct timeval loadingev_timer = { 0, 250000 };
122 48e9d457 2021-03-06 op
123 bcb0b073 2021-03-07 op static uint32_t tab_counter;
124 bcb0b073 2021-03-07 op
125 7c7d7bb7 2021-03-10 op static char keybuf[64];
126 9ca15951 2021-03-09 op
127 b3575139 2021-04-01 op static void (*yornp_cb)(int, unsigned int);
128 b3575139 2021-04-01 op static unsigned int yornp_data;
129 5d1bac73 2021-03-25 op
130 9ca15951 2021-03-09 op struct kmap global_map,
131 fa3fd864 2021-03-10 op minibuffer_map,
132 9ca15951 2021-03-09 op *current_map,
133 9ca15951 2021-03-09 op *base_map;
134 9ca15951 2021-03-09 op
135 3148eeac 2021-03-13 op static struct histhead eecmd_history,
136 22268e11 2021-03-11 op ir_history,
137 22268e11 2021-03-11 op lu_history;
138 22268e11 2021-03-11 op
139 9ca15951 2021-03-09 op static int in_minibuffer;
140 9ca15951 2021-03-09 op
141 9ca15951 2021-03-09 op static struct {
142 2ba66cea 2021-03-22 op char *curmesg;
143 9cb0f9ce 2021-03-10 op
144 5d1bac73 2021-03-25 op char prompt[64];
145 2ba66cea 2021-03-22 op void (*donefn)(void);
146 2ba66cea 2021-03-22 op void (*abortfn)(void);
147 22268e11 2021-03-11 op
148 2ba66cea 2021-03-22 op char buf[1025];
149 2ba66cea 2021-03-22 op struct line line;
150 2ba66cea 2021-03-22 op struct vline vline;
151 2ba66cea 2021-03-22 op struct window window;
152 2ba66cea 2021-03-22 op
153 3148eeac 2021-03-13 op struct histhead *history;
154 3148eeac 2021-03-13 op struct hist *hist_cur;
155 17e5106b 2021-03-21 op size_t hist_off;
156 9ca15951 2021-03-09 op } ministate;
157 9ca15951 2021-03-09 op
158 f95d82dd 2021-03-11 op struct lineprefix {
159 f95d82dd 2021-03-11 op const char *prfx1;
160 f95d82dd 2021-03-11 op const char *prfx2;
161 f95d82dd 2021-03-11 op } line_prefixes[] = {
162 f95d82dd 2021-03-11 op [LINE_TEXT] = { "", "" },
163 f95d82dd 2021-03-11 op [LINE_LINK] = { "=> ", " " },
164 f95d82dd 2021-03-11 op [LINE_TITLE_1] = { "# ", " " },
165 f95d82dd 2021-03-11 op [LINE_TITLE_2] = { "## ", " " },
166 f95d82dd 2021-03-11 op [LINE_TITLE_3] = { "### ", " " },
167 f95d82dd 2021-03-11 op [LINE_ITEM] = { "* ", " " },
168 71d5deec 2021-03-19 op [LINE_QUOTE] = { "> ", " " },
169 803ff456 2021-03-17 op [LINE_PRE_START] = { "```", " " },
170 f95d82dd 2021-03-11 op [LINE_PRE_CONTENT] = { "", "" },
171 f95d82dd 2021-03-11 op [LINE_PRE_END] = { "```", "```" },
172 d3d56d8a 2021-03-11 op };
173 d3d56d8a 2021-03-11 op
174 31840026 2021-03-16 op static struct line_face {
175 803ff456 2021-03-17 op int prefix_prop;
176 803ff456 2021-03-17 op int text_prop;
177 d3d56d8a 2021-03-11 op } line_faces[] = {
178 803ff456 2021-03-17 op [LINE_TEXT] = { 0, 0 },
179 803ff456 2021-03-17 op [LINE_LINK] = { 0, A_UNDERLINE },
180 803ff456 2021-03-17 op [LINE_TITLE_1] = { A_BOLD, A_BOLD },
181 803ff456 2021-03-17 op [LINE_TITLE_2] = { A_BOLD, A_BOLD },
182 803ff456 2021-03-17 op [LINE_TITLE_3] = { A_BOLD, A_BOLD },
183 803ff456 2021-03-17 op [LINE_ITEM] = { 0, 0 },
184 803ff456 2021-03-17 op [LINE_QUOTE] = { 0, A_DIM },
185 803ff456 2021-03-17 op [LINE_PRE_START] = { 0, 0 },
186 803ff456 2021-03-17 op [LINE_PRE_CONTENT] = { 0, 0 },
187 803ff456 2021-03-17 op [LINE_PRE_END] = { 0, 0 },
188 cb6c7aa0 2021-03-16 op };
189 cb6c7aa0 2021-03-16 op
190 cb6c7aa0 2021-03-16 op static struct tab_face {
191 cb6c7aa0 2021-03-16 op int background, tab, current_tab;
192 cb6c7aa0 2021-03-16 op } tab_face = {
193 cb6c7aa0 2021-03-16 op A_REVERSE, A_REVERSE, A_NORMAL
194 f95d82dd 2021-03-11 op };
195 65d9b3ca 2021-03-14 op
196 f832146f 2021-03-09 op static inline void
197 2ba66cea 2021-03-22 op global_set_key(const char *key, void (*fn)(struct window*))
198 f832146f 2021-03-09 op {
199 67c8ed7f 2021-03-13 op if (!kmap_define_key(&global_map, key, fn))
200 67c8ed7f 2021-03-13 op _exit(1);
201 f832146f 2021-03-09 op }
202 f832146f 2021-03-09 op
203 9ca15951 2021-03-09 op static inline void
204 2ba66cea 2021-03-22 op minibuffer_set_key(const char *key, void (*fn)(struct window*))
205 9ca15951 2021-03-09 op {
206 67c8ed7f 2021-03-13 op if (!kmap_define_key(&minibuffer_map, key, fn))
207 67c8ed7f 2021-03-13 op _exit(1);
208 9ca15951 2021-03-09 op }
209 9ca15951 2021-03-09 op
210 f832146f 2021-03-09 op static void
211 f832146f 2021-03-09 op load_default_keys(void)
212 f832146f 2021-03-09 op {
213 9ca15951 2021-03-09 op /* === global map === */
214 9ca15951 2021-03-09 op
215 f832146f 2021-03-09 op /* emacs */
216 f832146f 2021-03-09 op global_set_key("C-p", cmd_previous_line);
217 f832146f 2021-03-09 op global_set_key("C-n", cmd_next_line);
218 f832146f 2021-03-09 op global_set_key("C-f", cmd_forward_char);
219 f832146f 2021-03-09 op global_set_key("C-b", cmd_backward_char);
220 852d03e8 2021-03-17 op global_set_key("M-{", cmd_backward_paragraph);
221 852d03e8 2021-03-17 op global_set_key("M-}", cmd_forward_paragraph);
222 a329982b 2021-03-11 op global_set_key("C-a", cmd_move_beginning_of_line);
223 a329982b 2021-03-11 op global_set_key("C-e", cmd_move_end_of_line);
224 f832146f 2021-03-09 op
225 f832146f 2021-03-09 op global_set_key("M-v", cmd_scroll_up);
226 f832146f 2021-03-09 op global_set_key("C-v", cmd_scroll_down);
227 929c2d9f 2021-03-13 op global_set_key("M-space", cmd_scroll_up);
228 929c2d9f 2021-03-13 op global_set_key("space", cmd_scroll_down);
229 da191746 2021-03-28 op
230 da191746 2021-03-28 op global_set_key("M-<", cmd_beginning_of_buffer);
231 da191746 2021-03-28 op global_set_key("M->", cmd_end_of_buffer);
232 f832146f 2021-03-09 op
233 f832146f 2021-03-09 op global_set_key("C-x C-c", cmd_kill_telescope);
234 f832146f 2021-03-09 op
235 3b4f9e49 2021-03-11 op global_set_key("C-g", cmd_clear_minibuf);
236 3b4f9e49 2021-03-11 op
237 9ca15951 2021-03-09 op global_set_key("M-x", cmd_execute_extended_command);
238 5cd2ebb1 2021-03-11 op global_set_key("C-x C-f", cmd_load_url);
239 5cd2ebb1 2021-03-11 op global_set_key("C-x M-f", cmd_load_current_url);
240 9ca15951 2021-03-09 op
241 7c7d7bb7 2021-03-10 op global_set_key("C-x t 0", cmd_tab_close);
242 fea845b6 2021-03-15 op global_set_key("C-x t 1", cmd_tab_close_other);
243 7c7d7bb7 2021-03-10 op global_set_key("C-x t 2", cmd_tab_new);
244 7c7d7bb7 2021-03-10 op global_set_key("C-x t o", cmd_tab_next);
245 7c7d7bb7 2021-03-10 op global_set_key("C-x t O", cmd_tab_previous);
246 fad72201 2021-03-28 op global_set_key("C-x t m", cmd_tab_move);
247 fad72201 2021-03-28 op global_set_key("C-x t M", cmd_tab_move_to);
248 e19f9a04 2021-03-11 op
249 2051e653 2021-03-13 op global_set_key("C-M-b", cmd_previous_page);
250 2051e653 2021-03-13 op global_set_key("C-M-f", cmd_next_page);
251 2051e653 2021-03-13 op
252 740f578b 2021-03-15 op global_set_key("<f7> a", cmd_bookmark_page);
253 63875195 2021-04-01 op global_set_key("<f7> <f7>", cmd_list_bookmarks);
254 740f578b 2021-03-15 op
255 f832146f 2021-03-09 op /* vi/vi-like */
256 f832146f 2021-03-09 op global_set_key("k", cmd_previous_line);
257 f832146f 2021-03-09 op global_set_key("j", cmd_next_line);
258 f832146f 2021-03-09 op global_set_key("l", cmd_forward_char);
259 f832146f 2021-03-09 op global_set_key("h", cmd_backward_char);
260 852d03e8 2021-03-17 op global_set_key("{", cmd_backward_paragraph);
261 852d03e8 2021-03-17 op global_set_key("}", cmd_forward_paragraph);
262 a329982b 2021-03-11 op global_set_key("^", cmd_move_beginning_of_line);
263 a329982b 2021-03-11 op global_set_key("$", cmd_move_end_of_line);
264 f832146f 2021-03-09 op
265 ed44414d 2021-03-09 op global_set_key("K", cmd_scroll_line_up);
266 ed44414d 2021-03-09 op global_set_key("J", cmd_scroll_line_down);
267 da191746 2021-03-28 op
268 da191746 2021-03-28 op global_set_key("g g", cmd_beginning_of_buffer);
269 da191746 2021-03-28 op global_set_key("G", cmd_end_of_buffer);
270 f832146f 2021-03-09 op
271 a02335e9 2021-03-15 op global_set_key("g D", cmd_tab_close);
272 a02335e9 2021-03-15 op global_set_key("g N", cmd_tab_new);
273 a02335e9 2021-03-15 op global_set_key("g t", cmd_tab_next);
274 a02335e9 2021-03-15 op global_set_key("g T", cmd_tab_previous);
275 fad72201 2021-03-28 op global_set_key("g M-t", cmd_tab_move);
276 fad72201 2021-03-28 op global_set_key("g M-T", cmd_tab_move_to);
277 a02335e9 2021-03-15 op
278 2051e653 2021-03-13 op global_set_key("H", cmd_previous_page);
279 2051e653 2021-03-13 op global_set_key("L", cmd_next_page);
280 2051e653 2021-03-13 op
281 f832146f 2021-03-09 op /* tmp */
282 f832146f 2021-03-09 op global_set_key("q", cmd_kill_telescope);
283 f832146f 2021-03-09 op
284 3b4f9e49 2021-03-11 op global_set_key("esc", cmd_clear_minibuf);
285 3b4f9e49 2021-03-11 op
286 9ca15951 2021-03-09 op global_set_key(":", cmd_execute_extended_command);
287 9ca15951 2021-03-09 op
288 f832146f 2021-03-09 op /* cua */
289 f832146f 2021-03-09 op global_set_key("<up>", cmd_previous_line);
290 f832146f 2021-03-09 op global_set_key("<down>", cmd_next_line);
291 f832146f 2021-03-09 op global_set_key("<right>", cmd_forward_char);
292 f832146f 2021-03-09 op global_set_key("<left>", cmd_backward_char);
293 ed44414d 2021-03-09 op global_set_key("<prior>", cmd_scroll_up);
294 ed44414d 2021-03-09 op global_set_key("<next>", cmd_scroll_down);
295 f832146f 2021-03-09 op
296 2051e653 2021-03-13 op global_set_key("M-<left>", cmd_previous_page);
297 2051e653 2021-03-13 op global_set_key("M-<right>", cmd_next_page);
298 2051e653 2021-03-13 op
299 f832146f 2021-03-09 op /* "ncurses standard" */
300 f832146f 2021-03-09 op global_set_key("C-l", cmd_redraw);
301 f832146f 2021-03-09 op
302 f832146f 2021-03-09 op /* global */
303 bddc7bbd 2021-04-01 op global_set_key("<f1>", cmd_toggle_help);
304 f832146f 2021-03-09 op global_set_key("C-m", cmd_push_button);
305 b1df9b71 2021-03-12 op global_set_key("M-enter", cmd_push_button_new_tab);
306 a8c28919 2021-03-16 op global_set_key("M-tab", cmd_previous_button);
307 a8c28919 2021-03-16 op global_set_key("tab", cmd_next_button);
308 9ca15951 2021-03-09 op
309 fa3fd864 2021-03-10 op /* === minibuffer map === */
310 b360ebb3 2021-03-10 op minibuffer_set_key("ret", cmd_mini_complete_and_exit);
311 b360ebb3 2021-03-10 op minibuffer_set_key("C-g", cmd_mini_abort);
312 b360ebb3 2021-03-10 op minibuffer_set_key("esc", cmd_mini_abort);
313 d67133bf 2021-03-12 op minibuffer_set_key("C-d", cmd_mini_delete_char);
314 d67133bf 2021-03-12 op minibuffer_set_key("del", cmd_mini_delete_backward_char);
315 3c066721 2021-03-26 op minibuffer_set_key("backspace", cmd_mini_delete_backward_char);
316 3c066721 2021-03-26 op minibuffer_set_key("C-h", cmd_mini_delete_backward_char);
317 9ca15951 2021-03-09 op
318 2ba66cea 2021-03-22 op minibuffer_set_key("C-b", cmd_backward_char);
319 2ba66cea 2021-03-22 op minibuffer_set_key("C-f", cmd_forward_char);
320 2ba66cea 2021-03-22 op minibuffer_set_key("<left>", cmd_backward_char);
321 2ba66cea 2021-03-22 op minibuffer_set_key("<right>", cmd_forward_char);
322 2ba66cea 2021-03-22 op minibuffer_set_key("C-e", cmd_move_end_of_line);
323 2ba66cea 2021-03-22 op minibuffer_set_key("C-a", cmd_move_beginning_of_line);
324 2ba66cea 2021-03-22 op minibuffer_set_key("<end>", cmd_move_end_of_line);
325 2ba66cea 2021-03-22 op minibuffer_set_key("<home>", cmd_move_beginning_of_line);
326 fa3fd864 2021-03-10 op minibuffer_set_key("C-k", cmd_mini_kill_line);
327 22268e11 2021-03-11 op
328 22268e11 2021-03-11 op minibuffer_set_key("M-p", cmd_mini_previous_history_element);
329 22268e11 2021-03-11 op minibuffer_set_key("M-n", cmd_mini_next_history_element);
330 22268e11 2021-03-11 op minibuffer_set_key("<up>", cmd_mini_previous_history_element);
331 22268e11 2021-03-11 op minibuffer_set_key("<down>", cmd_mini_next_history_element);
332 1d08c280 2021-03-06 op }
333 1d08c280 2021-03-06 op
334 48e9d457 2021-03-06 op static void
335 2ba66cea 2021-03-22 op restore_cursor(struct window *window)
336 48e9d457 2021-03-06 op {
337 452589f7 2021-03-16 op struct vline *vl;
338 452589f7 2021-03-16 op const char *prfx;
339 452589f7 2021-03-16 op
340 2ba66cea 2021-03-22 op vl = window->current_line;
341 452589f7 2021-03-16 op if (vl == NULL || vl->line == NULL)
342 2ba66cea 2021-03-22 op window->curs_x = window->cpoff = 0;
343 452589f7 2021-03-16 op else
344 2ba66cea 2021-03-22 op window->curs_x = utf8_snwidth(vl->line, window->cpoff);
345 452589f7 2021-03-16 op
346 452589f7 2021-03-16 op if (vl != NULL) {
347 452589f7 2021-03-16 op prfx = line_prefixes[vl->parent->type].prfx1;
348 2ba66cea 2021-03-22 op window->curs_x += utf8_swidth(prfx);
349 452589f7 2021-03-16 op }
350 48e9d457 2021-03-06 op }
351 1d08c280 2021-03-06 op
352 09312eb3 2021-05-14 op void
353 2ba66cea 2021-03-22 op cmd_previous_line(struct window *window)
354 1d08c280 2021-03-06 op {
355 452589f7 2021-03-16 op struct vline *vl;
356 452589f7 2021-03-16 op
357 2ba66cea 2021-03-22 op if (window->current_line == NULL
358 2ba66cea 2021-03-22 op || (vl = TAILQ_PREV(window->current_line, vhead, vlines)) == NULL)
359 452589f7 2021-03-16 op return;
360 452589f7 2021-03-16 op
361 2ba66cea 2021-03-22 op if (--window->curs_y < 0) {
362 2ba66cea 2021-03-22 op window->curs_y = 0;
363 2ba66cea 2021-03-22 op cmd_scroll_line_up(window);
364 452589f7 2021-03-16 op return;
365 4dd664ce 2021-03-06 op }
366 4dd664ce 2021-03-06 op
367 2ba66cea 2021-03-22 op window->current_line = vl;
368 2ba66cea 2021-03-22 op restore_cursor(window);
369 1d08c280 2021-03-06 op }
370 1d08c280 2021-03-06 op
371 09312eb3 2021-05-14 op void
372 2ba66cea 2021-03-22 op cmd_next_line(struct window *window)
373 1d08c280 2021-03-06 op {
374 452589f7 2021-03-16 op struct vline *vl;
375 452589f7 2021-03-16 op
376 2ba66cea 2021-03-22 op if (window->current_line == NULL
377 2ba66cea 2021-03-22 op || (vl = TAILQ_NEXT(window->current_line, vlines)) == NULL)
378 fed61466 2021-03-11 op return;
379 fed61466 2021-03-11 op
380 2ba66cea 2021-03-22 op if (++window->curs_y > body_lines-1) {
381 2ba66cea 2021-03-22 op window->curs_y = body_lines-1;
382 2ba66cea 2021-03-22 op cmd_scroll_line_down(window);
383 452589f7 2021-03-16 op return;
384 4dd664ce 2021-03-06 op }
385 4dd664ce 2021-03-06 op
386 2ba66cea 2021-03-22 op window->current_line = vl;
387 2ba66cea 2021-03-22 op restore_cursor(window);
388 1d08c280 2021-03-06 op }
389 1d08c280 2021-03-06 op
390 09312eb3 2021-05-14 op void
391 2ba66cea 2021-03-22 op cmd_backward_char(struct window *window)
392 1d08c280 2021-03-06 op {
393 2ba66cea 2021-03-22 op if (window->cpoff != 0)
394 2ba66cea 2021-03-22 op window->cpoff--;
395 2ba66cea 2021-03-22 op restore_cursor(window);
396 1d08c280 2021-03-06 op }
397 1d08c280 2021-03-06 op
398 09312eb3 2021-05-14 op void
399 2ba66cea 2021-03-22 op cmd_forward_char(struct window *window)
400 1d08c280 2021-03-06 op {
401 3a20fb6c 2021-05-12 op size_t len = 0;
402 affb8144 2021-04-30 op
403 3a20fb6c 2021-05-12 op if (window->current_line->line != NULL)
404 3a20fb6c 2021-05-12 op len = utf8_cplen(window->current_line->line);
405 affb8144 2021-04-30 op if (++window->cpoff > len)
406 affb8144 2021-04-30 op window->cpoff = len;
407 2ba66cea 2021-03-22 op restore_cursor(window);
408 852d03e8 2021-03-17 op }
409 852d03e8 2021-03-17 op
410 09312eb3 2021-05-14 op void
411 2ba66cea 2021-03-22 op cmd_backward_paragraph(struct window *window)
412 852d03e8 2021-03-17 op {
413 852d03e8 2021-03-17 op do {
414 2ba66cea 2021-03-22 op if (window->current_line == NULL ||
415 2ba66cea 2021-03-22 op window->current_line == TAILQ_FIRST(&window->head)) {
416 852d03e8 2021-03-17 op message("No previous paragraph");
417 852d03e8 2021-03-17 op return;
418 852d03e8 2021-03-17 op }
419 2ba66cea 2021-03-22 op cmd_previous_line(window);
420 2ba66cea 2021-03-22 op } while (window->current_line->line != NULL ||
421 2ba66cea 2021-03-22 op window->current_line->parent->type != LINE_TEXT);
422 852d03e8 2021-03-17 op }
423 852d03e8 2021-03-17 op
424 09312eb3 2021-05-14 op void
425 2ba66cea 2021-03-22 op cmd_forward_paragraph(struct window *window)
426 852d03e8 2021-03-17 op {
427 852d03e8 2021-03-17 op do {
428 2ba66cea 2021-03-22 op if (window->current_line == NULL ||
429 2ba66cea 2021-03-22 op window->current_line == TAILQ_LAST(&window->head, vhead)) {
430 852d03e8 2021-03-17 op message("No next paragraph");
431 852d03e8 2021-03-17 op return;
432 852d03e8 2021-03-17 op }
433 2ba66cea 2021-03-22 op cmd_next_line(window);
434 2ba66cea 2021-03-22 op } while (window->current_line->line != NULL ||
435 2ba66cea 2021-03-22 op window->current_line->parent->type != LINE_TEXT);
436 a329982b 2021-03-11 op }
437 a329982b 2021-03-11 op
438 09312eb3 2021-05-14 op void
439 2ba66cea 2021-03-22 op cmd_move_beginning_of_line(struct window *window)
440 a329982b 2021-03-11 op {
441 2ba66cea 2021-03-22 op window->cpoff = 0;
442 2ba66cea 2021-03-22 op restore_cursor(window);
443 a329982b 2021-03-11 op }
444 a329982b 2021-03-11 op
445 09312eb3 2021-05-14 op void
446 2ba66cea 2021-03-22 op cmd_move_end_of_line(struct window *window)
447 a329982b 2021-03-11 op {
448 9a25f829 2021-03-14 op struct vline *vl;
449 a329982b 2021-03-11 op
450 2ba66cea 2021-03-22 op vl = window->current_line;
451 452589f7 2021-03-16 op if (vl->line == NULL)
452 452589f7 2021-03-16 op return;
453 2ba66cea 2021-03-22 op window->cpoff = utf8_cplen(vl->line);
454 2ba66cea 2021-03-22 op restore_cursor(window);
455 1d08c280 2021-03-06 op }
456 1d08c280 2021-03-06 op
457 09312eb3 2021-05-14 op void
458 2ba66cea 2021-03-22 op cmd_redraw(struct window *window)
459 1d08c280 2021-03-06 op {
460 b1738d2e 2021-03-06 op handle_resize(0, 0, NULL);
461 1d08c280 2021-03-06 op }
462 1d08c280 2021-03-06 op
463 09312eb3 2021-05-14 op void
464 2ba66cea 2021-03-22 op cmd_scroll_line_up(struct window *window)
465 1d08c280 2021-03-06 op {
466 9a25f829 2021-03-14 op struct vline *vl;
467 48e9d457 2021-03-06 op
468 2ba66cea 2021-03-22 op if (window->line_off == 0)
469 48e9d457 2021-03-06 op return;
470 48e9d457 2021-03-06 op
471 2ba66cea 2021-03-22 op vl = nth_line(window, --window->line_off);
472 48e9d457 2021-03-06 op wscrl(body, -1);
473 48e9d457 2021-03-06 op wmove(body, 0, 0);
474 bddc7bbd 2021-04-01 op print_vline(body, vl);
475 452589f7 2021-03-16 op
476 2ba66cea 2021-03-22 op window->current_line = TAILQ_PREV(window->current_line, vhead, vlines);
477 2ba66cea 2021-03-22 op restore_cursor(window);
478 1d08c280 2021-03-06 op }
479 1d08c280 2021-03-06 op
480 09312eb3 2021-05-14 op void
481 2ba66cea 2021-03-22 op cmd_scroll_line_down(struct window *window)
482 1d08c280 2021-03-06 op {
483 9a25f829 2021-03-14 op struct vline *vl;
484 48e9d457 2021-03-06 op
485 2ba66cea 2021-03-22 op vl = window->current_line;
486 452589f7 2021-03-16 op if ((vl = TAILQ_NEXT(vl, vlines)) == NULL)
487 48e9d457 2021-03-06 op return;
488 2ba66cea 2021-03-22 op window->current_line = vl;
489 48e9d457 2021-03-06 op
490 2ba66cea 2021-03-22 op window->line_off++;
491 48e9d457 2021-03-06 op wscrl(body, 1);
492 48e9d457 2021-03-06 op
493 2ba66cea 2021-03-22 op if (window->line_max - window->line_off < (size_t)body_lines)
494 48e9d457 2021-03-06 op return;
495 48e9d457 2021-03-06 op
496 2ba66cea 2021-03-22 op vl = nth_line(window, window->line_off + body_lines-1);
497 48e9d457 2021-03-06 op wmove(body, body_lines-1, 0);
498 bddc7bbd 2021-04-01 op print_vline(body, vl);
499 174b3cdf 2021-03-21 op
500 2ba66cea 2021-03-22 op restore_cursor(window);
501 ed44414d 2021-03-09 op }
502 ed44414d 2021-03-09 op
503 09312eb3 2021-05-14 op void
504 2ba66cea 2021-03-22 op cmd_scroll_up(struct window *window)
505 ed44414d 2021-03-09 op {
506 ed44414d 2021-03-09 op size_t off;
507 ed44414d 2021-03-09 op
508 cce62e74 2021-04-25 op off = body_lines-1;
509 ed44414d 2021-03-09 op
510 ed44414d 2021-03-09 op for (; off > 0; --off)
511 2ba66cea 2021-03-22 op cmd_scroll_line_up(window);
512 1d08c280 2021-03-06 op }
513 1d08c280 2021-03-06 op
514 09312eb3 2021-05-14 op void
515 2ba66cea 2021-03-22 op cmd_scroll_down(struct window *window)
516 ed44414d 2021-03-09 op {
517 338eecdc 2021-03-12 op size_t off;
518 ed44414d 2021-03-09 op
519 cce62e74 2021-04-25 op off = body_lines-1;
520 ed44414d 2021-03-09 op
521 338eecdc 2021-03-12 op for (; off > 0; --off)
522 2ba66cea 2021-03-22 op cmd_scroll_line_down(window);
523 e19f9a04 2021-03-11 op }
524 e19f9a04 2021-03-11 op
525 09312eb3 2021-05-14 op void
526 2ba66cea 2021-03-22 op cmd_beginning_of_buffer(struct window *window)
527 e19f9a04 2021-03-11 op {
528 2ba66cea 2021-03-22 op window->current_line = TAILQ_FIRST(&window->head);
529 2ba66cea 2021-03-22 op window->line_off = 0;
530 2ba66cea 2021-03-22 op window->curs_y = 0;
531 2ba66cea 2021-03-22 op window->cpoff = 0;
532 2ba66cea 2021-03-22 op restore_cursor(window);
533 e19f9a04 2021-03-11 op }
534 e19f9a04 2021-03-11 op
535 09312eb3 2021-05-14 op void
536 2ba66cea 2021-03-22 op cmd_end_of_buffer(struct window *window)
537 e19f9a04 2021-03-11 op {
538 e19f9a04 2021-03-11 op ssize_t off;
539 e19f9a04 2021-03-11 op
540 2ba66cea 2021-03-22 op off = window->line_max - body_lines;
541 e19f9a04 2021-03-11 op off = MAX(0, off);
542 e19f9a04 2021-03-11 op
543 2ba66cea 2021-03-22 op window->line_off = off;
544 2ba66cea 2021-03-22 op window->curs_y = MIN((size_t)body_lines, window->line_max-1);
545 e19f9a04 2021-03-11 op
546 2ba66cea 2021-03-22 op window->current_line = TAILQ_LAST(&window->head, vhead);
547 2ba66cea 2021-03-22 op window->cpoff = body_cols;
548 2ba66cea 2021-03-22 op restore_cursor(window);
549 ed44414d 2021-03-09 op }
550 ed44414d 2021-03-09 op
551 09312eb3 2021-05-14 op void
552 2ba66cea 2021-03-22 op cmd_kill_telescope(struct window *window)
553 1d08c280 2021-03-06 op {
554 c7107cec 2021-04-01 op save_session();
555 1d08c280 2021-03-06 op event_loopbreak();
556 1d08c280 2021-03-06 op }
557 1d08c280 2021-03-06 op
558 09312eb3 2021-05-14 op void
559 2ba66cea 2021-03-22 op cmd_push_button(struct window *window)
560 2a4ad912 2021-03-08 op {
561 9a25f829 2021-03-14 op struct vline *vl;
562 2a4ad912 2021-03-08 op size_t nth;
563 2a4ad912 2021-03-08 op
564 2ba66cea 2021-03-22 op nth = window->line_off + window->curs_y;
565 2ba66cea 2021-03-22 op if (nth >= window->line_max)
566 2a4ad912 2021-03-08 op return;
567 2ba66cea 2021-03-22 op vl = nth_line(window, nth);
568 9a25f829 2021-03-14 op if (vl->parent->type != LINE_LINK)
569 43a1b8d0 2021-03-09 op return;
570 43a1b8d0 2021-03-09 op
571 2ba66cea 2021-03-22 op load_url_in_tab(current_tab(), vl->parent->alt);
572 3b4f9e49 2021-03-11 op }
573 3b4f9e49 2021-03-11 op
574 09312eb3 2021-05-14 op void
575 2ba66cea 2021-03-22 op cmd_push_button_new_tab(struct window *window)
576 b1df9b71 2021-03-12 op {
577 9a25f829 2021-03-14 op struct vline *vl;
578 b1df9b71 2021-03-12 op size_t nth;
579 b1df9b71 2021-03-12 op
580 2ba66cea 2021-03-22 op nth = window->line_off + window->curs_y;
581 2ba66cea 2021-03-22 op if (nth > window->line_max)
582 b1df9b71 2021-03-12 op return;
583 2ba66cea 2021-03-22 op vl = nth_line(window, nth);
584 9a25f829 2021-03-14 op if (vl->parent->type != LINE_LINK)
585 b1df9b71 2021-03-12 op return;
586 b1df9b71 2021-03-12 op
587 1b81bc33 2021-03-15 op new_tab(vl->parent->alt);
588 a8c28919 2021-03-16 op }
589 a8c28919 2021-03-16 op
590 09312eb3 2021-05-14 op void
591 2ba66cea 2021-03-22 op cmd_previous_button(struct window *window)
592 a8c28919 2021-03-16 op {
593 a8c28919 2021-03-16 op do {
594 2ba66cea 2021-03-22 op if (window->current_line == NULL ||
595 2ba66cea 2021-03-22 op window->current_line == TAILQ_FIRST(&window->head)) {
596 a8c28919 2021-03-16 op message("No previous link");
597 a8c28919 2021-03-16 op return;
598 a8c28919 2021-03-16 op }
599 2ba66cea 2021-03-22 op cmd_previous_line(window);
600 2ba66cea 2021-03-22 op } while (window->current_line->parent->type != LINE_LINK);
601 a8c28919 2021-03-16 op }
602 a8c28919 2021-03-16 op
603 09312eb3 2021-05-14 op void
604 2ba66cea 2021-03-22 op cmd_next_button(struct window *window)
605 a8c28919 2021-03-16 op {
606 a8c28919 2021-03-16 op do {
607 2ba66cea 2021-03-22 op if (window->current_line == NULL ||
608 2ba66cea 2021-03-22 op window->current_line == TAILQ_LAST(&window->head, vhead)) {
609 a8c28919 2021-03-16 op message("No next link");
610 a8c28919 2021-03-16 op return;
611 a8c28919 2021-03-16 op }
612 2ba66cea 2021-03-22 op cmd_next_line(window);
613 2ba66cea 2021-03-22 op } while (window->current_line->parent->type != LINE_LINK);
614 2051e653 2021-03-13 op }
615 2051e653 2021-03-13 op
616 09312eb3 2021-05-14 op void
617 2ba66cea 2021-03-22 op cmd_previous_page(struct window *window)
618 2051e653 2021-03-13 op {
619 2ba66cea 2021-03-22 op struct tab *tab = current_tab();
620 2ba66cea 2021-03-22 op
621 2051e653 2021-03-13 op if (!load_previous_page(tab))
622 2051e653 2021-03-13 op message("No previous page");
623 c40c2250 2021-03-14 op else
624 c40c2250 2021-03-14 op start_loading_anim(tab);
625 2051e653 2021-03-13 op }
626 2051e653 2021-03-13 op
627 09312eb3 2021-05-14 op void
628 2ba66cea 2021-03-22 op cmd_next_page(struct window *window)
629 2051e653 2021-03-13 op {
630 2ba66cea 2021-03-22 op struct tab *tab = current_tab();
631 2ba66cea 2021-03-22 op
632 2051e653 2021-03-13 op if (!load_next_page(tab))
633 2051e653 2021-03-13 op message("No next page");
634 c40c2250 2021-03-14 op else
635 c40c2250 2021-03-14 op start_loading_anim(tab);
636 b1df9b71 2021-03-12 op }
637 b1df9b71 2021-03-12 op
638 09312eb3 2021-05-14 op void
639 2ba66cea 2021-03-22 op cmd_clear_minibuf(struct window *window)
640 3b4f9e49 2021-03-11 op {
641 3b4f9e49 2021-03-11 op handle_clear_minibuf(0, 0, NULL);
642 9ca15951 2021-03-09 op }
643 9ca15951 2021-03-09 op
644 09312eb3 2021-05-14 op void
645 2ba66cea 2021-03-22 op cmd_execute_extended_command(struct window *window)
646 9ca15951 2021-03-09 op {
647 9ca15951 2021-03-09 op size_t len;
648 9ca15951 2021-03-09 op
649 2ba66cea 2021-03-22 op if (in_minibuffer) {
650 2ba66cea 2021-03-22 op message("We don't have enable-recursive-minibuffers");
651 2ba66cea 2021-03-22 op return;
652 2ba66cea 2021-03-22 op }
653 2ba66cea 2021-03-22 op
654 22268e11 2021-03-11 op enter_minibuffer(eecmd_self_insert, eecmd_select, exit_minibuffer,
655 22268e11 2021-03-11 op &eecmd_history);
656 9ca15951 2021-03-09 op
657 9ca15951 2021-03-09 op len = sizeof(ministate.prompt);
658 9ca15951 2021-03-09 op strlcpy(ministate.prompt, "", len);
659 9ca15951 2021-03-09 op
660 9ca15951 2021-03-09 op if (thiskey.meta)
661 9ca15951 2021-03-09 op strlcat(ministate.prompt, "M-", len);
662 9ca15951 2021-03-09 op
663 9ca15951 2021-03-09 op strlcat(ministate.prompt, keyname(thiskey.key), len);
664 c3fcefe6 2021-05-12 op
665 c3fcefe6 2021-05-12 op if (thiskey.meta)
666 c3fcefe6 2021-05-12 op strlcat(ministate.prompt, " ", len);
667 9ca15951 2021-03-09 op }
668 9ca15951 2021-03-09 op
669 09312eb3 2021-05-14 op void
670 2ba66cea 2021-03-22 op cmd_tab_close(struct window *window)
671 7c7d7bb7 2021-03-10 op {
672 2ba66cea 2021-03-22 op struct tab *tab, *t;
673 a777f81f 2021-03-10 op
674 2ba66cea 2021-03-22 op tab = current_tab();
675 7c7d7bb7 2021-03-10 op if (TAILQ_PREV(tab, tabshead, tabs) == NULL &&
676 7c7d7bb7 2021-03-10 op TAILQ_NEXT(tab, tabs) == NULL) {
677 7c7d7bb7 2021-03-10 op message("Can't close the only tab.");
678 a777f81f 2021-03-10 op return;
679 a777f81f 2021-03-10 op }
680 a777f81f 2021-03-10 op
681 2ba66cea 2021-03-22 op if (evtimer_pending(&tab->loadingev, NULL))
682 2ba66cea 2021-03-22 op evtimer_del(&tab->loadingev);
683 d87ffbdc 2021-03-17 op
684 7c7d7bb7 2021-03-10 op stop_tab(tab);
685 7c7d7bb7 2021-03-10 op
686 9d24c64f 2021-03-17 op if ((t = TAILQ_PREV(tab, tabshead, tabs)) == NULL)
687 9d24c64f 2021-03-17 op t = TAILQ_NEXT(tab, tabs);
688 7c7d7bb7 2021-03-10 op TAILQ_REMOVE(&tabshead, tab, tabs);
689 7c7d7bb7 2021-03-10 op free(tab);
690 9d24c64f 2021-03-17 op
691 9d24c64f 2021-03-17 op switch_to_tab(t);
692 9ca15951 2021-03-09 op }
693 9ca15951 2021-03-09 op
694 09312eb3 2021-05-14 op void
695 2ba66cea 2021-03-22 op cmd_tab_close_other(struct window *window)
696 fea845b6 2021-03-15 op {
697 2ff75826 2021-03-26 op struct tab *t, *i;
698 fea845b6 2021-03-15 op
699 fea845b6 2021-03-15 op TAILQ_FOREACH_SAFE(t, &tabshead, tabs, i) {
700 fea845b6 2021-03-15 op if (t->flags & TAB_CURRENT)
701 fea845b6 2021-03-15 op continue;
702 fea845b6 2021-03-15 op
703 fea845b6 2021-03-15 op stop_tab(t);
704 fea845b6 2021-03-15 op TAILQ_REMOVE(&tabshead, t, tabs);
705 fea845b6 2021-03-15 op free(t);
706 fea845b6 2021-03-15 op }
707 fea845b6 2021-03-15 op }
708 fea845b6 2021-03-15 op
709 09312eb3 2021-05-14 op void
710 2ba66cea 2021-03-22 op cmd_tab_new(struct window *window)
711 7c7d7bb7 2021-03-10 op {
712 1b81bc33 2021-03-15 op new_tab(NEW_TAB_URL);
713 7c7d7bb7 2021-03-10 op }
714 7c7d7bb7 2021-03-10 op
715 09312eb3 2021-05-14 op void
716 2ba66cea 2021-03-22 op cmd_tab_next(struct window *window)
717 7c7d7bb7 2021-03-10 op {
718 2ba66cea 2021-03-22 op struct tab *tab, *t;
719 7c7d7bb7 2021-03-10 op
720 2ba66cea 2021-03-22 op tab = current_tab();
721 7c7d7bb7 2021-03-10 op tab->flags &= ~TAB_CURRENT;
722 7c7d7bb7 2021-03-10 op
723 7c7d7bb7 2021-03-10 op if ((t = TAILQ_NEXT(tab, tabs)) == NULL)
724 7c7d7bb7 2021-03-10 op t = TAILQ_FIRST(&tabshead);
725 7c7d7bb7 2021-03-10 op t->flags |= TAB_CURRENT;
726 e8a76665 2021-05-12 op t->flags &= ~TAB_URGENT;
727 7c7d7bb7 2021-03-10 op }
728 7c7d7bb7 2021-03-10 op
729 09312eb3 2021-05-14 op void
730 2ba66cea 2021-03-22 op cmd_tab_previous(struct window *window)
731 7c7d7bb7 2021-03-10 op {
732 2ba66cea 2021-03-22 op struct tab *tab, *t;
733 7c7d7bb7 2021-03-10 op
734 2ba66cea 2021-03-22 op tab = current_tab();
735 7c7d7bb7 2021-03-10 op tab->flags &= ~TAB_CURRENT;
736 7c7d7bb7 2021-03-10 op
737 7c7d7bb7 2021-03-10 op if ((t = TAILQ_PREV(tab, tabshead, tabs)) == NULL)
738 7c7d7bb7 2021-03-10 op t = TAILQ_LAST(&tabshead, tabshead);
739 7c7d7bb7 2021-03-10 op t->flags |= TAB_CURRENT;
740 e8a76665 2021-05-12 op t->flags &= ~TAB_URGENT;
741 5cd2ebb1 2021-03-11 op }
742 5cd2ebb1 2021-03-11 op
743 09312eb3 2021-05-14 op void
744 fad72201 2021-03-28 op cmd_tab_move(struct window *window)
745 fad72201 2021-03-28 op {
746 fad72201 2021-03-28 op struct tab *tab, *t;
747 fad72201 2021-03-28 op
748 fad72201 2021-03-28 op tab = current_tab();
749 fad72201 2021-03-28 op t = TAILQ_NEXT(tab, tabs);
750 fad72201 2021-03-28 op TAILQ_REMOVE(&tabshead, tab, tabs);
751 fad72201 2021-03-28 op
752 fad72201 2021-03-28 op if (t == NULL)
753 fad72201 2021-03-28 op TAILQ_INSERT_HEAD(&tabshead, tab, tabs);
754 fad72201 2021-03-28 op else
755 fad72201 2021-03-28 op TAILQ_INSERT_AFTER(&tabshead, t, tab, tabs);
756 fad72201 2021-03-28 op }
757 fad72201 2021-03-28 op
758 09312eb3 2021-05-14 op void
759 fad72201 2021-03-28 op cmd_tab_move_to(struct window *window)
760 fad72201 2021-03-28 op {
761 fad72201 2021-03-28 op struct tab *tab, *t;
762 fad72201 2021-03-28 op
763 fad72201 2021-03-28 op tab = current_tab();
764 fad72201 2021-03-28 op t = TAILQ_PREV(tab, tabshead, tabs);
765 fad72201 2021-03-28 op TAILQ_REMOVE(&tabshead, tab, tabs);
766 fad72201 2021-03-28 op
767 fad72201 2021-03-28 op if (t == NULL) {
768 fad72201 2021-03-28 op if (TAILQ_EMPTY(&tabshead))
769 fad72201 2021-03-28 op TAILQ_INSERT_HEAD(&tabshead, tab, tabs);
770 fad72201 2021-03-28 op else
771 fad72201 2021-03-28 op TAILQ_INSERT_TAIL(&tabshead, tab, tabs);
772 fad72201 2021-03-28 op } else
773 fad72201 2021-03-28 op TAILQ_INSERT_BEFORE(t, tab, tabs);
774 fad72201 2021-03-28 op }
775 fad72201 2021-03-28 op
776 09312eb3 2021-05-14 op void
777 2ba66cea 2021-03-22 op cmd_load_url(struct window *window)
778 5cd2ebb1 2021-03-11 op {
779 2ba66cea 2021-03-22 op if (in_minibuffer) {
780 2ba66cea 2021-03-22 op message("We don't have enable-recursive-minibuffers");
781 2ba66cea 2021-03-22 op return;
782 2ba66cea 2021-03-22 op }
783 2ba66cea 2021-03-22 op
784 22268e11 2021-03-11 op enter_minibuffer(lu_self_insert, lu_select, exit_minibuffer,
785 22268e11 2021-03-11 op &lu_history);
786 5cd2ebb1 2021-03-11 op strlcpy(ministate.prompt, "Load URL: ", sizeof(ministate.prompt));
787 2d13108f 2021-04-30 op strlcpy(ministate.buf, "gemini://", sizeof(ministate.buf));
788 2d13108f 2021-04-30 op cmd_move_end_of_line(&ministate.window);
789 7c7d7bb7 2021-03-10 op }
790 7c7d7bb7 2021-03-10 op
791 09312eb3 2021-05-14 op void
792 2ba66cea 2021-03-22 op cmd_load_current_url(struct window *window)
793 5cd2ebb1 2021-03-11 op {
794 2ba66cea 2021-03-22 op struct tab *tab = current_tab();
795 2ba66cea 2021-03-22 op
796 2ba66cea 2021-03-22 op if (in_minibuffer) {
797 2ba66cea 2021-03-22 op message("We don't have enable-recursive-minibuffers");
798 2ba66cea 2021-03-22 op return;
799 2ba66cea 2021-03-22 op }
800 2ba66cea 2021-03-22 op
801 22268e11 2021-03-11 op enter_minibuffer(lu_self_insert, lu_select, exit_minibuffer,
802 22268e11 2021-03-11 op &lu_history);
803 5cd2ebb1 2021-03-11 op strlcpy(ministate.prompt, "Load URL: ", sizeof(ministate.prompt));
804 740f578b 2021-03-15 op strlcpy(ministate.buf, tab->hist_cur->h, sizeof(ministate.buf));
805 2ba66cea 2021-03-22 op ministate.window.cpoff = utf8_cplen(ministate.buf);
806 740f578b 2021-03-15 op }
807 740f578b 2021-03-15 op
808 09312eb3 2021-05-14 op void
809 2ba66cea 2021-03-22 op cmd_bookmark_page(struct window *window)
810 740f578b 2021-03-15 op {
811 2ba66cea 2021-03-22 op struct tab *tab = current_tab();
812 2ba66cea 2021-03-22 op
813 740f578b 2021-03-15 op enter_minibuffer(lu_self_insert, bp_select, exit_minibuffer, NULL);
814 740f578b 2021-03-15 op strlcpy(ministate.prompt, "Bookmark URL: ", sizeof(ministate.prompt));
815 2051e653 2021-03-13 op strlcpy(ministate.buf, tab->hist_cur->h, sizeof(ministate.buf));
816 2ba66cea 2021-03-22 op ministate.window.cpoff = utf8_cplen(ministate.buf);
817 de0b2139 2021-03-15 op }
818 de0b2139 2021-03-15 op
819 09312eb3 2021-05-14 op void
820 63875195 2021-04-01 op cmd_list_bookmarks(struct window *window)
821 de0b2139 2021-03-15 op {
822 2ba66cea 2021-03-22 op load_url_in_tab(current_tab(), "about:bookmarks");
823 bddc7bbd 2021-04-01 op }
824 bddc7bbd 2021-04-01 op
825 09312eb3 2021-05-14 op void
826 bddc7bbd 2021-04-01 op cmd_toggle_help(struct window *window)
827 bddc7bbd 2021-04-01 op {
828 bddc7bbd 2021-04-01 op side_window = !side_window;
829 bddc7bbd 2021-04-01 op if (side_window)
830 bddc7bbd 2021-04-01 op recompute_help();
831 bddc7bbd 2021-04-01 op
832 7cdf12cb 2021-05-12 op /*
833 7cdf12cb 2021-05-12 op * ugly hack, but otherwise the window doesn't get updated
834 bddc7bbd 2021-04-01 op * until I call handle_resize a second time (i.e. C-l). I
835 7cdf12cb 2021-05-12 op * will be happy to know why something like this is needed.
836 7cdf12cb 2021-05-12 op */
837 bddc7bbd 2021-04-01 op handle_resize(0, 0, NULL);
838 bddc7bbd 2021-04-01 op handle_resize(0, 0, NULL);
839 5cd2ebb1 2021-03-11 op }
840 5cd2ebb1 2021-03-11 op
841 09312eb3 2021-05-14 op void
842 2ba66cea 2021-03-22 op cmd_mini_delete_char(struct window *window)
843 d67133bf 2021-03-12 op {
844 2ba66cea 2021-03-22 op char *c, *n;
845 17e5106b 2021-03-21 op
846 2ba66cea 2021-03-22 op if (!in_minibuffer) {
847 2ba66cea 2021-03-22 op message("text is read-only");
848 2ba66cea 2021-03-22 op return;
849 2ba66cea 2021-03-22 op }
850 2ba66cea 2021-03-22 op
851 3eecee9e 2021-03-12 op minibuffer_taint_hist();
852 3eecee9e 2021-03-12 op
853 2ba66cea 2021-03-22 op c = utf8_nth(window->current_line->line, window->cpoff);
854 2ba66cea 2021-03-22 op if (*c == '\0')
855 d67133bf 2021-03-12 op return;
856 2ba66cea 2021-03-22 op n = utf8_next_cp(c);
857 d67133bf 2021-03-12 op
858 2ba66cea 2021-03-22 op memmove(c, n, strlen(n)+1);
859 d67133bf 2021-03-12 op }
860 d67133bf 2021-03-12 op
861 09312eb3 2021-05-14 op void
862 2ba66cea 2021-03-22 op cmd_mini_delete_backward_char(struct window *window)
863 9ca15951 2021-03-09 op {
864 2ba66cea 2021-03-22 op char *c, *p, *start;
865 17e5106b 2021-03-21 op
866 2ba66cea 2021-03-22 op if (!in_minibuffer) {
867 2ba66cea 2021-03-22 op message("text is read-only");
868 2ba66cea 2021-03-22 op return;
869 2ba66cea 2021-03-22 op }
870 2ba66cea 2021-03-22 op
871 3eecee9e 2021-03-12 op minibuffer_taint_hist();
872 3eecee9e 2021-03-12 op
873 2ba66cea 2021-03-22 op c = utf8_nth(window->current_line->line, window->cpoff);
874 2ba66cea 2021-03-22 op start = window->current_line->line;
875 2ba66cea 2021-03-22 op if (c == start)
876 9ca15951 2021-03-09 op return;
877 2ba66cea 2021-03-22 op p = utf8_prev_cp(c-1, start);
878 9ca15951 2021-03-09 op
879 2ba66cea 2021-03-22 op memmove(p, c, strlen(c)+1);
880 2ba66cea 2021-03-22 op window->cpoff--;
881 9ca15951 2021-03-09 op }
882 9ca15951 2021-03-09 op
883 09312eb3 2021-05-14 op void
884 2ba66cea 2021-03-22 op cmd_mini_kill_line(struct window *window)
885 9ca15951 2021-03-09 op {
886 2ba66cea 2021-03-22 op char *c;
887 9ca15951 2021-03-09 op
888 2ba66cea 2021-03-22 op if (!in_minibuffer) {
889 2ba66cea 2021-03-22 op message("text is read-only");
890 9ca15951 2021-03-09 op return;
891 2ba66cea 2021-03-22 op }
892 9ca15951 2021-03-09 op
893 3eecee9e 2021-03-12 op minibuffer_taint_hist();
894 2ba66cea 2021-03-22 op c = utf8_nth(window->current_line->line, window->cpoff);
895 2ba66cea 2021-03-22 op *c = '\0';
896 fa3fd864 2021-03-10 op }
897 fa3fd864 2021-03-10 op
898 09312eb3 2021-05-14 op void
899 2ba66cea 2021-03-22 op cmd_mini_abort(struct window *window)
900 b360ebb3 2021-03-10 op {
901 2ba66cea 2021-03-22 op if (!in_minibuffer)
902 2ba66cea 2021-03-22 op return;
903 2ba66cea 2021-03-22 op
904 17e5106b 2021-03-21 op ministate.abortfn();
905 b360ebb3 2021-03-10 op }
906 b360ebb3 2021-03-10 op
907 09312eb3 2021-05-14 op void
908 2ba66cea 2021-03-22 op cmd_mini_complete_and_exit(struct window *window)
909 b360ebb3 2021-03-10 op {
910 2ba66cea 2021-03-22 op if (!in_minibuffer)
911 2ba66cea 2021-03-22 op return;
912 2ba66cea 2021-03-22 op
913 22268e11 2021-03-11 op minibuffer_taint_hist();
914 b360ebb3 2021-03-10 op ministate.donefn();
915 22268e11 2021-03-11 op }
916 22268e11 2021-03-11 op
917 09312eb3 2021-05-14 op void
918 2ba66cea 2021-03-22 op cmd_mini_previous_history_element(struct window *window)
919 22268e11 2021-03-11 op {
920 22268e11 2021-03-11 op if (ministate.history == NULL) {
921 22268e11 2021-03-11 op message("No history");
922 22268e11 2021-03-11 op return;
923 22268e11 2021-03-11 op }
924 22268e11 2021-03-11 op
925 22268e11 2021-03-11 op if (ministate.hist_cur == NULL ||
926 22268e11 2021-03-11 op (ministate.hist_cur = TAILQ_PREV(ministate.hist_cur, mhisthead, entries)) == NULL) {
927 22268e11 2021-03-11 op ministate.hist_cur = TAILQ_LAST(&ministate.history->head, mhisthead);
928 22268e11 2021-03-11 op ministate.hist_off = ministate.history->len - 1;
929 22268e11 2021-03-11 op if (ministate.hist_cur == NULL)
930 22268e11 2021-03-11 op message("No prev item");
931 22268e11 2021-03-11 op } else {
932 22268e11 2021-03-11 op ministate.hist_off--;
933 22268e11 2021-03-11 op }
934 22268e11 2021-03-11 op
935 17e5106b 2021-03-21 op if (ministate.hist_cur != NULL)
936 2ba66cea 2021-03-22 op window->current_line->line = ministate.hist_cur->h;
937 b360ebb3 2021-03-10 op }
938 b360ebb3 2021-03-10 op
939 09312eb3 2021-05-14 op void
940 2ba66cea 2021-03-22 op cmd_mini_next_history_element(struct window *window)
941 22268e11 2021-03-11 op {
942 22268e11 2021-03-11 op if (ministate.history == NULL) {
943 22268e11 2021-03-11 op message("No history");
944 22268e11 2021-03-11 op return;
945 22268e11 2021-03-11 op }
946 22268e11 2021-03-11 op
947 22268e11 2021-03-11 op if (ministate.hist_cur == NULL ||
948 22268e11 2021-03-11 op (ministate.hist_cur = TAILQ_NEXT(ministate.hist_cur, entries)) == NULL) {
949 22268e11 2021-03-11 op ministate.hist_cur = TAILQ_FIRST(&ministate.history->head);
950 22268e11 2021-03-11 op ministate.hist_off = 0;
951 22268e11 2021-03-11 op if (ministate.hist_cur == NULL)
952 22268e11 2021-03-11 op message("No next item");
953 22268e11 2021-03-11 op } else {
954 22268e11 2021-03-11 op ministate.hist_off++;
955 22268e11 2021-03-11 op }
956 22268e11 2021-03-11 op
957 17e5106b 2021-03-21 op if (ministate.hist_cur != NULL)
958 2ba66cea 2021-03-22 op window->current_line->line = ministate.hist_cur->h;
959 22268e11 2021-03-11 op }
960 22268e11 2021-03-11 op
961 22268e11 2021-03-11 op static void
962 870210fb 2021-03-26 op global_key_unbound(void)
963 870210fb 2021-03-26 op {
964 870210fb 2021-03-26 op message("%s is undefined", keybuf);
965 870210fb 2021-03-26 op }
966 870210fb 2021-03-26 op
967 870210fb 2021-03-26 op static void
968 22268e11 2021-03-11 op minibuffer_hist_save_entry(void)
969 22268e11 2021-03-11 op {
970 3148eeac 2021-03-13 op struct hist *hist;
971 22268e11 2021-03-11 op
972 22268e11 2021-03-11 op if (ministate.history == NULL)
973 22268e11 2021-03-11 op return;
974 22268e11 2021-03-11 op
975 22268e11 2021-03-11 op if ((hist = calloc(1, sizeof(*hist))) == NULL)
976 22268e11 2021-03-11 op abort();
977 22268e11 2021-03-11 op
978 22268e11 2021-03-11 op strlcpy(hist->h, ministate.buf, sizeof(hist->h));
979 22268e11 2021-03-11 op
980 22268e11 2021-03-11 op if (TAILQ_EMPTY(&ministate.history->head))
981 22268e11 2021-03-11 op TAILQ_INSERT_HEAD(&ministate.history->head, hist, entries);
982 22268e11 2021-03-11 op else
983 22268e11 2021-03-11 op TAILQ_INSERT_TAIL(&ministate.history->head, hist, entries);
984 22268e11 2021-03-11 op ministate.history->len++;
985 22268e11 2021-03-11 op }
986 22268e11 2021-03-11 op
987 22268e11 2021-03-11 op /*
988 22268e11 2021-03-11 op * taint the minibuffer cache: if we're currently showing a history
989 22268e11 2021-03-11 op * element, copy that to the current buf and reset the "history
990 22268e11 2021-03-11 op * navigation" thing.
991 22268e11 2021-03-11 op */
992 22268e11 2021-03-11 op static void
993 22268e11 2021-03-11 op minibuffer_taint_hist(void)
994 22268e11 2021-03-11 op {
995 22268e11 2021-03-11 op if (ministate.hist_cur == NULL)
996 22268e11 2021-03-11 op return;
997 22268e11 2021-03-11 op
998 22268e11 2021-03-11 op strlcpy(ministate.buf, ministate.hist_cur->h, sizeof(ministate.buf));
999 22268e11 2021-03-11 op ministate.hist_cur = NULL;
1000 22268e11 2021-03-11 op }
1001 22268e11 2021-03-11 op
1002 22268e11 2021-03-11 op static void
1003 5cd2ebb1 2021-03-11 op minibuffer_self_insert(void)
1004 9ca15951 2021-03-09 op {
1005 2ba66cea 2021-03-22 op char *c, tmp[5] = {0};
1006 17e5106b 2021-03-21 op size_t len;
1007 17e5106b 2021-03-21 op
1008 22268e11 2021-03-11 op minibuffer_taint_hist();
1009 22268e11 2021-03-11 op
1010 17e5106b 2021-03-21 op if (thiskey.cp == 0)
1011 9ca15951 2021-03-09 op return;
1012 9ca15951 2021-03-09 op
1013 17e5106b 2021-03-21 op len = utf8_encode(thiskey.cp, tmp);
1014 2ba66cea 2021-03-22 op c = utf8_nth(ministate.window.current_line->line, ministate.window.cpoff);
1015 2ba66cea 2021-03-22 op if (c + len > ministate.buf + sizeof(ministate.buf) - 1)
1016 17e5106b 2021-03-21 op return;
1017 9ca15951 2021-03-09 op
1018 2ba66cea 2021-03-22 op memmove(c + len, c, strlen(c)+1);
1019 2ba66cea 2021-03-22 op memcpy(c, tmp, len);
1020 2ba66cea 2021-03-22 op ministate.window.cpoff++;
1021 b360ebb3 2021-03-10 op }
1022 b360ebb3 2021-03-10 op
1023 b360ebb3 2021-03-10 op static void
1024 5cd2ebb1 2021-03-11 op eecmd_self_insert(void)
1025 5cd2ebb1 2021-03-11 op {
1026 17e5106b 2021-03-21 op if (thiskey.meta || unicode_isspace(thiskey.cp) ||
1027 17e5106b 2021-03-21 op !unicode_isgraph(thiskey.cp)) {
1028 5cd2ebb1 2021-03-11 op global_key_unbound();
1029 5cd2ebb1 2021-03-11 op return;
1030 5cd2ebb1 2021-03-11 op }
1031 5cd2ebb1 2021-03-11 op
1032 5cd2ebb1 2021-03-11 op minibuffer_self_insert();
1033 5cd2ebb1 2021-03-11 op }
1034 5cd2ebb1 2021-03-11 op
1035 5cd2ebb1 2021-03-11 op static void
1036 b360ebb3 2021-03-10 op eecmd_select(void)
1037 b360ebb3 2021-03-10 op {
1038 870210fb 2021-03-26 op struct cmds *cmd;
1039 870210fb 2021-03-26 op
1040 870210fb 2021-03-26 op for (cmd = cmds; cmd->cmd != NULL; ++cmd) {
1041 870210fb 2021-03-26 op if (!strcmp(cmd->cmd, ministate.buf)) {
1042 2fe98292 2021-03-26 op exit_minibuffer();
1043 2fe98292 2021-03-26 op minibuffer_hist_save_entry();
1044 870210fb 2021-03-26 op cmd->fn(current_window());
1045 870210fb 2021-03-26 op return;
1046 870210fb 2021-03-26 op }
1047 870210fb 2021-03-26 op }
1048 870210fb 2021-03-26 op
1049 2fe98292 2021-03-26 op message("No match");
1050 5cd2ebb1 2021-03-11 op }
1051 5cd2ebb1 2021-03-11 op
1052 5cd2ebb1 2021-03-11 op static void
1053 5cd2ebb1 2021-03-11 op ir_self_insert(void)
1054 5cd2ebb1 2021-03-11 op {
1055 5cd2ebb1 2021-03-11 op minibuffer_self_insert();
1056 5cd2ebb1 2021-03-11 op }
1057 5cd2ebb1 2021-03-11 op
1058 5cd2ebb1 2021-03-11 op static void
1059 5cd2ebb1 2021-03-11 op ir_select(void)
1060 5cd2ebb1 2021-03-11 op {
1061 5cd2ebb1 2021-03-11 op char buf[1025] = {0};
1062 31f1a758 2021-04-22 op struct phos_uri uri;
1063 5cd2ebb1 2021-03-11 op struct tab *tab;
1064 5cd2ebb1 2021-03-11 op
1065 5cd2ebb1 2021-03-11 op tab = current_tab();
1066 5cd2ebb1 2021-03-11 op
1067 5cd2ebb1 2021-03-11 op exit_minibuffer();
1068 22268e11 2021-03-11 op minibuffer_hist_save_entry();
1069 5cd2ebb1 2021-03-11 op
1070 5cd2ebb1 2021-03-11 op /* a bit ugly but... */
1071 31f1a758 2021-04-22 op memcpy(&uri, &tab->uri, sizeof(tab->uri));
1072 f7a80b3c 2021-04-24 op phos_uri_set_query(&uri, ministate.buf);
1073 31f1a758 2021-04-22 op phos_serialize_uri(&uri, buf, sizeof(buf));
1074 5cd2ebb1 2021-03-11 op load_url_in_tab(tab, buf);
1075 5cd2ebb1 2021-03-11 op }
1076 5cd2ebb1 2021-03-11 op
1077 5cd2ebb1 2021-03-11 op static void
1078 5cd2ebb1 2021-03-11 op lu_self_insert(void)
1079 5cd2ebb1 2021-03-11 op {
1080 17e5106b 2021-03-21 op if (thiskey.meta || unicode_isspace(thiskey.key) ||
1081 17e5106b 2021-03-21 op !unicode_isgraph(thiskey.key)) {
1082 5cd2ebb1 2021-03-11 op global_key_unbound();
1083 5cd2ebb1 2021-03-11 op return;
1084 5cd2ebb1 2021-03-11 op }
1085 5cd2ebb1 2021-03-11 op
1086 5cd2ebb1 2021-03-11 op minibuffer_self_insert();
1087 5cd2ebb1 2021-03-11 op }
1088 5cd2ebb1 2021-03-11 op
1089 5cd2ebb1 2021-03-11 op static void
1090 5cd2ebb1 2021-03-11 op lu_select(void)
1091 5cd2ebb1 2021-03-11 op {
1092 5cd2ebb1 2021-03-11 op exit_minibuffer();
1093 22268e11 2021-03-11 op minibuffer_hist_save_entry();
1094 5cd2ebb1 2021-03-11 op load_url_in_tab(current_tab(), ministate.buf);
1095 740f578b 2021-03-15 op }
1096 740f578b 2021-03-15 op
1097 740f578b 2021-03-15 op static void
1098 740f578b 2021-03-15 op bp_select(void)
1099 740f578b 2021-03-15 op {
1100 740f578b 2021-03-15 op exit_minibuffer();
1101 740f578b 2021-03-15 op if (*ministate.buf != '\0')
1102 740f578b 2021-03-15 op add_to_bookmarks(ministate.buf);
1103 740f578b 2021-03-15 op else
1104 740f578b 2021-03-15 op message("Abort.");
1105 5d1bac73 2021-03-25 op }
1106 5d1bac73 2021-03-25 op
1107 5d1bac73 2021-03-25 op static void
1108 5d1bac73 2021-03-25 op yornp_self_insert(void)
1109 5d1bac73 2021-03-25 op {
1110 5d1bac73 2021-03-25 op if (thiskey.key != 'y' && thiskey.key != 'n') {
1111 5d1bac73 2021-03-25 op message("Please answer y or n");
1112 5d1bac73 2021-03-25 op return;
1113 5d1bac73 2021-03-25 op }
1114 5d1bac73 2021-03-25 op
1115 5d1bac73 2021-03-25 op exit_minibuffer();
1116 fef9de8f 2021-04-25 op yornp_cb(thiskey.key == 'y', yornp_data);
1117 5d1bac73 2021-03-25 op }
1118 5d1bac73 2021-03-25 op
1119 5d1bac73 2021-03-25 op static void
1120 5d1bac73 2021-03-25 op yornp_abort(void)
1121 5d1bac73 2021-03-25 op {
1122 5d1bac73 2021-03-25 op exit_minibuffer();
1123 fef9de8f 2021-04-25 op yornp_cb(0, yornp_data);
1124 2a4ad912 2021-03-08 op }
1125 2a4ad912 2021-03-08 op
1126 9a25f829 2021-03-14 op static struct vline *
1127 2ba66cea 2021-03-22 op nth_line(struct window *window, size_t n)
1128 48e9d457 2021-03-06 op {
1129 9a25f829 2021-03-14 op struct vline *vl;
1130 48e9d457 2021-03-06 op size_t i;
1131 48e9d457 2021-03-06 op
1132 48e9d457 2021-03-06 op i = 0;
1133 2ba66cea 2021-03-22 op TAILQ_FOREACH(vl, &window->head, vlines) {
1134 48e9d457 2021-03-06 op if (i == n)
1135 9a25f829 2021-03-14 op return vl;
1136 48e9d457 2021-03-06 op i++;
1137 48e9d457 2021-03-06 op }
1138 48e9d457 2021-03-06 op
1139 48e9d457 2021-03-06 op /* unreachable */
1140 48e9d457 2021-03-06 op abort();
1141 48e9d457 2021-03-06 op }
1142 48e9d457 2021-03-06 op
1143 5e11c00c 2021-03-02 op static struct tab *
1144 5e11c00c 2021-03-02 op current_tab(void)
1145 5e11c00c 2021-03-02 op {
1146 5e11c00c 2021-03-02 op struct tab *t;
1147 5e11c00c 2021-03-02 op
1148 5e11c00c 2021-03-02 op TAILQ_FOREACH(t, &tabshead, tabs) {
1149 5e11c00c 2021-03-02 op if (t->flags & TAB_CURRENT)
1150 5e11c00c 2021-03-02 op return t;
1151 5e11c00c 2021-03-02 op }
1152 5e11c00c 2021-03-02 op
1153 5e11c00c 2021-03-02 op /* unreachable */
1154 5e11c00c 2021-03-02 op abort();
1155 5e11c00c 2021-03-02 op }
1156 5e11c00c 2021-03-02 op
1157 2ba66cea 2021-03-22 op static struct window *
1158 2ba66cea 2021-03-22 op current_window(void)
1159 2ba66cea 2021-03-22 op {
1160 2ba66cea 2021-03-22 op if (in_minibuffer)
1161 2ba66cea 2021-03-22 op return &ministate.window;
1162 2ba66cea 2021-03-22 op return &current_tab()->window;
1163 2ba66cea 2021-03-22 op }
1164 2ba66cea 2021-03-22 op
1165 8947c1f2 2021-03-21 op static int
1166 8947c1f2 2021-03-21 op readkey(void)
1167 5e11c00c 2021-03-02 op {
1168 8947c1f2 2021-03-21 op uint32_t state = 0;
1169 19f1448e 2021-03-08 op
1170 8947c1f2 2021-03-21 op if ((thiskey.key = wgetch(body)) == ERR)
1171 8947c1f2 2021-03-21 op return 0;
1172 5e11c00c 2021-03-02 op
1173 8947c1f2 2021-03-21 op thiskey.meta = thiskey.key == 27;
1174 8947c1f2 2021-03-21 op if (thiskey.meta) {
1175 9ca15951 2021-03-09 op thiskey.key = wgetch(body);
1176 c314a314 2021-03-11 op if (thiskey.key == ERR || thiskey.key == 27) {
1177 c314a314 2021-03-11 op thiskey.meta = 0;
1178 9ca15951 2021-03-09 op thiskey.key = 27;
1179 c314a314 2021-03-11 op }
1180 8947c1f2 2021-03-21 op }
1181 8947c1f2 2021-03-21 op
1182 8947c1f2 2021-03-21 op thiskey.cp = 0;
1183 8947c1f2 2021-03-21 op if ((unsigned int)thiskey.key < UINT8_MAX) {
1184 8947c1f2 2021-03-21 op while (1) {
1185 8947c1f2 2021-03-21 op if (!utf8_decode(&state, &thiskey.cp, (uint8_t)thiskey.key))
1186 8947c1f2 2021-03-21 op break;
1187 8947c1f2 2021-03-21 op if ((thiskey.key = wgetch(body)) == ERR) {
1188 8947c1f2 2021-03-21 op message("Error decoding user input");
1189 8947c1f2 2021-03-21 op return 0;
1190 8947c1f2 2021-03-21 op }
1191 8947c1f2 2021-03-21 op }
1192 8947c1f2 2021-03-21 op }
1193 19f1448e 2021-03-08 op
1194 8947c1f2 2021-03-21 op return 1;
1195 8947c1f2 2021-03-21 op }
1196 8947c1f2 2021-03-21 op
1197 8947c1f2 2021-03-21 op static void
1198 8947c1f2 2021-03-21 op dispatch_stdio(int fd, short ev, void *d)
1199 8947c1f2 2021-03-21 op {
1200 8947c1f2 2021-03-21 op struct keymap *k;
1201 8947c1f2 2021-03-21 op const char *keyname;
1202 8947c1f2 2021-03-21 op char tmp[5] = {0};
1203 8947c1f2 2021-03-21 op
1204 8947c1f2 2021-03-21 op if (!readkey())
1205 8947c1f2 2021-03-21 op return;
1206 8947c1f2 2021-03-21 op
1207 7c7d7bb7 2021-03-10 op if (keybuf[0] != '\0')
1208 7c7d7bb7 2021-03-10 op strlcat(keybuf, " ", sizeof(keybuf));
1209 7c7d7bb7 2021-03-10 op if (thiskey.meta)
1210 7c7d7bb7 2021-03-10 op strlcat(keybuf, "M-", sizeof(keybuf));
1211 8947c1f2 2021-03-21 op if (thiskey.cp != 0) {
1212 8947c1f2 2021-03-21 op utf8_encode(thiskey.cp, tmp);
1213 7c7d7bb7 2021-03-10 op strlcat(keybuf, tmp, sizeof(keybuf));
1214 8947c1f2 2021-03-21 op } else {
1215 8947c1f2 2021-03-21 op if ((keyname = unkbd(thiskey.key)) != NULL)
1216 8947c1f2 2021-03-21 op strlcat(keybuf, keyname, sizeof(keybuf));
1217 8947c1f2 2021-03-21 op else {
1218 8947c1f2 2021-03-21 op tmp[0] = thiskey.key;
1219 8947c1f2 2021-03-21 op strlcat(keybuf, tmp, sizeof(keybuf));
1220 8947c1f2 2021-03-21 op }
1221 7c7d7bb7 2021-03-10 op }
1222 7c7d7bb7 2021-03-10 op
1223 9ca15951 2021-03-09 op TAILQ_FOREACH(k, &current_map->m, keymaps) {
1224 9ca15951 2021-03-09 op if (k->meta == thiskey.meta &&
1225 9ca15951 2021-03-09 op k->key == thiskey.key) {
1226 f832146f 2021-03-09 op if (k->fn == NULL)
1227 f832146f 2021-03-09 op current_map = &k->map;
1228 f832146f 2021-03-09 op else {
1229 9ca15951 2021-03-09 op current_map = base_map;
1230 7c7d7bb7 2021-03-10 op strlcpy(keybuf, "", sizeof(keybuf));
1231 2ba66cea 2021-03-22 op k->fn(current_window());
1232 f832146f 2021-03-09 op }
1233 1d08c280 2021-03-06 op goto done;
1234 1d08c280 2021-03-06 op }
1235 eb259e66 2021-03-02 op }
1236 eb259e66 2021-03-02 op
1237 7c7d7bb7 2021-03-10 op if (current_map->unhandled_input != NULL)
1238 7c7d7bb7 2021-03-10 op current_map->unhandled_input();
1239 7c7d7bb7 2021-03-10 op else {
1240 7c7d7bb7 2021-03-10 op global_key_unbound();
1241 7c7d7bb7 2021-03-10 op }
1242 7c7d7bb7 2021-03-10 op
1243 7c7d7bb7 2021-03-10 op strlcpy(keybuf, "", sizeof(keybuf));
1244 9ca15951 2021-03-09 op current_map = base_map;
1245 1d08c280 2021-03-06 op
1246 1d08c280 2021-03-06 op done:
1247 bddc7bbd 2021-04-01 op if (side_window)
1248 bddc7bbd 2021-04-01 op recompute_help();
1249 bddc7bbd 2021-04-01 op
1250 179f0f58 2021-03-11 op redraw_tab(current_tab());
1251 a6d450c1 2021-03-06 op }
1252 48e9d457 2021-03-06 op
1253 a6d450c1 2021-03-06 op static void
1254 a6d450c1 2021-03-06 op handle_clear_minibuf(int fd, short ev, void *d)
1255 a6d450c1 2021-03-06 op {
1256 9cb0f9ce 2021-03-10 op free(ministate.curmesg);
1257 9cb0f9ce 2021-03-10 op ministate.curmesg = NULL;
1258 9cb0f9ce 2021-03-10 op
1259 9cb0f9ce 2021-03-10 op redraw_minibuffer();
1260 9cb0f9ce 2021-03-10 op if (in_minibuffer) {
1261 9cb0f9ce 2021-03-10 op wrefresh(body);
1262 9cb0f9ce 2021-03-10 op wrefresh(minibuf);
1263 9cb0f9ce 2021-03-10 op } else {
1264 9cb0f9ce 2021-03-10 op wrefresh(minibuf);
1265 9cb0f9ce 2021-03-10 op wrefresh(body);
1266 9cb0f9ce 2021-03-10 op }
1267 5e11c00c 2021-03-02 op }
1268 5e11c00c 2021-03-02 op
1269 5e11c00c 2021-03-02 op static void
1270 5e11c00c 2021-03-02 op handle_resize(int sig, short ev, void *d)
1271 831deb20 2021-05-12 op {
1272 831deb20 2021-05-12 op if (event_pending(&resizeev, EV_TIMEOUT, NULL)) {
1273 831deb20 2021-05-12 op event_del(&resizeev);
1274 831deb20 2021-05-12 op }
1275 831deb20 2021-05-12 op evtimer_set(&resizeev, handle_resize_timeout, NULL);
1276 831deb20 2021-05-12 op evtimer_add(&resizeev, &resize_timer);
1277 831deb20 2021-05-12 op }
1278 831deb20 2021-05-12 op
1279 831deb20 2021-05-12 op static void
1280 831deb20 2021-05-12 op handle_resize_timeout(int s, short ev, void *d)
1281 5e11c00c 2021-03-02 op {
1282 1d08c280 2021-03-06 op struct tab *tab;
1283 1d08c280 2021-03-06 op
1284 5e11c00c 2021-03-02 op endwin();
1285 5e11c00c 2021-03-02 op refresh();
1286 5e11c00c 2021-03-02 op clear();
1287 5e11c00c 2021-03-02 op
1288 48e9d457 2021-03-06 op /* move and resize the windows, in reverse order! */
1289 48e9d457 2021-03-06 op
1290 b1738d2e 2021-03-06 op mvwin(minibuf, LINES-1, 0);
1291 48e9d457 2021-03-06 op wresize(minibuf, 1, COLS);
1292 48e9d457 2021-03-06 op
1293 48e9d457 2021-03-06 op mvwin(modeline, LINES-2, 0);
1294 48e9d457 2021-03-06 op wresize(modeline, 1, COLS);
1295 48e9d457 2021-03-06 op
1296 48e9d457 2021-03-06 op body_lines = LINES-3;
1297 bd9637e9 2021-03-06 op body_cols = COLS;
1298 bddc7bbd 2021-04-01 op
1299 bddc7bbd 2021-04-01 op if (side_window) {
1300 bddc7bbd 2021-04-01 op help_cols = 0.3 * COLS;
1301 bddc7bbd 2021-04-01 op help_lines = LINES-3;
1302 bddc7bbd 2021-04-01 op mvwin(help, 1, 0);
1303 bddc7bbd 2021-04-01 op wresize(help, help_lines, help_cols);
1304 48e9d457 2021-03-06 op
1305 bddc7bbd 2021-04-01 op wrap_page(&helpwin, help_cols);
1306 bddc7bbd 2021-04-01 op
1307 bddc7bbd 2021-04-01 op body_cols = COLS - help_cols - 1;
1308 bddc7bbd 2021-04-01 op mvwin(body, 1, help_cols);
1309 bddc7bbd 2021-04-01 op } else
1310 bddc7bbd 2021-04-01 op mvwin(body, 1, 0);
1311 bddc7bbd 2021-04-01 op
1312 bddc7bbd 2021-04-01 op wresize(body, body_lines, body_cols);
1313 bddc7bbd 2021-04-01 op
1314 48e9d457 2021-03-06 op wresize(tabline, 1, COLS);
1315 48e9d457 2021-03-06 op
1316 1d08c280 2021-03-06 op tab = current_tab();
1317 1d08c280 2021-03-06 op
1318 bddc7bbd 2021-04-01 op wrap_page(&tab->window, body_cols);
1319 1d08c280 2021-03-06 op redraw_tab(tab);
1320 eb259e66 2021-03-02 op }
1321 eb259e66 2021-03-02 op
1322 1d08c280 2021-03-06 op static int
1323 bddc7bbd 2021-04-01 op wrap_page(struct window *window, int width)
1324 1d08c280 2021-03-06 op {
1325 452589f7 2021-03-16 op struct line *l;
1326 452589f7 2021-03-16 op const struct line *orig;
1327 d511dc85 2021-03-16 op struct vline *vl;
1328 452589f7 2021-03-16 op const char *prfx;
1329 452589f7 2021-03-16 op
1330 2ba66cea 2021-03-22 op orig = window->current_line == NULL
1331 452589f7 2021-03-16 op ? NULL
1332 2ba66cea 2021-03-22 op : window->current_line->parent;
1333 2ba66cea 2021-03-22 op window->current_line = NULL;
1334 452589f7 2021-03-16 op
1335 2ba66cea 2021-03-22 op window->curs_y = 0;
1336 2ba66cea 2021-03-22 op window->line_off = 0;
1337 1d08c280 2021-03-06 op
1338 2ba66cea 2021-03-22 op empty_vlist(window);
1339 1d08c280 2021-03-06 op
1340 2ba66cea 2021-03-22 op TAILQ_FOREACH(l, &window->page.head, lines) {
1341 0d568960 2021-03-11 op prfx = line_prefixes[l->type].prfx1;
1342 5e11c00c 2021-03-02 op switch (l->type) {
1343 5e11c00c 2021-03-02 op case LINE_TEXT:
1344 5e11c00c 2021-03-02 op case LINE_LINK:
1345 5e11c00c 2021-03-02 op case LINE_TITLE_1:
1346 5e11c00c 2021-03-02 op case LINE_TITLE_2:
1347 5e11c00c 2021-03-02 op case LINE_TITLE_3:
1348 5e11c00c 2021-03-02 op case LINE_ITEM:
1349 5e11c00c 2021-03-02 op case LINE_QUOTE:
1350 5e11c00c 2021-03-02 op case LINE_PRE_START:
1351 5e11c00c 2021-03-02 op case LINE_PRE_END:
1352 bddc7bbd 2021-04-01 op wrap_text(window, prfx, l, width);
1353 5e11c00c 2021-03-02 op break;
1354 5e11c00c 2021-03-02 op case LINE_PRE_CONTENT:
1355 bddc7bbd 2021-04-01 op hardwrap_text(window, l, width);
1356 5e11c00c 2021-03-02 op break;
1357 452589f7 2021-03-16 op }
1358 452589f7 2021-03-16 op
1359 2ba66cea 2021-03-22 op if (orig == l && window->current_line == NULL) {
1360 2ba66cea 2021-03-22 op window->line_off = window->line_max-1;
1361 2ba66cea 2021-03-22 op window->current_line = TAILQ_LAST(&window->head, vhead);
1362 d511dc85 2021-03-16 op
1363 d511dc85 2021-03-16 op while (1) {
1364 2ba66cea 2021-03-22 op vl = TAILQ_PREV(window->current_line, vhead, vlines);
1365 d511dc85 2021-03-16 op if (vl == NULL || vl->parent != orig)
1366 d511dc85 2021-03-16 op break;
1367 2ba66cea 2021-03-22 op window->current_line = vl;
1368 2ba66cea 2021-03-22 op window->line_off--;
1369 d511dc85 2021-03-16 op }
1370 5e11c00c 2021-03-02 op }
1371 5e11c00c 2021-03-02 op }
1372 452589f7 2021-03-16 op
1373 2ba66cea 2021-03-22 op if (window->current_line == NULL)
1374 2ba66cea 2021-03-22 op window->current_line = TAILQ_FIRST(&window->head);
1375 452589f7 2021-03-16 op
1376 1d08c280 2021-03-06 op return 1;
1377 1d08c280 2021-03-06 op }
1378 5e11c00c 2021-03-02 op
1379 754622a2 2021-03-15 op static void
1380 bddc7bbd 2021-04-01 op print_vline(WINDOW *window, struct vline *vl)
1381 1d08c280 2021-03-06 op {
1382 9a25f829 2021-03-14 op const char *text = vl->line;
1383 768db5bb 2021-03-12 op const char *prfx;
1384 803ff456 2021-03-17 op int prefix_face = line_faces[vl->parent->type].prefix_prop;
1385 803ff456 2021-03-17 op int text_face = line_faces[vl->parent->type].text_prop;
1386 bd9637e9 2021-03-06 op
1387 9a25f829 2021-03-14 op if (!vl->flags)
1388 9a25f829 2021-03-14 op prfx = line_prefixes[vl->parent->type].prfx1;
1389 768db5bb 2021-03-12 op else
1390 9a25f829 2021-03-14 op prfx = line_prefixes[vl->parent->type].prfx2;
1391 768db5bb 2021-03-12 op
1392 bd9637e9 2021-03-06 op if (text == NULL)
1393 bd9637e9 2021-03-06 op text = "";
1394 bd9637e9 2021-03-06 op
1395 bddc7bbd 2021-04-01 op wattron(window, prefix_face);
1396 bddc7bbd 2021-04-01 op wprintw(window, "%s", prfx);
1397 bddc7bbd 2021-04-01 op wattroff(window, prefix_face);
1398 803ff456 2021-03-17 op
1399 bddc7bbd 2021-04-01 op wattron(window, text_face);
1400 bddc7bbd 2021-04-01 op wprintw(window, "%s", text);
1401 bddc7bbd 2021-04-01 op wattroff(window, text_face);
1402 1d08c280 2021-03-06 op }
1403 1d08c280 2021-03-06 op
1404 1d08c280 2021-03-06 op static void
1405 8af5e5ed 2021-03-08 op redraw_tabline(void)
1406 8af5e5ed 2021-03-08 op {
1407 7c7d7bb7 2021-03-10 op struct tab *tab;
1408 8f127baa 2021-04-22 op size_t toskip, ots, tabwidth, space, x;
1409 8f127baa 2021-04-22 op int current, y, truncated;
1410 dc5df781 2021-03-13 op const char *title;
1411 119f393c 2021-03-16 op char buf[25];
1412 7c7d7bb7 2021-03-10 op
1413 8f127baa 2021-04-22 op tabwidth = sizeof(buf)+1;
1414 8f127baa 2021-04-22 op space = COLS-2;
1415 8f127baa 2021-04-22 op
1416 a636f50e 2021-03-16 op toskip = 0;
1417 a636f50e 2021-03-16 op TAILQ_FOREACH(tab, &tabshead, tabs) {
1418 a636f50e 2021-03-16 op toskip++;
1419 a636f50e 2021-03-16 op if (tab->flags & TAB_CURRENT)
1420 a636f50e 2021-03-16 op break;
1421 a636f50e 2021-03-16 op }
1422 8f127baa 2021-04-22 op
1423 8f127baa 2021-04-22 op if (toskip * tabwidth < space)
1424 8f127baa 2021-04-22 op toskip = 0;
1425 8f127baa 2021-04-22 op else {
1426 8f127baa 2021-04-22 op ots = toskip;
1427 a636f50e 2021-03-16 op toskip--;
1428 8f127baa 2021-04-22 op while (toskip != 0 &&
1429 8f127baa 2021-04-22 op (ots - toskip+1) * tabwidth < space)
1430 8f127baa 2021-04-22 op toskip--;
1431 8f127baa 2021-04-22 op }
1432 7c7d7bb7 2021-03-10 op
1433 a636f50e 2021-03-16 op werase(tabline);
1434 cb6c7aa0 2021-03-16 op wattron(tabline, tab_face.background);
1435 a636f50e 2021-03-16 op wprintw(tabline, toskip == 0 ? " " : "<");
1436 cb6c7aa0 2021-03-16 op wattroff(tabline, tab_face.background);
1437 a636f50e 2021-03-16 op
1438 a636f50e 2021-03-16 op truncated = 0;
1439 7c7d7bb7 2021-03-10 op TAILQ_FOREACH(tab, &tabshead, tabs) {
1440 a636f50e 2021-03-16 op if (truncated)
1441 a636f50e 2021-03-16 op break;
1442 a636f50e 2021-03-16 op if (toskip != 0) {
1443 a636f50e 2021-03-16 op toskip--;
1444 a636f50e 2021-03-16 op continue;
1445 a636f50e 2021-03-16 op }
1446 a636f50e 2021-03-16 op
1447 a636f50e 2021-03-16 op getyx(tabline, y, x);
1448 a636f50e 2021-03-16 op if (x + sizeof(buf)+2 >= (size_t)COLS)
1449 a636f50e 2021-03-16 op truncated = 1;
1450 a636f50e 2021-03-16 op
1451 7c7d7bb7 2021-03-10 op current = tab->flags & TAB_CURRENT;
1452 a329982b 2021-03-11 op
1453 2ba66cea 2021-03-22 op if (*(title = tab->window.page.title) == '\0')
1454 2051e653 2021-03-13 op title = tab->hist_cur->h;
1455 dc5df781 2021-03-13 op
1456 e8a76665 2021-05-12 op if (tab->flags & TAB_URGENT)
1457 e8a76665 2021-05-12 op strlcpy(buf, "!", sizeof(buf));
1458 e8a76665 2021-05-12 op else
1459 e8a76665 2021-05-12 op strlcpy(buf, " ", sizeof(buf));
1460 e8a76665 2021-05-12 op
1461 119f393c 2021-03-16 op if (strlcat(buf, title, sizeof(buf)) >= sizeof(buf)) {
1462 119f393c 2021-03-16 op /* truncation happens */
1463 119f393c 2021-03-16 op strlcpy(&buf[sizeof(buf)-4], "...", 4);
1464 119f393c 2021-03-16 op } else {
1465 119f393c 2021-03-16 op /* pad with spaces */
1466 119f393c 2021-03-16 op while (strlcat(buf, " ", sizeof(buf)) < sizeof(buf))
1467 119f393c 2021-03-16 op /* nop */ ;
1468 119f393c 2021-03-16 op }
1469 a329982b 2021-03-11 op
1470 a329982b 2021-03-11 op if (current)
1471 cb6c7aa0 2021-03-16 op wattron(tabline, tab_face.current_tab);
1472 cb6c7aa0 2021-03-16 op else
1473 cb6c7aa0 2021-03-16 op wattron(tabline, tab_face.tab);
1474 119f393c 2021-03-16 op
1475 119f393c 2021-03-16 op wprintw(tabline, "%s", buf);
1476 119f393c 2021-03-16 op if (TAILQ_NEXT(tab, tabs) != NULL)
1477 119f393c 2021-03-16 op wprintw(tabline, " ");
1478 cb6c7aa0 2021-03-16 op
1479 cb6c7aa0 2021-03-16 op if (current)
1480 cb6c7aa0 2021-03-16 op wattroff(tabline, tab_face.current_tab);
1481 cb6c7aa0 2021-03-16 op else
1482 cb6c7aa0 2021-03-16 op wattroff(tabline, tab_face.tab);
1483 7c7d7bb7 2021-03-10 op }
1484 119f393c 2021-03-16 op
1485 cb6c7aa0 2021-03-16 op wattron(tabline, tab_face.background);
1486 8f127baa 2021-04-22 op for (; x < (size_t)COLS; ++x)
1487 119f393c 2021-03-16 op waddch(tabline, ' ');
1488 a636f50e 2021-03-16 op if (truncated)
1489 a636f50e 2021-03-16 op mvwprintw(tabline, 0, COLS-1, ">");
1490 10346511 2021-03-17 op }
1491 10346511 2021-03-17 op
1492 bddc7bbd 2021-04-01 op static void
1493 bddc7bbd 2021-04-01 op redraw_window(WINDOW *win, int height, struct window *window)
1494 bddc7bbd 2021-04-01 op {
1495 bddc7bbd 2021-04-01 op struct vline *vl;
1496 bddc7bbd 2021-04-01 op int l;
1497 bddc7bbd 2021-04-01 op
1498 bddc7bbd 2021-04-01 op werase(win);
1499 bddc7bbd 2021-04-01 op
1500 bddc7bbd 2021-04-01 op window->line_off = MIN(window->line_max-1, window->line_off);
1501 bddc7bbd 2021-04-01 op if (TAILQ_EMPTY(&window->head))
1502 bddc7bbd 2021-04-01 op return;
1503 bddc7bbd 2021-04-01 op
1504 bddc7bbd 2021-04-01 op l = 0;
1505 bddc7bbd 2021-04-01 op vl = nth_line(window, window->line_off);
1506 bddc7bbd 2021-04-01 op for (; vl != NULL; vl = TAILQ_NEXT(vl, vlines)) {
1507 bddc7bbd 2021-04-01 op wmove(win, l, 0);
1508 bddc7bbd 2021-04-01 op print_vline(win, vl);
1509 bddc7bbd 2021-04-01 op l++;
1510 bddc7bbd 2021-04-01 op if (l == height)
1511 bddc7bbd 2021-04-01 op break;
1512 bddc7bbd 2021-04-01 op }
1513 bddc7bbd 2021-04-01 op
1514 bddc7bbd 2021-04-01 op wmove(win, window->curs_y, window->curs_x);
1515 bddc7bbd 2021-04-01 op }
1516 bddc7bbd 2021-04-01 op
1517 bddc7bbd 2021-04-01 op static void
1518 bddc7bbd 2021-04-01 op redraw_help(void)
1519 bddc7bbd 2021-04-01 op {
1520 bddc7bbd 2021-04-01 op redraw_window(help, help_lines, &helpwin);
1521 bddc7bbd 2021-04-01 op }
1522 bddc7bbd 2021-04-01 op
1523 bddc7bbd 2021-04-01 op static void
1524 bddc7bbd 2021-04-01 op redraw_body(struct tab *tab)
1525 bddc7bbd 2021-04-01 op {
1526 bddc7bbd 2021-04-01 op redraw_window(body, body_lines, &tab->window);
1527 bddc7bbd 2021-04-01 op }
1528 bddc7bbd 2021-04-01 op
1529 10346511 2021-03-17 op static inline char
1530 10346511 2021-03-17 op trust_status_char(enum trust_state ts)
1531 10346511 2021-03-17 op {
1532 10346511 2021-03-17 op switch (ts) {
1533 10346511 2021-03-17 op case TS_UNKNOWN: return 'u';
1534 10346511 2021-03-17 op case TS_UNTRUSTED: return '!';
1535 10346511 2021-03-17 op case TS_TRUSTED: return 'v';
1536 10346511 2021-03-17 op case TS_VERIFIED: return 'V';
1537 10346511 2021-03-17 op }
1538 8af5e5ed 2021-03-08 op }
1539 8af5e5ed 2021-03-08 op
1540 8af5e5ed 2021-03-08 op static void
1541 48e9d457 2021-03-06 op redraw_modeline(struct tab *tab)
1542 1d08c280 2021-03-06 op {
1543 481340cc 2021-03-11 op double pct;
1544 48e9d457 2021-03-06 op int x, y, max_x, max_y;
1545 2ba66cea 2021-03-22 op const char *mode = tab->window.page.name;
1546 8af5e5ed 2021-03-08 op const char *spin = "-\\|/";
1547 1d08c280 2021-03-06 op
1548 156f1501 2021-03-13 op werase(modeline);
1549 48e9d457 2021-03-06 op wattron(modeline, A_REVERSE);
1550 48e9d457 2021-03-06 op wmove(modeline, 0, 0);
1551 1d08c280 2021-03-06 op
1552 10346511 2021-03-17 op wprintw(modeline, "-%c%c %s ",
1553 2ba66cea 2021-03-22 op spin[tab->loading_anim_step],
1554 10346511 2021-03-17 op trust_status_char(tab->trust),
1555 58c75fab 2021-03-16 op mode == NULL ? "(none)" : mode);
1556 481340cc 2021-03-11 op
1557 2ba66cea 2021-03-22 op pct = (tab->window.line_off + tab->window.curs_y) * 100.0 / tab->window.line_max;
1558 481340cc 2021-03-11 op
1559 2ba66cea 2021-03-22 op if (tab->window.line_max <= (size_t)body_lines)
1560 481340cc 2021-03-11 op wprintw(modeline, "All ");
1561 2ba66cea 2021-03-22 op else if (tab->window.line_off == 0)
1562 481340cc 2021-03-11 op wprintw(modeline, "Top ");
1563 2ba66cea 2021-03-22 op else if (tab->window.line_off + body_lines >= tab->window.line_max)
1564 481340cc 2021-03-11 op wprintw(modeline, "Bottom ");
1565 481340cc 2021-03-11 op else
1566 481340cc 2021-03-11 op wprintw(modeline, "%.0f%% ", pct);
1567 481340cc 2021-03-11 op
1568 481340cc 2021-03-11 op wprintw(modeline, "%d/%d %s ",
1569 2ba66cea 2021-03-22 op tab->window.line_off + tab->window.curs_y,
1570 2ba66cea 2021-03-22 op tab->window.line_max,
1571 2051e653 2021-03-13 op tab->hist_cur->h);
1572 481340cc 2021-03-11 op
1573 48e9d457 2021-03-06 op getyx(modeline, y, x);
1574 48e9d457 2021-03-06 op getmaxyx(modeline, max_y, max_x);
1575 48e9d457 2021-03-06 op
1576 48e9d457 2021-03-06 op (void)y;
1577 48e9d457 2021-03-06 op (void)max_y;
1578 48e9d457 2021-03-06 op
1579 48e9d457 2021-03-06 op for (; x < max_x; ++x)
1580 48e9d457 2021-03-06 op waddstr(modeline, "-");
1581 9ca15951 2021-03-09 op }
1582 9ca15951 2021-03-09 op
1583 9ca15951 2021-03-09 op static void
1584 9ca15951 2021-03-09 op redraw_minibuffer(void)
1585 9ca15951 2021-03-09 op {
1586 8300dd3c 2021-03-18 op struct tab *tab;
1587 17e5106b 2021-03-21 op size_t off_y, off_x = 0;
1588 2ba66cea 2021-03-22 op char *start, *c;
1589 9ca15951 2021-03-09 op
1590 156f1501 2021-03-13 op werase(minibuf);
1591 17e5106b 2021-03-21 op
1592 22268e11 2021-03-11 op if (in_minibuffer) {
1593 22268e11 2021-03-11 op mvwprintw(minibuf, 0, 0, "%s", ministate.prompt);
1594 22268e11 2021-03-11 op if (ministate.hist_cur != NULL)
1595 22268e11 2021-03-11 op wprintw(minibuf, "(%zu/%zu) ",
1596 22268e11 2021-03-11 op ministate.hist_off + 1,
1597 22268e11 2021-03-11 op ministate.history->len);
1598 9ca15951 2021-03-09 op
1599 22268e11 2021-03-11 op getyx(minibuf, off_y, off_x);
1600 9cb0f9ce 2021-03-10 op
1601 17e5106b 2021-03-21 op start = ministate.hist_cur != NULL
1602 17e5106b 2021-03-21 op ? ministate.hist_cur->h
1603 17e5106b 2021-03-21 op : ministate.buf;
1604 2ba66cea 2021-03-22 op c = utf8_nth(ministate.window.current_line->line,
1605 2ba66cea 2021-03-22 op ministate.window.cpoff);
1606 2ba66cea 2021-03-22 op while (utf8_swidth_between(start, c) > (size_t)COLS/2) {
1607 17e5106b 2021-03-21 op start = utf8_next_cp(start);
1608 22268e11 2021-03-11 op }
1609 9cb0f9ce 2021-03-10 op
1610 17e5106b 2021-03-21 op waddstr(minibuf, start);
1611 22268e11 2021-03-11 op }
1612 9cb0f9ce 2021-03-10 op
1613 17e5106b 2021-03-21 op if (ministate.curmesg != NULL)
1614 17e5106b 2021-03-21 op wprintw(minibuf, in_minibuffer ? " [%s]" : "%s",
1615 17e5106b 2021-03-21 op ministate.curmesg);
1616 9cb0f9ce 2021-03-10 op
1617 91a72220 2021-03-13 op if (!in_minibuffer && ministate.curmesg == NULL)
1618 17e5106b 2021-03-21 op waddstr(minibuf, keybuf);
1619 8300dd3c 2021-03-18 op
1620 8300dd3c 2021-03-18 op /* If nothing else, show the URL at point */
1621 8300dd3c 2021-03-18 op if (!in_minibuffer && ministate.curmesg == NULL && *keybuf == '\0') {
1622 8300dd3c 2021-03-18 op tab = current_tab();
1623 2ba66cea 2021-03-22 op if (tab->window.current_line != NULL &&
1624 2ba66cea 2021-03-22 op tab->window.current_line->parent->type == LINE_LINK)
1625 2ba66cea 2021-03-22 op waddstr(minibuf, tab->window.current_line->parent->alt);
1626 8300dd3c 2021-03-18 op }
1627 91a72220 2021-03-13 op
1628 2aaf475e 2021-03-13 op if (in_minibuffer)
1629 2ba66cea 2021-03-22 op wmove(minibuf, 0, off_x + utf8_swidth_between(start, c));
1630 48e9d457 2021-03-06 op }
1631 48e9d457 2021-03-06 op
1632 48e9d457 2021-03-06 op static void
1633 48e9d457 2021-03-06 op redraw_tab(struct tab *tab)
1634 48e9d457 2021-03-06 op {
1635 bddc7bbd 2021-04-01 op if (side_window) {
1636 bddc7bbd 2021-04-01 op redraw_help();
1637 bddc7bbd 2021-04-01 op wnoutrefresh(help);
1638 bddc7bbd 2021-04-01 op }
1639 bddc7bbd 2021-04-01 op
1640 e19f9a04 2021-03-11 op redraw_tabline();
1641 e19f9a04 2021-03-11 op redraw_body(tab);
1642 e19f9a04 2021-03-11 op redraw_modeline(tab);
1643 e19f9a04 2021-03-11 op redraw_minibuffer();
1644 e19f9a04 2021-03-11 op
1645 bddc7bbd 2021-04-01 op wnoutrefresh(tabline);
1646 bddc7bbd 2021-04-01 op wnoutrefresh(modeline);
1647 e19f9a04 2021-03-11 op
1648 e19f9a04 2021-03-11 op if (in_minibuffer) {
1649 bddc7bbd 2021-04-01 op wnoutrefresh(body);
1650 bddc7bbd 2021-04-01 op wnoutrefresh(minibuf);
1651 e19f9a04 2021-03-11 op } else {
1652 bddc7bbd 2021-04-01 op wnoutrefresh(minibuf);
1653 bddc7bbd 2021-04-01 op wnoutrefresh(body);
1654 e19f9a04 2021-03-11 op }
1655 bddc7bbd 2021-04-01 op
1656 bddc7bbd 2021-04-01 op doupdate();
1657 e19f9a04 2021-03-11 op }
1658 e19f9a04 2021-03-11 op
1659 e19f9a04 2021-03-11 op static void
1660 bddc7bbd 2021-04-01 op emit_help_item(char *prfx, void *fn)
1661 e19f9a04 2021-03-11 op {
1662 bddc7bbd 2021-04-01 op struct line *l;
1663 bddc7bbd 2021-04-01 op struct cmds *cmd;
1664 48e9d457 2021-03-06 op
1665 bddc7bbd 2021-04-01 op for (cmd = cmds; cmd->cmd != NULL; ++cmd) {
1666 bddc7bbd 2021-04-01 op if (fn == cmd->fn)
1667 bddc7bbd 2021-04-01 op break;
1668 bddc7bbd 2021-04-01 op }
1669 bddc7bbd 2021-04-01 op assert(cmd != NULL);
1670 48e9d457 2021-03-06 op
1671 bddc7bbd 2021-04-01 op if ((l = calloc(1, sizeof(*l))) == NULL)
1672 bddc7bbd 2021-04-01 op abort();
1673 48e9d457 2021-03-06 op
1674 bddc7bbd 2021-04-01 op l->type = LINE_TEXT;
1675 bddc7bbd 2021-04-01 op l->alt = NULL;
1676 bddc7bbd 2021-04-01 op
1677 bddc7bbd 2021-04-01 op asprintf(&l->line, "%s %s", prfx, cmd->cmd);
1678 bddc7bbd 2021-04-01 op
1679 bddc7bbd 2021-04-01 op if (TAILQ_EMPTY(&helpwin.page.head))
1680 bddc7bbd 2021-04-01 op TAILQ_INSERT_HEAD(&helpwin.page.head, l, lines);
1681 bddc7bbd 2021-04-01 op else
1682 bddc7bbd 2021-04-01 op TAILQ_INSERT_TAIL(&helpwin.page.head, l, lines);
1683 bddc7bbd 2021-04-01 op }
1684 bddc7bbd 2021-04-01 op
1685 bddc7bbd 2021-04-01 op static void
1686 bddc7bbd 2021-04-01 op rec_compute_help(struct kmap *keymap, char *prfx, size_t len)
1687 bddc7bbd 2021-04-01 op {
1688 bddc7bbd 2021-04-01 op struct keymap *k;
1689 bddc7bbd 2021-04-01 op char p[32];
1690 bddc7bbd 2021-04-01 op const char *kn;
1691 bddc7bbd 2021-04-01 op
1692 bddc7bbd 2021-04-01 op TAILQ_FOREACH(k, &keymap->m, keymaps) {
1693 bddc7bbd 2021-04-01 op strlcpy(p, prfx, sizeof(p));
1694 bddc7bbd 2021-04-01 op if (*p != '\0')
1695 bddc7bbd 2021-04-01 op strlcat(p, " ", sizeof(p));
1696 bddc7bbd 2021-04-01 op if (k->meta)
1697 bddc7bbd 2021-04-01 op strlcat(p, "M-", sizeof(p));
1698 bddc7bbd 2021-04-01 op if ((kn = unkbd(k->key)) != NULL)
1699 bddc7bbd 2021-04-01 op strlcat(p, kn, sizeof(p));
1700 bddc7bbd 2021-04-01 op else
1701 bddc7bbd 2021-04-01 op strlcat(p, keyname(k->key), sizeof(p));
1702 bddc7bbd 2021-04-01 op
1703 bddc7bbd 2021-04-01 op if (k->fn == NULL)
1704 bddc7bbd 2021-04-01 op rec_compute_help(&k->map, p, sizeof(p));
1705 bddc7bbd 2021-04-01 op else
1706 bddc7bbd 2021-04-01 op emit_help_item(p, k->fn);
1707 9ca15951 2021-03-09 op }
1708 bddc7bbd 2021-04-01 op }
1709 174b3cdf 2021-03-21 op
1710 bddc7bbd 2021-04-01 op static void
1711 bddc7bbd 2021-04-01 op recompute_help(void)
1712 bddc7bbd 2021-04-01 op {
1713 bddc7bbd 2021-04-01 op char p[32] = { 0 };
1714 bddc7bbd 2021-04-01 op
1715 bddc7bbd 2021-04-01 op empty_vlist(&helpwin);
1716 bddc7bbd 2021-04-01 op empty_linelist(&helpwin);
1717 bddc7bbd 2021-04-01 op rec_compute_help(current_map, p, sizeof(p));
1718 bddc7bbd 2021-04-01 op wrap_page(&helpwin, help_cols);
1719 7953dd72 2021-03-07 op }
1720 7953dd72 2021-03-07 op
1721 7953dd72 2021-03-07 op static void
1722 740f578b 2021-03-15 op vmessage(const char *fmt, va_list ap)
1723 7953dd72 2021-03-07 op {
1724 e0e26735 2021-03-16 op if (evtimer_pending(&clminibufev, NULL))
1725 7953dd72 2021-03-07 op evtimer_del(&clminibufev);
1726 7953dd72 2021-03-07 op evtimer_set(&clminibufev, handle_clear_minibuf, NULL);
1727 7953dd72 2021-03-07 op evtimer_add(&clminibufev, &clminibufev_timer);
1728 bf992581 2021-03-12 op
1729 bf992581 2021-03-12 op free(ministate.curmesg);
1730 7953dd72 2021-03-07 op
1731 9cb0f9ce 2021-03-10 op /* TODO: what to do if the allocation fails here? */
1732 9cb0f9ce 2021-03-10 op if (vasprintf(&ministate.curmesg, fmt, ap) == -1)
1733 9cb0f9ce 2021-03-10 op ministate.curmesg = NULL;
1734 9cb0f9ce 2021-03-10 op
1735 9cb0f9ce 2021-03-10 op redraw_minibuffer();
1736 9cb0f9ce 2021-03-10 op if (in_minibuffer) {
1737 9cb0f9ce 2021-03-10 op wrefresh(body);
1738 9cb0f9ce 2021-03-10 op wrefresh(minibuf);
1739 9cb0f9ce 2021-03-10 op } else {
1740 9cb0f9ce 2021-03-10 op wrefresh(minibuf);
1741 9cb0f9ce 2021-03-10 op wrefresh(body);
1742 9cb0f9ce 2021-03-10 op }
1743 bcb0b073 2021-03-07 op }
1744 bcb0b073 2021-03-07 op
1745 bcb0b073 2021-03-07 op static void
1746 740f578b 2021-03-15 op message(const char *fmt, ...)
1747 740f578b 2021-03-15 op {
1748 740f578b 2021-03-15 op va_list ap;
1749 740f578b 2021-03-15 op
1750 740f578b 2021-03-15 op va_start(ap, fmt);
1751 740f578b 2021-03-15 op vmessage(fmt, ap);
1752 740f578b 2021-03-15 op va_end(ap);
1753 740f578b 2021-03-15 op }
1754 740f578b 2021-03-15 op
1755 740f578b 2021-03-15 op static void
1756 8af5e5ed 2021-03-08 op start_loading_anim(struct tab *tab)
1757 8af5e5ed 2021-03-08 op {
1758 2ba66cea 2021-03-22 op if (tab->loading_anim)
1759 8af5e5ed 2021-03-08 op return;
1760 2ba66cea 2021-03-22 op tab->loading_anim = 1;
1761 2ba66cea 2021-03-22 op evtimer_set(&tab->loadingev, update_loading_anim, tab);
1762 2ba66cea 2021-03-22 op evtimer_add(&tab->loadingev, &loadingev_timer);
1763 8af5e5ed 2021-03-08 op }
1764 8af5e5ed 2021-03-08 op
1765 8af5e5ed 2021-03-08 op static void
1766 8af5e5ed 2021-03-08 op update_loading_anim(int fd, short ev, void *d)
1767 8af5e5ed 2021-03-08 op {
1768 8af5e5ed 2021-03-08 op struct tab *tab = d;
1769 8af5e5ed 2021-03-08 op
1770 2ba66cea 2021-03-22 op tab->loading_anim_step = (tab->loading_anim_step+1)%4;
1771 9ca15951 2021-03-09 op
1772 6347dcd0 2021-03-16 op if (tab->flags & TAB_CURRENT) {
1773 6347dcd0 2021-03-16 op redraw_modeline(tab);
1774 6347dcd0 2021-03-16 op wrefresh(modeline);
1775 6347dcd0 2021-03-16 op wrefresh(body);
1776 6347dcd0 2021-03-16 op if (in_minibuffer)
1777 6347dcd0 2021-03-16 op wrefresh(minibuf);
1778 6347dcd0 2021-03-16 op }
1779 8af5e5ed 2021-03-08 op
1780 2ba66cea 2021-03-22 op evtimer_add(&tab->loadingev, &loadingev_timer);
1781 8af5e5ed 2021-03-08 op }
1782 8af5e5ed 2021-03-08 op
1783 8af5e5ed 2021-03-08 op static void
1784 8af5e5ed 2021-03-08 op stop_loading_anim(struct tab *tab)
1785 8af5e5ed 2021-03-08 op {
1786 2ba66cea 2021-03-22 op if (!tab->loading_anim)
1787 8af5e5ed 2021-03-08 op return;
1788 2ba66cea 2021-03-22 op evtimer_del(&tab->loadingev);
1789 2ba66cea 2021-03-22 op tab->loading_anim = 0;
1790 2ba66cea 2021-03-22 op tab->loading_anim_step = 0;
1791 3d8c2326 2021-03-18 op
1792 3d8c2326 2021-03-18 op if (!(tab->flags & TAB_CURRENT))
1793 3d8c2326 2021-03-18 op return;
1794 43a1b8d0 2021-03-09 op
1795 43a1b8d0 2021-03-09 op redraw_modeline(tab);
1796 9ca15951 2021-03-09 op
1797 43a1b8d0 2021-03-09 op wrefresh(modeline);
1798 43a1b8d0 2021-03-09 op wrefresh(body);
1799 9ca15951 2021-03-09 op if (in_minibuffer)
1800 9ca15951 2021-03-09 op wrefresh(minibuf);
1801 8af5e5ed 2021-03-08 op }
1802 8af5e5ed 2021-03-08 op
1803 8af5e5ed 2021-03-08 op static void
1804 43a1b8d0 2021-03-09 op load_url_in_tab(struct tab *tab, const char *url)
1805 8af5e5ed 2021-03-08 op {
1806 8af5e5ed 2021-03-08 op message("Loading %s...", url);
1807 8af5e5ed 2021-03-08 op start_loading_anim(tab);
1808 8af5e5ed 2021-03-08 op load_url(tab, url);
1809 43a1b8d0 2021-03-09 op
1810 2ba66cea 2021-03-22 op tab->window.curs_x = 0;
1811 2ba66cea 2021-03-22 op tab->window.curs_y = 0;
1812 43a1b8d0 2021-03-09 op redraw_tab(tab);
1813 8af5e5ed 2021-03-08 op }
1814 8af5e5ed 2021-03-08 op
1815 8af5e5ed 2021-03-08 op static void
1816 b360ebb3 2021-03-10 op enter_minibuffer(void (*self_insert_fn)(void), void (*donefn)(void),
1817 3148eeac 2021-03-13 op void (*abortfn)(void), struct histhead *hist)
1818 9ca15951 2021-03-09 op {
1819 9ca15951 2021-03-09 op in_minibuffer = 1;
1820 fa3fd864 2021-03-10 op base_map = &minibuffer_map;
1821 fa3fd864 2021-03-10 op current_map = &minibuffer_map;
1822 9ca15951 2021-03-09 op
1823 fa3fd864 2021-03-10 op base_map->unhandled_input = self_insert_fn;
1824 b360ebb3 2021-03-10 op
1825 b360ebb3 2021-03-10 op ministate.donefn = donefn;
1826 b360ebb3 2021-03-10 op ministate.abortfn = abortfn;
1827 9ca15951 2021-03-09 op memset(ministate.buf, 0, sizeof(ministate.buf));
1828 2ba66cea 2021-03-22 op ministate.window.current_line = &ministate.vline;
1829 2ba66cea 2021-03-22 op ministate.window.current_line->line = ministate.buf;
1830 2ba66cea 2021-03-22 op ministate.window.cpoff = 0;
1831 9ca15951 2021-03-09 op strlcpy(ministate.buf, "", sizeof(ministate.prompt));
1832 22268e11 2021-03-11 op
1833 22268e11 2021-03-11 op ministate.history = hist;
1834 22268e11 2021-03-11 op ministate.hist_cur = NULL;
1835 22268e11 2021-03-11 op ministate.hist_off = 0;
1836 9ca15951 2021-03-09 op }
1837 9ca15951 2021-03-09 op
1838 9ca15951 2021-03-09 op static void
1839 b360ebb3 2021-03-10 op exit_minibuffer(void)
1840 9ca15951 2021-03-09 op {
1841 156f1501 2021-03-13 op werase(minibuf);
1842 9ca15951 2021-03-09 op
1843 9ca15951 2021-03-09 op in_minibuffer = 0;
1844 9ca15951 2021-03-09 op base_map = &global_map;
1845 9ca15951 2021-03-09 op current_map = &global_map;
1846 9ca15951 2021-03-09 op }
1847 9ca15951 2021-03-09 op
1848 9ca15951 2021-03-09 op static void
1849 5cd2ebb1 2021-03-11 op switch_to_tab(struct tab *tab)
1850 5cd2ebb1 2021-03-11 op {
1851 5cd2ebb1 2021-03-11 op struct tab *t;
1852 5cd2ebb1 2021-03-11 op
1853 5cd2ebb1 2021-03-11 op TAILQ_FOREACH(t, &tabshead, tabs) {
1854 5cd2ebb1 2021-03-11 op t->flags &= ~TAB_CURRENT;
1855 5cd2ebb1 2021-03-11 op }
1856 5cd2ebb1 2021-03-11 op
1857 5cd2ebb1 2021-03-11 op tab->flags |= TAB_CURRENT;
1858 2eef3403 2021-04-22 op }
1859 2eef3403 2021-04-22 op
1860 2eef3403 2021-04-22 op unsigned int
1861 2eef3403 2021-04-22 op tab_new_id(void)
1862 2eef3403 2021-04-22 op {
1863 2eef3403 2021-04-22 op return tab_counter++;
1864 5cd2ebb1 2021-03-11 op }
1865 5cd2ebb1 2021-03-11 op
1866 b1df9b71 2021-03-12 op static struct tab *
1867 1b81bc33 2021-03-15 op new_tab(const char *url)
1868 bcb0b073 2021-03-07 op {
1869 754622a2 2021-03-15 op struct tab *tab;
1870 bcb0b073 2021-03-07 op
1871 71105afa 2021-03-16 op if ((tab = calloc(1, sizeof(*tab))) == NULL) {
1872 71105afa 2021-03-16 op event_loopbreak();
1873 71105afa 2021-03-16 op return NULL;
1874 71105afa 2021-03-16 op }
1875 bcb0b073 2021-03-07 op
1876 2051e653 2021-03-13 op TAILQ_INIT(&tab->hist.head);
1877 2051e653 2021-03-13 op
1878 2ba66cea 2021-03-22 op TAILQ_INIT(&tab->window.head);
1879 bcb0b073 2021-03-07 op
1880 2eef3403 2021-04-22 op tab->id = tab_new_id();
1881 5cd2ebb1 2021-03-11 op switch_to_tab(tab);
1882 bcb0b073 2021-03-07 op
1883 bcb0b073 2021-03-07 op if (TAILQ_EMPTY(&tabshead))
1884 bcb0b073 2021-03-07 op TAILQ_INSERT_HEAD(&tabshead, tab, tabs);
1885 bcb0b073 2021-03-07 op else
1886 bcb0b073 2021-03-07 op TAILQ_INSERT_TAIL(&tabshead, tab, tabs);
1887 bcb0b073 2021-03-07 op
1888 43a1b8d0 2021-03-09 op load_url_in_tab(tab, url);
1889 b1df9b71 2021-03-12 op return tab;
1890 c7107cec 2021-04-01 op }
1891 c7107cec 2021-04-01 op
1892 c7107cec 2021-04-01 op static void
1893 c7107cec 2021-04-01 op session_new_tab_cb(const char *url)
1894 c7107cec 2021-04-01 op {
1895 c7107cec 2021-04-01 op new_tab(url);
1896 5e11c00c 2021-03-02 op }
1897 5e11c00c 2021-03-02 op
1898 941b3761 2021-03-18 op static void
1899 941b3761 2021-03-18 op usage(void)
1900 5e11c00c 2021-03-02 op {
1901 941b3761 2021-03-18 op fprintf(stderr, "USAGE: %s [url]\n", getprogname());
1902 941b3761 2021-03-18 op }
1903 941b3761 2021-03-18 op
1904 941b3761 2021-03-18 op int
1905 6cd6a9e1 2021-03-20 op ui_init(int argc, char * const *argv)
1906 941b3761 2021-03-18 op {
1907 941b3761 2021-03-18 op const char *url = NEW_TAB_URL;
1908 941b3761 2021-03-18 op int ch;
1909 941b3761 2021-03-18 op
1910 941b3761 2021-03-18 op while ((ch = getopt(argc, argv, "")) != -1) {
1911 941b3761 2021-03-18 op switch (ch) {
1912 941b3761 2021-03-18 op default:
1913 941b3761 2021-03-18 op usage();
1914 941b3761 2021-03-18 op return 0;
1915 941b3761 2021-03-18 op }
1916 941b3761 2021-03-18 op }
1917 941b3761 2021-03-18 op argc -= optind;
1918 941b3761 2021-03-18 op argv += optind;
1919 941b3761 2021-03-18 op
1920 941b3761 2021-03-18 op if (argc != 0)
1921 941b3761 2021-03-18 op url = argv[0];
1922 941b3761 2021-03-18 op
1923 5e11c00c 2021-03-02 op setlocale(LC_ALL, "");
1924 5e11c00c 2021-03-02 op
1925 9ca15951 2021-03-09 op TAILQ_INIT(&global_map.m);
1926 9ca15951 2021-03-09 op global_map.unhandled_input = global_key_unbound;
1927 9ca15951 2021-03-09 op
1928 fa3fd864 2021-03-10 op TAILQ_INIT(&minibuffer_map.m);
1929 9ca15951 2021-03-09 op
1930 22268e11 2021-03-11 op TAILQ_INIT(&eecmd_history.head);
1931 22268e11 2021-03-11 op TAILQ_INIT(&ir_history.head);
1932 22268e11 2021-03-11 op TAILQ_INIT(&lu_history.head);
1933 22268e11 2021-03-11 op
1934 2ba66cea 2021-03-22 op ministate.line.type = LINE_TEXT;
1935 2ba66cea 2021-03-22 op ministate.vline.parent = &ministate.line;
1936 2ba66cea 2021-03-22 op ministate.window.current_line = &ministate.vline;
1937 2ba66cea 2021-03-22 op
1938 bddc7bbd 2021-04-01 op /* initialize help window */
1939 bddc7bbd 2021-04-01 op TAILQ_INIT(&helpwin.head);
1940 bddc7bbd 2021-04-01 op
1941 9ca15951 2021-03-09 op base_map = &global_map;
1942 f832146f 2021-03-09 op current_map = &global_map;
1943 f832146f 2021-03-09 op load_default_keys();
1944 f832146f 2021-03-09 op
1945 5e11c00c 2021-03-02 op initscr();
1946 15e1b108 2021-03-02 op raw();
1947 5e11c00c 2021-03-02 op noecho();
1948 5e11c00c 2021-03-02 op
1949 5e11c00c 2021-03-02 op nonl();
1950 5e11c00c 2021-03-02 op intrflush(stdscr, FALSE);
1951 5e11c00c 2021-03-02 op
1952 48e9d457 2021-03-06 op if ((tabline = newwin(1, COLS, 0, 0)) == NULL)
1953 48e9d457 2021-03-06 op return 0;
1954 48e9d457 2021-03-06 op if ((body = newwin(LINES - 3, COLS, 1, 0)) == NULL)
1955 48e9d457 2021-03-06 op return 0;
1956 48e9d457 2021-03-06 op if ((modeline = newwin(1, COLS, LINES-2, 0)) == NULL)
1957 48e9d457 2021-03-06 op return 0;
1958 48e9d457 2021-03-06 op if ((minibuf = newwin(1, COLS, LINES-1, 0)) == NULL)
1959 48e9d457 2021-03-06 op return 0;
1960 bddc7bbd 2021-04-01 op if ((help = newwin(1, 1, 1, 0)) == NULL)
1961 bddc7bbd 2021-04-01 op return 0;
1962 1d08c280 2021-03-06 op
1963 48e9d457 2021-03-06 op body_lines = LINES-3;
1964 48e9d457 2021-03-06 op body_cols = COLS;
1965 48e9d457 2021-03-06 op
1966 43a1b8d0 2021-03-09 op keypad(body, TRUE);
1967 48e9d457 2021-03-06 op scrollok(body, TRUE);
1968 48e9d457 2021-03-06 op
1969 5e11c00c 2021-03-02 op /* non-blocking input */
1970 48e9d457 2021-03-06 op wtimeout(body, 0);
1971 5e11c00c 2021-03-02 op
1972 48e9d457 2021-03-06 op mvwprintw(body, 0, 0, "");
1973 5e11c00c 2021-03-02 op
1974 5e11c00c 2021-03-02 op event_set(&stdioev, 0, EV_READ | EV_PERSIST, dispatch_stdio, NULL);
1975 5e11c00c 2021-03-02 op event_add(&stdioev, NULL);
1976 5e11c00c 2021-03-02 op
1977 5e11c00c 2021-03-02 op signal_set(&winchev, SIGWINCH, handle_resize, NULL);
1978 5e11c00c 2021-03-02 op signal_add(&winchev, NULL);
1979 5e11c00c 2021-03-02 op
1980 c7107cec 2021-04-01 op load_last_session(session_new_tab_cb);
1981 c7107cec 2021-04-01 op if (strcmp(url, NEW_TAB_URL) || TAILQ_EMPTY(&tabshead))
1982 c7107cec 2021-04-01 op new_tab(url);
1983 5e11c00c 2021-03-02 op
1984 1d08c280 2021-03-06 op return 1;
1985 5e11c00c 2021-03-02 op }
1986 5e11c00c 2021-03-02 op
1987 5e11c00c 2021-03-02 op void
1988 8af5e5ed 2021-03-08 op ui_on_tab_loaded(struct tab *tab)
1989 8af5e5ed 2021-03-08 op {
1990 8af5e5ed 2021-03-08 op stop_loading_anim(tab);
1991 2051e653 2021-03-13 op message("Loaded %s", tab->hist_cur->h);
1992 3d8c2326 2021-03-18 op
1993 3d8c2326 2021-03-18 op redraw_tabline();
1994 3d8c2326 2021-03-18 op wrefresh(tabline);
1995 3d8c2326 2021-03-18 op if (in_minibuffer)
1996 3d8c2326 2021-03-18 op wrefresh(minibuf);
1997 3d8c2326 2021-03-18 op else
1998 3d8c2326 2021-03-18 op wrefresh(body);
1999 8af5e5ed 2021-03-08 op }
2000 8af5e5ed 2021-03-08 op
2001 8af5e5ed 2021-03-08 op void
2002 5e11c00c 2021-03-02 op ui_on_tab_refresh(struct tab *tab)
2003 5e11c00c 2021-03-02 op {
2004 bddc7bbd 2021-04-01 op wrap_page(&tab->window, body_cols);
2005 3e433958 2021-03-21 op if (tab->flags & TAB_CURRENT) {
2006 2ba66cea 2021-03-22 op restore_cursor(&tab->window);
2007 3d8c2326 2021-03-18 op redraw_tab(tab);
2008 e8a76665 2021-05-12 op } else
2009 e8a76665 2021-05-12 op tab->flags |= TAB_URGENT;
2010 5e11c00c 2021-03-02 op }
2011 5e11c00c 2021-03-02 op
2012 5e11c00c 2021-03-02 op void
2013 5cd2ebb1 2021-03-11 op ui_require_input(struct tab *tab, int hide)
2014 5cd2ebb1 2021-03-11 op {
2015 5cd2ebb1 2021-03-11 op /* TODO: hard-switching to another tab is ugly */
2016 5cd2ebb1 2021-03-11 op switch_to_tab(tab);
2017 5cd2ebb1 2021-03-11 op
2018 22268e11 2021-03-11 op enter_minibuffer(ir_self_insert, ir_select, exit_minibuffer,
2019 22268e11 2021-03-11 op &ir_history);
2020 5cd2ebb1 2021-03-11 op strlcpy(ministate.prompt, "Input required: ",
2021 5cd2ebb1 2021-03-11 op sizeof(ministate.prompt));
2022 5cd2ebb1 2021-03-11 op redraw_tab(tab);
2023 5cd2ebb1 2021-03-11 op }
2024 5cd2ebb1 2021-03-11 op
2025 5cd2ebb1 2021-03-11 op void
2026 b3575139 2021-04-01 op ui_yornp(const char *prompt, void (*fn)(int, unsigned int),
2027 b3575139 2021-04-01 op unsigned int data)
2028 5d1bac73 2021-03-25 op {
2029 5d1bac73 2021-03-25 op size_t len;
2030 5d1bac73 2021-03-25 op
2031 6f66d9bf 2021-04-24 op if (in_minibuffer) {
2032 6f66d9bf 2021-04-24 op fn(0, data);
2033 6f66d9bf 2021-04-24 op return;
2034 6f66d9bf 2021-04-24 op }
2035 6f66d9bf 2021-04-24 op
2036 5d1bac73 2021-03-25 op yornp_cb = fn;
2037 e49ce157 2021-04-01 op yornp_data = data;
2038 5d1bac73 2021-03-25 op enter_minibuffer(yornp_self_insert, yornp_self_insert,
2039 5d1bac73 2021-03-25 op yornp_abort, NULL);
2040 5d1bac73 2021-03-25 op
2041 5d1bac73 2021-03-25 op len = sizeof(ministate.prompt);
2042 5d1bac73 2021-03-25 op strlcpy(ministate.prompt, prompt, len);
2043 5d1bac73 2021-03-25 op strlcat(ministate.prompt, " (y or n) ", len);
2044 5d1bac73 2021-03-25 op redraw_tab(current_tab());
2045 5d1bac73 2021-03-25 op }
2046 5d1bac73 2021-03-25 op
2047 5d1bac73 2021-03-25 op void
2048 740f578b 2021-03-15 op ui_notify(const char *fmt, ...)
2049 740f578b 2021-03-15 op {
2050 740f578b 2021-03-15 op va_list ap;
2051 740f578b 2021-03-15 op
2052 740f578b 2021-03-15 op va_start(ap, fmt);
2053 740f578b 2021-03-15 op vmessage(fmt, ap);
2054 740f578b 2021-03-15 op va_end(ap);
2055 740f578b 2021-03-15 op }
2056 740f578b 2021-03-15 op
2057 740f578b 2021-03-15 op void
2058 5e11c00c 2021-03-02 op ui_end(void)
2059 5e11c00c 2021-03-02 op {
2060 5e11c00c 2021-03-02 op endwin();
2061 5e11c00c 2021-03-02 op }