kirkenda@eecs.cs.pdx.edu (Steve Kirkendall) (12/04/90)
# --------------------------- cut here ---------------------------- # This is a shar archive. To unpack it, save it to a file, and delete # anything above the "cut here" line. Then run sh on the file. # # -rw-r--r-- 1 kirkenda 9893 Dec 2 17:57 modify.c # -rw-r--r-- 1 kirkenda 10230 Dec 2 17:57 move1.c # -rw-r--r-- 1 kirkenda 4445 Dec 2 17:57 move2.c # -rw-r--r-- 1 kirkenda 2697 Dec 2 17:57 move3.c # -rw-r--r-- 1 kirkenda 3699 Dec 2 17:57 move4.c # -rw-r--r-- 1 kirkenda 3940 Dec 2 17:57 move5.c # -rw-r--r-- 1 kirkenda 14362 Dec 2 17:57 opts.c # -rw-r--r-- 1 kirkenda 2306 Dec 2 17:57 recycle.c # -rw-r--r-- 1 kirkenda 19816 Dec 2 17:57 redraw.c # if test -f modify.c -a "$1" != -f then echo Will not overwrite modify.c else echo Extracting modify.c sed 's/^X//' >modify.c <<\eof X/* modify.c */ X X/* This file contains the low-level file modification functions: X * delete(frommark, tomark) - removes line or portions of lines X * add(frommark, text) - inserts new text X * change(frommark, tomark, text) - delete, then add X */ X X#include "config.h" X#include "vi.h" X X#ifdef DEBUG X# include <stdio.h> Xstatic FILE *dbg; X X/*VARARGS1*/ Xdebout(msg, arg1, arg2, arg3, arg4, arg5) X char *msg, *arg1, *arg2, *arg3, *arg4, *arg5; X{ X if (!dbg) X { X dbg = fopen("debug.out", "w"); X setbuf(dbg, (FILE *)0); X } X fprintf(dbg, msg, arg1, arg2, arg3, arg4, arg5); X} X#endif /* DEBUG */ X X/* delete a range of text from the file */ Xvoid delete(frommark, tomark) X MARK frommark; /* first char to be deleted */ X MARK tomark; /* AFTER last char to be deleted */ X{ X int i; /* used to move thru logical blocks */ X REG char *scan; /* used to scan thru text of the blk */ X REG char *cpy; /* used when copying chars */ X BLK *blk; /* a text block */ X long l; /* a line number */ X MARK m; /* a traveling version of frommark */ X X#ifdef DEBUG X debout("delete(%ld.%d, %ld.%d)\n", markline(frommark), markidx(frommark), markline(tomark), markidx(tomark)); X#endif X X /* if not deleting anything, quit now */ X if (frommark == tomark) X { X return; X } X X /* This is a change */ X changes++; X significant = TRUE; X X /* if this is a multi-line change, then we'll have to redraw */ X if (markline(frommark) != markline(tomark)) X { X mustredraw = TRUE; X redrawrange(markline(frommark), markline(tomark), markline(frommark)); X } X X /* adjust marks 'a through 'z and '' as needed */ X l = markline(tomark); X for (i = 0; i < NMARKS; i++) X { X if (mark[i] < frommark) X { X continue; X } X else if (mark[i] < tomark) X { X mark[i] = MARK_UNSET; X } X else if (markline(mark[i]) == l) X { X if (markline(frommark) == l) X { X mark[i] -= markidx(tomark) - markidx(frommark); X } X else X { X mark[i] -= markidx(tomark); X } X } X else X { X mark[i] -= MARK_AT_LINE(l - markline(frommark)); X } X } X X /* Reporting... */ X if (markidx(frommark) == 0 && markidx(tomark) == 0) X { X rptlines = markline(tomark) - markline(frommark); X rptlabel = "deleted"; X } X X /* find the block containing frommark */ X l = markline(frommark); X for (i = 1; lnum[i] < l; i++) X { X } X X /* process each affected block... */ X for (m = frommark; X m < tomark && lnum[i] < INFINITY; X m = MARK_AT_LINE(lnum[i - 1] + 1)) X { X /* fetch the block */ X blk = blkget(i); X X /* find the mark in the block */ X scan = blk->c; X for (l = markline(m) - lnum[i - 1] - 1; l > 0; l--) X { X while (*scan++ != '\n') X { X } X } X scan += markidx(m); X X /* figure out where the changes to this block end */ X if (markline(tomark) > lnum[i]) X { X cpy = blk->c + BLKSIZE; X } X else if (markline(tomark) == markline(m)) X { X cpy = scan - markidx(m) + markidx(tomark); X } X else X { X cpy = scan; X for (l = markline(tomark) - markline(m); X l > 0; X l--) X { X while (*cpy++ != '\n') X { X } X } X cpy += markidx(tomark); X } X X /* delete the stuff by moving chars within this block */ X while (cpy < blk->c + BLKSIZE) X { X *scan++ = *cpy++; X } X while (scan < blk->c + BLKSIZE) X { X *scan++ = '\0'; X } X X /* adjust tomark to allow for lines deleted from this block */ X tomark -= MARK_AT_LINE(lnum[i] + 1 - markline(m)); X X /* if this block isn't empty now, then advance i */ X if (*blk->c) X { X i++; X } X X /* the buffer has changed. Update hdr and lnum. */ X blkdirty(blk); X } X X /* must have at least 1 line */ X if (nlines == 0) X { X blk = blkadd(1); X blk->c[0] = '\n'; X blkdirty(blk); X cursor = MARK_FIRST; X } X} X X X/* add some text at a specific place in the file */ Xvoid add(atmark, newtext) X MARK atmark; /* where to insert the new text */ X char *newtext; /* NUL-terminated string to insert */ X{ X REG char *scan; /* used to move through string */ X REG char *build; /* used while copying chars */ X int addlines; /* number of lines we're adding */ X int lastpart; /* size of last partial line */ X BLK *blk; /* the block to be modified */ X int blkno; /* the logical block# of (*blk) */ X REG char *newptr; /* where new text starts in blk */ X BLK buf; /* holds chars from orig blk */ X BLK linebuf; /* holds part of line that didn't fit */ X BLK *following; /* the BLK following the last BLK */ X int i; X long l; X X#ifdef DEBUG X debout("add(%ld.%d, \"%s\")\n", markline(atmark), markidx(atmark), newtext); X#endif X#ifdef lint X buf.c[0] = 0; X#endif X /* if not adding anything, return now */ X if (!*newtext) X { X return; X } X X /* This is a change */ X changes++; X significant = TRUE; X X /* count the number of lines in the new text */ X for (scan = newtext, lastpart = addlines = 0; *scan; ) X { X if (*scan++ == '\n') X { X addlines++; X lastpart = 0; X } X else X { X lastpart++; X } X } X X /* Reporting... */ X if (lastpart == 0 && markidx(atmark) == 0) X { X rptlines = addlines; X rptlabel = "added"; X } X X /* extract the line# from atmark */ X l = markline(atmark); X X /* if more than 0 lines, then we'll have to redraw the screen */ X if (addlines > 0) X { X mustredraw = TRUE; X if (markidx(atmark) == 0 && lastpart == 0) X { X redrawrange(l, l, l + addlines); X } X else X { X /* make sure the last line gets redrawn -- it was X * split, so its appearance has changed X */ X redrawrange(l, l + 1L, l + addlines + 1L); X } X } X X /* adjust marks 'a through 'z and '' as needed */ X for (i = 0; i < NMARKS; i++) X { X if (mark[i] < atmark) X { X /* earlier line, or earlier in same line: no change */ X continue; X } X else if (markline(mark[i]) > l) X { X /* later line: move down a whole number of lines */ X mark[i] += MARK_AT_LINE(addlines); X } X else X { X /* later in same line */ X if (addlines > 0) X { X /* multi-line add, which split this line: X * move down, and possibly left or right, X * depending on where the split was and how X * much text was inserted after the last \n X */ X mark[i] += MARK_AT_LINE(addlines) + lastpart - markidx(atmark); X } X else X { X /* totally within this line: move right */ X mark[i] += lastpart; X } X } X } X X /* get the block to be modified */ X for (blkno = 1; lnum[blkno] < l && lnum[blkno + 1] < INFINITY; blkno++) X { X } X blk = blkget(blkno); X buf = *blk; X X /* figure out where the new text starts */ X for (newptr = buf.c, l = markline(atmark) - lnum[blkno - 1] - 1; X l > 0; X l--) X { X while (*newptr++ != '\n') X { X } X } X newptr += markidx(atmark); X X /* keep start of old block */ X build = blk->c + (newptr - buf.c); X X /* fill this block (or blocks) from the newtext string */ X while (*newtext) X { X while (*newtext && build < blk->c + BLKSIZE - 1) X { X *build++ = *newtext++; X } X if (*newtext) X { X /* save the excess */ X for (scan = linebuf.c + BLKSIZE; X build > blk->c && build[-1] != '\n'; X ) X { X *--scan = *--build; X } X X /* write the block */ X while (build < blk->c + BLKSIZE) X { X *build++ = '\0'; X } X blkdirty(blk); X X /* add another block */ X blkno++; X blk = blkadd(blkno); X X /* copy in the excess from last time */ X for (build = blk->c; scan < linebuf.c + BLKSIZE; ) X { X *build++ = *scan++; X } X } X } X X /* fill this block(s) from remainder of orig block */ X while (newptr < buf.c + BLKSIZE && *newptr) X { X while (newptr < buf.c + BLKSIZE X && *newptr X && build < blk->c + BLKSIZE - 1) X { X *build++ = *newptr++; X } X if (newptr < buf.c + BLKSIZE && *newptr) X { X /* save the excess */ X for (scan = linebuf.c + BLKSIZE; X build > blk->c && build[-1] != '\n'; X ) X { X *--scan = *--build; X } X X /* write the block */ X while (build < blk->c + BLKSIZE) X { X *build++ = '\0'; X } X blkdirty(blk); X X /* add another block */ X blkno++; X blk = blkadd(blkno); X X /* copy in the excess from last time */ X for (build = blk->c; scan < linebuf.c + BLKSIZE; ) X { X *build++ = *scan++; X } X } X } X X /* see if we can combine our last block with the following block */ X if (lnum[blkno] < nlines && lnum[blkno + 1] - lnum[blkno] < (BLKSIZE >> 6)) X { X /* hey, we probably can! Get the following block & see... */ X following = blkget(blkno + 1); X if (strlen(following->c) + (build - blk->c) < BLKSIZE - 1) X { X /* we can! Copy text from following to blk */ X for (scan = following->c; *scan; ) X { X *build++ = *scan++; X } X while (build < blk->c + BLKSIZE) X { X *build++ = '\0'; X } X blkdirty(blk); X X /* pretend the following was the last blk */ X blk = following; X build = blk->c; X } X } X X /* that last block is dirty by now */ X while (build < blk->c + BLKSIZE) X { X *build++ = '\0'; X } X blkdirty(blk); X} X X X/* change the text of a file */ Xvoid change(frommark, tomark, newtext) X MARK frommark, tomark; X char *newtext; X{ X int i; X long l; X char *text; X BLK *blk; X X#ifdef DEBUG X debout("change(%ld.%d, %ld.%d, \"%s\")\n", markline(frommark), markidx(frommark), markline(tomark), markidx(tomark), newtext); X#endif X X /* optimize for single-character replacement */ X if (frommark + 1 == tomark && newtext[0] && !newtext[1] && newtext[0] != '\n') X { X /* find the block containing frommark */ X l = markline(frommark); X for (i = 1; lnum[i] < l; i++) X { X } X X /* get the block */ X blk = blkget(i); X X /* find the line within the block */ X for (text = blk->c, i = l - lnum[i - 1] - 1; i > 0; text++) X { X if (*text == '\n') X { X i--; X } X } X X /* replace the char */ X text += markidx(frommark); X if (*text == newtext[0]) X { X /* no change was needed - same char */ X return; X } X else if (*text != '\n') X { X /* This is a change */ X changes++; X significant = TRUE; X ChangeText X { X *text = newtext[0]; X blkdirty(blk); X } X return; X } X /* else it is a complex change involving newline... */ X } X X /* couldn't optimize, so do delete & add */ X ChangeText X { X delete(frommark, tomark); X add(frommark, newtext); X rptlabel = "changed"; X } X} eof if test `wc -c <modify.c` -ne 9893 then echo modify.c damaged! fi fi if test -f move1.c -a "$1" != -f then echo Will not overwrite move1.c else echo Extracting move1.c sed 's/^X//' >move1.c <<\eof X/* move1.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains most movement functions */ X X#include "config.h" X#include <ctype.h> X#include "vi.h" X X#ifndef isascii X# define isascii(c) !((c) & ~0x7f) X#endif X XMARK m_updnto(m, cnt, cmd) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X DEFAULT(cmd == 'G' ? nlines : 1L); X X /* move up or down 'cnt' lines */ X switch (cmd) X { X case ('P'&0x1f): X case '-': X case 'k': X m -= MARK_AT_LINE(cnt); X break; X X case 'G': X if (cnt < 1L || cnt > nlines) X { X msg("Only %ld lines", nlines); X return MARK_UNSET; X } X m = MARK_AT_LINE(cnt); X break; X X default: X m += MARK_AT_LINE(cnt); X } X X /* if that left us screwed up, then fail */ X if (m < MARK_FIRST || markline(m) > nlines) X { X return MARK_UNSET; X } X X return m; X} X X/*ARGSUSED*/ XMARK m_right(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X int idx; /* index of the new cursor position */ X X DEFAULT(1); X X /* move to right, if that's OK */ X pfetch(markline(m)); X idx = markidx(m) + cnt; X if (idx < plen) X { X m += cnt; X } X else X { X return MARK_UNSET; X } X X return m; X} X X/*ARGSUSED*/ XMARK m_left(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X DEFAULT(1); X X /* move to the left, if that's OK */ X if (markidx(m) >= cnt) X { X m -= cnt; X } X else X { X return MARK_UNSET; X } X X return m; X} X X/*ARGSUSED*/ XMARK m_tocol(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X char *text; /* text of the line */ X int col; /* column number */ X int idx; /* index into the line */ X X DEFAULT(1); X X /* internally, columns are numbered 0..COLS-1, not 1..COLS */ X cnt--; X X /* if 0, that's easy */ X if (cnt == 0) X { X m &= ~(BLKSIZE - 1); X return m; X } X X /* find that column within the line */ X pfetch(markline(m)); X text = ptext; X for (col = idx = 0; col < cnt && *text; text++, idx++) X { X if (*text == '\t' && !*o_list) X { X col += *o_tabstop; X col -= col % *o_tabstop; X } X else if (UCHAR(*text) < ' ' || *text == '\177') X { X col += 2; X } X#ifndef NO_CHARATTR X else if (text[0] == '\\' && text[1] == 'f' && text[2] && *o_charattr) X { X text += 2; /* plus one more as part of for loop */ X } X#endif X else X { X col++; X } X } X if (!*text) X { X return MARK_UNSET; X } X else X { X m = (m & ~(BLKSIZE - 1)) + idx; X } X return m; X} X X/*ARGSUSED*/ XMARK m_front(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument (ignored) */ X{ X char *scan; X X /* move to the first non-whitespace character */ X pfetch(markline(m)); X scan = ptext; X m &= ~(BLKSIZE - 1); X while (*scan == ' ' || *scan == '\t') X { X scan++; X m++; X } X X return m; X} X X/*ARGSUSED*/ XMARK m_rear(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument (ignored) */ X{ X /* Try to move *EXTREMELY* far to the right. It is fervently hoped X * that other code will convert this to a more reasonable MARK before X * anything tries to actually use it. (See adjmove() in vi.c) X */ X return m | (BLKSIZE - 1); X} X X#ifndef NO_SENTENCE X/*ARGSUSED*/ XMARK m_fsentence(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X REG char *text; X REG long l; X X DEFAULT(1); X X /* get the current line */ X l = markline(m); X pfetch(l); X text = ptext + markidx(m); X X /* for each requested sentence... */ X while (cnt-- > 0) X { X /* search forward for one of [.?!] followed by spaces or EOL */ X do X { X /* wrap at end of line */ X if (!text[0]) X { X if (l >= nlines) X { X return MARK_UNSET; X } X l++; X pfetch(l); X text = ptext; X } X else X { X text++; X } X } while (text[0] != '.' && text[0] != '?' && text[0] != '!' X || text[1] && (text[1] != ' ' || text[2] && text[2] != ' ')); X } X X /* construct a mark for this location */ X m = buildmark(text); X X /* move forward to the first word of the next sentence */ X m = m_fword(m, 1L); X X return m; X} X X/*ARGSUSED*/ XMARK m_bsentence(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X REG char *text; /* used to scan thru text */ X REG long l; /* current line number */ X int flag; /* have we passed at least one word? */ X X DEFAULT(1); X X /* get the current line */ X l = markline(m); X pfetch(l); X text = ptext + markidx(m); X X /* for each requested sentence... */ X flag = TRUE; X while (cnt-- > 0) X { X /* search backward for one of [.?!] followed by spaces or EOL */ X do X { X /* wrap at beginning of line */ X if (text == ptext) X { X do X { X if (l <= 1) X { X return MARK_UNSET; X } X l--; X pfetch(l); X } while (!*ptext); X text = ptext + plen - 1; X } X else X { X text--; X } X X /* are we moving past a "word"? */ X if (text[0] >= '0') X { X flag = FALSE; X } X } while (flag || text[0] != '.' && text[0] != '?' && text[0] != '!' X || text[1] && (text[1] != ' ' || text[2] && text[2] != ' ')); X } X X /* construct a mark for this location */ X m = buildmark(text); X X /* move to the front of the following sentence */ X m = m_fword(m, 1L); X X return m; X} X#endif X X/*ARGSUSED*/ XMARK m_fparagraph(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X char *text; X char *pscn; /* used to scan thru value of "paragraphs" option */ X long l; X X DEFAULT(1); X X for (l = markline(m); cnt > 0 && l++ < nlines; ) X { X text = fetchline(l); X if (!*text) X { X cnt--; X } X#ifndef NO_SENTENCE X else if (*text == '.') X { X for (pscn = o_paragraphs; pscn[0] && pscn[1]; pscn += 2) X { X if (pscn[0] == text[1] && pscn[1] == text[2]) X { X cnt--; X break; X } X } X } X#endif X } X if (l <= nlines) X { X m = MARK_AT_LINE(l); X } X else X { X m = MARK_LAST; X } X return m; X} X X/*ARGSUSED*/ XMARK m_bparagraph(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X{ X char *text; X char *pscn; /* used to scan thru value of "paragraph" option */ X long l; X X DEFAULT(1); X X for (l = markline(m); cnt > 0 && l-- > 1; ) X { X text = fetchline(l); X if (!*text) X { X cnt--; X } X#ifndef NO_SENTENCE X else if (*text == '.') X { X for (pscn = o_paragraphs; pscn[0] && pscn[1]; pscn += 2) X { X if (pscn[0] == text[1] && pscn[1] == text[2]) X { X cnt--; X break; X } X } X } X#endif X } X if (l >= 1) X { X m = MARK_AT_LINE(l); X } X else X { X m = MARK_FIRST; X } X return m; X} X X/*ARGSUSED*/ XMARK m_fsection(m, cnt, key) X MARK m; /* movement is relative to this mark */ X long cnt; /* (ignored) */ X int key; /* second key stroke - must be ']' */ X{ X char *text; X char *sscn; /* used to scan thru value of "sections" option */ X long l; X X /* make sure second key was ']' */ X if (key != ']') X { X return MARK_UNSET; X } X X for (l = markline(m); l++ < nlines; ) X { X text = fetchline(l); X if (*text == '{') X { X break; X } X#ifndef NO_SENTENCE X else if (*text == '.') X { X for (sscn = o_sections; sscn[0] && sscn[1]; sscn += 2) X { X if (sscn[0] == text[1] && sscn[1] == text[2]) X { X goto BreakBreak; X } X } X } X#endif X } XBreakBreak: X if (l <= nlines) X { X m = MARK_AT_LINE(l); X } X else X { X m = MARK_LAST; X } X return m; X} X X/*ARGSUSED*/ XMARK m_bsection(m, cnt, key) X MARK m; /* movement is relative to this mark */ X long cnt; /* (ignored) */ X int key; /* second key stroke - must be '[' */ X{ X char *text; X char *sscn; /* used to scan thru value of "sections" option */ X long l; X X /* make sure second key was '[' */ X if (key != '[') X { X return MARK_UNSET; X } X X for (l = markline(m); l-- > 1; ) X { X text = fetchline(l); X if (*text == '{') X { X break; X } X#ifndef NO_SENTENCE X else if (*text == '.') X { X for (sscn = o_sections; sscn[0] && sscn[1]; sscn += 2) X { X if (sscn[0] == text[1] && sscn[1] == text[2]) X { X goto BreakBreak; X } X } X } X#endif X } XBreakBreak: X if (l >= 1) X { X m = MARK_AT_LINE(l); X } X else X { X m = MARK_FIRST; X } X return m; X} X X X/*ARGSUSED*/ XMARK m_match(m, cnt) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument (ignored) */ X{ X long l; X REG char *text; X REG char match; X REG char nest; X REG int count; X X /* get the current line */ X l = markline(m); X pfetch(l); X text = ptext + markidx(m); X X /* search forward within line for one of "[](){}" */ X for (match = '\0'; !match && *text; text++) X { X /* tricky way to recognize 'em in ASCII */ X nest = *text; X if ((nest & 0xdf) == ']' || (nest & 0xdf) == '[') X { X match = nest ^ ('[' ^ ']'); X } X else if ((nest & 0xfe) == '(') X { X match = nest ^ ('(' ^ ')'); X } X else X { X match = 0; X } X } X if (!match) X { X return MARK_UNSET; X } X text--; X X /* search forward or backward for match */ X if (match == '(' || match == '[' || match == '{') X { X /* search backward */ X for (count = 1; count > 0; ) X { X /* wrap at beginning of line */ X if (text == ptext) X { X do X { X if (l <= 1L) X { X return MARK_UNSET; X } X l--; X pfetch(l); X } while (!*ptext); X text = ptext + plen - 1; X } X else X { X text--; X } X X /* check the char */ X if (*text == match) X count--; X else if (*text == nest) X count++; X } X } X else X { X /* search forward */ X for (count = 1; count > 0; ) X { X /* wrap at end of line */ X if (!*text) X { X if (l >= nlines) X { X return MARK_UNSET; X } X l++; X pfetch(l); X text = ptext; X } X else X { X text++; X } X X /* check the char */ X if (*text == match) X count--; X else if (*text == nest) X count++; X } X } X X /* construct a mark for this place */ X m = buildmark(text); X return m; X} X X/*ARGSUSED*/ XMARK m_tomark(m, cnt, key) X MARK m; /* movement is relative to this mark */ X long cnt; /* (ignored) */ X int key; /* keystroke - the mark to move to */ X{ X /* mark '' is a special case */ X if (key == '\'' || key == '`') X { X if (mark[26] == MARK_UNSET) X { X return MARK_FIRST; X } X else X { X return mark[26]; X } X } X X /* if not a valid mark number, don't move */ X if (key < 'a' || key > 'z') X { X return MARK_UNSET; X } X X /* return the selected mark -- may be MARK_UNSET */ X if (!mark[key - 'a']) X { X msg("mark '%c is unset", key); X } X return mark[key - 'a']; X} X eof if test `wc -c <move1.c` -ne 10230 then echo move1.c damaged! fi fi if test -f move2.c -a "$1" != -f then echo Will not overwrite move2.c else echo Extracting move2.c sed 's/^X//' >move2.c <<\eof X/* move2.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This function contains the movement functions that perform RE searching */ X X#include "config.h" X#include "vi.h" X#include "regexp.h" X Xextern long atol(); X Xstatic regexp *re; /* compiled version of the pattern to search for */ Xstatic prevsf; /* boolean: previous search direction was forward? */ X XMARK m_nsrch(m) X MARK m; /* where to start searching */ X{ X if (prevsf) X { X m = m_fsrch(m, (char *)0); X prevsf = TRUE; X } X else X { X m = m_bsrch(m, (char *)0); X prevsf = FALSE; X } X return m; X} X XMARK m_Nsrch(m) X MARK m; /* where to start searching */ X{ X if (prevsf) X { X m = m_bsrch(m, (char *)0); X prevsf = TRUE; X } X else X { X m = m_fsrch(m, (char *)0); X prevsf = FALSE; X } X return m; X} X XMARK m_fsrch(m, ptrn) X MARK m; /* where to start searching */ X char *ptrn; /* pattern to search for */ X{ X long l; /* line# of line to be searched */ X char *line; /* text of line to be searched */ X int wrapped;/* boolean: has our search wrapped yet? */ X int pos; /* where we are in the line */ X long delta; /* line offset, for things like "/foo/+1" */ X X /* remember: "previous search was forward" */ X prevsf = TRUE; X X delta = 0L; X if (ptrn && *ptrn) X { X /* locate the closing '/', if any */ X line = parseptrn(ptrn); X if (*line) X { X delta = atol(line); X } X ptrn++; X X /* free the previous pattern */ X if (re) free(re); X X /* compile the pattern */ X re = regcomp(ptrn); X if (!re) X { X return MARK_UNSET; X } X } X else if (!re) X { X msg("No previous expression"); X return MARK_UNSET; X } X X /* search forward for the pattern */ X pos = markidx(m) + 1; X pfetch(markline(m)); X if (pos >= plen) X { X pos = 0; X m = (m | (BLKSIZE - 1)) + 1; X } X wrapped = FALSE; X for (l = markline(m); l != markline(m) + 1 || !wrapped; l++) X { X /* wrap search */ X if (l > nlines) X { X /* if we wrapped once already, then the search failed */ X if (wrapped) X { X break; X } X X /* else maybe we should wrap now? */ X if (*o_wrapscan) X { X l = 0; X wrapped = TRUE; X continue; X } X else X { X break; X } X } X X /* get this line */ X line = fetchline(l); X X /* check this line */ X if (regexec(re, &line[pos], (pos == 0))) X { X /* match! */ X if (wrapped && *o_warn) X msg("(wrapped)"); X if (delta != 0L) X { X l += delta; X if (l < 1 || l > nlines) X { X msg("search offset too big"); X return MARK_UNSET; X } X return m_front(MARK_AT_LINE(l), 0L); X } X return MARK_AT_LINE(l) + (int)(re->startp[0] - line); X } X pos = 0; X } X X /* not found */ X msg(*o_wrapscan ? "Not found" : "Hit bottom without finding RE"); X return MARK_UNSET; X} X XMARK m_bsrch(m, ptrn) X MARK m; /* where to start searching */ X char *ptrn; /* pattern to search for */ X{ X long l; /* line# of line to be searched */ X char *line; /* text of line to be searched */ X int wrapped;/* boolean: has our search wrapped yet? */ X int pos; /* last acceptable idx for a match on this line */ X int last; /* remembered idx of the last acceptable match on this line */ X int try; /* an idx at which we strat searching for another match */ X X /* remember: "previous search was not forward" */ X prevsf = FALSE; X X if (ptrn && *ptrn) X { X /* locate the closing '?', if any */ X line = parseptrn(ptrn); X ptrn++; X X /* free the previous pattern, if any */ X if (re) free(re); X X /* compile the pattern */ X re = regcomp(ptrn); X if (!re) X { X return MARK_UNSET; X } X } X else if (!re) X { X msg("No previous expression"); X return MARK_UNSET; X } X X /* search backward for the pattern */ X pos = markidx(m); X wrapped = FALSE; X for (l = markline(m); l != markline(m) - 1 || !wrapped; l--) X { X /* wrap search */ X if (l < 1) X { X if (*o_wrapscan) X { X l = nlines + 1; X wrapped = TRUE; X continue; X } X else X { X break; X } X } X X /* get this line */ X line = fetchline(l); X X /* check this line */ X if (regexec(re, line, 1) && (int)(re->startp[0] - line) < pos) X { X /* match! now find the last acceptable one in this line */ X do X { X last = (int)(re->startp[0] - line); X try = (int)(re->endp[0] - line); X } while (try > 0 X && regexec(re, &line[try], FALSE) X && (int)(re->startp[0] - line) < pos); X X if (wrapped && *o_warn) X msg("(wrapped)"); X return MARK_AT_LINE(l) + last; X } X pos = BLKSIZE; X } X X /* not found */ X msg(*o_wrapscan ? "Not found" : "Hit top without finding RE"); X return MARK_UNSET; X} X eof if test `wc -c <move2.c` -ne 4445 then echo move2.c damaged! fi fi if test -f move3.c -a "$1" != -f then echo Will not overwrite move3.c else echo Extracting move3.c sed 's/^X//' >move3.c <<\eof X/* move3.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains movement functions that perform character searches */ X X#include "config.h" X#include "vi.h" X X#ifndef NO_CHARSEARCH Xstatic MARK (*prevfwdfn)(); /* function to search in same direction */ Xstatic MARK (*prevrevfn)(); /* function to search in opposite direction */ Xstatic char prev_key; /* sought cvhar from previous [fFtT] */ X XMARK m__ch(m, cnt, cmd) X MARK m; /* current position */ X long cnt; X char cmd; /* command: either ',' or ';' */ X{ X MARK (*tmp)(); X X if (!prevfwdfn) X { X msg("No previous f, F, t, or T command"); X return MARK_UNSET; X } X X if (cmd == ',') X { X m = (*prevrevfn)(m, cnt, prev_key); X X /* Oops! we didn't want to change the prev*fn vars! */ X tmp = prevfwdfn; X prevfwdfn = prevrevfn; X prevrevfn = tmp; X X return m; X } X else X { X return (*prevfwdfn)(m, cnt, prev_key); X } X} X X/* move forward within this line to next occurrence of key */ XMARK m_fch(m, cnt, key) X MARK m; /* where to search from */ X long cnt; X char key; /* what to search for */ X{ X REG char *text; X X DEFAULT(1); X X prevfwdfn = m_fch; X prevrevfn = m_Fch; X prev_key = key; X X pfetch(markline(m)); X text = ptext + markidx(m); X while (cnt-- > 0) X { X do X { X m++; X text++; X } while (*text && *text != key); X } X if (!*text) X { X return MARK_UNSET; X } X return m; X} X X/* move backward within this line to previous occurrence of key */ XMARK m_Fch(m, cnt, key) X MARK m; /* where to search from */ X long cnt; X char key; /* what to search for */ X{ X REG char *text; X X DEFAULT(1); X X prevfwdfn = m_Fch; X prevrevfn = m_fch; X prev_key = key; X X pfetch(markline(m)); X text = ptext + markidx(m); X while (cnt-- > 0) X { X do X { X m--; X text--; X } while (text >= ptext && *text != key); X } X if (text < ptext) X { X return MARK_UNSET; X } X return m; X} X X/* move forward within this line almost to next occurrence of key */ XMARK m_tch(m, cnt, key) X MARK m; /* where to search from */ X long cnt; X char key; /* what to search for */ X{ X /* skip the adjacent char */ X pfetch(markline(m)); X if (plen <= markidx(m)) X { X return MARK_UNSET; X } X m++; X X m = m_fch(m, cnt, key); X if (m == MARK_UNSET) X { X return MARK_UNSET; X } X X prevfwdfn = m_tch; X prevrevfn = m_Tch; X X return m - 1; X} X X/* move backward within this line almost to previous occurrence of key */ XMARK m_Tch(m, cnt, key) X MARK m; /* where to search from */ X long cnt; X char key; /* what to search for */ X{ X /* skip the adjacent char */ X if (markidx(m) == 0) X { X return MARK_UNSET; X } X m--; X X m = m_Fch(m, cnt, key); X if (m == MARK_UNSET) X { X return MARK_UNSET; X } X X prevfwdfn = m_Tch; X prevrevfn = m_tch; X X return m + 1; X} X#endif eof if test `wc -c <move3.c` -ne 2697 then echo move3.c damaged! fi fi if test -f move4.c -a "$1" != -f then echo Will not overwrite move4.c else echo Extracting move4.c sed 's/^X//' >move4.c <<\eof X/* move4.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains movement functions which are screen-relative */ X X#include "config.h" X#include "vi.h" X X/* This moves the cursor to a particular row on the screen */ X/*ARGSUSED*/ XMARK m_row(m, cnt, key) X MARK m; /* the cursor position */ X long cnt; /* the row we'll move to */ X int key; /* the keystroke of this move - H/L/M */ X{ X DEFAULT(1); X X /* calculate destination line based on key */ X cnt--; X switch (key) X { X case 'H': X cnt = topline + cnt; X break; X X case 'M': X cnt = topline + (LINES - 1) / 2; X break; X X case 'L': X cnt = botline - cnt; X break; X } X X /* return the mark of the destination line */ X return MARK_AT_LINE(cnt); X} X X X/* This function repositions the current line to show on a given row */ X/*ARGSUSED*/ XMARK m_z(m, cnt, key) X MARK m; /* the cursor */ X long cnt; /* the line number we're repositioning */ X int key; /* key struck after the z */ X{ X long newtop; X X /* Which line are we talking about? */ X if (cnt < 0 || cnt > nlines) X { X return MARK_UNSET; X } X if (cnt) X { X m = MARK_AT_LINE(cnt); X newtop = cnt; X } X else X { X newtop = markline(m); X } X X /* allow a "window size" number to be entered, but ignore it */ X while (key >= '0' && key <= '9') X { X key = getkey(0); X } X X /* figure out which line will have to be at the top of the screen */ X switch (key) X { X case '\n': X#if OSK X case '\l': X#else X case '\r': X#endif X case '+': X break; X X case '.': X case 'z': X newtop -= LINES / 2; X break; X X case '-': X newtop -= LINES - 1; X break; X X default: X return MARK_UNSET; X } X X /* make the new topline take effect */ X if (newtop >= 1) X { X topline = newtop; X } X else X { X topline = 1L; X } X mustredraw = TRUE; X X /* The cursor doesn't move */ X return m; X} X X X/* This function scrolls the screen. It does this by calling redraw() with X * an off-screen line as the argument. It will move the cursor if necessary X * so that the cursor is on the new screen. X */ X/*ARGSUSED*/ XMARK m_scroll(m, cnt, key) X MARK m; /* the cursor position */ X long cnt; /* for some keys: the number of lines to scroll */ X int key; /* keystroke that causes this movement */ X{ X MARK tmp; /* a temporary mark, used as arg to redraw() */ X X /* adjust cnt, and maybe *o_scroll, depending of key */ X switch (key) X { X case ctrl('F'): X case ctrl('B'): X DEFAULT(1); X mustredraw = TRUE; X cnt = cnt * (LINES - 1) - 1; /* keeps one old line on screen */ X break; X X case ctrl('E'): X case ctrl('Y'): X DEFAULT(1); X break; X X case ctrl('U'): X case ctrl('D'): X if (cnt == 0) /* default */ X { X cnt = *o_scroll; X } X else X { X if (cnt > LINES - 1) X { X cnt = LINES - 1; X } X *o_scroll = cnt; X } X break; X } X X /* scroll up or down, depending on key */ X switch (key) X { X case ctrl('B'): X case ctrl('Y'): X case ctrl('U'): X cnt = topline - cnt; X if (cnt < 1L) X { X cnt = 1L; X m = MARK_FIRST; X } X tmp = MARK_AT_LINE(cnt) + markidx(m); X redraw(tmp, FALSE); X if (markline(m) > botline) X { X m = MARK_AT_LINE(botline); X } X break; X X case ctrl('F'): X case ctrl('E'): X case ctrl('D'): X cnt = botline + cnt; X if (cnt > nlines) X { X cnt = nlines; X m = MARK_LAST; X } X tmp = MARK_AT_LINE(cnt) + markidx(m); X redraw(tmp, FALSE); X if (markline(m) < topline) X { X m = MARK_AT_LINE(topline); X } X break; X } X X /* arrange for ctrl-B and ctrl-F to redraw the smart line */ X if (key == ctrl('B') || key == ctrl('F')) X { X changes++; X X /* also, erase the statusline. This happens naturally for X * the scrolling commands, but the paging commands need to X * explicitly clear the statusline. X */ X msg(""); X } X X return m; X} eof if test `wc -c <move4.c` -ne 3699 then echo move4.c damaged! fi fi if test -f move5.c -a "$1" != -f then echo Will not overwrite move5.c else echo Extracting move5.c sed 's/^X//' >move5.c <<\eof X/* move5.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains the word-oriented movement functions */ X X#include <ctype.h> X#include "config.h" X#include "vi.h" X X#ifndef isascii X# define isascii(c) !((c) & ~0x7f) X#endif X X XMARK m_fword(m, cnt, cmd) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X int cmd; /* either 'w' or 'W' */ X{ X REG long l; X REG char *text; X REG int i; X X DEFAULT(1); X X l = markline(m); X pfetch(l); X text = ptext + markidx(m); X while (cnt-- > 0) /* yes, ASSIGNMENT! */ X { X i = *text++; X X if (cmd == 'W') X { X /* include any non-whitespace */ X while (i && (!isascii(i) || !isspace(i))) X { X i = *text++; X } X } X else if (!isascii(i) || isalnum(i) || i == '_') X { X /* include an alphanumeric word */ X while (i && (!isascii(i) || isalnum(i) || i == '_')) X { X i = *text++; X } X } X else X { X /* include contiguous punctuation */ X while (i && isascii(i) && !isalnum(i) && !isspace(i)) X { X i = *text++; X } X } X X /* include trailing whitespace */ X while (!i || isascii(i) && isspace(i)) X { X /* did we hit the end of this line? */ X if (!i) X { X /* move to next line, if there is one */ X l++; X if (l > nlines) X { X return MARK_UNSET; X } X pfetch(l); X text = ptext; X } X X i = *text++; X } X text--; X } X X /* construct a MARK for this place */ X m = buildmark(text); X return m; X} X X XMARK m_bword(m, cnt, cmd) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X int cmd; /* either 'b' or 'B' */ X{ X REG long l; X REG char *text; X X DEFAULT(1); X X l = markline(m); X pfetch(l); X text = ptext + markidx(m); X while (cnt-- > 0) /* yes, ASSIGNMENT! */ X { X text--; X X /* include preceding whitespace */ X while (text < ptext || isascii(*text) && isspace(*text)) X { X /* did we hit the end of this line? */ X if (text < ptext) X { X /* move to preceding line, if there is one */ X l--; X if (l <= 0) X { X return MARK_UNSET; X } X pfetch(l); X text = ptext + plen - 1; X } X else X { X text--; X } X } X X if (cmd == 'B') X { X /* include any non-whitespace */ X while (text >= ptext && (!isascii(*text) || !isspace(*text))) X { X text--; X } X } X else if (!isascii(*text) || isalnum(*text) || *text == '_') X { X /* include an alphanumeric word */ X while (text >= ptext && (!isascii(*text) || isalnum(*text) || *text == '_')) X { X text--; X } X } X else X { X /* include contiguous punctuation */ X while (text >= ptext && isascii(*text) && !isalnum(*text) && !isspace(*text)) X { X text--; X } X } X text++; X } X X /* construct a MARK for this place */ X m = buildmark(text); X return m; X} X XMARK m_eword(m, cnt, cmd) X MARK m; /* movement is relative to this mark */ X long cnt; /* a numeric argument */ X int cmd; /* either 'e' or 'E' */ X{ X REG long l; X REG char *text; X REG int i; X X DEFAULT(1); X X l = markline(m); X pfetch(l); X text = ptext + markidx(m); X while (cnt-- > 0) /* yes, ASSIGNMENT! */ X { X text++; X i = *text++; X X /* include preceding whitespace */ X while (!i || isascii(i) && isspace(i)) X { X /* did we hit the end of this line? */ X if (!i) X { X /* move to next line, if there is one */ X l++; X if (l > nlines) X { X return MARK_UNSET; X } X pfetch(l); X text = ptext; X } X X i = *text++; X } X X if (cmd == 'E') X { X /* include any non-whitespace */ X while (i && (!isascii(i) || !isspace(i))) X { X i = *text++; X } X } X else if (!isascii(i) || isalnum(i) || i == '_') X { X /* include an alphanumeric word */ X while (i && (!isascii(i) || isalnum(i) || i == '_')) X { X i = *text++; X } X } X else X { X /* include contiguous punctuation */ X while (i && isascii(i) && !isalnum(i) && !isspace(i)) X { X i = *text++; X } X } X text -= 2; X } X X /* construct a MARK for this place */ X m = buildmark(text); X return m; X} eof if test `wc -c <move5.c` -ne 3940 then echo move5.c damaged! fi fi if test -f opts.c -a "$1" != -f then echo Will not overwrite opts.c else echo Extracting opts.c sed 's/^X//' >opts.c <<\eof X/* opts.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains the code that manages the run-time options -- The X * values that can be modified via the "set" command. X */ X X#include "config.h" X#include "vi.h" X#ifndef NULL X#define NULL (char *)0 X#endif Xextern char *getenv(); X X/* maximum width to permit for strings, including ="" */ X#define MAXWIDTH 20 X X/* These are the default values of all options */ Xchar o_autoindent[1] = {FALSE}; Xchar o_autoprint[1] = {TRUE}; Xchar o_autowrite[1] = {FALSE}; X#ifndef NO_ERRLIST Xchar o_cc[30] = {CC_COMMAND}; X#endif X#ifndef NO_CHARATTR Xchar o_charattr[1] = {FALSE}; X#endif Xchar o_columns[3] = {80, 32, 255}; X#ifndef NO_DIGRAPH Xchar o_digraph[1] = {FALSE}; X#endif Xchar o_directory[30] = TMPDIR; Xchar o_edcompatible[1] = {FALSE}; Xchar o_errorbells[1] = {TRUE}; Xchar o_exrefresh[1] = {TRUE}; X#ifndef NO_DIGRAPH Xchar o_flipcase[80] X# if CS_IBMPC X = {"\207\200\201\232\202\220\204\216\206\217\221\222\224\231\244\245"} X# endif X# if CS_LATIN1 X /* initialized by initopts() */ X# endif X ; X#endif X#ifndef NO_SENTENCE Xchar o_hideformat[1] = {FALSE}; X#endif Xchar o_ignorecase[1] = {FALSE}; X#ifndef NO_EXTENSIONS Xchar o_inputmode[1] = {FALSE}; X#endif Xchar o_keytime[3] = {2, 0, 5}; Xchar o_keywordprg[80] = {KEYWORDPRG}; Xchar o_lines[3] = {25, 2, 50}; /* More lines? Enlarge kbuf */ Xchar o_list[1] = {FALSE}; X#ifndef NO_MAGIC Xchar o_magic[1] = {TRUE}; X#endif X#ifndef NO_ERRLIST Xchar o_make[30] = {MAKE_COMMAND}; X#endif X#ifndef NO_MODELINE Xchar o_modeline[1] = {FALSE}; X#endif X#ifndef NO_SENTENCE Xchar o_paragraphs[30] = "PPppIPLPQP"; X#endif X#if MSDOS Xchar o_pcbios[1] = {TRUE}; X#endif Xchar o_readonly[1] = {FALSE}; Xchar o_report[3] = {5, 1, 127}; Xchar o_scroll[3] = {12, 1, 127}; X#ifndef NO_SENTENCE Xchar o_sections[30] = "NHSHSSSEse"; X#endif Xchar o_shell[60] = SHELL; Xchar o_shiftwidth[3] = {8, 1, 255}; X#ifndef NO_SHOWMATCH Xchar o_showmatch[1] = {FALSE}; X#endif X#ifndef NO_SHOWMODE Xchar o_smd[1] = {FALSE}; X#endif Xchar o_sidescroll[3] = {8, 1, 40}; Xchar o_sync[1] = {NEEDSYNC}; Xchar o_tabstop[3] = {8, 1, 40}; Xchar o_term[30] = "?"; Xchar o_vbell[1] = {TRUE}; Xchar o_warn[1] = {TRUE}; Xchar o_wrapmargin[3] = {0, 0, 255}; Xchar o_wrapscan[1] = {TRUE}; X X X/* The following describes the names & types of all options */ X#define BOOL 0 X#define NUM 1 X#define STR 2 X#define SET 0x01 /* this option has had its value altered */ X#define CANSET 0x02 /* this option can be set at any time */ X#define RCSET 0x06 /* this option can be set in a .exrc file only */ X#define MR 0x40 /* does this option affect the way text is displayed? */ Xstruct X{ X char *name; /* name of an option */ X char *nm; /* short name of an option */ X char type; /* type of an option */ X char flags; /* boolean: has this option been set? */ X char *value; /* value */ X} X opts[] = X{ X /* name type flags redraw value */ X { "autoindent", "ai", BOOL, CANSET , o_autoindent }, X { "autoprint", "ap", BOOL, CANSET , o_autoprint }, X { "autowrite", "aw", BOOL, CANSET , o_autowrite }, X#ifndef NO_ERRLIST X { "cc", "cc", STR, CANSET , o_cc }, X#endif X#ifndef NO_CHARATTR X { "charattr", "ca", BOOL, CANSET | MR, o_charattr }, X#endif X { "columns", "co", NUM, SET , o_columns }, X#ifndef NO_DIGRAPH X { "digraph", "dig", BOOL, CANSET , o_digraph }, X#endif X { "directory", "dir", STR, RCSET , o_directory }, X { "edcompatible","ed", BOOL, CANSET , o_edcompatible }, X { "errorbells", "eb", BOOL, CANSET , o_errorbells }, X { "exrefresh", "er", BOOL, CANSET , o_exrefresh }, X#ifndef NO_DIGRAPH X { "flipcase", "fc", STR, CANSET , o_flipcase }, X#endif X#ifndef NO_SENTENCE X { "hideformat", "hf", BOOL, CANSET | MR, o_hideformat }, X#endif X { "ignorecase", "ic", BOOL, CANSET , o_ignorecase }, X#ifndef NO_EXTENSIONS X { "inputmode", "im", BOOL, CANSET , o_inputmode }, X#endif X { "keytime", "kt", NUM, CANSET , o_keytime }, X { "keywordprg", "kp", STR, CANSET , o_keywordprg }, X { "lines", "ls", NUM, SET , o_lines }, X { "list", "li", BOOL, CANSET | MR, o_list }, X#ifndef NO_MAGIC X { "magic", "ma", BOOL, CANSET , o_magic }, X#endif X#ifndef NO_ERRLIST X { "make", "mk", STR, CANSET , o_make }, X#endif X#ifndef NO_MODELINE X { "modeline", "ml", BOOL, CANSET , o_modeline }, X#endif X#ifndef NO_SENTENCE X { "paragraphs", "pa", STR, CANSET , o_paragraphs }, X#endif X#if MSDOS X { "pcbios", "pc", BOOL, SET , o_pcbios }, X#endif X { "readonly", "ro", BOOL, CANSET , o_readonly }, X { "report", "re", NUM, CANSET , o_report }, X { "scroll", "sc", NUM, CANSET , o_scroll }, X#ifndef NO_SENTENCE X { "sections", "se", STR, CANSET , o_sections }, X#endif X { "shell", "sh", STR, CANSET , o_shell }, X#ifndef NO_SHOWMATCH X { "showmatch", "sm", BOOL, CANSET , o_showmatch }, X#endif X#ifndef NO_SHOWMODE X { "showmode", "smd", BOOL, CANSET , o_smd }, X#endif X { "shiftwidth", "sw", NUM, CANSET , o_shiftwidth }, X { "sidescroll", "ss", NUM, CANSET , o_sidescroll }, X { "sync", "sy", BOOL, CANSET , o_sync }, X { "tabstop", "ts", NUM, CANSET | MR, o_tabstop }, X { "term", "te", STR, SET , o_term }, X { "vbell", "vb", BOOL, CANSET , o_vbell }, X { "warn", "wa", BOOL, CANSET , o_warn }, X { "wrapmargin", "wm", NUM, CANSET , o_wrapmargin }, X { "wrapscan", "ws", BOOL, CANSET , o_wrapscan }, X { NULL, NULL, 0, CANSET, NULL } X}; X X X/* This function initializes certain options from environment variables, etc. */ Xvoid initopts() X{ X char *val; X int i; X X /* set some stuff from environment variables */ X#if MSDOS X if (val = getenv("COMSPEC")) /* yes, ASSIGNMENT! */ X#else X if (val = getenv("SHELL")) /* yes, ASSIGNMENT! */ X#endif X { X strcpy(o_shell, val); X } X X#if ANY_UNIX X if (val = getenv("TERM")) /* yes, ASSIGNMENT! */ X { X strcpy(o_term, val); X } X#endif X#if TOS X val = "vt52"; X strcpy(o_term, val); X#endif X#if MSDOS X if ((val = getenv("TERM")) /* yes, ASSIGNMENT! */ X && strcmp(val, "pcbios")) X { X strcpy(o_term, val); X o_pcbios[0] = 0; X } X else X { X strcpy(o_term, "pcbios"); X o_pcbios[0] = 1; X } X#endif X#if MSDOS || TOS X if ((val = getenv("TMP")) /* yes, ASSIGNMENT! */ X || (val = getenv("TEMP"))) X strcpy(o_directory, val); X#endif X X *o_scroll = LINES / 2 - 1; X X /* disable the vbell option if we don't know how to do a vbell */ X if (!has_VB) X { X for (i = 0; opts[i].value != o_vbell; i++) X { X } X opts[i].flags &= ~CANSET; X *o_vbell = FALSE; X } X X#ifndef NO_DIGRAPH X# ifdef CS_LATIN1 X for (i = 0, val = o_flipcase; i < 32; i++) X { X /* leave out the multiply/divide symbols */ X if (i == 23) X continue; X X /* add upper/lowercase pair */ X *val++ = i + 0xc0; X *val++ = i + 0xe0; X } X *val = '\0'; X# endif /* CS_LATIN1 */ X#endif /* not NO_DIGRAPH */ X} X X/* This function lists the current values of all options */ Xvoid dumpopts(all) X int all; /* boolean: dump all options, or just set ones? */ X{ X#ifndef NO_OPTCOLS X int i, j, k; X char nbuf[4]; /* used for converting numbers to ASCII */ X int widths[5]; /* width of each column, including gap */ X int ncols; /* number of columns */ X int nrows; /* number of options per column */ X int nset; /* number of options to be output */ X int width; /* width of a particular option */ X int todump[50]; /* indicies of options to be dumped */ X X /* step 1: count the number of set options */ X for (nset = i = 0; opts[i].name; i++) X { X if (all || (opts[i].flags & SET)) X { X todump[nset++] = i; X } X } X X /* step two: try to use as many columns as possible */ X for (ncols = (nset > 5 ? 5 : nset); ncols > 1; ncols--) X { X /* how many would go in this column? */ X nrows = (nset + ncols - 1) / ncols; X X /* figure out the width of each column */ X for (i = 0; i < ncols; i++) X { X widths[i] = 0; X for (j = 0, k = nrows * i; j < nrows && k < nset; j++, k++) X { X /* figure out the width of a particular option */ X switch (opts[todump[k]].type) X { X case BOOL: X if (!*opts[todump[k]].value) X width = 2; X else X width = 0; X break; X X case STR: X width = 3 + strlen(opts[todump[k]].value); X if (width > MAXWIDTH) X width = MAXWIDTH; X break; X X case NUM: X width = 4; X break; X } X width += strlen(opts[todump[k]].name); X X /* if this is the widest so far, widen col */ X if (width > widths[i]) X { X widths[i] = width; X } X } X X } X X /* if the total width is narrow enough, then use it */ X for (width = -2, i = 0; i < ncols; i++) X { X width += widths[i] + 2; X } X if (width < COLS - 1) X { X break; X } X } X X /* step 3: output the columns */ X nrows = (nset + ncols - 1) / ncols; X for (i = 0; i < nrows; i++) X { X for (j = 0; j < ncols; j++) X { X /* if we hit the end of the options, quit */ X k = i + j * nrows; X if (k >= nset) X { X break; X } X X /* output this option's value */ X width = 0; X switch (opts[todump[k]].type) X { X case BOOL: X if (!*opts[todump[k]].value) X { X qaddch('n'); X qaddch('o'); X width = 2; X } X qaddstr(opts[todump[k]].name); X width += strlen(opts[todump[k]].name); X break; X X case NUM: X sprintf(nbuf, "%-3d", UCHAR(*opts[todump[k]].value)); X qaddstr(opts[todump[k]].name); X qaddch('='); X qaddstr(nbuf); X width = 4 + strlen(opts[todump[k]].name); X break; X X case STR: X qaddstr(opts[todump[k]].name); X qaddch('='); X qaddch('"'); X strcpy(tmpblk.c, opts[todump[k]].value); X width = 3 + strlen(tmpblk.c); X if (width > MAXWIDTH) X { X width = MAXWIDTH; X strcpy(tmpblk.c + MAXWIDTH - 6, "..."); X } X qaddstr(tmpblk.c); X qaddch('"'); X width += strlen(opts[todump[k]].name); X break; X } X X /* pad the field to the correct size */ X if (k + nrows <= nset) X { X while (width < widths[j] + 2) X { X qaddch(' '); X width++; X } X } X } X addch('\n'); X exrefresh(); X } X#else X int i; X int col; X char nbuf[4]; X X for (i = col = 0; opts[i].name; i++) X { X /* if not set and not all, ignore this option */ X if (!all && !(opts[i].flags & SET)) X { X continue; X } X X /* align this option in one of the columns */ X if (col > 52) X { X addch('\n'); X col = 0; X } X else if (col > 26) X { X while (col < 52) X { X qaddch(' '); X col++; X } X } X else if (col > 0) X { X while (col < 26) X { X qaddch(' '); X col++; X } X } X X switch (opts[i].type) X { X case BOOL: X if (!*opts[i].value) X { X qaddch('n'); X qaddch('o'); X col += 2; X } X qaddstr(opts[i].name); X col += strlen(opts[i].name); X break; X X case NUM: X sprintf(nbuf, "%-3d", UCHAR(*opts[i].value)); X qaddstr(opts[i].name); X qaddch('='); X qaddstr(nbuf); X col += 4 + strlen(opts[i].name); X break; X X case STR: X qaddstr(opts[i].name); X qaddch('='); X qaddch('"'); X qaddstr(opts[i].value); X qaddch('"'); X col += 3 + strlen(opts[i].name) + strlen(opts[i].value); X break; X } X exrefresh(); X } X if (col > 0) X { X addch('\n'); X exrefresh(); X } X#endif X} X X#ifndef NO_MKEXRC X/* This function saves the current configuarion of options to a file */ Xvoid saveopts(fd) X int fd; /* file descriptor to write to */ X{ X int i; X char buf[256], *pos; X X /* write each set options */ X for (i = 0; opts[i].name; i++) X { X /* if unset or unsettable, ignore this option */ X if (!(opts[i].flags & SET) || !(opts[i].flags & CANSET)) X { X continue; X } X X strcpy(buf, "set "); X pos = &buf[4]; X switch (opts[i].type) X { X case BOOL: X if (!*opts[i].value) X { X *pos++='n'; X *pos++='o'; X } X strcpy(pos, opts[i].name); X strcat(pos, "\n"); X break; X X case NUM: X sprintf(pos, "%s=%-3d\n", opts[i].name, *opts[i].value & 0xff); X break; X X case STR: X sprintf(pos, "%s=\"%s\"\n", opts[i].name, opts[i].value); X break; X } X twrite(fd, buf, strlen(buf)); X } X} X#endif X X X/* This function changes the values of one or more options. */ Xvoid setopts(assignments) X char *assignments; /* a string containing option assignments */ X{ X char *name; /* name of variable in assignments */ X char *value; /* value of the variable */ X char *scan; /* used for moving through strings */ X int i, j; X X /* for each assignment... */ X for (name = assignments; *name; ) X { X /* skip whitespace */ X if (*name == ' ' || *name == '\t') X { X name++; X continue; X } X X /* find the value, if any */ X for (scan = name; *scan >= 'a' && *scan <= 'z'; scan++) X { X } X if (*scan == '=') X { X *scan++ = '\0'; X if (*scan == '"') X { X value = ++scan; X while (*scan && *scan != '"') X { X scan++; X } X if (*scan) X { X *scan++ = '\0'; X } X } X else X { X value = scan; X while (*scan && *scan != ' ' && *scan != '\t') X { X scan++; X } X if (*scan) X { X *scan++ = '\0'; X } X } X } X else X { X if (*scan) X { X *scan++ = '\0'; X } X value = NULL; X if (name[0] == 'n' && name[1] == 'o') X { X name += 2; X } X } X X /* find the variable */ X for (i = 0; X opts[i].name && strcmp(opts[i].name, name) && strcmp(opts[i].nm, name); X i++) X { X } X X /* change the variable */ X if (!opts[i].name) X { X msg("invalid option name \"%s\"", name); X } X else if ((opts[i].flags & CANSET) != CANSET) X { X msg("option \"%s\" can't be altered", name); X } X else if ((opts[i].flags & RCSET) != CANSET && nlines >= 1L) X { X msg("option \"%s\" can only be set in a %s file", name, EXRC); X } X else if (value) X { X switch (opts[i].type) X { X case BOOL: X msg("option \"[no]%s\" is boolean", name); X break; X X case NUM: X j = atoi(value); X if (j == 0 && *value != '0') X { X msg("option \"%s\" must have a numeric value", name); X } X else if (j < opts[i].value[1] || j > (opts[i].value[2] & 0xff)) X { X msg("option \"%s\" must have a value between %d and %d", X name, opts[i].value[1], opts[i].value[2] & 0xff); X } X else X { X *opts[i].value = atoi(value); X opts[i].flags |= SET; X } X break; X X case STR: X strcpy(opts[i].value, value); X opts[i].flags |= SET; X break; X } X if (opts[i].flags & MR) X { X mustredraw = TRUE; X } X } X else /* valid option, no value */ X { X if (opts[i].type == BOOL) X { X *opts[i].value = (name[-1] != 'o'); X opts[i].flags |= SET; X if (opts[i].flags & MR) X { X mustredraw = TRUE; X } X } X else X { X msg("option \"%s\" must be given a value", name); X } X } X X /* move on to the next option */ X name = scan; X } X X /* special processing ... */ X X /* if "readonly" then set the READONLY flag for this file */ X if (*o_readonly) X { X setflag(file, READONLY); X } X} eof if test `wc -c <opts.c` -ne 14362 then echo opts.c damaged! fi fi if test -f recycle.c -a "$1" != -f then echo Will not overwrite recycle.c else echo Extracting recycle.c sed 's/^X//' >recycle.c <<\eof X/* recycle.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains the functions perform garbage collection and allocate X * reusable blocks. X */ X X#include "config.h" X#include "vi.h" X X#ifndef NO_RECYCLE X/* this whole file would have be skipped if NO_RECYCLE is defined */ X Xextern long lseek(); X X#define BTST(bitno, byte) ((byte) & (1 << (bitno))) X#define BSET(bitno, byte) ((byte) |= (1 << (bitno))) X#define BCLR(bitno, byte) ((byte) &= ~(1 << (bitno))) X X#define TST(blkno) ((blkno) < MAXBIT ? BTST((blkno) & 7, bitmap[(blkno) >> 3]) : 1) X#define SET(blkno) if ((blkno) < MAXBIT) BSET((blkno) & 7, bitmap[(blkno) >> 3]) X#define CLR(blkno) if ((blkno) < MAXBIT) BCLR((blkno) & 7, bitmap[(blkno) >> 3]) X X/* bitmap of free blocks in first 4096k of tmp file */ Xstatic unsigned char bitmap[512]; X#define MAXBIT (sizeof bitmap << 3) X X/* this function locates all free blocks in the current tmp file */ Xvoid garbage() X{ X int i; X BLK oldhdr; X X /* start by assuming every block is free */ X for (i = 0; i < sizeof bitmap; i++) X { X bitmap[i] = 255; X } X X /* header block isn't free */ X#ifndef lint X CLR(0); X#endif X X /* blocks needed for current hdr aren't free */ X for (i = 1; i < MAXBLKS; i++) X { X CLR(hdr.n[i]); X } X X /* blocks needed for undo version aren't free */ X lseek(tmpfd, 0L, 0); X if (read(tmpfd, &oldhdr, (unsigned)sizeof oldhdr) != sizeof oldhdr) X { X msg("garbage() failed to read oldhdr??"); X for (i = 0; i < sizeof bitmap; i++) X { X bitmap[i] = 0; X } X return; X } X for (i = 1; i < MAXBLKS; i++) X { X CLR(oldhdr.n[i]); X } X X /* blocks needed for cut buffers aren't free */ X for (i = cutneeds(&oldhdr) - 1; i >= 0; i--) X { X CLR(oldhdr.n[i]); X } X} X X/* This function allocates the first available block in the tmp file */ Xlong allocate() X{ X int i; X long offset; X X /* search for the first byte with a free bit set */ X for (i = 0; i < sizeof bitmap && bitmap[i] == 0; i++) X { X } X X /* if we hit the end of the bitmap, return the end of the file */ X if (i == sizeof bitmap) X { X offset = lseek(tmpfd, 0L, 2); X } X else /* compute the offset for the free block */ X { X for (i <<= 3; TST(i) == 0; i++) X { X } X offset = (long)i * (long)BLKSIZE; X X /* mark the block as "allocated" */ X CLR(i); X } X X return offset; X} X X#endif eof if test `wc -c <recycle.c` -ne 2306 then echo recycle.c damaged! fi fi if test -f redraw.c -a "$1" != -f then echo Will not overwrite redraw.c else echo Extracting redraw.c sed 's/^X//' >redraw.c <<\eof X/* redraw.c */ X X/* Author: X * Steve Kirkendall X * 14407 SW Teal Blvd. #C X * Beaverton, OR 97005 X * kirkenda@cs.pdx.edu X */ X X X/* This file contains functions that draw text on the screen. The major entry X * points are: X * redrawrange() - called from modify.c to give hints about what parts X * of the screen need to be redrawn. X * redraw() - redraws the screen (or part of it) and positions X * the cursor where it belongs. X * idx2col() - converts a markidx() value to a logical column number. X */ X X#include "config.h" X#include "vi.h" X X/* This variable contains the line number that smartdrawtext() knows best */ Xstatic long smartlno; X X/* This function remebers where changes were made, so that the screen can be X * redraw in a more efficient manner. X */ Xstatic long redrawafter; /* line# of first line that must be redrawn */ Xstatic long preredraw; /* line# of last line changed, before change */ Xstatic long postredraw; /* line# of last line changed, after change */ Xvoid redrawrange(after, pre, post) X long after; /* lower bound of redrawafter */ X long pre; /* upper bound of preredraw */ X long post; /* upper bound of postredraw */ X{ X if (after == redrawafter) X { X /* multiple insertions/deletions at the same place -- combine X * them X */ X preredraw -= (post - pre); X if (postredraw < post) X { X preredraw += (post - postredraw); X postredraw = post; X } X if (redrawafter > preredraw) X { X redrawafter = preredraw; X } X if (redrawafter < 1L) X { X redrawafter = 0L; X preredraw = postredraw = INFINITY; X } X } X else if (postredraw > 0L) X { X /* multiple changes in different places -- redraw everything X * after "after". X */ X postredraw = preredraw = INFINITY; X if (after < redrawafter) X redrawafter = after; X } X else X { X /* first change */ X redrawafter = after; X preredraw = pre; X postredraw = post; X } X} X X X#ifndef NO_CHARATTR X/* see if a given line uses character attribute strings */ Xstatic int hasattr(lno, text) X long lno; /* the line# of the cursor */ X REG char *text; /* the text of the line, from fetchline */ X{ X static long plno; /* previous line number */ X static long chgs; /* previous value of changes counter */ X static int panswer;/* previous answer */ X char *scan; X X /* if charattr is off, then the answer is "no, it doesn't" */ X if (!*o_charattr) X { X chgs = 0; /* <- forces us to check if charattr is later set */ X return FALSE; X } X X /* if we already know the answer, return it... */ X if (lno == plno && chgs == changes) X { X return panswer; X } X X /* get the line & look for "\fX" */ X if (!text[0] || !text[1] || !text[2]) X { X panswer = FALSE; X } X else X { X for (scan = text; scan[2] && !(scan[0] == '\\' && scan[1] == 'f'); scan++) X { X } X panswer = (scan[2] != '\0'); X } X X /* save the results */ X plno = lno; X chgs = changes; X X /* return the results */ X return panswer; X} X#endif X X X X/* This function converts a MARK to a column number. It doesn't automatically X * adjust for leftcol; that must be done by the calling function X */ Xint idx2col(curs, text, inputting) X MARK curs; /* the line# & index# of the cursor */ X REG char *text; /* the text of the line, from fetchline */ X int inputting; /* boolean: called from input() ? */ X{ X static MARK pcursor;/* previous cursor, for possible shortcut */ X static MARK pcol; /* column number for pcol */ X static long chgs; /* previous value of changes counter */ X REG int col; /* used to count column numbers */ X REG int idx; /* used to count down the index */ X REG int i; X X /* for now, assume we have to start counting at the left edge */ X col = 0; X idx = markidx(curs); X X /* if the file hasn't changed & line number is the same & it has no X * embedded character attribute strings, can we do shortcuts? X */ X if (chgs == changes X && !((curs ^ pcursor) & ~(BLKSIZE - 1)) X#ifndef NO_CHARATTR X && !hasattr(markline(curs), text) X#endif X ) X { X /* no movement? */ X if (curs == pcursor) X { X /* return the column of the char; for tabs, return its last column */ X if (text[idx] == '\t' && !inputting && !*o_list) X { X return pcol + *o_tabstop - (pcol % *o_tabstop) - 1; X } X else X { X return pcol; X } X } X X /* movement to right? */ X if (curs > pcursor) X { X /* start counting from previous place */ X col = pcol; X idx = markidx(curs) - markidx(pcursor); X text += markidx(pcursor); X } X } X X /* count over to the char after the idx position */ X while (idx > 0 && (i = *text)) /* yes, ASSIGNMENT! */ X { X if (i == '\t' && !*o_list) X { X col += *o_tabstop; X col -= col % *o_tabstop; X } X else if (i >= '\0' && i < ' ' || i == '\177') X { X col += 2; X } X#ifndef NO_CHARATTR X else if (i == '\\' && text[1] == 'f' && text[2] && *o_charattr) X { X text += 2; /* plus one more at bottom of loop */ X idx -= 2; X } X#endif X else X { X col++; X } X text++; X idx--; X } X X /* save stuff to speed next call */ X pcursor = curs; X pcol = col; X chgs = changes; X X /* return the column of the char; for tabs, return its last column */ X if (*text == '\t' && !inputting && !*o_list) X { X return col + *o_tabstop - (col % *o_tabstop) - 1; X } X else X { X return col; X } X} X X X/* This function is similar to idx2col except that it takes care of sideways X * scrolling - for the given line, at least. X */ Xint mark2phys(m, text, inputting) X MARK m; /* a mark to convert */ X char *text; /* the line that m refers to */ X int inputting; /* boolean: caled from input() ? */ X{ X int i; X X i = idx2col(m, text, inputting); X while (i < leftcol) X { X leftcol -= *o_sidescroll; X mustredraw = TRUE; X redrawrange(1L, INFINITY, INFINITY); X qaddch('\r'); X /* drawtext(text); */ X } X while (i > rightcol) X { X leftcol += *o_sidescroll; X mustredraw = TRUE; X redrawrange(1L, INFINITY, INFINITY); X qaddch('\r'); X /* drawtext(text); */ X } X physcol = i - leftcol; X physrow = markline(m) - topline; X X return physcol; X} X X/* This function draws a single line of text on the screen. The screen's X * cursor is assumed to be located at the leftmost column of the appropriate X * row. X */ Xstatic void drawtext(text, clr) X REG char *text; /* the text to draw */ X int clr; /* boolean: do a clrtoeol? */ X{ X REG int col; /* column number */ X REG int i; X REG int tabstop; /* *o_tabstop */ X REG int limitcol; /* leftcol or leftcol + COLS */ X int abnormal; /* boolean: charattr != A_NORMAL? */ X X#ifndef NO_SENTENCE X /* if we're hiding format lines, and this is one of them, then hide it */ X if (*o_hideformat && *text == '.') X { X clrtoeol(); X#if OSK X qaddch('\l'); X#else X qaddch('\n'); X#endif X return; X } X#endif X X /* move some things into registers... */ X limitcol = leftcol; X tabstop = *o_tabstop; X abnormal = FALSE; X X#ifndef CRUNCH X if (clr) X clrtoeol(); X#endif X /* skip stuff that was scrolled off left edge */ X for (col = 0; X (i = *text) && col < limitcol; /* yes, ASSIGNMENT! */ X text++) X { X if (i == '\t' && !*o_list) X { X col = col + tabstop - (col % tabstop); X } X else if (i >= 0 && i < ' ' || i == '\177') X { X col += 2; X } X#ifndef NO_CHARATTR X else if (i == '\\' && text[1] == 'f' && text[2] && *o_charattr) X { X text += 2; /* plus one more as part of "for" loop */ X X /* since this attribute might carry over, we need it */ X switch (*text) X { X case 'R': X case 'P': X attrset(A_NORMAL); X abnormal = FALSE; X break; X X case 'B': X attrset(A_BOLD); X abnormal = TRUE; X break; X X case 'U': X attrset(A_UNDERLINE); X abnormal = TRUE; X break; X X case 'I': X attrset(A_ALTCHARSET); X abnormal = TRUE; X break; X } X } X#endif X else X { X col++; X } X } X X /* adjust for control char that was partially visible */ X while (col > limitcol) X { X qaddch(' '); X limitcol++; X } X X /* now for the visible characters */ X for (limitcol = leftcol + COLS; X (i = *text) && col < limitcol; X text++) X { X if (i == '\t' && !*o_list) X { X i = col + tabstop - (col % tabstop); X if (i < limitcol) X { X#ifdef CRUNCH X if (!clr && has_PT && !((i - leftcol) & 7)) X#else X if (has_PT && !((i - leftcol) & 7)) X#endif X { X do X { X qaddch('\t'); X col += 8; /* not exact! */ X } while (col < i); X col = i; /* NOW it is exact */ X } X else X { X do X { X qaddch(' '); X col++; X } while (col < i); X } X } X else /* tab ending after screen? next line! */ X { X col = limitcol; X if (has_AM) X { X addch('\n'); /* GB */ X } X } X } X else if (i >= 0 && i < ' ' || i == '\177') X { X col += 2; X qaddch('^'); X if (col <= limitcol) X { X qaddch(i ^ '@'); X } X } X#ifndef NO_CHARATTR X else if (i == '\\' && text[1] == 'f' && text[2] && *o_charattr) X { X text += 2; /* plus one more as part of "for" loop */ X switch (*text) X { X case 'R': X case 'P': X attrset(A_NORMAL); X abnormal = FALSE; X break; X X case 'B': X attrset(A_BOLD); X abnormal = TRUE; X break; X X case 'U': X attrset(A_UNDERLINE); X abnormal = TRUE; X break; X X case 'I': X attrset(A_ALTCHARSET); X abnormal = TRUE; X break; X } X } X#endif X else X { X col++; X qaddch(i); X } X } X X /* get ready for the next line */ X#ifndef NO_CHARATTR X if (abnormal) X { X attrset(A_NORMAL); X } X#endif X if (*o_list && col < limitcol) X { X qaddch('$'); X col++; X } X#ifdef CRUNCH X if (clr && col < limitcol) X { X clrtoeol(); X } X#endif X if (!has_AM || col < limitcol) X { X addch('\n'); X } X} X X X#ifndef CRUNCH Xstatic void nudgecursor(same, scan, new, lno) X int same; /* number of chars to be skipped over */ X char *scan; /* where the same chars end */ X char *new; /* where the visible part of the line starts */ X long lno; /* line number of this line */ X{ X if (same > 0) X { X if (same < 5) X { X /* move the cursor by overwriting */ X while (same > 0) X { X qaddch(scan[-same]); X same--; X } X } X else X { X /* move the cursor by calling move() */ X move((int)(lno - topline), (int)(scan - new)); X } X } X} X#endif /* not CRUNCH */ X X/* This function draws a single line of text on the screen, possibly with X * some cursor optimization. The cursor is repositioned before drawing X * begins, so its position before doesn't really matter. X */ Xstatic void smartdrawtext(text, lno) X REG char *text; /* the text to draw */ X long lno; /* line number of the text */ X{ X#ifdef CRUNCH X move((int)(lno - topline), 0); X drawtext(text, TRUE); X#else /* not CRUNCH */ X static char old[256]; /* how the line looked last time */ X char new[256]; /* how it looks now */ X char *build; /* used to put chars into new[] */ X char *scan; /* used for moving thru new[] or old[] */ X char *end; /* last non-blank changed char */ X char *shift; /* used to insert/delete chars */ X int same; /* length of a run of unchanged chars */ X int limitcol; X int col; X int i; X X# ifndef NO_CHARATTR X /* if this line has attributes, do it the dumb way instead */ X if (hasattr(lno, text)) X { X move((int)(lno - topline), 0); X drawtext(text, TRUE); X return; X } X# endif X# ifndef NO_SENTENCE X /* if this line is a format line, & we're hiding format lines, then X * let the dumb drawtext() function handle it X */ X if (*o_hideformat && *text == '.') X { X move((int)(lno - topline), 0); X drawtext(text, TRUE); X return; X } X# endif X X /* skip stuff that was scrolled off left edge */ X limitcol = leftcol; X for (col = 0; X (i = *text) && col < limitcol; /* yes, ASSIGNMENT! */ X text++) X { X if (i == '\t' && !*o_list) X { X col = col + *o_tabstop - (col % *o_tabstop); X } X else if (i >= 0 && i < ' ' || i == '\177') X { X col += 2; X } X else X { X col++; X } X } X X /* adjust for control char that was partially visible */ X build = new; X while (col > limitcol) X { X *build++ = ' '; X limitcol++; X } X X /* now for the visible characters */ X for (limitcol = leftcol + COLS; X (i = *text) && col < limitcol; X text++) X { X if (i == '\t' && !*o_list) X { X i = col + *o_tabstop - (col % *o_tabstop); X while (col < i && col < limitcol) X { X *build++ = ' '; X col++; X } X } X else if (i >= 0 && i < ' ' || i == '\177') X { X col += 2; X *build++ = '^'; X if (col <= limitcol) X { X *build++ = (i ^ '@'); X } X } X else X { X col++; X *build++ = i; X } X } X if (col < limitcol && *o_list) X { X *build++ = '$'; X col++; X } X end = build; X while (col < limitcol) X { X *build++ = ' '; X col++; X } X X /* locate the last non-blank character */ X while (end > new && end[-1] == ' ') X { X end--; X } X X /* can we optimize the displaying of this line? */ X if (lno != smartlno) X { X /* nope, can't optimize - different line */ X move((int)(lno - topline), 0); X for (scan = new, build = old; scan < end; ) X { X qaddch(*scan); X *build++ = *scan++; X } X if (end < new + COLS) X { X clrtoeol(); X while (build < old + COLS) X { X *build++ = ' '; X } X } X smartlno = lno; X return; X } X X /* skip any initial unchanged characters */ X for (scan = new, build = old; scan < end && *scan == *build; scan++, build++) X { X } X move((int)(lno - topline), (int)(scan - new)); X X /* The in-between characters must be changed */ X same = 0; X while (scan < end) X { X /* is this character a match? */ X if (scan[0] == build[0]) X { X same++; X } X else /* do we want to insert? */ X if (scan < end - 1 && scan[1] == build[0] && (has_IC || has_IM)) X { X nudgecursor(same, scan, new, lno); X same = 0; X X insch(*scan); X for (shift = old + COLS; --shift > build; ) X { X shift[0] = shift[-1]; X } X *build = *scan; X } X else /* do we want to delete? */ X if (build < old + COLS - 1 && scan[0] == build[1] && has_DC) X { X nudgecursor(same, scan, new, lno); X same = 0; X X delch(); X same++; X for (shift = build; shift < old + COLS - 1; shift++) X { X shift[0] = shift[1]; X } X *shift = ' '; X } X else /* we must overwrite */ X { X nudgecursor(same, scan, new, lno); X same = 0; X X addch(*scan); X *build = *scan; X } X X build++; X scan++; X } X X /* maybe clear to EOL */ X while (build < old + COLS && *build == ' ') X { X build++; X } X if (build < old + COLS) X { X nudgecursor(same, scan, new, lno); X same = 0; X X clrtoeol(); X while (build < old + COLS) X { X *build++ = ' '; X } X } X#endif /* not CRUNCH */ X} X X X/* This function is used in visual mode for drawing the screen (or just parts X * of the screen, if that's all thats needed). It also takes care of X * scrolling. X */ Xvoid redraw(curs, inputting) X MARK curs; /* where to leave the screen's cursor */ X int inputting; /* boolean: being called from input() ? */ X{ X char *text; /* a line of text to display */ X static long chgs; /* previous changes level */ X long l; X int i; X X /* if curs == MARK_UNSET, then we should reset internal vars */ X if (curs == MARK_UNSET) X { X if (topline < 1 || topline > nlines) X { X topline = 1L; X } X else X { X move(LINES - 1, 0); X clrtoeol(); X } X leftcol = 0; X mustredraw = TRUE; X redrawafter = INFINITY; X preredraw = 0L; X postredraw = 0L; X chgs = 0; X smartlno = 0L; X return; X } X X /* figure out which column the cursor will be in */ X l = markline(curs); X text = fetchline(l); X mark2phys(curs, text, inputting); X X /* adjust topline, if necessary, to get the cursor on the screen */ X if (l >= topline && l <= botline) X { X /* it is on the screen already */ X X /* if the file was changed but !mustredraw, then redraw line */ X if (chgs != changes && !mustredraw) X { X smartdrawtext(text, l); X } X } X else if (l < topline && l > topline - LINES && (has_SR || has_AL)) X { X /* near top - scroll down */ X if (!mustredraw) X { X move(0,0); X while (l < topline) X { X topline--; X if (has_SR) X { X do_SR(); X } X else X { X insertln(); X } X text = fetchline(topline); X drawtext(text, FALSE); X do_UP(); X } X X /* blank out the last line */ X move(LINES - 1, 0); X clrtoeol(); X } X else X { X topline = l; X redrawafter = INFINITY; X preredraw = 0L; X postredraw = 0L; X } X } X else if (l > topline && l < botline + LINES) X { X /* near bottom -- scroll up */ X if (!mustredraw X#if 1 X || redrawafter == preredraw && preredraw == botline && postredraw == l X#endif X ) X { X move(LINES - 1,0); X clrtoeol(); X while (l > botline) X { X topline++; /* <-- also adjusts botline */ X text = fetchline(botline); X drawtext(text, FALSE); X } X mustredraw = FALSE; X } X else X { X topline = l - (LINES - 2); X redrawafter = INFINITY; X preredraw = 0L; X postredraw = 0L; X } X } X else X { X /* distant line - center it & force a redraw */ X topline = l - (LINES / 2) - 1; X if (topline < 1) X { X topline = 1; X } X mustredraw = TRUE; X redrawafter = INFINITY; X preredraw = 0L; X postredraw = 0L; X } X X /* Now... do we really have to redraw? */ X if (mustredraw) X { X /* If redrawfter (and friends) aren't set, assume we should X * redraw everything. X */ X if (redrawafter == INFINITY) X { X redrawafter = 0L; X preredraw = postredraw = INFINITY; X } X X /* adjust smartlno to correspond with inserted/deleted lines */ X if (smartlno >= redrawafter) X { X if (smartlno < preredraw) X { X smartlno = 0L; X } X else X { X smartlno += (postredraw - preredraw); X } X } X X /* should we insert some lines into the screen? */ X if (preredraw < postredraw && preredraw <= botline) X { X /* lines were inserted into the file */ X X /* decide where insertion should start */ X if (preredraw < topline) X { X l = topline; X } X else X { X l = preredraw; X } X X /* insert the lines... maybe */ X if (l + postredraw - preredraw > botline || !has_AL) X { X /* Whoa! a whole screen full - just redraw */ X preredraw = postredraw = INFINITY; X } X else X { X /* really insert 'em */ X move((int)(l - topline), 0); X for (i = postredraw - preredraw; i > 0; i--) X { X insertln(); X } X X /* NOTE: the contents of those lines will be X * drawn as part of the regular redraw loop. X */ X X /* clear the last line */ X move(LINES - 1, 0); X clrtoeol(); X } X } X X /* do we want to delete some lines from the screen? */ X if (preredraw > postredraw && postredraw <= botline) X { X if (preredraw > botline || !has_DL) X { X postredraw = preredraw = INFINITY; X } X else /* we'd best delete some lines from the screen */ X { X /* clear the last line, so it doesn't look X * ugly as it gets pulled up into the screen X */ X move(LINES - 1, 0); X clrtoeol(); X X /* delete the lines */ X move((int)(postredraw - topline), 0); X for (l = postredraw; X l < preredraw && l <= botline; X l++) X { X deleteln(); X } X X /* draw the lines that are now newly visible X * at the bottom of the screen X */ X i = LINES - 1 + (postredraw - preredraw); X move(i, 0); X for (l = topline + i; l <= botline; l++) X { X /* clear this line */ X clrtoeol(); X X /* draw the line, or ~ for non-lines */ X if (l <= nlines) X { X text = fetchline(l); X drawtext(text, FALSE); X } X else X { X addstr("~\n"); X } X } X } X } X X /* redraw the current line */ X l = markline(curs); X pfetch(l); X smartdrawtext(ptext, l); X X /* decide where we should start redrawing from */ X if (redrawafter < topline) X { X l = topline; X } X else X { X l = redrawafter; X } X move((int)(l - topline), 0); X X /* draw the other lines */ X for (; l <= botline && l < postredraw; l++) X { X /* we already drew the current line, so skip it now */ X if (l == smartlno) X { X#if OSK X qaddch('\l'); X#else X qaddch('\n'); X#endif X continue; X } X X /* draw the line, or ~ for non-lines */ X if (l <= nlines) X { X text = fetchline(l); X drawtext(text, TRUE); X } X else X { X qaddch('~'); X clrtoeol(); X addch('\n'); X } X } X X mustredraw = FALSE; X } X X /* force total (non-partial) redraw next time if not set */ X redrawafter = INFINITY; X preredraw = 0L; X postredraw = 0L; X X /* move the cursor to where it belongs */ X move((int)(markline(curs) - topline), physcol); X wqrefresh(stdscr); X X chgs = changes; X} eof if test `wc -c <redraw.c` -ne 19816 then echo redraw.c damaged! fi fi exit 0 ------------------------------------------------------------------------------- Steve Kirkendall kirkenda@cs.pdx.edu Grad student at Portland State U.