1 /* $NetBSD: v_replace.c,v 1.2 2013/11/22 15:52:06 christos Exp $ */
3 * Copyright (c) 1992, 1993, 1994
4 * The Regents of the University of California. All rights reserved.
5 * Copyright (c) 1992, 1993, 1994, 1995, 1996
6 * Keith Bostic. All rights reserved.
8 * See the LICENSE file for redistribution information.
14 static const char sccsid
[] = "Id: v_replace.c,v 10.24 2001/06/25 15:19:34 skimo Exp (Berkeley) Date: 2001/06/25 15:19:34 ";
17 #include <sys/types.h>
18 #include <sys/queue.h>
21 #include <bitstring.h>
29 #include "../common/common.h"
33 * v_replace -- [count]r<char>
36 * The r command in historic vi was almost beautiful in its badness. For
37 * example, "r<erase>" and "r<word erase>" beeped the terminal and deleted
38 * a single character. "Nr<carriage return>", where N was greater than 1,
39 * inserted a single carriage return. "r<escape>" did cancel the command,
40 * but "r<literal><escape>" erased a single character. To enter a literal
41 * <literal> character, it required three <literal> characters after the
42 * command. This may not be right, but at least it's not insane.
44 * PUBLIC: int v_replace __P((SCR *, VICMD *));
47 v_replace(SCR
*sp
, VICMD
*vp
)
61 * If the line doesn't exist, or it's empty, replacement isn't
62 * allowed. It's not hard to implement, but:
64 * 1: It's historic practice (vi beeped before the replacement
65 * character was even entered).
66 * 2: For consistency, this change would require that the more
67 * general case, "Nr", when the user is < N characters from
68 * the end of the line, also work, which would be a bit odd.
69 * 3: Replacing with a <newline> has somewhat odd semantics.
71 if (db_get(sp
, vp
->m_start
.lno
, DBG_FATAL
, &p
, &len
))
74 msgq(sp
, M_BERR
, "186|No characters to replace");
79 * Figure out how many characters to be replace. For no particular
80 * reason (other than that the semantics of replacing the newline
81 * are confusing) only permit the replacement of the characters in
82 * the current line. I suppose we could append replacement characters
83 * to the line, but I see no compelling reason to do so. Check this
84 * before we get the character to match historic practice, where Nr
85 * failed immediately if there were less than N characters from the
86 * cursor to the end of the line.
88 cnt
= F_ISSET(vp
, VC_C1SET
) ? vp
->count
: 1;
89 vp
->m_stop
.lno
= vp
->m_start
.lno
;
90 vp
->m_stop
.cno
= vp
->m_start
.cno
+ cnt
- 1;
91 if (vp
->m_stop
.cno
> len
- 1) {
92 v_eol(sp
, &vp
->m_start
);
97 * If it's not a repeat, reset the current mode and get a replacement
101 if (!F_ISSET(vp
, VC_ISDOT
)) {
102 sp
->showmode
= SM_REPLACE
;
103 if (vs_refresh(sp
, 0))
105 next
: if (v_event_get(sp
, &ev
, 0, 0))
108 switch (ev
.e_event
) {
111 * <literal_next> means escape the next character.
112 * <escape> means they changed their minds.
115 if (ev
.e_value
== K_VLNEXT
) {
119 if (ev
.e_value
== K_ESCAPE
)
123 vip
->rvalue
= ev
.e_value
;
127 F_SET(sp
, SC_EXIT_FORCE
);
130 /* <interrupt> means they changed their minds. */
133 /* <resize> interrupts the input mode. */
134 v_emsg(sp
, NULL
, VIM_WRESIZE
);
137 if (v_erepaint(sp
, &ev
))
141 v_event_err(sp
, &ev
);
147 GET_SPACE_RETW(sp
, bp
, blen
, len
);
152 * Versions of nvi before 1.57 created N new lines when they replaced
153 * N characters with <carriage-return> or <newline> characters. This
154 * is different from the historic vi, which replaced N characters with
155 * a single new line. Users complained, so we match historic practice.
157 if ((!quote
&& vip
->rvalue
== K_CR
) || vip
->rvalue
== K_NL
) {
158 /* Set return line. */
159 vp
->m_stop
.lno
= vp
->m_start
.lno
+ 1;
162 /* The first part of the current line. */
163 if (db_set(sp
, vp
->m_start
.lno
, p
, vp
->m_start
.cno
))
167 * The rest of the current line. And, of course, now it gets
168 * tricky. If there are characters left in the line and if
169 * the autoindent edit option is set, white space after the
170 * replaced character is discarded, autoindent is applied, and
171 * the cursor moves to the last indent character.
173 p
+= vp
->m_start
.cno
+ cnt
;
174 len
-= vp
->m_start
.cno
+ cnt
;
175 if (len
!= 0 && O_ISSET(sp
, O_AUTOINDENT
))
176 for (; len
&& ISBLANK((UCHAR_T
)*p
); --len
, ++p
);
178 if ((tp
= text_init(sp
, p
, len
, len
)) == NULL
)
181 if (len
!= 0 && O_ISSET(sp
, O_AUTOINDENT
)) {
182 if (v_txt_auto(sp
, vp
->m_start
.lno
, NULL
, 0, tp
))
184 vp
->m_stop
.cno
= tp
->ai
? tp
->ai
- 1 : 0;
188 vp
->m_stop
.cno
= tp
->ai
? tp
->ai
- 1 : 0;
189 if (db_append(sp
, 1, vp
->m_start
.lno
, tp
->lb
, tp
->len
))
196 STRSET(bp
+ vp
->m_start
.cno
, vip
->rlast
, cnt
);
197 rval
= db_set(sp
, vp
->m_start
.lno
, bp
, len
);
199 FREE_SPACEW(sp
, bp
, blen
);
201 vp
->m_final
= vp
->m_stop
;