4 * Copyright (C) 2015-2019 Ali Gholami Rudi <ali at rudi dot ir>
6 * Permission to use, copy, modify, and/or distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the above
8 * copyright notice and this permission notice appear in all copies.
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
25 static char vi_msg
[EXLEN
]; /* current message */
26 static char vi_charlast
[8]; /* the last character searched via f, t, F, or T */
27 static int vi_charcmd
; /* the character finding command */
28 static int vi_arg1
, vi_arg2
; /* the first and second arguments */
29 static int vi_ybuf
; /* current yank buffer */
30 static int vi_pcol
; /* the column requested by | command */
31 static int vi_printed
; /* ex_print() calls since the last command */
32 static int vi_scroll
; /* scroll amount for ^f and ^d*/
33 static int vi_soset
, vi_so
; /* search offset; 1 in "/kw/1" */
35 static void vi_wait(void)
38 free(ex_read("[enter to continue]"));
44 static void vi_drawmsg(void)
48 led_printmsg(vi_msg
, xrows
, "---");
53 static void vi_drawrow(int row
)
55 char *s
= lbuf_get(xb
, row
);
56 led_print(s
? s
: (row
? "~" : ""), row
- xtop
, ex_filetype());
59 /* redraw the screen */
60 static void vi_drawagain(int xcol
, int lineonly
)
64 for (i
= xtop
; i
< xtop
+ xrows
; i
++)
65 if (!lineonly
|| i
== xrow
)
68 term_pos(xrow
, led_pos(lbuf_get(xb
, i
), xcol
));
72 /* update the screen */
73 static void vi_drawupdate(int xcol
, int otop
)
79 term_room(otop
- xtop
);
81 int n
= MIN(xtop
- otop
, xrows
);
82 for (i
= 0; i
< n
; i
++)
83 vi_drawrow(xtop
+ xrows
- n
+ i
);
85 int n
= MIN(otop
- xtop
, xrows
);
86 for (i
= 0; i
< n
; i
++)
89 term_pos(xrow
, led_pos(lbuf_get(xb
, i
), xcol
));
93 term_pos(xrow
, led_pos(lbuf_get(xb
, i
), xcol
));
96 /* update the screen by removing lines r1 to r2 before an input command */
97 static void vi_drawrm(int r1
, int r2
, int newln
)
99 r1
= MIN(MAX(r1
, xtop
), xtop
+ xrows
);
100 r2
= MIN(MAX(r2
, xtop
), xtop
+ xrows
);
101 term_pos(r1
- xtop
, 0);
102 term_room(r1
- r2
+ newln
);
105 static int vi_buf
[128];
106 static int vi_buflen
;
108 static int vi_read(void)
110 return vi_buflen
? vi_buf
[--vi_buflen
] : term_read();
113 static void vi_back(int c
)
115 if (vi_buflen
< sizeof(vi_buf
))
116 vi_buf
[vi_buflen
++] = c
;
119 static char *vi_char(void)
121 return led_read(&xkmap
);
124 static char *vi_prompt(char *msg
, int *kmap
)
127 term_pos(xrows
, led_pos(msg
, 0));
129 s
= led_prompt(msg
, "", kmap
, "---");
132 r
= uc_dup(strlen(s
) >= strlen(msg
) ? s
+ strlen(msg
) : s
);
137 /* read an ex input line */
138 char *ex_read(char *msg
)
144 char *s
= led_prompt(msg
, "", &xkmap
, "---");
151 while ((c
= getchar()) != EOF
&& c
!= '\n')
157 return sbuf_done(sb
);
160 /* show an ex message */
161 void ex_show(char *msg
)
164 snprintf(vi_msg
, sizeof(vi_msg
), "%s", msg
);
166 led_print(msg
, -1, "---");
173 /* print an ex output line */
174 void ex_print(char *line
)
177 vi_printed
+= line
? 1 : 2;
179 snprintf(vi_msg
, sizeof(vi_msg
), "%s", line
);
181 led_print(line
, -1, "");
189 static int vi_yankbuf(void)
198 static int vi_prefix(void)
202 if ((c
>= '1' && c
<= '9')) {
204 n
= n
* 10 + c
- '0';
212 static int vi_col2off(struct lbuf
*lb
, int row
, int col
)
214 char *ln
= lbuf_get(lb
, row
);
215 return ln
? ren_off(ln
, col
) : 0;
218 static int vi_off2col(struct lbuf
*lb
, int row
, int off
)
220 char *ln
= lbuf_get(lb
, row
);
221 return ln
? ren_pos(ln
, off
) : 0;
224 static int vi_nextoff(struct lbuf
*lb
, int dir
, int *row
, int *off
)
227 if (o
< 0 || !lbuf_get(lb
, *row
) || o
>= uc_slen(lbuf_get(lb
, *row
)))
233 static int vi_nextcol(struct lbuf
*lb
, int dir
, int *row
, int *off
)
235 char *ln
= lbuf_get(lb
, *row
);
236 int col
= ln
? ren_pos(ln
, *off
) : 0;
237 int o
= ln
? ren_next(ln
, col
, dir
) : -1;
240 *off
= ren_off(ln
, o
);
244 static int vi_findchar(struct lbuf
*lb
, char *cs
, int cmd
, int n
, int *row
, int *off
)
246 strcpy(vi_charlast
, cs
);
248 return lbuf_findchar(lb
, cs
, cmd
, n
, row
, off
);
251 static int vi_search(int cmd
, int cnt
, int *row
, int *off
)
259 if (cmd
== '/' || cmd
== '?') {
260 char sign
[4] = {cmd
};
262 char *kw
= vi_prompt(sign
, &xkmap
);
271 if ((re
= re_read(&kw
))) {
272 ex_kwdset(re
[0] ? re
: NULL
, cmd
== '/' ? +1 : -1);
281 if (!lbuf_len(xb
) || ex_kwd(&kwd
, &dir
))
283 dir
= cmd
== 'N' ? -dir
: dir
;
285 for (i
= 0; i
< cnt
; i
++) {
286 if (lbuf_search(xb
, kwd
, dir
, &r
, &o
, &len
)) {
290 if (i
+ 1 < cnt
&& cmd
== '/')
298 if (*row
+ vi_so
< 0 || *row
+ vi_so
>= lbuf_len(xb
))
305 snprintf(vi_msg
, sizeof(vi_msg
), "\"%s\" not found\n", kwd
);
309 /* read a line motion */
310 static int vi_motionln(int *row
, int cmd
)
312 int cnt
= (vi_arg1
? vi_arg1
: 1) * (vi_arg2
? vi_arg2
: 1);
314 int mark
, mark_row
, mark_off
;
318 *row
= MIN(*row
+ cnt
, lbuf_len(xb
) - 1);
321 *row
= MAX(*row
- cnt
, 0);
324 *row
= MIN(*row
+ cnt
- 1, lbuf_len(xb
) - 1);
327 if ((mark
= vi_read()) <= 0)
329 if (lbuf_jump(xb
, mark
, &mark_row
, &mark_off
))
334 *row
= MIN(*row
+ cnt
, lbuf_len(xb
) - 1);
337 *row
= MAX(*row
- cnt
, 0);
340 *row
= (vi_arg1
|| vi_arg2
) ? cnt
- 1 : lbuf_len(xb
) - 1;
343 *row
= MIN(xtop
+ cnt
- 1, lbuf_len(xb
) - 1);
346 *row
= MIN(xtop
+ xrows
- 1 - cnt
+ 1, lbuf_len(xb
) - 1);
349 *row
= MIN(xtop
+ xrows
/ 2, lbuf_len(xb
) - 1);
353 *row
= MIN(*row
+ cnt
- 1, lbuf_len(xb
) - 1);
356 if (c
== '%' && (vi_arg1
|| vi_arg2
)) {
359 *row
= MAX(0, lbuf_len(xb
) - 1) * cnt
/ 100;
370 static char *vi_curword(struct lbuf
*lb
, int row
, int off
)
373 char *ln
= lbuf_get(lb
, row
);
377 beg
= uc_chr(ln
, ren_noeol(ln
, off
));
379 while (*end
&& uc_kind(end
) == 1)
381 while (beg
> ln
&& uc_kind(uc_beg(ln
, beg
- 1)) == 1)
382 beg
= uc_beg(ln
, beg
- 1);
387 sbuf_mem(sb
, beg
, end
- beg
);
389 return sbuf_done(sb
);
393 static int vi_motion(int *row
, int *off
)
395 int cnt
= (vi_arg1
? vi_arg1
: 1) * (vi_arg2
? vi_arg2
: 1);
396 char *ln
= lbuf_get(xb
, *row
);
397 int dir
= dir_context(ln
? ln
: "");
398 int mark
, mark_row
, mark_off
;
402 if ((mv
= vi_motionln(row
, 0))) {
409 if (!(cs
= vi_char()))
411 if (vi_findchar(xb
, cs
, mv
, cnt
, row
, off
))
415 if (!(cs
= vi_char()))
417 if (vi_findchar(xb
, cs
, mv
, cnt
, row
, off
))
423 if (vi_findchar(xb
, vi_charlast
, vi_charcmd
, cnt
, row
, off
))
429 if (vi_findchar(xb
, vi_charlast
, vi_charcmd
, -cnt
, row
, off
))
433 for (i
= 0; i
< cnt
; i
++)
434 if (vi_nextcol(xb
, -1 * dir
, row
, off
))
438 for (i
= 0; i
< cnt
; i
++)
439 if (vi_nextcol(xb
, +1 * dir
, row
, off
))
443 if (!(cs
= vi_char()))
445 if (vi_findchar(xb
, cs
, mv
, cnt
, row
, off
))
449 if (!(cs
= vi_char()))
451 if (vi_findchar(xb
, cs
, mv
, cnt
, row
, off
))
455 for (i
= 0; i
< cnt
; i
++)
456 if (lbuf_wordend(xb
, 1, -1, row
, off
))
460 for (i
= 0; i
< cnt
; i
++)
461 if (lbuf_wordend(xb
, 1, +1, row
, off
))
465 for (i
= 0; i
< cnt
; i
++)
466 if (lbuf_wordbeg(xb
, 1, +1, row
, off
))
470 for (i
= 0; i
< cnt
; i
++)
471 if (lbuf_wordend(xb
, 0, -1, row
, off
))
475 for (i
= 0; i
< cnt
; i
++)
476 if (lbuf_wordend(xb
, 0, +1, row
, off
))
480 for (i
= 0; i
< cnt
; i
++)
481 if (lbuf_wordbeg(xb
, 0, +1, row
, off
))
485 for (i
= 0; i
< cnt
; i
++)
486 if (lbuf_paragraphbeg(xb
, -1, row
, off
))
490 for (i
= 0; i
< cnt
; i
++)
491 if (lbuf_paragraphbeg(xb
, +1, row
, off
))
495 if (vi_read() != '[')
497 for (i
= 0; i
< cnt
; i
++)
498 if (lbuf_sectionbeg(xb
, -1, row
, off
))
502 if (vi_read() != ']')
504 for (i
= 0; i
< cnt
; i
++)
505 if (lbuf_sectionbeg(xb
, +1, row
, off
))
512 *off
= lbuf_indents(xb
, *row
);
515 *off
= lbuf_eol(xb
, *row
);
518 *off
= vi_col2off(xb
, *row
, cnt
- 1);
522 if (vi_search(mv
, cnt
, row
, off
))
526 if (vi_search(mv
, cnt
, row
, off
))
530 if (vi_search(mv
, cnt
, row
, off
))
534 if (vi_search(mv
, cnt
, row
, off
))
538 if (!(cs
= vi_curword(xb
, *row
, *off
)))
542 if (vi_search('n', cnt
, row
, off
))
546 for (i
= 0; i
< cnt
; i
++)
547 if (vi_nextoff(xb
, +1, row
, off
))
552 for (i
= 0; i
< cnt
; i
++)
553 if (vi_nextoff(xb
, -1, row
, off
))
557 if ((mark
= vi_read()) <= 0)
559 if (lbuf_jump(xb
, mark
, &mark_row
, &mark_off
))
565 if (lbuf_pair(xb
, row
, off
))
575 static void swap(int *a
, int *b
)
582 static char *lbuf_region(struct lbuf
*lb
, int r1
, int o1
, int r2
, int o2
)
587 return uc_sub(lbuf_get(lb
, r1
), o1
, o2
);
589 s1
= uc_sub(lbuf_get(lb
, r1
), o1
, -1);
590 s3
= uc_sub(lbuf_get(lb
, r2
), 0, o2
);
591 s2
= lbuf_cp(lb
, r1
+ 1, r2
);
598 return sbuf_done(sb
);
601 static void vi_yank(int r1
, int o1
, int r2
, int o2
, int lnmode
)
604 region
= lbuf_region(xb
, r1
, lnmode
? 0 : o1
, r2
, lnmode
? -1 : o2
);
605 reg_put(vi_ybuf
, region
, lnmode
);
608 xoff
= lnmode
? xoff
: o1
;
611 static void vi_delete(int r1
, int o1
, int r2
, int o2
, int lnmode
)
615 region
= lbuf_region(xb
, r1
, lnmode
? 0 : o1
, r2
, lnmode
? -1 : o2
);
616 reg_put(vi_ybuf
, region
, lnmode
);
618 pref
= lnmode
? uc_dup("") : uc_sub(lbuf_get(xb
, r1
), 0, o1
);
619 post
= lnmode
? uc_dup("\n") : uc_sub(lbuf_get(xb
, r2
), o2
, -1);
621 struct sbuf
*sb
= sbuf_make();
624 lbuf_edit(xb
, sbuf_buf(sb
), r1
, r2
+ 1);
627 lbuf_edit(xb
, NULL
, r1
, r2
+ 1);
630 xoff
= lnmode
? lbuf_indents(xb
, xrow
) : o1
;
635 static int linecount(char *s
)
639 if ((s
= strchr(s
, '\n')))
644 static int charcount(char *text
, char *post
)
646 int tlen
= strlen(text
);
647 int plen
= strlen(post
);
652 for (i
= 0; i
< tlen
- plen
; i
++)
655 return uc_slen(nl
) - uc_slen(post
);
658 static char *vi_input(char *pref
, char *post
, int *row
, int *off
)
660 char *rep
= led_input(pref
, post
, &xkmap
, ex_filetype());
663 *row
= linecount(rep
) - 1;
664 *off
= charcount(rep
, post
) - 1;
670 static char *vi_indents(char *ln
)
672 struct sbuf
*sb
= sbuf_make();
673 while (xai
&& ln
&& (*ln
== ' ' || *ln
== '\t'))
675 return sbuf_done(sb
);
678 static void vi_change(int r1
, int o1
, int r2
, int o2
, int lnmode
)
684 region
= lbuf_region(xb
, r1
, lnmode
? 0 : o1
, r2
, lnmode
? -1 : o2
);
685 reg_put(vi_ybuf
, region
, lnmode
);
687 pref
= lnmode
? vi_indents(lbuf_get(xb
, r1
)) : uc_sub(lbuf_get(xb
, r1
), 0, o1
);
688 post
= lnmode
? uc_dup("\n") : uc_sub(lbuf_get(xb
, r2
), o2
, -1);
689 vi_drawrm(r1
, r2
, 0);
690 rep
= vi_input(pref
, post
, &row
, &off
);
692 lbuf_edit(xb
, rep
, r1
, r2
+ 1);
701 static void vi_case(int r1
, int o1
, int r2
, int o2
, int lnmode
, int cmd
)
705 region
= lbuf_region(xb
, r1
, lnmode
? 0 : o1
, r2
, lnmode
? -1 : o2
);
708 int c
= (unsigned char) s
[0];
715 s
[0] = islower(c
) ? toupper(c
) : tolower(c
);
719 pref
= lnmode
? uc_dup("") : uc_sub(lbuf_get(xb
, r1
), 0, o1
);
720 post
= lnmode
? uc_dup("\n") : uc_sub(lbuf_get(xb
, r2
), o2
, -1);
722 struct sbuf
*sb
= sbuf_make();
724 sbuf_str(sb
, region
);
726 lbuf_edit(xb
, sbuf_buf(sb
), r1
, r2
+ 1);
729 lbuf_edit(xb
, region
, r1
, r2
+ 1);
732 xoff
= lnmode
? lbuf_indents(xb
, r2
) : o2
;
738 static void vi_pipe(int r1
, int r2
)
743 char *cmd
= vi_prompt("!", &kmap
);
746 text
= lbuf_cp(xb
, r1
, r2
+ 1);
747 rep
= cmd_pipe(cmd
, text
, 1, 1);
749 lbuf_edit(xb
, rep
, r1
, r2
+ 1);
755 static void vi_shift(int r1
, int r2
, int dir
)
760 for (i
= r1
; i
<= r2
; i
++) {
761 if (!(ln
= lbuf_get(xb
, i
)))
767 ln
= ln
[0] == ' ' || ln
[0] == '\t' ? ln
+ 1 : ln
;
769 lbuf_edit(xb
, sbuf_buf(sb
), i
, i
+ 1);
773 xoff
= lbuf_indents(xb
, xrow
);
776 static int vc_motion(int cmd
)
778 int r1
= xrow
, r2
= xrow
; /* region rows */
779 int o1
= xoff
, o2
= xoff
; /* visual region columns */
780 int lnmode
= 0; /* line-based region */
782 vi_arg2
= vi_prefix();
785 o1
= ren_noeol(lbuf_get(xb
, r1
), o1
);
787 if ((mv
= vi_motionln(&r2
, cmd
))) {
789 } else if (!(mv
= vi_motion(&r2
, &o2
))) {
798 o2
= lbuf_eol(xb
, r2
);
804 if (r1
== r2
&& o1
> o2
)
806 o1
= ren_noeol(lbuf_get(xb
, r1
), o1
);
807 if (!lnmode
&& strchr("fFtTeE%", mv
))
808 if (o2
< lbuf_eol(xb
, r2
))
809 o2
= ren_noeol(lbuf_get(xb
, r2
), o2
) + 1;
811 vi_yank(r1
, o1
, r2
, o2
, lnmode
);
813 vi_delete(r1
, o1
, r2
, o2
, lnmode
);
815 vi_change(r1
, o1
, r2
, o2
, lnmode
);
816 if (cmd
== '~' || cmd
== 'u' || cmd
== 'U')
817 vi_case(r1
, o1
, r2
, o2
, lnmode
, cmd
);
820 if (cmd
== '>' || cmd
== '<')
821 vi_shift(r1
, r2
, cmd
== '>' ? +1 : -1);
825 static int vc_insert(int cmd
)
828 char *ln
= lbuf_get(xb
, xrow
);
832 xoff
= lbuf_indents(xb
, xrow
);
834 xoff
= lbuf_eol(xb
, xrow
);
835 xoff
= ren_noeol(ln
, xoff
);
838 if (cmd
== 'i' || cmd
== 'I')
840 if (cmd
== 'a' || cmd
== 'A')
842 if (ln
&& ln
[0] == '\n')
844 pref
= ln
&& cmd
!= 'o' && cmd
!= 'O' ? uc_sub(ln
, 0, off
) : vi_indents(ln
);
845 post
= ln
&& cmd
!= 'o' && cmd
!= 'O' ? uc_sub(ln
, off
, -1) : uc_dup("\n");
846 vi_drawrm(xrow
, xrow
, cmd
== 'o' || cmd
== 'O');
847 rep
= vi_input(pref
, post
, &row
, &off
);
848 if ((cmd
== 'o' || cmd
== 'O') && !lbuf_len(xb
))
849 lbuf_edit(xb
, "\n", 0, 0);
851 lbuf_edit(xb
, rep
, xrow
, xrow
+ (cmd
!= 'o' && cmd
!= 'O'));
861 static int vc_put(int cmd
)
863 int cnt
= MAX(1, vi_arg1
);
865 char *buf
= reg_get(vi_ybuf
, &lnmode
);
868 snprintf(vi_msg
, sizeof(vi_msg
), "yank buffer empty\n");
872 struct sbuf
*sb
= sbuf_make();
873 for (i
= 0; i
< cnt
; i
++)
876 lbuf_edit(xb
, "\n", 0, 0);
879 lbuf_edit(xb
, sbuf_buf(sb
), xrow
, xrow
);
880 xoff
= lbuf_indents(xb
, xrow
);
883 struct sbuf
*sb
= sbuf_make();
884 char *ln
= xrow
< lbuf_len(xb
) ? lbuf_get(xb
, xrow
) : "\n";
885 int off
= ren_noeol(ln
, xoff
) + (ln
[0] != '\n' && cmd
== 'p');
886 char *s
= uc_sub(ln
, 0, off
);
889 for (i
= 0; i
< cnt
; i
++)
891 s
= uc_sub(ln
, off
, -1);
894 lbuf_edit(xb
, sbuf_buf(sb
), xrow
, xrow
+ 1);
895 xoff
= off
+ uc_slen(buf
) * cnt
- 1;
901 static int join_spaces(char *prev
, char *next
)
903 int prevlen
= strlen(prev
);
906 if (prev
[prevlen
- 1] == ' ' || next
[0] == ')')
908 return prev
[prevlen
- 1] == '.' ? 2 : 1;
911 static int vc_join(void)
914 int cnt
= vi_arg1
<= 1 ? 2 : vi_arg1
;
916 int end
= xrow
+ cnt
;
919 if (!lbuf_get(xb
, beg
) || !lbuf_get(xb
, end
- 1))
922 for (i
= beg
; i
< end
; i
++) {
923 char *ln
= lbuf_get(xb
, i
);
924 char *lnend
= strchr(ln
, '\n');
927 while (ln
[0] == ' ' || ln
[0] == '\t')
929 spaces
= i
> beg
? join_spaces(sbuf_buf(sb
), ln
) : 0;
930 off
= uc_slen(sbuf_buf(sb
));
933 sbuf_mem(sb
, ln
, lnend
- ln
);
936 lbuf_edit(xb
, sbuf_buf(sb
), beg
, end
);
942 static int vi_scrollforeward(int cnt
)
944 if (xtop
>= lbuf_len(xb
) - 1)
946 xtop
= MIN(lbuf_len(xb
) - 1, xtop
+ cnt
);
947 xrow
= MAX(xrow
, xtop
);
951 static int vi_scrollbackward(int cnt
)
955 xtop
= MAX(0, xtop
- cnt
);
956 xrow
= MIN(xrow
, xtop
+ xrows
- 1);
960 static void vc_status(void)
962 int col
= vi_off2col(xb
, xrow
, xoff
);
963 snprintf(vi_msg
, sizeof(vi_msg
),
964 "\"%s\"%c %d lines L%d C%d\n",
965 ex_path()[0] ? ex_path() : "unnamed",
966 lbuf_modified(xb
) ? '*' : ' ',
967 lbuf_len(xb
), xrow
+ 1,
968 ren_cursor(lbuf_get(xb
, xrow
), col
) + 1);
971 static void vc_charinfo(void)
973 char *c
= uc_chr(lbuf_get(xb
, xrow
), xoff
);
976 memcpy(cbuf
, c
, uc_len(c
));
977 snprintf(vi_msg
, sizeof(vi_msg
), "<%s> %04x\n", cbuf
, uc_code(c
));
981 static int vc_replace(void)
983 int cnt
= MAX(1, vi_arg1
);
984 char *cs
= vi_char();
985 char *ln
= lbuf_get(xb
, xrow
);
992 off
= ren_noeol(ln
, xoff
);
994 for (i
= 0; s
[0] != '\n' && i
< cnt
; i
++)
998 pref
= uc_sub(ln
, 0, off
);
999 post
= uc_sub(ln
, off
+ cnt
, -1);
1002 for (i
= 0; i
< cnt
; i
++)
1005 lbuf_edit(xb
, sbuf_buf(sb
), xrow
, xrow
+ 1);
1014 static char rep_cmd
[4096]; /* the last command */
1017 static void vc_repeat(void)
1020 for (i
= 0; i
< MAX(1, vi_arg1
); i
++)
1021 term_push(rep_cmd
, rep_len
);
1024 static void vc_execute(void)
1026 static int exec_buf
;
1036 buf
= reg_get(exec_buf
, &lnmode
);
1038 for (i
= 0; i
< MAX(1, vi_arg1
); i
++)
1039 term_push(buf
, strlen(buf
));
1042 static void vi(void)
1048 xtop
= MAX(0, xrow
- xrows
/ 2);
1050 xcol
= vi_off2col(xb
, xrow
, xoff
);
1051 vi_drawagain(xcol
, 0);
1052 term_pos(xrow
- xtop
, led_pos(lbuf_get(xb
, xrow
), xcol
));
1054 int mod
= 0; /* screen should be redrawn (1: the whole screen, 2: the current line) */
1056 int noff
= ren_noeol(lbuf_get(xb
, xrow
), xoff
);
1063 vi_ybuf
= vi_yankbuf();
1064 vi_arg1
= vi_prefix();
1066 vi_ybuf
= vi_yankbuf();
1067 mv
= vi_motion(&nrow
, &noff
);
1069 if (strchr("\'`GHML/?{}[]nN", mv
) ||
1070 (mv
== '%' && noff
< 0)) {
1071 lbuf_mark(xb
, '\'', xrow
, xoff
);
1072 lbuf_mark(xb
, '`', xrow
, xoff
);
1075 if (noff
< 0 && !strchr("jk", mv
))
1076 noff
= lbuf_indents(xb
, xrow
);
1077 if (strchr("jk", mv
))
1078 noff
= vi_col2off(xb
, xrow
, xcol
);
1079 xoff
= ren_noeol(lbuf_get(xb
, xrow
), noff
);
1080 if (!strchr("|jk", mv
))
1081 xcol
= vi_off2col(xb
, xrow
, xoff
);
1084 } else if (mv
== 0) {
1090 lbuf_mark(xb
, '*', xrow
, xoff
);
1093 if (vi_scrollbackward(MAX(1, vi_arg1
) * (xrows
- 1)))
1095 xoff
= lbuf_indents(xb
, xrow
);
1099 if (vi_scrollforeward(MAX(1, vi_arg1
) * (xrows
- 1)))
1101 xoff
= lbuf_indents(xb
, xrow
);
1105 if (vi_scrollforeward(MAX(1, vi_arg1
)))
1109 if (vi_scrollbackward(MAX(1, vi_arg1
)))
1116 vi_scroll
= vi_arg1
;
1117 n
= vi_scroll
? vi_scroll
: xrows
/ 2;
1118 xrow
= MAX(0, xrow
- n
);
1120 xtop
= MAX(0, xtop
- n
);
1121 xoff
= lbuf_indents(xb
, xrow
);
1125 if (xrow
== lbuf_len(xb
) - 1)
1128 vi_scroll
= vi_arg1
;
1129 n
= vi_scroll
? vi_scroll
: xrows
/ 2;
1130 xrow
= MIN(MAX(0, lbuf_len(xb
) - 1), xrow
+ n
);
1131 if (xtop
< lbuf_len(xb
) - xrows
)
1132 xtop
= MIN(lbuf_len(xb
) - xrows
, xtop
+ n
);
1133 xoff
= lbuf_indents(xb
, xrow
);
1142 if (!lbuf_undo(xb
)) {
1143 lbuf_jump(xb
, '*', &xrow
, &xoff
);
1146 snprintf(vi_msg
, sizeof(vi_msg
), "undo failed\n");
1150 if (!lbuf_redo(xb
)) {
1151 lbuf_jump(xb
, '*', &xrow
, &xoff
);
1154 snprintf(vi_msg
, sizeof(vi_msg
), "redo failed\n");
1165 ln
= vi_prompt(":", &kmap
);
1202 if ((mark
= vi_read()) > 0 && islower(mark
))
1203 lbuf_mark(xb
, mark
, xrow
, xoff
);
1214 xtop
= vi_arg1
? vi_arg1
: xrow
;
1217 n
= vi_arg1
? vi_arg1
: xrow
;
1218 xtop
= MAX(0, n
- xrows
/ 2);
1221 n
= vi_arg1
? vi_arg1
: xrow
;
1222 xtop
= MAX(0, n
- xrows
+ 1);
1226 xtd
= k
== 'r' ? -1 : +1;
1230 xtd
= k
== 'R' ? -2 : +2;
1234 xkmap
= k
== 'e' ? 0 : xkmap_alt
;
1241 if (k
== '~' || k
== 'u' || k
== 'U')
1249 if (!vc_motion('d'))
1253 vi_back(TK_CTL('h'));
1254 if (!vc_motion('d'))
1259 if (!vc_motion('c'))
1264 if (!vc_motion('d'))
1273 if (!vc_motion('c'))
1278 if (!vc_motion('c'))
1292 if (!vc_motion('~'))
1305 if (strchr("!<>ACDIJOPRSXYacdioprsxy~", c
) ||
1306 (c
== 'g' && strchr("uU~", k
))) {
1307 if (n
< sizeof(rep_cmd
)) {
1308 memcpy(rep_cmd
, cmd
, n
);
1313 if (xrow
< 0 || xrow
>= lbuf_len(xb
))
1314 xrow
= lbuf_len(xb
) ? lbuf_len(xb
) - 1 : 0;
1316 xtop
= xtop
- xrows
/ 2 > xrow
?
1317 MAX(0, xrow
- xrows
/ 2) : xrow
;
1318 if (xtop
+ xrows
<= xrow
)
1319 xtop
= xtop
+ xrows
+ xrows
/ 2 <= xrow
?
1320 xrow
- xrows
/ 2 : xrow
- xrows
+ 1;
1321 xoff
= ren_noeol(lbuf_get(xb
, xrow
), xoff
);
1323 xcol
= vi_off2col(xb
, xrow
, xoff
);
1324 if (xcol
>= xleft
+ xcols
)
1325 xleft
= xcol
- xcols
/ 2;
1327 xleft
= xcol
< xcols
? 0 : xcol
- xcols
/ 2;
1329 if (mod
|| xleft
!= oleft
)
1330 vi_drawagain(xcol
, mod
== 2 && xleft
== oleft
&& xrow
== orow
);
1331 else if (xtop
!= otop
)
1332 vi_drawupdate(xcol
, otop
);
1335 term_pos(xrow
- xtop
, led_pos(lbuf_get(xb
, xrow
),
1336 ren_cursor(lbuf_get(xb
, xrow
), xcol
)));
1343 int main(int argc
, char *argv
[])
1346 char *prog
= strchr(argv
[0], '/') ? strrchr(argv
[0], '/') + 1 : argv
[0];
1347 xvis
= strcmp("ex", prog
) && strcmp("neatex", prog
);
1348 for (i
= 1; i
< argc
&& argv
[i
][0] == '-'; i
++) {
1349 if (argv
[i
][1] == 's')
1351 if (argv
[i
][1] == 'e')
1353 if (argv
[i
][1] == 'v')
1360 if (!ex_init(argv
+ i
)) {