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