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