📄 vi.c
字号:
return ed_argument_digit(el, c); el->el_line.cursor = el->el_line.buffer; if (el->el_chared.c_vcmd.action != NOP) { cv_delfini(el); return (CC_REFRESH); } return (CC_CURSOR);}/* vi_delete_prev_char(): * Vi move to previous character (backspace) * [^H] in insert mode only */protected el_action_t/*ARGSUSED*/vi_delete_prev_char(EditLine *el, int c __attribute__((__unused__))){ if (el->el_line.cursor <= el->el_line.buffer) return (CC_ERROR); c_delbefore1(el); el->el_line.cursor--; return (CC_REFRESH);}/* vi_list_or_eof(): * Vi list choices for completion or indicate end of file if empty line * [^D] */protected el_action_t/*ARGSUSED*/vi_list_or_eof(EditLine *el, int c __attribute__((__unused__))){ if (el->el_line.cursor == el->el_line.lastchar) { if (el->el_line.cursor == el->el_line.buffer) { term_overwrite(el, STReof, 4); /* then do a EOF */ term__flush(); return (CC_EOF); } else { /* * Here we could list completions, but it is an * error right now */ term_beep(el); return (CC_ERROR); } } else {#ifdef notyet re_goto_bottom(el); *el->el_line.lastchar = '\0'; /* just in case */ return (CC_LIST_CHOICES);#else /* * Just complain for now. */ term_beep(el); return (CC_ERROR);#endif }}/* vi_kill_line_prev(): * Vi cut from beginning of line to cursor * [^U] */protected el_action_t/*ARGSUSED*/vi_kill_line_prev(EditLine *el, int c __attribute__((__unused__))){ char *kp, *cp; cp = el->el_line.buffer; kp = el->el_chared.c_kill.buf; while (cp < el->el_line.cursor) *kp++ = *cp++; /* copy it */ el->el_chared.c_kill.last = kp; c_delbefore(el, el->el_line.cursor - el->el_line.buffer); el->el_line.cursor = el->el_line.buffer; /* zap! */ return (CC_REFRESH);}/* vi_search_prev(): * Vi search history previous * [?] */protected el_action_t/*ARGSUSED*/vi_search_prev(EditLine *el, int c __attribute__((__unused__))){ return (cv_search(el, ED_SEARCH_PREV_HISTORY));}/* vi_search_next(): * Vi search history next * [/] */protected el_action_t/*ARGSUSED*/vi_search_next(EditLine *el, int c __attribute__((__unused__))){ return (cv_search(el, ED_SEARCH_NEXT_HISTORY));}/* vi_repeat_search_next(): * Vi repeat current search in the same search direction * [n] */protected el_action_t/*ARGSUSED*/vi_repeat_search_next(EditLine *el, int c __attribute__((__unused__))){ if (el->el_search.patlen == 0) return (CC_ERROR); else return (cv_repeat_srch(el, el->el_search.patdir));}/* vi_repeat_search_prev(): * Vi repeat current search in the opposite search direction * [N] *//*ARGSUSED*/protected el_action_tvi_repeat_search_prev(EditLine *el, int c __attribute__((__unused__))){ if (el->el_search.patlen == 0) return (CC_ERROR); else return (cv_repeat_srch(el, el->el_search.patdir == ED_SEARCH_PREV_HISTORY ? ED_SEARCH_NEXT_HISTORY : ED_SEARCH_PREV_HISTORY));}/* vi_next_char(): * Vi move to the character specified next * [f] */protected el_action_t/*ARGSUSED*/vi_next_char(EditLine *el, int c __attribute__((__unused__))){ return cv_csearch(el, CHAR_FWD, -1, el->el_state.argument, 0);}/* vi_prev_char(): * Vi move to the character specified previous * [F] */protected el_action_t/*ARGSUSED*/vi_prev_char(EditLine *el, int c __attribute__((__unused__))){ return cv_csearch(el, CHAR_BACK, -1, el->el_state.argument, 0);}/* vi_to_next_char(): * Vi move up to the character specified next * [t] */protected el_action_t/*ARGSUSED*/vi_to_next_char(EditLine *el, int c __attribute__((__unused__))){ return cv_csearch(el, CHAR_FWD, -1, el->el_state.argument, 1);}/* vi_to_prev_char(): * Vi move up to the character specified previous * [T] */protected el_action_t/*ARGSUSED*/vi_to_prev_char(EditLine *el, int c __attribute__((__unused__))){ return cv_csearch(el, CHAR_BACK, -1, el->el_state.argument, 1);}/* vi_repeat_next_char(): * Vi repeat current character search in the same search direction * [;] */protected el_action_t/*ARGSUSED*/vi_repeat_next_char(EditLine *el, int c __attribute__((__unused__))){ return cv_csearch(el, el->el_search.chadir, el->el_search.chacha, el->el_state.argument, el->el_search.chatflg);}/* vi_repeat_prev_char(): * Vi repeat current character search in the opposite search direction * [,] */protected el_action_t/*ARGSUSED*/vi_repeat_prev_char(EditLine *el, int c __attribute__((__unused__))){ el_action_t r; int dir = el->el_search.chadir; r = cv_csearch(el, -dir, el->el_search.chacha, el->el_state.argument, el->el_search.chatflg); el->el_search.chadir = dir; return r;}/* vi_match(): * Vi go to matching () {} or [] * [%] */protected el_action_t/*ARGSUSED*/vi_match(EditLine *el, int c){ const char match_chars[] = "()[]{}"; char *cp; int delta, i, count; char o_ch, c_ch; *el->el_line.lastchar = '\0'; /* just in case */ i = strcspn(el->el_line.cursor, match_chars); o_ch = el->el_line.cursor[i]; if (o_ch == 0) return CC_ERROR; delta = strchr(match_chars, o_ch) - match_chars; c_ch = match_chars[delta ^ 1]; count = 1; delta = 1 - (delta & 1) * 2; for (cp = &el->el_line.cursor[i]; count; ) { cp += delta; if (cp < el->el_line.buffer || cp >= el->el_line.lastchar) return CC_ERROR; if (*cp == o_ch) count++; else if (*cp == c_ch) count--; } el->el_line.cursor = cp; if (el->el_chared.c_vcmd.action != NOP) { /* NB posix says char under cursor should NOT be deleted for -ve delta - this is different to netbsd vi. */ if (delta > 0) el->el_line.cursor++; cv_delfini(el); return (CC_REFRESH); } return (CC_CURSOR);}/* vi_undo_line(): * Vi undo all changes to line * [U] */protected el_action_t/*ARGSUSED*/vi_undo_line(EditLine *el, int c){ cv_undo(el); return hist_get(el);}/* vi_to_column(): * Vi go to specified column * [|] * NB netbsd vi goes to screen column 'n', posix says nth character */protected el_action_t/*ARGSUSED*/vi_to_column(EditLine *el, int c){ el->el_line.cursor = el->el_line.buffer; el->el_state.argument--; return ed_next_char(el, 0);}/* vi_yank_end(): * Vi yank to end of line * [Y] */protected el_action_t/*ARGSUSED*/vi_yank_end(EditLine *el, int c){ cv_yank(el, el->el_line.cursor, el->el_line.lastchar - el->el_line.cursor); return CC_REFRESH;}/* vi_yank(): * Vi yank * [y] */protected el_action_t/*ARGSUSED*/vi_yank(EditLine *el, int c){ return cv_action(el, YANK);}/* vi_comment_out(): * Vi comment out current command * [c] */protected el_action_t/*ARGSUSED*/vi_comment_out(EditLine *el, int c){ el->el_line.cursor = el->el_line.buffer; c_insert(el, 1); *el->el_line.cursor = '#'; re_refresh(el); return ed_newline(el, 0);}/* vi_alias(): * Vi include shell alias * [@] * NB: posix impiles that we should enter insert mode, however * this is against historical precedent... */protected el_action_t/*ARGSUSED*/vi_alias(EditLine *el, int c){#ifdef __weak_extern char alias_name[3]; char *alias_text; extern char *get_alias_text(const char *); __weak_extern(get_alias_text); if (get_alias_text == 0) { return CC_ERROR; } alias_name[0] = '_'; alias_name[2] = 0; if (el_getc(el, &alias_name[1]) != 1) return CC_ERROR; alias_text = get_alias_text(alias_name); if (alias_text != NULL) el_push(el, alias_text); return CC_NORM;#else return CC_ERROR;#endif}/* vi_to_history_line(): * Vi go to specified history file line. * [G] */protected el_action_t/*ARGSUSED*/vi_to_history_line(EditLine *el, int c){ int sv_event_no = el->el_history.eventno; el_action_t rval; if (el->el_history.eventno == 0) { (void) strncpy(el->el_history.buf, el->el_line.buffer, EL_BUFSIZ); el->el_history.last = el->el_history.buf + (el->el_line.lastchar - el->el_line.buffer); } /* Lack of a 'count' means oldest, not 1 */ if (!el->el_state.doingarg) { el->el_history.eventno = 0x7fffffff; hist_get(el); } else { /* This is brain dead, all the rest of this code counts * upwards going into the past. Here we need count in the * other direction (to match the output of fc -l). * I could change the world, but this seems to suffice. */ el->el_history.eventno = 1; if (hist_get(el) == CC_ERROR) return CC_ERROR; el->el_history.eventno = 1 + el->el_history.ev.num - el->el_state.argument; if (el->el_history.eventno < 0) { el->el_history.eventno = sv_event_no; return CC_ERROR; } } rval = hist_get(el); if (rval == CC_ERROR) el->el_history.eventno = sv_event_no; return rval;}/* vi_histedit(): * Vi edit history line with vi * [v] */protected el_action_t/*ARGSUSED*/vi_histedit(EditLine *el, int c){ int fd; pid_t pid; int st; char tempfile[] = "/tmp/histedit.XXXXXXXXXX"; char *cp; if (el->el_state.doingarg) { if (vi_to_history_line(el, 0) == CC_ERROR) return CC_ERROR; } fd = mkstemp(tempfile); if (fd < 0) return CC_ERROR; cp = el->el_line.buffer; write(fd, cp, el->el_line.lastchar - cp +0u); write(fd, "\n", 1); pid = fork(); switch (pid) { case -1: close(fd); unlink(tempfile); return CC_ERROR; case 0: close(fd); execlp("vi", "vi", tempfile, (char *) NULL); exit(0); /*NOTREACHED*/ default: while (waitpid(pid, &st, 0) != pid) continue; lseek(fd, 0ll, SEEK_SET); st = read(fd, cp, el->el_line.limit - cp +0u); if (st > 0 && cp[st - 1] == '\n') st--; el->el_line.cursor = cp; el->el_line.lastchar = cp + st; break; } close(fd); unlink(tempfile); /* return CC_REFRESH; */ return ed_newline(el, 0);}/* vi_history_word(): * Vi append word from previous input line * [_] * Who knows where this one came from! * '_' in vi means 'entire current line', so 'cc' is a synonym for 'c_' */protected el_action_t/*ARGSUSED*/vi_history_word(EditLine *el, int c){ const char *wp = HIST_FIRST(el); const char *wep, *wsp; int len; char *cp; const char *lim; if (wp == NULL) return CC_ERROR; wep = wsp = 0; do { while (isspace((unsigned char)*wp)) wp++; if (*wp == 0) break; wsp = wp; while (*wp && !isspace((unsigned char)*wp)) wp++; wep = wp; } while ((!el->el_state.doingarg || --el->el_state.argument > 0) && *wp != 0); if (wsp == 0 || (el->el_state.doingarg && el->el_state.argument != 0)) return CC_ERROR; cv_undo(el); len = wep - wsp; if (el->el_line.cursor < el->el_line.lastchar) el->el_line.cursor++; c_insert(el, len + 1); cp = el->el_line.cursor; lim = el->el_line.limit; if (cp < lim) *cp++ = ' '; while (wsp < wep && cp < lim) *cp++ = *wsp++; el->el_line.cursor = cp; el->el_map.current = el->el_map.key; return CC_REFRESH;}/* vi_redo(): * Vi redo last non-motion command * [.] */protected el_action_t/*ARGSUSED*/vi_redo(EditLine *el, int c){ c_redo_t *r = &el->el_chared.c_redo; if (!el->el_state.doingarg && r->count) { el->el_state.doingarg = 1; el->el_state.argument = r->count; } el->el_chared.c_vcmd.pos = el->el_line.cursor; el->el_chared.c_vcmd.action = r->action; if (r->pos != r->buf) { if (r->pos + 1 > r->lim) /* sanity */ r->pos = r->lim - 1; r->pos[0] = 0; el_push(el, r->buf); } el->el_state.thiscmd = r->cmd; el->el_state.thisch = r->ch; return (*el->el_map.func[r->cmd])(el, r->ch);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -