xref: /freebsd/contrib/nvi/ex/ex_visual.c (revision c271fa9295c13b3cc926562c9b204fa597dba7e6)
1b8ba871bSPeter Wemm /*-
2b8ba871bSPeter Wemm  * Copyright (c) 1992, 1993, 1994
3b8ba871bSPeter Wemm  *	The Regents of the University of California.  All rights reserved.
4b8ba871bSPeter Wemm  * Copyright (c) 1992, 1993, 1994, 1995, 1996
5b8ba871bSPeter Wemm  *	Keith Bostic.  All rights reserved.
6b8ba871bSPeter Wemm  *
7b8ba871bSPeter Wemm  * See the LICENSE file for redistribution information.
8b8ba871bSPeter Wemm  */
9b8ba871bSPeter Wemm 
10b8ba871bSPeter Wemm #include "config.h"
11b8ba871bSPeter Wemm 
12b8ba871bSPeter Wemm #ifndef lint
13f0957ccaSPeter Wemm static const char sccsid[] = "$Id: ex_visual.c,v 10.16 2001/08/29 11:04:13 skimo Exp $";
14b8ba871bSPeter Wemm #endif /* not lint */
15b8ba871bSPeter Wemm 
16b8ba871bSPeter Wemm #include <sys/types.h>
17b8ba871bSPeter Wemm #include <sys/queue.h>
18b8ba871bSPeter Wemm #include <sys/time.h>
19b8ba871bSPeter Wemm 
20b8ba871bSPeter Wemm #include <bitstring.h>
21b8ba871bSPeter Wemm #include <limits.h>
22b8ba871bSPeter Wemm #include <stdio.h>
23b8ba871bSPeter Wemm #include <stdlib.h>
24b8ba871bSPeter Wemm #include <string.h>
25b8ba871bSPeter Wemm #include <unistd.h>
26b8ba871bSPeter Wemm 
27b8ba871bSPeter Wemm #include "../common/common.h"
28b8ba871bSPeter Wemm #include "../vi/vi.h"
29b8ba871bSPeter Wemm 
30b8ba871bSPeter Wemm /*
31b8ba871bSPeter Wemm  * ex_visual -- :[line] vi[sual] [^-.+] [window_size] [flags]
32b8ba871bSPeter Wemm  *	Switch to visual mode.
33b8ba871bSPeter Wemm  *
34*c271fa92SBaptiste Daroussin  * PUBLIC: int ex_visual(SCR *, EXCMD *);
35b8ba871bSPeter Wemm  */
36b8ba871bSPeter Wemm int
37f0957ccaSPeter Wemm ex_visual(SCR *sp, EXCMD *cmdp)
38b8ba871bSPeter Wemm {
39b8ba871bSPeter Wemm 	SCR *tsp;
40b8ba871bSPeter Wemm 	size_t len;
41b8ba871bSPeter Wemm 	int pos;
42b8ba871bSPeter Wemm 	char buf[256];
43f0957ccaSPeter Wemm 	size_t wlen;
44f0957ccaSPeter Wemm 	CHAR_T *wp;
45b8ba871bSPeter Wemm 
46b8ba871bSPeter Wemm 	/* If open option off, disallow visual command. */
47b8ba871bSPeter Wemm 	if (!O_ISSET(sp, O_OPEN)) {
48b8ba871bSPeter Wemm 		msgq(sp, M_ERR,
49b8ba871bSPeter Wemm 	    "175|The visual command requires that the open option be set");
50b8ba871bSPeter Wemm 		return (1);
51b8ba871bSPeter Wemm 	}
52b8ba871bSPeter Wemm 
53b8ba871bSPeter Wemm 	/* Move to the address. */
54b8ba871bSPeter Wemm 	sp->lno = cmdp->addr1.lno == 0 ? 1 : cmdp->addr1.lno;
55b8ba871bSPeter Wemm 
56b8ba871bSPeter Wemm 	/*
57b8ba871bSPeter Wemm 	 * Push a command based on the line position flags.  If no
58b8ba871bSPeter Wemm 	 * flag specified, the line goes at the top of the screen.
59b8ba871bSPeter Wemm 	 */
60b8ba871bSPeter Wemm 	switch (FL_ISSET(cmdp->iflags,
61b8ba871bSPeter Wemm 	    E_C_CARAT | E_C_DASH | E_C_DOT | E_C_PLUS)) {
62b8ba871bSPeter Wemm 	case E_C_CARAT:
63b8ba871bSPeter Wemm 		pos = '^';
64b8ba871bSPeter Wemm 		break;
65b8ba871bSPeter Wemm 	case E_C_DASH:
66b8ba871bSPeter Wemm 		pos = '-';
67b8ba871bSPeter Wemm 		break;
68b8ba871bSPeter Wemm 	case E_C_DOT:
69b8ba871bSPeter Wemm 		pos = '.';
70b8ba871bSPeter Wemm 		break;
71b8ba871bSPeter Wemm 	case E_C_PLUS:
72b8ba871bSPeter Wemm 		pos = '+';
73b8ba871bSPeter Wemm 		break;
74b8ba871bSPeter Wemm 	default:
75b8ba871bSPeter Wemm 		sp->frp->lno = sp->lno;
76b8ba871bSPeter Wemm 		sp->frp->cno = 0;
77b8ba871bSPeter Wemm 		(void)nonblank(sp, sp->lno, &sp->cno);
78b8ba871bSPeter Wemm 		F_SET(sp->frp, FR_CURSORSET);
79b8ba871bSPeter Wemm 		goto nopush;
80b8ba871bSPeter Wemm 	}
81b8ba871bSPeter Wemm 
82b8ba871bSPeter Wemm 	if (FL_ISSET(cmdp->iflags, E_C_COUNT))
83b8ba871bSPeter Wemm 		len = snprintf(buf, sizeof(buf),
84c513aafeSBruce Evans 		     "%luz%c%lu", (u_long)sp->lno, pos, cmdp->count);
85b8ba871bSPeter Wemm 	else
86f0957ccaSPeter Wemm 		len = snprintf(buf, sizeof(buf), "%luz%c", (u_long)sp->lno, pos);
87f0957ccaSPeter Wemm 	CHAR2INT(sp, buf, len, wp, wlen);
88f0957ccaSPeter Wemm 	(void)v_event_push(sp, NULL, wp, wlen, CH_NOMAP | CH_QUOTED);
89b8ba871bSPeter Wemm 
90b8ba871bSPeter Wemm 	/*
91b8ba871bSPeter Wemm 	 * !!!
92b8ba871bSPeter Wemm 	 * Historically, if no line address was specified, the [p#l] flags
93b8ba871bSPeter Wemm 	 * caused the cursor to be moved to the last line of the file, which
94b8ba871bSPeter Wemm 	 * was then positioned as described above.  This seems useless, so
95b8ba871bSPeter Wemm 	 * I haven't implemented it.
96b8ba871bSPeter Wemm 	 */
97b8ba871bSPeter Wemm 	switch (FL_ISSET(cmdp->iflags, E_C_HASH | E_C_LIST | E_C_PRINT)) {
98b8ba871bSPeter Wemm 	case E_C_HASH:
99b8ba871bSPeter Wemm 		O_SET(sp, O_NUMBER);
100b8ba871bSPeter Wemm 		break;
101b8ba871bSPeter Wemm 	case E_C_LIST:
102b8ba871bSPeter Wemm 		O_SET(sp, O_LIST);
103b8ba871bSPeter Wemm 		break;
104b8ba871bSPeter Wemm 	case E_C_PRINT:
105b8ba871bSPeter Wemm 		break;
106b8ba871bSPeter Wemm 	}
107b8ba871bSPeter Wemm 
108b8ba871bSPeter Wemm nopush:	/*
109b8ba871bSPeter Wemm 	 * !!!
110b8ba871bSPeter Wemm 	 * You can call the visual part of the editor from within an ex
111b8ba871bSPeter Wemm 	 * global command.
112b8ba871bSPeter Wemm 	 *
113b8ba871bSPeter Wemm 	 * XXX
114b8ba871bSPeter Wemm 	 * Historically, undoing a visual session was a single undo command,
115b8ba871bSPeter Wemm 	 * i.e. you could undo all of the changes you made in visual mode.
116b8ba871bSPeter Wemm 	 * We don't get this right; I'm waiting for the new logging code to
117b8ba871bSPeter Wemm 	 * be available.
118b8ba871bSPeter Wemm 	 *
119b8ba871bSPeter Wemm 	 * It's explicit, don't have to wait for the user, unless there's
120b8ba871bSPeter Wemm 	 * already a reason to wait.
121b8ba871bSPeter Wemm 	 */
122b8ba871bSPeter Wemm 	if (!F_ISSET(sp, SC_SCR_EXWROTE))
123b8ba871bSPeter Wemm 		F_SET(sp, SC_EX_WAIT_NO);
124b8ba871bSPeter Wemm 
125b8ba871bSPeter Wemm 	if (F_ISSET(sp, SC_EX_GLOBAL)) {
126b8ba871bSPeter Wemm 		/*
127b8ba871bSPeter Wemm 		 * When the vi screen(s) exit, we don't want to lose our hold
128b8ba871bSPeter Wemm 		 * on this screen or this file, otherwise we're going to fail
129b8ba871bSPeter Wemm 		 * fairly spectacularly.
130b8ba871bSPeter Wemm 		 */
131b8ba871bSPeter Wemm 		++sp->refcnt;
132b8ba871bSPeter Wemm 		++sp->ep->refcnt;
133f0957ccaSPeter Wemm 		/* XXXX where is this decremented ? */
134b8ba871bSPeter Wemm 
135b8ba871bSPeter Wemm 		/*
136b8ba871bSPeter Wemm 		 * Fake up a screen pointer -- vi doesn't get to change our
137b8ba871bSPeter Wemm 		 * underlying file, regardless.
138b8ba871bSPeter Wemm 		 */
139b8ba871bSPeter Wemm 		tsp = sp;
140b8ba871bSPeter Wemm 		if (vi(&tsp))
141b8ba871bSPeter Wemm 			return (1);
142b8ba871bSPeter Wemm 
143b8ba871bSPeter Wemm 		/*
144b8ba871bSPeter Wemm 		 * !!!
145b8ba871bSPeter Wemm 		 * Historically, if the user exited the vi screen(s) using an
146b8ba871bSPeter Wemm 		 * ex quit command (e.g. :wq, :q) ex/vi exited, it was only if
147b8ba871bSPeter Wemm 		 * they exited vi using the Q command that ex continued.  Some
148b8ba871bSPeter Wemm 		 * early versions of nvi continued in ex regardless, but users
149b8ba871bSPeter Wemm 		 * didn't like the semantic.
150b8ba871bSPeter Wemm 		 *
151b8ba871bSPeter Wemm 		 * Reset the screen.
152b8ba871bSPeter Wemm 		 */
153b8ba871bSPeter Wemm 		if (ex_init(sp))
154b8ba871bSPeter Wemm 			return (1);
155b8ba871bSPeter Wemm 
156b8ba871bSPeter Wemm 		/* Move out of the vi screen. */
157b8ba871bSPeter Wemm 		(void)ex_puts(sp, "\n");
158b8ba871bSPeter Wemm 	} else {
159b8ba871bSPeter Wemm 		F_CLR(sp, SC_EX | SC_SCR_EX);
160b8ba871bSPeter Wemm 		F_SET(sp, SC_VI);
161b8ba871bSPeter Wemm 	}
162b8ba871bSPeter Wemm 	return (0);
163b8ba871bSPeter Wemm }
164