/* * 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.
*/
/* * A really simple-minded text editor based on cords. * Things it does right: * No size bounds. * Unbounded 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.
*/
#include <stdio.h> #include <stdlib.h> /* for exit() */
#include"gc.h" #include"cord.h"
#ifdef THINK_C #define MACINTOSH #endif #include <ctype.h>
#if (defined(__BORLANDC__) || defined(__CYGWIN__) || defined(__MINGW32__) \
|| defined(__NT__) || defined(_WIN32)) && !defined(WIN32) /* If this is DOS or win16, we'll fail anyway. */ /* Might as well assume win32. */ # define WIN32 #endif
/* List of line number to position mappings, in descending order. */ /* There may be holes. */ typedefstruct LineMapRep { int line;
size_t pos; struct LineMapRep * previous;
} * line_map;
/* List of file versions, one per edit operation */ typedefstruct 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(void)
{
line_map map = current_map; int start_line = map -> line;
current_map_size = 0; do {
current_map_size++; if (map -> line < start_line - LINES && map -> previous != 0) {
line_map pred = map -> previous -> previous;
if (NULL == new_map) OUT_OF_MEMORY; if (current_map_size >= MAX_MAP_SIZE) prune_map();
new_map -> line = line_arg;
new_map -> pos = pos;
cur_map = current_map;
GC_PTR_STORE_AND_DIRTY(&new_map->previous, cur_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;
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) {
size_t next = CORD_chr(current, cur, '\n');
if (next == CORD_NOT_FOUND) next = current_len - 1; if (next < cur + *c) {
*c = (int)(next - cur);
}
cur += *c;
} return(cur);
}
void add_hist(CORD s)
{
history new_file = GC_NEW(struct HistoryRep);
if (NULL == new_file) OUT_OF_MEMORY;
new_file -> file_contents = current = s;
current_len = CORD_len(s);
new_file -> previous = now;
GC_END_STUBBORN_CHANGE(new_file); if (now != NULL) {
now -> map = current_map;
GC_END_STUBBORN_CHANGE(now);
}
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)
{
CORD_pos p; # if !defined(MACINTOSH)
size_t len = CORD_len(s); # endif
if (screen == 0 || LINES > screen_size) {
screen_size = LINES;
screen = (CORD *)GC_MALLOC(screen_size * sizeof(CORD)); if (NULL == screen) OUT_OF_MEMORY;
} # if !defined(MACINTOSH) /* A gross workaround for an apparent curses bug: */ if (i == LINES-1 && len == (unsigned)COLS) {
s = CORD_substr(s, 0, len - 1);
} # endif if (CORD_cmp(screen[i], s) != 0) {
move(i, 0); clrtoeol(); move(i,0);
CORD_FOR (p, s) { int c = CORD_pos_fetch(p) & 0x7f;
/* 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 */
size_t 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));
}
/* Display the visible section of the current file */ void redisplay(void)
{ int i;
invalidate_map(dis_line + LINES); /* Prune search */ for (i = 0; i < LINES; i++) { if (need_redisplay == ALL || need_redisplay == i) {
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(void)
{ 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;
}
}
/* 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(void)
{ int my_col = col;
#ifdefined(WIN32) # define beep() Beep(1000 /* Hz */, 300 /* ms */) #elifdefined(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.
*/ int beep(void)
{
putc('\007', stderr); return(0);
} #endif/* !WIN32 && !MACINTOSH */
# 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 = (int)(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;
} elseif (c < 0x100 && isdigit(c)){ if (repeat_count == BARE_PREFIX) {
repeat_count = c - '0'; return;
} elseif (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 = 0;
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--; /* FALLTHRU */ 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");
initial = CORD_from_file(f);
len = CORD_len(initial); if (0 == len || CORD_fetch(initial, len - 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 */
GC_END_STUBBORN_CHANGE(now);
need_redisplay = ALL;
fix_cursor();
}
#ifndef WIN32
int main(int argc, char **argv)
{ int c; void *buf;
# ifdefined(MACINTOSH)
console_options.title = "\pDumb Editor";
cshow(stdout);
argc = ccommand(&argv); # endif
GC_set_find_leak(0); /* app is not for testing leak detection mode */
GC_INIT(); # ifndef NO_INCREMENTAL
GC_enable_incremental(); # endif
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.