build: fix _XOPEN_SOURCE redefinition warning
[vis.git] / vis-cmds.c
blobf2551d034d3640e9426e91e104167ad53f7ead4e
1 /* this file is included from sam.c */
3 #include <termkey.h>
4 #include "vis-lua.h"
6 // FIXME: avoid this redirection?
7 typedef struct {
8 CommandDef def;
9 VisCommandFunction *func;
10 void *data;
11 } CmdUser;
13 static void cmdfree(CmdUser *cmd) {
14 if (!cmd)
15 return;
16 free((char*)cmd->def.name);
17 free(VIS_HELP_USE((char*)cmd->def.help));
18 free(cmd);
21 bool vis_cmd_register(Vis *vis, const char *name, const char *help, void *data, VisCommandFunction *func) {
22 if (!name)
23 return false;
24 if (!vis->usercmds && !(vis->usercmds = map_new()))
25 return false;
26 CmdUser *cmd = calloc(1, sizeof *cmd);
27 if (!cmd)
28 return false;
29 if (!(cmd->def.name = strdup(name)))
30 goto err;
31 #if CONFIG_HELP
32 if (help && !(cmd->def.help = strdup(help)))
33 goto err;
34 #endif
35 cmd->def.flags = CMD_ARGV|CMD_FORCE|CMD_ONCE|CMD_ADDRESS_ALL;
36 cmd->def.func = cmd_user;
37 cmd->func = func;
38 cmd->data = data;
39 if (!map_put(vis->cmds, name, &cmd->def))
40 goto err;
41 if (!map_put(vis->usercmds, name, cmd)) {
42 map_delete(vis->cmds, name);
43 goto err;
45 return true;
46 err:
47 cmdfree(cmd);
48 return false;
51 bool vis_cmd_unregister(Vis *vis, const char *name) {
52 if (!name)
53 return true;
54 CmdUser *cmd = map_get(vis->usercmds, name);
55 if (!cmd)
56 return false;
57 if (!map_delete(vis->cmds, name))
58 return false;
59 if (!map_delete(vis->usercmds, name))
60 return false;
61 cmdfree(cmd);
62 return true;
65 static void option_free(OptionDef *opt) {
66 if (!opt)
67 return;
68 for (size_t i = 0; i < LENGTH(options); i++) {
69 if (opt == &options[i])
70 return;
73 for (const char **name = opt->names; *name; name++)
74 free((char*)*name);
75 free(VIS_HELP_USE((char*)opt->help));
76 free(opt);
79 bool vis_option_register(Vis *vis, const char *names[], enum VisOption flags,
80 VisOptionFunction *func, void *context, const char *help) {
82 if (!names || !names[0])
83 return false;
85 for (const char **name = names; *name; name++) {
86 if (map_get(vis->options, *name))
87 return false;
89 OptionDef *opt = calloc(1, sizeof *opt);
90 if (!opt)
91 return false;
92 for (size_t i = 0; i < LENGTH(opt->names)-1 && names[i]; i++) {
93 if (!(opt->names[i] = strdup(names[i])))
94 goto err;
96 opt->flags = flags;
97 opt->func = func;
98 opt->context = context;
99 #if CONFIG_HELP
100 if (help && !(opt->help = strdup(help)))
101 goto err;
102 #endif
103 for (const char **name = names; *name; name++)
104 map_put(vis->options, *name, opt);
105 return true;
106 err:
107 option_free(opt);
108 return false;
111 bool vis_option_unregister(Vis *vis, const char *name) {
112 OptionDef *opt = map_get(vis->options, name);
113 if (!opt)
114 return false;
115 for (const char **alias = opt->names; *alias; alias++) {
116 if (!map_delete(vis->options, *alias))
117 return false;
119 option_free(opt);
120 return true;
123 static bool cmd_user(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
124 CmdUser *user = map_get(vis->usercmds, argv[0]);
125 return user && user->func(vis, win, user->data, cmd->flags == '!', argv, sel, range);
128 static void windows_arrange(Vis *vis, enum UiLayout layout) {
129 vis->ui->arrange(vis->ui, layout);
132 static void tabwidth_set(Vis *vis, int tabwidth) {
133 if (tabwidth < 1 || tabwidth > 8)
134 return;
135 for (Win *win = vis->windows; win; win = win->next)
136 view_tabwidth_set(win->view, tabwidth);
137 vis->tabwidth = tabwidth;
140 /* parse human-readable boolean value in s. If successful, store the result in
141 * outval and return true. Else return false and leave outval alone. */
142 static bool parse_bool(const char *s, bool *outval) {
143 for (const char **t = (const char*[]){"1", "true", "yes", "on", NULL}; *t; t++) {
144 if (!strcasecmp(s, *t)) {
145 *outval = true;
146 return true;
149 for (const char **f = (const char*[]){"0", "false", "no", "off", NULL}; *f; f++) {
150 if (!strcasecmp(s, *f)) {
151 *outval = false;
152 return true;
155 return false;
158 static bool cmd_set(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
160 if (!argv[1] || !argv[1][0] || argv[3]) {
161 vis_info_show(vis, "Expecting: set option [value]");
162 return false;
165 char name[256];
166 strncpy(name, argv[1], sizeof(name)-1);
167 char *lastchar = &name[strlen(name)-1];
168 bool toggle = (*lastchar == '!');
169 if (toggle)
170 *lastchar = '\0';
172 OptionDef *opt = map_closest(vis->options, name);
173 if (!opt) {
174 vis_info_show(vis, "Unknown option: `%s'", name);
175 return false;
178 if (!win && (opt->flags & VIS_OPTION_NEED_WINDOW)) {
179 vis_info_show(vis, "Need active window for `:set %s'", name);
180 return false;
183 if (toggle) {
184 if (!(opt->flags & VIS_OPTION_TYPE_BOOL)) {
185 vis_info_show(vis, "Only boolean options can be toggled");
186 return false;
188 if (argv[2]) {
189 vis_info_show(vis, "Can not specify option value when toggling");
190 return false;
194 Arg arg;
195 if (opt->flags & VIS_OPTION_TYPE_STRING) {
196 if (!(opt->flags & VIS_OPTION_VALUE_OPTIONAL) && !argv[2]) {
197 vis_info_show(vis, "Expecting string option value");
198 return false;
200 arg.s = argv[2];
201 } else if (opt->flags & VIS_OPTION_TYPE_BOOL) {
202 if (!argv[2]) {
203 arg.b = !toggle;
204 } else if (!parse_bool(argv[2], &arg.b)) {
205 vis_info_show(vis, "Expecting boolean option value not: `%s'", argv[2]);
206 return false;
208 } else if (opt->flags & VIS_OPTION_TYPE_NUMBER) {
209 if (!argv[2]) {
210 vis_info_show(vis, "Expecting number");
211 return false;
213 char *ep;
214 errno = 0;
215 long lval = strtol(argv[2], &ep, 10);
216 if (argv[2][0] == '\0' || *ep != '\0') {
217 vis_info_show(vis, "Invalid number");
218 return false;
221 if ((errno == ERANGE && (lval == LONG_MAX || lval == LONG_MIN)) ||
222 (lval > INT_MAX || lval < INT_MIN)) {
223 vis_info_show(vis, "Number overflow");
224 return false;
227 if (lval < 0) {
228 vis_info_show(vis, "Expecting positive number");
229 return false;
231 arg.i = lval;
232 } else {
233 return false;
236 size_t opt_index = 0;
237 for (; opt_index < LENGTH(options); opt_index++) {
238 if (opt == &options[opt_index])
239 break;
242 switch (opt_index) {
243 case OPTION_SHELL:
245 char *shell = strdup(arg.s);
246 if (!shell) {
247 vis_info_show(vis, "Failed to change shell");
248 return false;
250 free(vis->shell);
251 vis->shell = shell;
252 break;
254 case OPTION_ESCDELAY:
256 TermKey *termkey = vis->ui->termkey_get(vis->ui);
257 termkey_set_waittime(termkey, arg.i);
258 break;
260 case OPTION_EXPANDTAB:
261 vis->expandtab = toggle ? !vis->expandtab : arg.b;
262 break;
263 case OPTION_AUTOINDENT:
264 vis->autoindent = toggle ? !vis->autoindent : arg.b;
265 break;
266 case OPTION_TABWIDTH:
267 tabwidth_set(vis, arg.i);
268 break;
269 case OPTION_SHOW_SPACES:
270 case OPTION_SHOW_TABS:
271 case OPTION_SHOW_NEWLINES:
272 case OPTION_SHOW_EOF:
274 const int values[] = {
275 [OPTION_SHOW_SPACES] = UI_OPTION_SYMBOL_SPACE,
276 [OPTION_SHOW_TABS] = UI_OPTION_SYMBOL_TAB|UI_OPTION_SYMBOL_TAB_FILL,
277 [OPTION_SHOW_NEWLINES] = UI_OPTION_SYMBOL_EOL,
278 [OPTION_SHOW_EOF] = UI_OPTION_SYMBOL_EOF,
280 int flags = view_options_get(win->view);
281 if (arg.b || (toggle && !(flags & values[opt_index])))
282 flags |= values[opt_index];
283 else
284 flags &= ~values[opt_index];
285 view_options_set(win->view, flags);
286 break;
288 case OPTION_NUMBER: {
289 enum UiOption opt = view_options_get(win->view);
290 if (arg.b || (toggle && !(opt & UI_OPTION_LINE_NUMBERS_ABSOLUTE))) {
291 opt &= ~UI_OPTION_LINE_NUMBERS_RELATIVE;
292 opt |= UI_OPTION_LINE_NUMBERS_ABSOLUTE;
293 } else {
294 opt &= ~UI_OPTION_LINE_NUMBERS_ABSOLUTE;
296 view_options_set(win->view, opt);
297 break;
299 case OPTION_NUMBER_RELATIVE: {
300 enum UiOption opt = view_options_get(win->view);
301 if (arg.b || (toggle && !(opt & UI_OPTION_LINE_NUMBERS_RELATIVE))) {
302 opt &= ~UI_OPTION_LINE_NUMBERS_ABSOLUTE;
303 opt |= UI_OPTION_LINE_NUMBERS_RELATIVE;
304 } else {
305 opt &= ~UI_OPTION_LINE_NUMBERS_RELATIVE;
307 view_options_set(win->view, opt);
308 break;
310 case OPTION_CURSOR_LINE: {
311 enum UiOption opt = view_options_get(win->view);
312 if (arg.b || (toggle && !(opt & UI_OPTION_CURSOR_LINE)))
313 opt |= UI_OPTION_CURSOR_LINE;
314 else
315 opt &= ~UI_OPTION_CURSOR_LINE;
316 view_options_set(win->view, opt);
317 break;
319 case OPTION_COLOR_COLUMN:
320 view_colorcolumn_set(win->view, arg.i);
321 break;
322 case OPTION_SAVE_METHOD:
323 if (strcmp("auto", arg.s) == 0) {
324 win->file->save_method = TEXT_SAVE_AUTO;
325 } else if (strcmp("atomic", arg.s) == 0) {
326 win->file->save_method = TEXT_SAVE_ATOMIC;
327 } else if (strcmp("inplace", arg.s) == 0) {
328 win->file->save_method = TEXT_SAVE_INPLACE;
329 } else {
330 vis_info_show(vis, "Invalid save method `%s', expected "
331 "'auto', 'atomic' or 'inplace'", arg.s);
332 return false;
334 break;
335 case OPTION_LOAD_METHOD:
336 if (strcmp("auto", arg.s) == 0) {
337 vis->load_method = TEXT_LOAD_AUTO;
338 } else if (strcmp("read", arg.s) == 0) {
339 vis->load_method = TEXT_LOAD_READ;
340 } else if (strcmp("mmap", arg.s) == 0) {
341 vis->load_method = TEXT_LOAD_MMAP;
342 } else {
343 vis_info_show(vis, "Invalid load method `%s', expected "
344 "'auto', 'read' or 'mmap'", arg.s);
345 return false;
347 break;
348 case OPTION_CHANGE_256COLORS:
349 vis->change_colors = toggle ? !vis->change_colors : arg.b;
350 break;
351 case OPTION_LAYOUT: {
352 enum UiLayout layout;
353 if (strcmp("h", arg.s) == 0) {
354 layout = UI_LAYOUT_HORIZONTAL;
355 } else if (strcmp("v", arg.s) == 0) {
356 layout = UI_LAYOUT_VERTICAL;
357 } else {
358 vis_info_show(vis, "Invalid layout `%s', expected 'h' or 'v'", arg.s);
359 return false;
361 windows_arrange(vis, layout);
362 break;
364 default:
365 if (!opt->func)
366 return false;
367 return opt->func(vis, win, opt->context, toggle, opt->flags, name, &arg);
370 return true;
373 static bool is_file_pattern(const char *pattern) {
374 if (!pattern)
375 return false;
376 struct stat meta;
377 if (stat(pattern, &meta) == 0 && S_ISDIR(meta.st_mode))
378 return true;
379 for (char special[] = "*?[{$~", *s = special; *s; s++) {
380 if (strchr(pattern, *s))
381 return true;
383 return false;
386 static const char *file_open_dialog(Vis *vis, const char *pattern) {
387 static char name[PATH_MAX];
388 name[0] = '\0';
390 if (!is_file_pattern(pattern))
391 return pattern;
393 Buffer bufcmd, bufout, buferr;
394 buffer_init(&bufcmd);
395 buffer_init(&bufout);
396 buffer_init(&buferr);
398 if (!buffer_put0(&bufcmd, VIS_OPEN " ") || !buffer_append0(&bufcmd, pattern ? pattern : ""))
399 return NULL;
401 Filerange empty = text_range_new(0,0);
402 int status = vis_pipe(vis, vis->win->file, &empty,
403 (const char*[]){ buffer_content0(&bufcmd), NULL },
404 &bufout, read_buffer, &buferr, read_buffer);
406 if (status == 0)
407 strncpy(name, buffer_content0(&bufout), sizeof(name)-1);
408 else if (status != 1)
409 vis_info_show(vis, "Command failed %s", buffer_content0(&buferr));
411 buffer_release(&bufcmd);
412 buffer_release(&bufout);
413 buffer_release(&buferr);
415 for (char *end = name+strlen(name)-1; end >= name && isspace((unsigned char)*end); end--)
416 *end = '\0';
418 return name[0] ? name : NULL;
421 static bool openfiles(Vis *vis, const char **files) {
422 for (; *files; files++) {
423 const char *file = file_open_dialog(vis, *files);
424 if (!file)
425 return false;
426 errno = 0;
427 if (!vis_window_new(vis, file)) {
428 vis_info_show(vis, "Could not open `%s' %s", file,
429 errno ? strerror(errno) : "");
430 return false;
433 return true;
436 static bool cmd_open(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
437 if (!argv[1])
438 return vis_window_new(vis, NULL);
439 return openfiles(vis, &argv[1]);
442 static void info_unsaved_changes(Vis *vis) {
443 vis_info_show(vis, "No write since last change (add ! to override)");
446 static bool cmd_edit(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
447 if (argv[2]) {
448 vis_info_show(vis, "Only 1 filename allowed");
449 return false;
451 Win *oldwin = win;
452 if (!oldwin)
453 return false;
454 if (cmd->flags != '!' && !vis_window_closable(oldwin)) {
455 info_unsaved_changes(vis);
456 return false;
458 if (!argv[1]) {
459 if (oldwin->file->refcount > 1) {
460 vis_info_show(vis, "Can not reload file being opened multiple times");
461 return false;
463 return vis_window_reload(oldwin);
465 if (!openfiles(vis, &argv[1]))
466 return false;
467 if (vis->win != oldwin) {
468 Win *newwin = vis->win;
469 vis_window_swap(oldwin, newwin);
470 vis_window_close(oldwin);
471 vis_window_focus(newwin);
473 return vis->win != oldwin;
476 static bool cmd_read(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
477 bool ret = false;
478 const size_t first_file = 3;
479 const char *args[MAX_ARGV] = { argv[0], "cat", "--" };
480 const char **name = argv[1] ? &argv[1] : (const char*[]){ ".", NULL };
481 for (size_t i = first_file; *name && i < LENGTH(args)-1; name++, i++) {
482 const char *file = file_open_dialog(vis, *name);
483 if (!file || !(args[i] = strdup(file)))
484 goto err;
486 args[LENGTH(args)-1] = NULL;
487 ret = cmd_pipein(vis, win, cmd, args, sel, range);
488 err:
489 for (size_t i = first_file; i < LENGTH(args); i++)
490 free((char*)args[i]);
491 return ret;
494 static bool has_windows(Vis *vis) {
495 for (Win *win = vis->windows; win; win = win->next) {
496 if (!win->file->internal)
497 return true;
499 return false;
502 static bool cmd_quit(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
503 if (cmd->flags != '!' && !vis_window_closable(win)) {
504 info_unsaved_changes(vis);
505 return false;
507 vis_window_close(win);
508 if (!has_windows(vis))
509 vis_exit(vis, argv[1] ? atoi(argv[1]) : EXIT_SUCCESS);
510 return true;
513 static bool cmd_qall(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
514 for (Win *next, *win = vis->windows; win; win = next) {
515 next = win->next;
516 if (!win->file->internal && (!text_modified(win->file->text) || cmd->flags == '!'))
517 vis_window_close(win);
519 if (!has_windows(vis)) {
520 vis_exit(vis, argv[1] ? atoi(argv[1]) : EXIT_SUCCESS);
521 return true;
522 } else {
523 info_unsaved_changes(vis);
524 return false;
528 static bool cmd_split(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
529 if (!win)
530 return false;
531 enum UiOption options = view_options_get(win->view);
532 windows_arrange(vis, UI_LAYOUT_HORIZONTAL);
533 if (!argv[1])
534 return vis_window_split(win);
535 bool ret = openfiles(vis, &argv[1]);
536 if (ret)
537 view_options_set(vis->win->view, options);
538 return ret;
541 static bool cmd_vsplit(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
542 if (!win)
543 return false;
544 enum UiOption options = view_options_get(win->view);
545 windows_arrange(vis, UI_LAYOUT_VERTICAL);
546 if (!argv[1])
547 return vis_window_split(win);
548 bool ret = openfiles(vis, &argv[1]);
549 if (ret)
550 view_options_set(vis->win->view, options);
551 return ret;
554 static bool cmd_new(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
555 windows_arrange(vis, UI_LAYOUT_HORIZONTAL);
556 return vis_window_new(vis, NULL);
559 static bool cmd_vnew(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
560 windows_arrange(vis, UI_LAYOUT_VERTICAL);
561 return vis_window_new(vis, NULL);
564 static bool cmd_wq(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
565 if (!win)
566 return false;
567 File *file = win->file;
568 bool unmodified = file->fd == -1 && !file->name && !text_modified(file->text);
569 if (unmodified || cmd_write(vis, win, cmd, argv, sel, range))
570 return cmd_quit(vis, win, cmd, (const char*[]){argv[0], NULL}, sel, range);
571 return false;
574 static bool cmd_earlier_later(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
575 if (!win)
576 return false;
577 Text *txt = win->file->text;
578 char *unit = "";
579 long count = 1;
580 size_t pos = EPOS;
581 if (argv[1]) {
582 errno = 0;
583 count = strtol(argv[1], &unit, 10);
584 if (errno || unit == argv[1] || count < 0) {
585 vis_info_show(vis, "Invalid number");
586 return false;
589 if (*unit) {
590 while (*unit && isspace((unsigned char)*unit))
591 unit++;
592 switch (*unit) {
593 case 'd': count *= 24; /* fall through */
594 case 'h': count *= 60; /* fall through */
595 case 'm': count *= 60; /* fall through */
596 case 's': break;
597 default:
598 vis_info_show(vis, "Unknown time specifier (use: s,m,h or d)");
599 return false;
602 if (argv[0][0] == 'e')
603 count = -count; /* earlier, move back in time */
605 pos = text_restore(txt, text_state(txt) + count);
609 if (!*unit) {
610 VisCountIterator it = vis_count_iterator_init(vis, count);
611 while (vis_count_iterator_next(&it)) {
612 if (argv[0][0] == 'e')
613 pos = text_earlier(txt);
614 else
615 pos = text_later(txt);
619 time_t state = text_state(txt);
620 char buf[32];
621 strftime(buf, sizeof buf, "State from %H:%M", localtime(&state));
622 vis_info_show(vis, "%s", buf);
624 return pos != EPOS;
627 static bool print_keylayout(const char *key, void *value, void *data) {
628 return text_appendf(data, " %-18s\t%s\n", key[0] == ' ' ? "␣" : key, (char*)value);
631 static bool print_keybinding(const char *key, void *value, void *data) {
632 KeyBinding *binding = value;
633 const char *desc = binding->alias;
634 if (!desc && binding->action)
635 desc = VIS_HELP_USE(binding->action->help);
636 return text_appendf(data, " %-18s\t%s\n", key[0] == ' ' ? "␣" : key, desc ? desc : "");
639 static void print_mode(Mode *mode, Text *txt) {
640 if (!map_empty(mode->bindings))
641 text_appendf(txt, "\n %s\n\n", mode->name);
642 map_iterate(mode->bindings, print_keybinding, txt);
645 static bool print_action(const char *key, void *value, void *data) {
646 const char *help = VIS_HELP_USE(((KeyAction*)value)->help);
647 return text_appendf(data, " %-30s\t%s\n", key, help ? help : "");
650 static bool print_cmd(const char *key, void *value, void *data) {
651 CommandDef *cmd = value;
652 const char *help = VIS_HELP_USE(cmd->help);
653 char usage[256];
654 snprintf(usage, sizeof usage, "%s%s%s%s%s%s%s",
655 cmd->name,
656 (cmd->flags & CMD_FORCE) ? "[!]" : "",
657 (cmd->flags & CMD_TEXT) ? "/text/" : "",
658 (cmd->flags & CMD_REGEX) ? "/regexp/" : "",
659 (cmd->flags & CMD_CMD) ? " command" : "",
660 (cmd->flags & CMD_SHELL) ? (!strcmp(cmd->name, "s") ? "/regexp/text/" : " shell-command") : "",
661 (cmd->flags & CMD_ARGV) ? " [args...]" : "");
662 return text_appendf(data, " %-30s %s\n", usage, help ? help : "");
665 static bool print_option(const char *key, void *value, void *txt) {
666 char desc[256];
667 const OptionDef *opt = value;
668 const char *help = VIS_HELP_USE(opt->help);
669 if (strcmp(key, opt->names[0]))
670 return true;
671 snprintf(desc, sizeof desc, "%s%s%s%s%s",
672 opt->names[0],
673 opt->names[1] ? "|" : "",
674 opt->names[1] ? opt->names[1] : "",
675 opt->flags & VIS_OPTION_TYPE_BOOL ? " on|off" : "",
676 opt->flags & VIS_OPTION_TYPE_NUMBER ? " nn" : "");
677 return text_appendf(txt, " %-30s %s\n", desc, help ? help : "");
680 static void print_symbolic_keys(Vis *vis, Text *txt) {
681 static const int keys[] = {
682 TERMKEY_SYM_BACKSPACE,
683 TERMKEY_SYM_TAB,
684 TERMKEY_SYM_ENTER,
685 TERMKEY_SYM_ESCAPE,
686 //TERMKEY_SYM_SPACE,
687 TERMKEY_SYM_DEL,
688 TERMKEY_SYM_UP,
689 TERMKEY_SYM_DOWN,
690 TERMKEY_SYM_LEFT,
691 TERMKEY_SYM_RIGHT,
692 TERMKEY_SYM_BEGIN,
693 TERMKEY_SYM_FIND,
694 TERMKEY_SYM_INSERT,
695 TERMKEY_SYM_DELETE,
696 TERMKEY_SYM_SELECT,
697 TERMKEY_SYM_PAGEUP,
698 TERMKEY_SYM_PAGEDOWN,
699 TERMKEY_SYM_HOME,
700 TERMKEY_SYM_END,
701 TERMKEY_SYM_CANCEL,
702 TERMKEY_SYM_CLEAR,
703 TERMKEY_SYM_CLOSE,
704 TERMKEY_SYM_COMMAND,
705 TERMKEY_SYM_COPY,
706 TERMKEY_SYM_EXIT,
707 TERMKEY_SYM_HELP,
708 TERMKEY_SYM_MARK,
709 TERMKEY_SYM_MESSAGE,
710 TERMKEY_SYM_MOVE,
711 TERMKEY_SYM_OPEN,
712 TERMKEY_SYM_OPTIONS,
713 TERMKEY_SYM_PRINT,
714 TERMKEY_SYM_REDO,
715 TERMKEY_SYM_REFERENCE,
716 TERMKEY_SYM_REFRESH,
717 TERMKEY_SYM_REPLACE,
718 TERMKEY_SYM_RESTART,
719 TERMKEY_SYM_RESUME,
720 TERMKEY_SYM_SAVE,
721 TERMKEY_SYM_SUSPEND,
722 TERMKEY_SYM_UNDO,
723 TERMKEY_SYM_KP0,
724 TERMKEY_SYM_KP1,
725 TERMKEY_SYM_KP2,
726 TERMKEY_SYM_KP3,
727 TERMKEY_SYM_KP4,
728 TERMKEY_SYM_KP5,
729 TERMKEY_SYM_KP6,
730 TERMKEY_SYM_KP7,
731 TERMKEY_SYM_KP8,
732 TERMKEY_SYM_KP9,
733 TERMKEY_SYM_KPENTER,
734 TERMKEY_SYM_KPPLUS,
735 TERMKEY_SYM_KPMINUS,
736 TERMKEY_SYM_KPMULT,
737 TERMKEY_SYM_KPDIV,
738 TERMKEY_SYM_KPCOMMA,
739 TERMKEY_SYM_KPPERIOD,
740 TERMKEY_SYM_KPEQUALS,
743 TermKey *termkey = vis->ui->termkey_get(vis->ui);
744 text_appendf(txt, " ␣ (a literal \" \" space symbol must be used to refer to <Space>)\n");
745 for (size_t i = 0; i < LENGTH(keys); i++) {
746 text_appendf(txt, " <%s>\n", termkey_get_keyname(termkey, keys[i]));
750 static bool cmd_help(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
751 if (!vis_window_new(vis, NULL))
752 return false;
754 Text *txt = vis->win->file->text;
756 text_appendf(txt, "vis %s (PID: %ld)\n\n", VERSION, (long)getpid());
758 text_appendf(txt, " Modes\n\n");
759 for (int i = 0; i < LENGTH(vis_modes); i++) {
760 Mode *mode = &vis_modes[i];
761 if (mode->help)
762 text_appendf(txt, " %-18s\t%s\n", mode->name, mode->help);
765 if (!map_empty(vis->keymap)) {
766 text_appendf(txt, "\n Layout specific mappings (affects all modes except INSERT/REPLACE)\n\n");
767 map_iterate(vis->keymap, print_keylayout, txt);
770 print_mode(&vis_modes[VIS_MODE_NORMAL], txt);
771 print_mode(&vis_modes[VIS_MODE_OPERATOR_PENDING], txt);
772 print_mode(&vis_modes[VIS_MODE_VISUAL], txt);
773 print_mode(&vis_modes[VIS_MODE_INSERT], txt);
775 text_appendf(txt, "\n :-Commands\n\n");
776 map_iterate(vis->cmds, print_cmd, txt);
778 text_appendf(txt, "\n Marks\n\n");
779 text_appendf(txt, " a-z General purpose marks\n");
780 for (size_t i = 0; i < LENGTH(vis_marks); i++) {
781 const char *help = VIS_HELP_USE(vis_marks[i].help);
782 text_appendf(txt, " %c %s\n", vis_marks[i].name, help ? help : "");
785 text_appendf(txt, "\n Registers\n\n");
786 text_appendf(txt, " a-z General purpose registers\n");
787 text_appendf(txt, " A-Z Append to corresponding general purpose register\n");
788 for (size_t i = 0; i < LENGTH(vis_registers); i++) {
789 const char *help = VIS_HELP_USE(vis_registers[i].help);
790 text_appendf(txt, " %c %s\n", vis_registers[i].name, help ? help : "");
793 text_appendf(txt, "\n :set command options\n\n");
794 map_iterate(vis->options, print_option, txt);
796 text_appendf(txt, "\n Key binding actions\n\n");
797 map_iterate(vis->actions, print_action, txt);
799 text_appendf(txt, "\n Symbolic keys usable for key bindings "
800 "(prefix with C-, S-, and M- for Ctrl, Shift and Alt respectively)\n\n");
801 print_symbolic_keys(vis, txt);
803 char *paths[] = { NULL, NULL };
804 char *paths_description[] = {
805 "Lua paths used to load runtime files (? will be replaced by filename):",
806 "Lua paths used to load C libraries (? will be replaced by filename):",
809 if (vis_lua_paths_get(vis, &paths[0], &paths[1])) {
810 for (size_t i = 0; i < LENGTH(paths); i++) {
811 text_appendf(txt, "\n %s\n\n", paths_description[i]);
812 for (char *elem = paths[i], *next; elem; elem = next) {
813 if ((next = strstr(elem, ";")))
814 *next++ = '\0';
815 if (*elem)
816 text_appendf(txt, " %s\n", elem);
818 free(paths[i]);
822 text_appendf(txt, "\n Compile time configuration\n\n");
824 const struct {
825 const char *name;
826 bool enabled;
827 } configs[] = {
828 { "Curses support: ", CONFIG_CURSES },
829 { "Lua support: ", CONFIG_LUA },
830 { "Lua LPeg statically built-in: ", CONFIG_LPEG },
831 { "TRE based regex support: ", CONFIG_TRE },
832 { "POSIX ACL support: ", CONFIG_ACL },
833 { "SELinux support: ", CONFIG_SELINUX },
836 for (size_t i = 0; i < LENGTH(configs); i++)
837 text_appendf(txt, " %-32s\t%s\n", configs[i].name, configs[i].enabled ? "yes" : "no");
839 text_save(txt, NULL);
840 view_cursor_to(vis->win->view, 0);
842 if (argv[1])
843 vis_motion(vis, VIS_MOVE_SEARCH_FORWARD, argv[1]);
844 return true;
847 static bool cmd_langmap(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
848 const char *nonlatin = argv[1];
849 const char *latin = argv[2];
850 bool mapped = true;
852 if (!latin || !nonlatin) {
853 vis_info_show(vis, "usage: langmap <non-latin keys> <latin keys>");
854 return false;
857 while (*latin && *nonlatin) {
858 size_t i = 0, j = 0;
859 char latin_key[8], nonlatin_key[8];
860 do {
861 if (i < sizeof(latin_key)-1)
862 latin_key[i++] = *latin;
863 latin++;
864 } while (!ISUTF8(*latin));
865 do {
866 if (j < sizeof(nonlatin_key)-1)
867 nonlatin_key[j++] = *nonlatin;
868 nonlatin++;
869 } while (!ISUTF8(*nonlatin));
870 latin_key[i] = '\0';
871 nonlatin_key[j] = '\0';
872 mapped &= vis_keymap_add(vis, nonlatin_key, strdup(latin_key));
875 return mapped;
878 static bool cmd_map(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
879 bool mapped = false;
880 bool local = strstr(argv[0], "-") != NULL;
881 enum VisMode mode = vis_mode_from(vis, argv[1]);
883 if (local && !win) {
884 vis_info_show(vis, "Invalid window for :%s", argv[0]);
885 return false;
888 if (mode == VIS_MODE_INVALID || !argv[2] || !argv[3]) {
889 vis_info_show(vis, "usage: %s mode lhs rhs", argv[0]);
890 return false;
893 const char *lhs = argv[2];
894 KeyBinding *binding = vis_binding_new(vis);
895 if (!binding || !(binding->alias = strdup(argv[3])))
896 goto err;
898 if (local)
899 mapped = vis_window_mode_map(win, mode, cmd->flags == '!', lhs, binding);
900 else
901 mapped = vis_mode_map(vis, mode, cmd->flags == '!', lhs, binding);
903 err:
904 if (!mapped) {
905 vis_info_show(vis, "Failed to map `%s' in %s mode%s", lhs, argv[1],
906 cmd->flags != '!' ? ", mapping already exists, "
907 "override with `!'" : "");
908 vis_binding_free(vis, binding);
910 return mapped;
913 static bool cmd_unmap(Vis *vis, Win *win, Command *cmd, const char *argv[], Selection *sel, Filerange *range) {
914 bool unmapped = false;
915 bool local = strstr(argv[0], "-") != NULL;
916 enum VisMode mode = vis_mode_from(vis, argv[1]);
917 const char *lhs = argv[2];
919 if (local && !win) {
920 vis_info_show(vis, "Invalid window for :%s", argv[0]);
921 return false;
924 if (mode == VIS_MODE_INVALID || !lhs) {
925 vis_info_show(vis, "usage: %s mode lhs", argv[0]);
926 return false;
929 if (local)
930 unmapped = vis_window_mode_unmap(win, mode, lhs);
931 else
932 unmapped = vis_mode_unmap(vis, mode, lhs);
933 if (!unmapped)
934 vis_info_show(vis, "Failed to unmap `%s' in %s mode", lhs, argv[1]);
935 return unmapped;