604 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			604 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 1993-1994 by Xerox Corporation.  All rights reserved.
 | |
|  *
 | |
|  * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
 | |
|  * OR IMPLIED.  ANY USE IS AT YOUR OWN RISK.
 | |
|  *
 | |
|  * Permission is hereby granted to use or copy this program
 | |
|  * for any purpose,  provided the above notices are retained on all copies.
 | |
|  * Permission to modify the code and to distribute modified code is granted,
 | |
|  * provided the above notices are retained, and a notice that the code was
 | |
|  * modified is included with the above copyright notice.
 | |
|  *
 | |
|  * Author: Hans-J. Boehm (boehm@parc.xerox.com)
 | |
|  */
 | |
| /*
 | |
|  * A really simple-minded text editor based on cords.
 | |
|  * Things it does right:
 | |
|  * 	No size bounds.
 | |
|  *	Inbounded undo.
 | |
|  *	Shouldn't crash no matter what file you invoke it on (e.g. /vmunix)
 | |
|  *		(Make sure /vmunix is not writable before you try this.)
 | |
|  *	Scrolls horizontally.
 | |
|  * Things it does wrong:
 | |
|  *	It doesn't handle tabs reasonably (use "expand" first).
 | |
|  *	The command set is MUCH too small.
 | |
|  *	The redisplay algorithm doesn't let curses do the scrolling.
 | |
|  *	The rule for moving the window over the file is suboptimal.
 | |
|  */
 | |
| /* Boehm, February 6, 1995 12:27 pm PST */
 | |
| 
 | |
| /* Boehm, May 19, 1994 2:20 pm PDT */
 | |
| #include <stdio.h>
 | |
| #include "gc.h"
 | |
| #include "cord.h"
 | |
| 
 | |
| #ifdef THINK_C
 | |
| #define MACINTOSH
 | |
| #include <ctype.h>
 | |
| #endif
 | |
| 
 | |
| #if defined(__BORLANDC__) && !defined(WIN32)
 | |
|     /* If this is DOS or win16, we'll fail anyway.	*/
 | |
|     /* Might as well assume win32.			*/
 | |
| #   define WIN32
 | |
| #endif
 | |
| 
 | |
| #if defined(WIN32)
 | |
| #  include <windows.h>
 | |
| #  include "de_win.h"
 | |
| #elif defined(MACINTOSH)
 | |
| #	include <console.h>
 | |
| /* curses emulation. */
 | |
| #	define initscr()
 | |
| #	define endwin()
 | |
| #	define nonl()
 | |
| #	define noecho() csetmode(C_NOECHO, stdout)
 | |
| #	define cbreak() csetmode(C_CBREAK, stdout)
 | |
| #	define refresh()
 | |
| #	define addch(c) putchar(c)
 | |
| #	define standout() cinverse(1, stdout)
 | |
| #	define standend() cinverse(0, stdout)
 | |
| #	define move(line,col) cgotoxy(col + 1, line + 1, stdout)
 | |
| #	define clrtoeol() ccleol(stdout)
 | |
| #	define de_error(s) { fprintf(stderr, s); getchar(); }
 | |
| #	define LINES 25
 | |
| #	define COLS 80
 | |
| #else
 | |
| #  include <curses.h>
 | |
| #  define de_error(s) { fprintf(stderr, s); sleep(2); }
 | |
| #endif
 | |
| #include "de_cmds.h"
 | |
| 
 | |
| /* List of line number to position mappings, in descending order. */
 | |
| /* There may be holes.						  */
 | |
| typedef struct LineMapRep {
 | |
|     int line;
 | |
|     size_t pos;
 | |
|     struct LineMapRep * previous;
 | |
| } * line_map;
 | |
| 
 | |
| /* List of file versions, one per edit operation */
 | |
| typedef struct HistoryRep {
 | |
|     CORD file_contents;
 | |
|     struct HistoryRep * previous;
 | |
|     line_map map;	/* Invalid for first record "now" */
 | |
| } * history;
 | |
| 
 | |
| history now = 0;
 | |
| CORD current;		/* == now -> file_contents.	*/
 | |
| size_t current_len;	/* Current file length.		*/
 | |
| line_map current_map = 0;	/* Current line no. to pos. map	 */
 | |
| size_t current_map_size = 0;	/* Number of current_map entries.	*/
 | |
| 				/* Not always accurate, but reset	*/
 | |
| 				/* by prune_map.			*/
 | |
| # define MAX_MAP_SIZE 3000
 | |
| 
 | |
| /* Current display position */
 | |
| int dis_line = 0;
 | |
| int dis_col = 0;
 | |
| 
 | |
| # define ALL -1
 | |
| # define NONE - 2
 | |
| int need_redisplay = 0;	/* Line that needs to be redisplayed.	*/
 | |
| 
 | |
| 
 | |
| /* Current cursor position. Always within file. */
 | |
| int line = 0; 
 | |
| int col = 0;
 | |
| size_t file_pos = 0;	/* Character position corresponding to cursor.	*/
 | |
| 
 | |
| /* Invalidate line map for lines > i */
 | |
| void invalidate_map(int i)
 | |
| {
 | |
|     while(current_map -> line > i) {
 | |
|         current_map = current_map -> previous;
 | |
|         current_map_size--;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Reduce the number of map entries to save space for huge files. */
 | |
| /* This also affects maps in histories.				  */
 | |
| void prune_map()
 | |
| {
 | |
|     line_map map = current_map;
 | |
|     int start_line = map -> line;
 | |
|     
 | |
|     current_map_size = 0;
 | |
|     for(; map != 0; map = map -> previous) {
 | |
|     	current_map_size++;
 | |
|     	if (map -> line < start_line - LINES && map -> previous != 0) {
 | |
|     	    map -> previous = map -> previous -> previous;
 | |
|     	}
 | |
|     }
 | |
| }
 | |
| /* Add mapping entry */
 | |
| void add_map(int line, size_t pos)
 | |
| {
 | |
|     line_map new_map = GC_NEW(struct LineMapRep);
 | |
|     
 | |
|     if (current_map_size >= MAX_MAP_SIZE) prune_map();
 | |
|     new_map -> line = line;
 | |
|     new_map -> pos = pos;
 | |
|     new_map -> previous = current_map;
 | |
|     current_map = new_map;
 | |
|     current_map_size++;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /* Return position of column *c of ith line in   */
 | |
| /* current file. Adjust *c to be within the line.*/
 | |
| /* A 0 pointer is taken as 0 column.		 */
 | |
| /* Returns CORD_NOT_FOUND if i is too big.	 */
 | |
| /* Assumes i > dis_line.			 */
 | |
| size_t line_pos(int i, int *c)
 | |
| {
 | |
|     int j;
 | |
|     size_t cur;
 | |
|     size_t next;
 | |
|     line_map map = current_map;
 | |
|     
 | |
|     while (map -> line > i) map = map -> previous;
 | |
|     if (map -> line < i - 2) /* rebuild */ invalidate_map(i);
 | |
|     for (j = map -> line, cur = map -> pos; j < i;) {
 | |
| 	cur = CORD_chr(current, cur, '\n');
 | |
|         if (cur == current_len-1) return(CORD_NOT_FOUND);
 | |
|         cur++;
 | |
|         if (++j > current_map -> line) add_map(j, cur);
 | |
|     }
 | |
|     if (c != 0) {
 | |
|         next = CORD_chr(current, cur, '\n');
 | |
|         if (next == CORD_NOT_FOUND) next = current_len - 1;
 | |
|         if (next < cur + *c) {
 | |
|             *c = next - cur;
 | |
|         }
 | |
|         cur += *c;
 | |
|     }
 | |
|     return(cur);
 | |
| }
 | |
| 
 | |
| void add_hist(CORD s)
 | |
| {
 | |
|     history new_file = GC_NEW(struct HistoryRep);
 | |
|     
 | |
|     new_file -> file_contents = current = s;
 | |
|     current_len = CORD_len(s);
 | |
|     new_file -> previous = now;
 | |
|     if (now != 0) now -> map = current_map;
 | |
|     now = new_file;
 | |
| }
 | |
| 
 | |
| void del_hist(void)
 | |
| {
 | |
|     now = now -> previous;
 | |
|     current = now -> file_contents;
 | |
|     current_map = now -> map;
 | |
|     current_len = CORD_len(current);
 | |
| }
 | |
| 
 | |
| /* Current screen_contents; a dynamically allocated array of CORDs	*/
 | |
| CORD * screen = 0;
 | |
| int screen_size = 0;
 | |
| 
 | |
| # ifndef WIN32
 | |
| /* Replace a line in the curses stdscr.	All control characters are	*/
 | |
| /* displayed as upper case characters in standout mode.  This isn't	*/
 | |
| /* terribly appropriate for tabs.									*/
 | |
| void replace_line(int i, CORD s)
 | |
| {
 | |
|     register int c;
 | |
|     CORD_pos p;
 | |
|     size_t len = CORD_len(s);
 | |
|     
 | |
|     if (screen == 0 || LINES > screen_size) {
 | |
|         screen_size = LINES;
 | |
|     	screen = (CORD *)GC_MALLOC(screen_size * sizeof(CORD));
 | |
|     }
 | |
| #   if !defined(MACINTOSH)
 | |
|         /* A gross workaround for an apparent curses bug: */
 | |
|         if (i == LINES-1 && len == COLS) {
 | |
|             s = CORD_substr(s, 0, CORD_len(s) - 1);
 | |
|         }
 | |
| #   endif
 | |
|     if (CORD_cmp(screen[i], s) != 0) {
 | |
|         move(i, 0); clrtoeol(); move(i,0);
 | |
| 
 | |
|         CORD_FOR (p, s) {
 | |
|             c = CORD_pos_fetch(p) & 0x7f;
 | |
|             if (iscntrl(c)) {
 | |
|             	standout(); addch(c + 0x40); standend();
 | |
|             } else {
 | |
|     	        addch(c);
 | |
|     	    }
 | |
|     	}
 | |
|     	screen[i] = s;
 | |
|     }
 | |
| }
 | |
| #else
 | |
| # define replace_line(i,s) invalidate_line(i)
 | |
| #endif
 | |
| 
 | |
| /* Return up to COLS characters of the line of s starting at pos,	*/
 | |
| /* returning only characters after the given column.			*/
 | |
| CORD retrieve_line(CORD s, size_t pos, unsigned column)
 | |
| {
 | |
|     CORD candidate = CORD_substr(s, pos, column + COLS);
 | |
|     			/* avoids scanning very long lines	*/
 | |
|     int eol = CORD_chr(candidate, 0, '\n');
 | |
|     int len;
 | |
|     
 | |
|     if (eol == CORD_NOT_FOUND) eol = CORD_len(candidate);
 | |
|     len = (int)eol - (int)column;
 | |
|     if (len < 0) len = 0;
 | |
|     return(CORD_substr(s, pos + column, len));
 | |
| }
 | |
| 
 | |
| # ifdef WIN32
 | |
| #   define refresh();
 | |
| 
 | |
|     CORD retrieve_screen_line(int i)
 | |
|     {
 | |
|     	register size_t pos;
 | |
|     	
 | |
|     	invalidate_map(dis_line + LINES);	/* Prune search */
 | |
|     	pos = line_pos(dis_line + i, 0);
 | |
|     	if (pos == CORD_NOT_FOUND) return(CORD_EMPTY);
 | |
|     	return(retrieve_line(current, pos, dis_col));
 | |
|     }
 | |
| # endif
 | |
| 
 | |
| /* Display the visible section of the current file	 */
 | |
| void redisplay(void)
 | |
| {
 | |
|     register int i;
 | |
|     
 | |
|     invalidate_map(dis_line + LINES);	/* Prune search */
 | |
|     for (i = 0; i < LINES; i++) {
 | |
|         if (need_redisplay == ALL || need_redisplay == i) {
 | |
|             register size_t pos = line_pos(dis_line + i, 0);
 | |
|             
 | |
|             if (pos == CORD_NOT_FOUND) break;
 | |
|             replace_line(i, retrieve_line(current, pos, dis_col));
 | |
|             if (need_redisplay == i) goto done;
 | |
|         }
 | |
|     }
 | |
|     for (; i < LINES; i++) replace_line(i, CORD_EMPTY);
 | |
| done:
 | |
|     refresh();
 | |
|     need_redisplay = NONE;
 | |
| }
 | |
| 
 | |
| int dis_granularity;
 | |
| 
 | |
| /* Update dis_line, dis_col, and dis_pos to make cursor visible.	*/
 | |
| /* Assumes line, col, dis_line, dis_pos are in bounds.			*/
 | |
| void normalize_display()
 | |
| {
 | |
|     int old_line = dis_line;
 | |
|     int old_col = dis_col;
 | |
|     
 | |
|     dis_granularity = 1;
 | |
|     if (LINES > 15 && COLS > 15) dis_granularity = 2;
 | |
|     while (dis_line > line) dis_line -= dis_granularity;
 | |
|     while (dis_col > col) dis_col -= dis_granularity;
 | |
|     while (line >= dis_line + LINES) dis_line += dis_granularity;
 | |
|     while (col >= dis_col + COLS) dis_col += dis_granularity;
 | |
|     if (old_line != dis_line || old_col != dis_col) {
 | |
|         need_redisplay = ALL;
 | |
|     }
 | |
| }
 | |
| 
 | |
| # if defined(WIN32)
 | |
| # elif defined(MACINTOSH)
 | |
| #		define move_cursor(x,y) cgotoxy(x + 1, y + 1, stdout)
 | |
| # else
 | |
| #		define move_cursor(x,y) move(y,x)
 | |
| # endif
 | |
| 
 | |
| /* Adjust display so that cursor is visible; move cursor into position	*/
 | |
| /* Update screen if necessary.						*/
 | |
| void fix_cursor(void)
 | |
| {
 | |
|     normalize_display();
 | |
|     if (need_redisplay != NONE) redisplay();
 | |
|     move_cursor(col - dis_col, line - dis_line);
 | |
|     refresh();
 | |
| #   ifndef WIN32
 | |
|       fflush(stdout);
 | |
| #   endif
 | |
| }
 | |
| 
 | |
| /* Make sure line, col, and dis_pos are somewhere inside file.	*/
 | |
| /* Recompute file_pos.	Assumes dis_pos is accurate or past eof	*/
 | |
| void fix_pos()
 | |
| {
 | |
|     int my_col = col;
 | |
|     
 | |
|     if ((size_t)line > current_len) line = current_len;
 | |
|     file_pos = line_pos(line, &my_col);
 | |
|     if (file_pos == CORD_NOT_FOUND) {
 | |
|         for (line = current_map -> line, file_pos = current_map -> pos;
 | |
|              file_pos < current_len;
 | |
|              line++, file_pos = CORD_chr(current, file_pos, '\n') + 1);
 | |
|     	line--;
 | |
|         file_pos = line_pos(line, &col);
 | |
|     } else {
 | |
|     	col = my_col;
 | |
|     }
 | |
| }
 | |
| 
 | |
| #if defined(WIN32)
 | |
| #  define beep() Beep(1000 /* Hz */, 300 /* msecs */) 
 | |
| #elif defined(MACINTOSH)
 | |
| #	define beep() SysBeep(1)
 | |
| #else
 | |
| /*
 | |
|  * beep() is part of some curses packages and not others.
 | |
|  * We try to match the type of the builtin one, if any.
 | |
|  */
 | |
| #ifdef __STDC__
 | |
|     int beep(void)
 | |
| #else
 | |
|     int beep()
 | |
| #endif
 | |
| {
 | |
|     putc('\007', stderr);
 | |
|     return(0);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #   define NO_PREFIX -1
 | |
| #   define BARE_PREFIX -2
 | |
| int repeat_count = NO_PREFIX;	/* Current command prefix. */
 | |
| 
 | |
| int locate_mode = 0;			/* Currently between 2 ^Ls	*/
 | |
| CORD locate_string = CORD_EMPTY;	/* Current search string.	*/
 | |
| 
 | |
| char * arg_file_name;
 | |
| 
 | |
| #ifdef WIN32
 | |
| /* Change the current position to whatever is currently displayed at	*/
 | |
| /* the given SCREEN coordinates.					*/
 | |
| void set_position(int c, int l)
 | |
| {
 | |
|     line = l + dis_line;
 | |
|     col = c + dis_col;
 | |
|     fix_pos();
 | |
|     move_cursor(col - dis_col, line - dis_line);
 | |
| }
 | |
| #endif /* WIN32 */
 | |
| 
 | |
| /* Perform the command associated with character c.  C may be an	*/
 | |
| /* integer > 256 denoting a windows command, one of the above control	*/
 | |
| /* characters, or another ASCII character to be used as either a 	*/
 | |
| /* character to be inserted, a repeat count, or a search string, 	*/
 | |
| /* depending on the current state.					*/
 | |
| void do_command(int c)
 | |
| {
 | |
|     int i;
 | |
|     int need_fix_pos;
 | |
|     FILE * out;
 | |
|     
 | |
|     if ( c == '\r') c = '\n';
 | |
|     if (locate_mode) {
 | |
|         size_t new_pos;
 | |
|           
 | |
|         if (c == LOCATE) {
 | |
|               locate_mode = 0;
 | |
|               locate_string = CORD_EMPTY;
 | |
|               return;
 | |
|         }
 | |
|         locate_string = CORD_cat_char(locate_string, (char)c);
 | |
|         new_pos = CORD_str(current, file_pos - CORD_len(locate_string) + 1,
 | |
|           		   locate_string);
 | |
|         if (new_pos != CORD_NOT_FOUND) {
 | |
|             need_redisplay = ALL;
 | |
|             new_pos += CORD_len(locate_string);
 | |
|             for (;;) {
 | |
|               	file_pos = line_pos(line + 1, 0);
 | |
|               	if (file_pos > new_pos) break;
 | |
|               	line++;
 | |
|             }
 | |
|             col = new_pos - line_pos(line, 0);
 | |
|             file_pos = new_pos;
 | |
|             fix_cursor();
 | |
|         } else {
 | |
|             locate_string = CORD_substr(locate_string, 0,
 | |
|               				CORD_len(locate_string) - 1);
 | |
|             beep();
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
|     if (c == REPEAT) {
 | |
|       	repeat_count = BARE_PREFIX; return;
 | |
|     } else if (c < 0x100 && isdigit(c)){
 | |
|         if (repeat_count == BARE_PREFIX) {
 | |
|           repeat_count = c - '0'; return;
 | |
|         } else if (repeat_count != NO_PREFIX) {
 | |
|           repeat_count = 10 * repeat_count + c - '0'; return;
 | |
|         }
 | |
|     }
 | |
|     if (repeat_count == NO_PREFIX) repeat_count = 1;
 | |
|     if (repeat_count == BARE_PREFIX && (c == UP || c == DOWN)) {
 | |
|       	repeat_count = LINES - dis_granularity;
 | |
|     }
 | |
|     if (repeat_count == BARE_PREFIX) repeat_count = 8;
 | |
|     need_fix_pos = 0;
 | |
|     for (i = 0; i < repeat_count; i++) {
 | |
|         switch(c) {
 | |
|           case LOCATE:
 | |
|             locate_mode = 1;
 | |
|             break;
 | |
|           case TOP:
 | |
|             line = col = file_pos = 0;
 | |
|             break;
 | |
|      	  case UP:
 | |
|      	    if (line != 0) {
 | |
|      	        line--;
 | |
|      	        need_fix_pos = 1;
 | |
|      	    }
 | |
|      	    break;
 | |
|      	  case DOWN:
 | |
|      	    line++;
 | |
|      	    need_fix_pos = 1;
 | |
|      	    break;
 | |
|      	  case LEFT:
 | |
|      	    if (col != 0) {
 | |
|      	        col--; file_pos--;
 | |
|      	    }
 | |
|      	    break;
 | |
|      	  case RIGHT:
 | |
|      	    if (CORD_fetch(current, file_pos) == '\n') break;
 | |
|      	    col++; file_pos++;
 | |
|      	    break;
 | |
|      	  case UNDO:
 | |
|      	    del_hist();
 | |
|      	    need_redisplay = ALL; need_fix_pos = 1;
 | |
|      	    break;
 | |
|      	  case BS:
 | |
|      	    if (col == 0) {
 | |
|      	        beep();
 | |
|      	        break;
 | |
|      	    }
 | |
|      	    col--; file_pos--;
 | |
|      	    /* fall through: */
 | |
|      	  case DEL:
 | |
|      	    if (file_pos == current_len-1) break;
 | |
|      	    	/* Can't delete trailing newline */
 | |
|      	    if (CORD_fetch(current, file_pos) == '\n') {
 | |
|      	        need_redisplay = ALL; need_fix_pos = 1;
 | |
|      	    } else {
 | |
|      	        need_redisplay = line - dis_line;
 | |
|      	    }
 | |
|      	    add_hist(CORD_cat(
 | |
|      	    		CORD_substr(current, 0, file_pos),
 | |
|      	    		CORD_substr(current, file_pos+1, current_len)));
 | |
|      	    invalidate_map(line);
 | |
|      	    break;
 | |
|      	  case WRITE:
 | |
| 	    {
 | |
|   		CORD name = CORD_cat(CORD_from_char_star(arg_file_name),
 | |
| 				     ".new");
 | |
| 
 | |
|     	        if ((out = fopen(CORD_to_const_char_star(name), "wb")) == NULL
 | |
|   	            || CORD_put(current, out) == EOF) {
 | |
|         	    de_error("Write failed\n");
 | |
|         	    need_redisplay = ALL;
 | |
|                 } else {
 | |
|                     fclose(out);
 | |
|                 }
 | |
| 	    }
 | |
|             break;
 | |
|      	  default:
 | |
|      	    {
 | |
|      	        CORD left_part = CORD_substr(current, 0, file_pos);
 | |
|      	        CORD right_part = CORD_substr(current, file_pos, current_len);
 | |
|      	        
 | |
|      	        add_hist(CORD_cat(CORD_cat_char(left_part, (char)c),
 | |
|      	        		  right_part));
 | |
|      	        invalidate_map(line);
 | |
|      	        if (c == '\n') {
 | |
|      	            col = 0; line++; file_pos++;
 | |
|      	            need_redisplay = ALL;
 | |
|      	        } else {
 | |
|      	            col++; file_pos++;
 | |
|      	            need_redisplay = line - dis_line;
 | |
|      	    	}
 | |
|      	        break;
 | |
|      	    }
 | |
|         }
 | |
|     }
 | |
|     if (need_fix_pos) fix_pos();
 | |
|     fix_cursor();
 | |
|     repeat_count = NO_PREFIX;
 | |
| }
 | |
| 
 | |
| /* OS independent initialization */
 | |
| 
 | |
| void generic_init(void)
 | |
| {
 | |
|     FILE * f;
 | |
|     CORD initial;
 | |
|     
 | |
|     if ((f = fopen(arg_file_name, "rb")) == NULL) {
 | |
|      	initial = "\n";
 | |
|     } else {
 | |
|         initial = CORD_from_file(f);
 | |
|         if (initial == CORD_EMPTY
 | |
|             || CORD_fetch(initial, CORD_len(initial)-1) != '\n') {
 | |
|             initial = CORD_cat(initial, "\n");
 | |
|         }
 | |
|     }
 | |
|     add_map(0,0);
 | |
|     add_hist(initial);
 | |
|     now -> map = current_map;
 | |
|     now -> previous = now;  /* Can't back up further: beginning of the world */
 | |
|     need_redisplay = ALL;
 | |
|     fix_cursor();
 | |
| }
 | |
| 
 | |
| #ifndef WIN32
 | |
| 
 | |
| main(argc, argv)
 | |
| int argc;
 | |
| char ** argv;
 | |
| {
 | |
|     int c;
 | |
| 
 | |
| #if defined(MACINTOSH)
 | |
| 	console_options.title = "\pDumb Editor";
 | |
| 	cshow(stdout);
 | |
| 	GC_init();
 | |
| 	argc = ccommand(&argv);
 | |
| #endif
 | |
|     
 | |
|     if (argc != 2) goto usage;
 | |
|     arg_file_name = argv[1];
 | |
|     setvbuf(stdout, GC_MALLOC_ATOMIC(8192), _IOFBF, 8192);
 | |
|     initscr();
 | |
|     noecho(); nonl(); cbreak();
 | |
|     generic_init();
 | |
|     while ((c = getchar()) != QUIT) {
 | |
| 		if (c == EOF) break;
 | |
| 	    do_command(c);
 | |
|     }
 | |
| done:
 | |
|     move(LINES-1, 0);
 | |
|     clrtoeol();
 | |
|     refresh();
 | |
|     nl();
 | |
|     echo();
 | |
|     endwin();
 | |
|     exit(0);
 | |
| usage:
 | |
|     fprintf(stderr, "Usage: %s file\n", argv[0]);
 | |
|     fprintf(stderr, "Cursor keys: ^B(left) ^F(right) ^P(up) ^N(down)\n");
 | |
|     fprintf(stderr, "Undo: ^U    Write to <file>.new: ^W");
 | |
|     fprintf(stderr, "Quit:^D  Repeat count: ^R[n]\n");
 | |
|     fprintf(stderr, "Top: ^T   Locate (search, find): ^L text ^L\n");
 | |
|     exit(1);
 | |
| }
 | |
| 
 | |
| #endif  /* !WIN32 */
 |