2 * Copyright (c) 2021 Omar Polo <op@omarpolo.com>
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
23 #include "minibuffer.h"
24 #include "telescope.h"
28 /* return 1 if moved, 0 otherwise */
30 forward_line(struct buffer *buffer, int n)
35 if (buffer->current_line == NULL)
37 vl = buffer->current_line;
42 vl = TAILQ_NEXT(vl, vlines);
45 if (vl->parent->flags & L_HIDDEN)
47 buffer->current_line = vl;
50 vl = TAILQ_PREV(vl, vhead, vlines);
53 if (vl->parent->flags & L_HIDDEN)
55 if (buffer->current_line == buffer->top_line) {
57 buffer->top_line = vl;
59 buffer->current_line = vl;
70 cmd_previous_line(struct buffer *buffer)
72 forward_line(buffer, -1);
76 cmd_next_line(struct buffer *buffer)
78 forward_line(buffer, +1);
82 cmd_backward_char(struct buffer *buffer)
84 if (buffer->cpoff != 0)
89 cmd_forward_char(struct buffer *buffer)
93 if (buffer->current_line->line != NULL)
94 len = utf8_cplen(buffer->current_line->line);
95 if (++buffer->cpoff > len)
100 cmd_backward_paragraph(struct buffer *buffer)
103 if (!forward_line(buffer, -1)) {
104 message("No previous paragraph");
107 } while (buffer->current_line->line != NULL ||
108 buffer->current_line->parent->type != LINE_TEXT);
112 cmd_forward_paragraph(struct buffer *buffer)
115 if (!forward_line(buffer, +1)) {
116 message("No next paragraph");
119 } while (buffer->current_line->line != NULL ||
120 buffer->current_line->parent->type != LINE_TEXT);
124 cmd_move_beginning_of_line(struct buffer *buffer)
130 cmd_move_end_of_line(struct buffer *buffer)
134 vl = buffer->current_line;
135 if (vl->line == NULL)
137 buffer->cpoff = utf8_cplen(vl->line);
141 cmd_redraw(struct buffer *buffer)
143 ui_schedule_redraw();
147 cmd_scroll_line_up(struct buffer *buffer)
151 if ((vl = TAILQ_PREV(buffer->top_line, vhead, vlines))
155 forward_line(buffer, -1);
157 buffer->top_line = vl;
161 cmd_scroll_line_down(struct buffer *buffer)
163 if (!forward_line(buffer, +1))
166 buffer->top_line = TAILQ_NEXT(buffer->top_line, vlines);
171 cmd_scroll_up(struct buffer *buffer)
173 forward_line(buffer, -1*body_lines);
177 cmd_scroll_down(struct buffer *buffer)
179 forward_line(buffer, body_lines);
183 cmd_beginning_of_buffer(struct buffer *buffer)
185 buffer->current_line = TAILQ_FIRST(&buffer->head);
187 buffer->top_line = buffer->current_line;
188 buffer->line_off = 0;
192 cmd_end_of_buffer(struct buffer *buffer)
194 buffer->current_line = TAILQ_LAST(&buffer->head, vhead);
195 buffer->cpoff = body_cols;
199 cmd_kill_telescope(struct buffer *buffer)
206 cmd_push_button(struct buffer *buffer)
211 vl = buffer->current_line;
212 switch (vl->parent->type) {
214 load_url_in_tab(current_tab(), vl->parent->meta.alt);
217 l = TAILQ_NEXT(vl->parent, lines);
218 for (; l != NULL; l = TAILQ_NEXT(l, lines)) {
219 if (l->type == LINE_PRE_END)
221 l->flags ^= L_HIDDEN;
222 if (l->flags & L_HIDDEN)
234 cmd_push_button_new_tab(struct buffer *buffer)
238 vl = buffer->current_line;
239 if (vl->parent->type != LINE_LINK)
242 new_tab(vl->parent->meta.alt);
246 cmd_previous_button(struct buffer *buffer)
248 struct excursion place;
250 save_excursion(&place, buffer);
253 if (!forward_line(buffer, -1)) {
254 restore_excursion(&place, buffer);
255 message("No previous link");
258 } while (buffer->current_line->parent->type != LINE_LINK);
262 cmd_next_button(struct buffer *buffer)
264 struct excursion place;
266 save_excursion(&place, buffer);
269 if (!forward_line(buffer, +1)){
270 restore_excursion(&place, buffer);
271 message("No next link");
274 } while (buffer->current_line->parent->type != LINE_LINK);
278 is_heading(const struct line *l)
280 return l->type == LINE_TITLE_1 ||
281 l->type == LINE_TITLE_2 ||
282 l->type == LINE_TITLE_3;
286 cmd_previous_heading(struct buffer *buffer)
288 struct excursion place;
290 save_excursion(&place, buffer);
293 if (!forward_line(buffer, -1)) {
294 restore_excursion(&place, buffer);
295 message("No previous heading");
298 } while (!is_heading(buffer->current_line->parent));
302 cmd_next_heading(struct buffer *buffer)
304 struct excursion place;
306 save_excursion(&place, buffer);
309 if (!forward_line(buffer, +1)) {
310 restore_excursion(&place, buffer);
311 message("No next heading");
314 } while (!is_heading(buffer->current_line->parent));
318 cmd_previous_page(struct buffer *buffer)
320 struct tab *tab = current_tab();
322 if (!load_previous_page(tab))
323 message("No previous page");
325 start_loading_anim(tab);
329 cmd_next_page(struct buffer *buffer)
331 struct tab *tab = current_tab();
333 if (!load_next_page(tab))
334 message("No next page");
336 start_loading_anim(tab);
340 cmd_clear_minibuf(struct buffer *buffer)
346 cmd_execute_extended_command(struct buffer *buffer)
351 message("We don't have enable-recursive-minibuffers");
355 enter_minibuffer(eecmd_self_insert, eecmd_select, exit_minibuffer,
356 &eecmd_history, compl_eecmd, NULL);
358 len = sizeof(ministate.prompt);
359 strlcpy(ministate.prompt, "", len);
362 strlcat(ministate.prompt, "M-", len);
364 strlcat(ministate.prompt, ui_keyname(thiskey.key), len);
367 strlcat(ministate.prompt, " ", len);
371 cmd_tab_close(struct buffer *buffer)
376 if (TAILQ_PREV(tab, tabshead, tabs) == NULL &&
377 TAILQ_NEXT(tab, tabs) == NULL) {
378 message("Can't close the only tab.");
382 if (evtimer_pending(&tab->loadingev, NULL))
383 evtimer_del(&tab->loadingev);
387 if ((t = TAILQ_PREV(tab, tabshead, tabs)) == NULL)
388 t = TAILQ_NEXT(tab, tabs);
389 TAILQ_REMOVE(&tabshead, tab, tabs);
396 cmd_tab_close_other(struct buffer *buffer)
400 TAILQ_FOREACH_SAFE(t, &tabshead, tabs, i) {
401 if (t->flags & TAB_CURRENT)
405 TAILQ_REMOVE(&tabshead, t, tabs);
411 cmd_tab_new(struct buffer *buffer)
415 if ((url = new_tab_url) == NULL)
422 cmd_tab_next(struct buffer *buffer)
427 tab->flags &= ~TAB_CURRENT;
429 if ((t = TAILQ_NEXT(tab, tabs)) == NULL)
430 t = TAILQ_FIRST(&tabshead);
431 t->flags |= TAB_CURRENT;
432 t->flags &= ~TAB_URGENT;
436 cmd_tab_previous(struct buffer *buffer)
441 tab->flags &= ~TAB_CURRENT;
443 if ((t = TAILQ_PREV(tab, tabshead, tabs)) == NULL)
444 t = TAILQ_LAST(&tabshead, tabshead);
445 t->flags |= TAB_CURRENT;
446 t->flags &= ~TAB_URGENT;
450 cmd_tab_move(struct buffer *buffer)
455 t = TAILQ_NEXT(tab, tabs);
456 TAILQ_REMOVE(&tabshead, tab, tabs);
459 TAILQ_INSERT_HEAD(&tabshead, tab, tabs);
461 TAILQ_INSERT_AFTER(&tabshead, t, tab, tabs);
465 cmd_tab_move_to(struct buffer *buffer)
470 t = TAILQ_PREV(tab, tabshead, tabs);
471 TAILQ_REMOVE(&tabshead, tab, tabs);
474 if (TAILQ_EMPTY(&tabshead))
475 TAILQ_INSERT_HEAD(&tabshead, tab, tabs);
477 TAILQ_INSERT_TAIL(&tabshead, tab, tabs);
479 TAILQ_INSERT_BEFORE(t, tab, tabs);
483 cmd_tab_select(struct buffer *buffer)
486 message("We don't have enable-recursive-minibuffers");
490 enter_minibuffer(sensible_self_insert, ts_select, exit_minibuffer,
491 NULL, compl_ts, NULL);
492 strlcpy(ministate.prompt, "Select tab: ", sizeof(ministate.prompt));
496 cmd_load_url(struct buffer *buffer)
499 message("We don't have enable-recursive-minibuffers");
503 enter_minibuffer(sensible_self_insert, lu_select, exit_minibuffer,
504 &lu_history, NULL, NULL);
505 strlcpy(ministate.prompt, "Load URL: ", sizeof(ministate.prompt));
506 strlcpy(ministate.buf, "gemini://", sizeof(ministate.buf));
507 cmd_move_end_of_line(&ministate.buffer);
511 cmd_load_current_url(struct buffer *buffer)
513 struct tab *tab = current_tab();
516 message("We don't have enable-recursive-minibuffers");
520 enter_minibuffer(sensible_self_insert, lu_select, exit_minibuffer,
521 &lu_history, NULL, NULL);
522 strlcpy(ministate.prompt, "Load URL: ", sizeof(ministate.prompt));
523 strlcpy(ministate.buf, tab->hist_cur->h, sizeof(ministate.buf));
524 ministate.buffer.cpoff = utf8_cplen(ministate.buf);
528 cmd_reload_page(struct buffer *buffer)
533 load_url_in_tab(tab, tab->hist_cur->h);
537 cmd_bookmark_page(struct buffer *buffer)
539 struct tab *tab = current_tab();
541 enter_minibuffer(sensible_self_insert, bp_select, exit_minibuffer, NULL,
543 strlcpy(ministate.prompt, "Bookmark URL: ", sizeof(ministate.prompt));
544 strlcpy(ministate.buf, tab->hist_cur->h, sizeof(ministate.buf));
545 ministate.buffer.cpoff = utf8_cplen(ministate.buf);
549 cmd_list_bookmarks(struct buffer *buffer)
551 load_url_in_tab(current_tab(), "about:bookmarks");
555 cmd_toggle_help(struct buffer *buffer)
557 ui_toggle_side_window();
561 cmd_link_select(struct buffer *buffer)
564 message("We don't have enable-recursive-minibuffers");
568 enter_minibuffer(sensible_self_insert, ls_select, exit_minibuffer,
569 NULL, compl_ls, TAILQ_FIRST(&buffer->page.head));
570 strlcpy(ministate.prompt, "Select link: ", sizeof(ministate.prompt));
574 cmd_swiper(struct buffer *buffer)
577 message("We don't have enable-recursive-minibuffers");
581 enter_minibuffer(sensible_self_insert, swiper_select, exit_minibuffer,
582 NULL, compl_swiper, TAILQ_FIRST(&buffer->page.head));
583 strlcpy(ministate.prompt, "Select line: ", sizeof(ministate.prompt));
587 cmd_toc(struct buffer *buffer)
590 message("We don't have enable-recursive-minibuffers");
594 enter_minibuffer(sensible_self_insert, toc_select, exit_minibuffer,
595 NULL, compl_toc, TAILQ_FIRST(&buffer->page.head));
596 strlcpy(ministate.prompt, "Select heading: ",
597 sizeof(ministate.prompt));
601 cmd_inc_fill_column(struct buffer *buffer)
603 if (fill_column == INT_MAX)
607 message("fill-column: %d", fill_column);
609 ui_schedule_redraw();
613 cmd_dec_fill_column(struct buffer *buffer)
615 if (fill_column == INT_MAX || fill_column < 8)
619 message("fill-column: %d", fill_column);
621 ui_schedule_redraw();
625 cmd_olivetti_mode(struct buffer *buffer)
627 olivetti_mode = !olivetti_mode;
629 message("olivetti-mode enabled");
631 message("olivetti-mode disabled");
633 ui_schedule_redraw();
637 cmd_mini_delete_char(struct buffer *buffer)
641 if (!in_minibuffer) {
642 message("text is read-only");
646 minibuffer_taint_hist();
648 c = utf8_nth(buffer->current_line->line, buffer->cpoff);
653 memmove(c, n, strlen(n)+1);
655 recompute_completions(0);
659 cmd_mini_delete_backward_char(struct buffer *buffer)
663 if (!in_minibuffer) {
664 message("text is read-only");
668 minibuffer_taint_hist();
670 c = utf8_nth(buffer->current_line->line, buffer->cpoff);
671 start = buffer->current_line->line;
674 p = utf8_prev_cp(c-1, start);
676 memmove(p, c, strlen(c)+1);
679 recompute_completions(0);
683 cmd_mini_kill_line(struct buffer *buffer)
687 if (!in_minibuffer) {
688 message("text is read-only");
692 minibuffer_taint_hist();
693 c = utf8_nth(buffer->current_line->line, buffer->cpoff);
696 recompute_completions(0);
700 cmd_mini_abort(struct buffer *buffer)
709 cmd_mini_complete_and_exit(struct buffer *buffer)
714 minibuffer_taint_hist();
719 cmd_mini_previous_history_element(struct buffer *buffer)
721 if (ministate.history == NULL) {
722 message("No history");
726 if (ministate.hist_cur == NULL ||
727 (ministate.hist_cur = TAILQ_PREV(ministate.hist_cur, mhisthead, entries)) == NULL) {
728 ministate.hist_cur = TAILQ_LAST(&ministate.history->head, mhisthead);
729 ministate.hist_off = ministate.history->len - 1;
730 if (ministate.hist_cur == NULL)
731 message("No prev history item");
733 ministate.hist_off--;
736 if (ministate.hist_cur != NULL)
737 buffer->current_line->line = ministate.hist_cur->h;
741 cmd_mini_next_history_element(struct buffer *buffer)
743 if (ministate.history == NULL) {
744 message("No history");
748 if (ministate.hist_cur == NULL ||
749 (ministate.hist_cur = TAILQ_NEXT(ministate.hist_cur, entries)) == NULL) {
750 ministate.hist_cur = TAILQ_FIRST(&ministate.history->head);
751 ministate.hist_off = 0;
752 if (ministate.hist_cur == NULL)
753 message("No next history item");
755 ministate.hist_off++;
758 if (ministate.hist_cur != NULL)
759 buffer->current_line->line = ministate.hist_cur->h;
763 cmd_previous_completion(struct buffer *buffer)
765 if (in_minibuffer != MB_COMPREAD)
768 buffer = &ministate.compl.buffer;
770 if (buffer->current_line != NULL)
771 buffer->current_line->parent->type = LINE_COMPL;
773 forward_line(buffer, -1);
775 if (buffer->current_line != NULL)
776 buffer->current_line->parent->type = LINE_COMPL_CURRENT;
780 cmd_next_completion(struct buffer *buffer)
782 if (in_minibuffer != MB_COMPREAD)
785 buffer = &ministate.compl.buffer;
787 if (buffer->current_line != NULL)
788 buffer->current_line->parent->type = LINE_COMPL;
790 forward_line(buffer, +1);
792 if (buffer->current_line != NULL)
793 buffer->current_line->parent->type = LINE_COMPL_CURRENT;
797 cmd_insert_current_candidate(struct buffer *buffer)
801 if (in_minibuffer != MB_COMPREAD)
804 buffer = &ministate.compl.buffer;
805 if ((vl = buffer->current_line) == NULL)
808 minibuffer_taint_hist();
809 strlcpy(ministate.buf, vl->parent->line, sizeof(ministate.buf));
810 ministate.buffer.cpoff = utf8_cplen(ministate.buf);