Annotation of OpenXM/src/kan96xx/gc-4.14/cord/de.c, Revision 1.1
1.1 ! maekawa 1: /*
! 2: * Copyright (c) 1993-1994 by Xerox Corporation. All rights reserved.
! 3: *
! 4: * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
! 5: * OR IMPLIED. ANY USE IS AT YOUR OWN RISK.
! 6: *
! 7: * Permission is hereby granted to use or copy this program
! 8: * for any purpose, provided the above notices are retained on all copies.
! 9: * Permission to modify the code and to distribute modified code is granted,
! 10: * provided the above notices are retained, and a notice that the code was
! 11: * modified is included with the above copyright notice.
! 12: *
! 13: * Author: Hans-J. Boehm (boehm@parc.xerox.com)
! 14: */
! 15: /*
! 16: * A really simple-minded text editor based on cords.
! 17: * Things it does right:
! 18: * No size bounds.
! 19: * Inbounded undo.
! 20: * Shouldn't crash no matter what file you invoke it on (e.g. /vmunix)
! 21: * (Make sure /vmunix is not writable before you try this.)
! 22: * Scrolls horizontally.
! 23: * Things it does wrong:
! 24: * It doesn't handle tabs reasonably (use "expand" first).
! 25: * The command set is MUCH too small.
! 26: * The redisplay algorithm doesn't let curses do the scrolling.
! 27: * The rule for moving the window over the file is suboptimal.
! 28: */
! 29: /* Boehm, February 6, 1995 12:27 pm PST */
! 30:
! 31: /* Boehm, May 19, 1994 2:20 pm PDT */
! 32: #include <stdio.h>
! 33: #include "gc.h"
! 34: #include "cord.h"
! 35:
! 36: #ifdef THINK_C
! 37: #define MACINTOSH
! 38: #include <ctype.h>
! 39: #endif
! 40:
! 41: #if defined(__BORLANDC__) && !defined(WIN32)
! 42: /* If this is DOS or win16, we'll fail anyway. */
! 43: /* Might as well assume win32. */
! 44: # define WIN32
! 45: #endif
! 46:
! 47: #if defined(WIN32)
! 48: # include <windows.h>
! 49: # include "de_win.h"
! 50: #elif defined(MACINTOSH)
! 51: # include <console.h>
! 52: /* curses emulation. */
! 53: # define initscr()
! 54: # define endwin()
! 55: # define nonl()
! 56: # define noecho() csetmode(C_NOECHO, stdout)
! 57: # define cbreak() csetmode(C_CBREAK, stdout)
! 58: # define refresh()
! 59: # define addch(c) putchar(c)
! 60: # define standout() cinverse(1, stdout)
! 61: # define standend() cinverse(0, stdout)
! 62: # define move(line,col) cgotoxy(col + 1, line + 1, stdout)
! 63: # define clrtoeol() ccleol(stdout)
! 64: # define de_error(s) { fprintf(stderr, s); getchar(); }
! 65: # define LINES 25
! 66: # define COLS 80
! 67: #else
! 68: # include <curses.h>
! 69: # define de_error(s) { fprintf(stderr, s); sleep(2); }
! 70: #endif
! 71: #include "de_cmds.h"
! 72:
! 73: /* List of line number to position mappings, in descending order. */
! 74: /* There may be holes. */
! 75: typedef struct LineMapRep {
! 76: int line;
! 77: size_t pos;
! 78: struct LineMapRep * previous;
! 79: } * line_map;
! 80:
! 81: /* List of file versions, one per edit operation */
! 82: typedef struct HistoryRep {
! 83: CORD file_contents;
! 84: struct HistoryRep * previous;
! 85: line_map map; /* Invalid for first record "now" */
! 86: } * history;
! 87:
! 88: history now = 0;
! 89: CORD current; /* == now -> file_contents. */
! 90: size_t current_len; /* Current file length. */
! 91: line_map current_map = 0; /* Current line no. to pos. map */
! 92: size_t current_map_size = 0; /* Number of current_map entries. */
! 93: /* Not always accurate, but reset */
! 94: /* by prune_map. */
! 95: # define MAX_MAP_SIZE 3000
! 96:
! 97: /* Current display position */
! 98: int dis_line = 0;
! 99: int dis_col = 0;
! 100:
! 101: # define ALL -1
! 102: # define NONE - 2
! 103: int need_redisplay = 0; /* Line that needs to be redisplayed. */
! 104:
! 105:
! 106: /* Current cursor position. Always within file. */
! 107: int line = 0;
! 108: int col = 0;
! 109: size_t file_pos = 0; /* Character position corresponding to cursor. */
! 110:
! 111: /* Invalidate line map for lines > i */
! 112: void invalidate_map(int i)
! 113: {
! 114: while(current_map -> line > i) {
! 115: current_map = current_map -> previous;
! 116: current_map_size--;
! 117: }
! 118: }
! 119:
! 120: /* Reduce the number of map entries to save space for huge files. */
! 121: /* This also affects maps in histories. */
! 122: void prune_map()
! 123: {
! 124: line_map map = current_map;
! 125: int start_line = map -> line;
! 126:
! 127: current_map_size = 0;
! 128: for(; map != 0; map = map -> previous) {
! 129: current_map_size++;
! 130: if (map -> line < start_line - LINES && map -> previous != 0) {
! 131: map -> previous = map -> previous -> previous;
! 132: }
! 133: }
! 134: }
! 135: /* Add mapping entry */
! 136: void add_map(int line, size_t pos)
! 137: {
! 138: line_map new_map = GC_NEW(struct LineMapRep);
! 139:
! 140: if (current_map_size >= MAX_MAP_SIZE) prune_map();
! 141: new_map -> line = line;
! 142: new_map -> pos = pos;
! 143: new_map -> previous = current_map;
! 144: current_map = new_map;
! 145: current_map_size++;
! 146: }
! 147:
! 148:
! 149:
! 150: /* Return position of column *c of ith line in */
! 151: /* current file. Adjust *c to be within the line.*/
! 152: /* A 0 pointer is taken as 0 column. */
! 153: /* Returns CORD_NOT_FOUND if i is too big. */
! 154: /* Assumes i > dis_line. */
! 155: size_t line_pos(int i, int *c)
! 156: {
! 157: int j;
! 158: size_t cur;
! 159: size_t next;
! 160: line_map map = current_map;
! 161:
! 162: while (map -> line > i) map = map -> previous;
! 163: if (map -> line < i - 2) /* rebuild */ invalidate_map(i);
! 164: for (j = map -> line, cur = map -> pos; j < i;) {
! 165: cur = CORD_chr(current, cur, '\n');
! 166: if (cur == current_len-1) return(CORD_NOT_FOUND);
! 167: cur++;
! 168: if (++j > current_map -> line) add_map(j, cur);
! 169: }
! 170: if (c != 0) {
! 171: next = CORD_chr(current, cur, '\n');
! 172: if (next == CORD_NOT_FOUND) next = current_len - 1;
! 173: if (next < cur + *c) {
! 174: *c = next - cur;
! 175: }
! 176: cur += *c;
! 177: }
! 178: return(cur);
! 179: }
! 180:
! 181: void add_hist(CORD s)
! 182: {
! 183: history new_file = GC_NEW(struct HistoryRep);
! 184:
! 185: new_file -> file_contents = current = s;
! 186: current_len = CORD_len(s);
! 187: new_file -> previous = now;
! 188: if (now != 0) now -> map = current_map;
! 189: now = new_file;
! 190: }
! 191:
! 192: void del_hist(void)
! 193: {
! 194: now = now -> previous;
! 195: current = now -> file_contents;
! 196: current_map = now -> map;
! 197: current_len = CORD_len(current);
! 198: }
! 199:
! 200: /* Current screen_contents; a dynamically allocated array of CORDs */
! 201: CORD * screen = 0;
! 202: int screen_size = 0;
! 203:
! 204: # ifndef WIN32
! 205: /* Replace a line in the curses stdscr. All control characters are */
! 206: /* displayed as upper case characters in standout mode. This isn't */
! 207: /* terribly appropriate for tabs. */
! 208: void replace_line(int i, CORD s)
! 209: {
! 210: register int c;
! 211: CORD_pos p;
! 212: size_t len = CORD_len(s);
! 213:
! 214: if (screen == 0 || LINES > screen_size) {
! 215: screen_size = LINES;
! 216: screen = (CORD *)GC_MALLOC(screen_size * sizeof(CORD));
! 217: }
! 218: # if !defined(MACINTOSH)
! 219: /* A gross workaround for an apparent curses bug: */
! 220: if (i == LINES-1 && len == COLS) {
! 221: s = CORD_substr(s, 0, CORD_len(s) - 1);
! 222: }
! 223: # endif
! 224: if (CORD_cmp(screen[i], s) != 0) {
! 225: move(i, 0); clrtoeol(); move(i,0);
! 226:
! 227: CORD_FOR (p, s) {
! 228: c = CORD_pos_fetch(p) & 0x7f;
! 229: if (iscntrl(c)) {
! 230: standout(); addch(c + 0x40); standend();
! 231: } else {
! 232: addch(c);
! 233: }
! 234: }
! 235: screen[i] = s;
! 236: }
! 237: }
! 238: #else
! 239: # define replace_line(i,s) invalidate_line(i)
! 240: #endif
! 241:
! 242: /* Return up to COLS characters of the line of s starting at pos, */
! 243: /* returning only characters after the given column. */
! 244: CORD retrieve_line(CORD s, size_t pos, unsigned column)
! 245: {
! 246: CORD candidate = CORD_substr(s, pos, column + COLS);
! 247: /* avoids scanning very long lines */
! 248: int eol = CORD_chr(candidate, 0, '\n');
! 249: int len;
! 250:
! 251: if (eol == CORD_NOT_FOUND) eol = CORD_len(candidate);
! 252: len = (int)eol - (int)column;
! 253: if (len < 0) len = 0;
! 254: return(CORD_substr(s, pos + column, len));
! 255: }
! 256:
! 257: # ifdef WIN32
! 258: # define refresh();
! 259:
! 260: CORD retrieve_screen_line(int i)
! 261: {
! 262: register size_t pos;
! 263:
! 264: invalidate_map(dis_line + LINES); /* Prune search */
! 265: pos = line_pos(dis_line + i, 0);
! 266: if (pos == CORD_NOT_FOUND) return(CORD_EMPTY);
! 267: return(retrieve_line(current, pos, dis_col));
! 268: }
! 269: # endif
! 270:
! 271: /* Display the visible section of the current file */
! 272: void redisplay(void)
! 273: {
! 274: register int i;
! 275:
! 276: invalidate_map(dis_line + LINES); /* Prune search */
! 277: for (i = 0; i < LINES; i++) {
! 278: if (need_redisplay == ALL || need_redisplay == i) {
! 279: register size_t pos = line_pos(dis_line + i, 0);
! 280:
! 281: if (pos == CORD_NOT_FOUND) break;
! 282: replace_line(i, retrieve_line(current, pos, dis_col));
! 283: if (need_redisplay == i) goto done;
! 284: }
! 285: }
! 286: for (; i < LINES; i++) replace_line(i, CORD_EMPTY);
! 287: done:
! 288: refresh();
! 289: need_redisplay = NONE;
! 290: }
! 291:
! 292: int dis_granularity;
! 293:
! 294: /* Update dis_line, dis_col, and dis_pos to make cursor visible. */
! 295: /* Assumes line, col, dis_line, dis_pos are in bounds. */
! 296: void normalize_display()
! 297: {
! 298: int old_line = dis_line;
! 299: int old_col = dis_col;
! 300:
! 301: dis_granularity = 1;
! 302: if (LINES > 15 && COLS > 15) dis_granularity = 2;
! 303: while (dis_line > line) dis_line -= dis_granularity;
! 304: while (dis_col > col) dis_col -= dis_granularity;
! 305: while (line >= dis_line + LINES) dis_line += dis_granularity;
! 306: while (col >= dis_col + COLS) dis_col += dis_granularity;
! 307: if (old_line != dis_line || old_col != dis_col) {
! 308: need_redisplay = ALL;
! 309: }
! 310: }
! 311:
! 312: # if defined(WIN32)
! 313: # elif defined(MACINTOSH)
! 314: # define move_cursor(x,y) cgotoxy(x + 1, y + 1, stdout)
! 315: # else
! 316: # define move_cursor(x,y) move(y,x)
! 317: # endif
! 318:
! 319: /* Adjust display so that cursor is visible; move cursor into position */
! 320: /* Update screen if necessary. */
! 321: void fix_cursor(void)
! 322: {
! 323: normalize_display();
! 324: if (need_redisplay != NONE) redisplay();
! 325: move_cursor(col - dis_col, line - dis_line);
! 326: refresh();
! 327: # ifndef WIN32
! 328: fflush(stdout);
! 329: # endif
! 330: }
! 331:
! 332: /* Make sure line, col, and dis_pos are somewhere inside file. */
! 333: /* Recompute file_pos. Assumes dis_pos is accurate or past eof */
! 334: void fix_pos()
! 335: {
! 336: int my_col = col;
! 337:
! 338: if ((size_t)line > current_len) line = current_len;
! 339: file_pos = line_pos(line, &my_col);
! 340: if (file_pos == CORD_NOT_FOUND) {
! 341: for (line = current_map -> line, file_pos = current_map -> pos;
! 342: file_pos < current_len;
! 343: line++, file_pos = CORD_chr(current, file_pos, '\n') + 1);
! 344: line--;
! 345: file_pos = line_pos(line, &col);
! 346: } else {
! 347: col = my_col;
! 348: }
! 349: }
! 350:
! 351: #if defined(WIN32)
! 352: # define beep() Beep(1000 /* Hz */, 300 /* msecs */)
! 353: #elif defined(MACINTOSH)
! 354: # define beep() SysBeep(1)
! 355: #else
! 356: /*
! 357: * beep() is part of some curses packages and not others.
! 358: * We try to match the type of the builtin one, if any.
! 359: */
! 360: #ifdef __STDC__
! 361: int beep(void)
! 362: #else
! 363: int beep()
! 364: #endif
! 365: {
! 366: putc('\007', stderr);
! 367: return(0);
! 368: }
! 369: #endif
! 370:
! 371: # define NO_PREFIX -1
! 372: # define BARE_PREFIX -2
! 373: int repeat_count = NO_PREFIX; /* Current command prefix. */
! 374:
! 375: int locate_mode = 0; /* Currently between 2 ^Ls */
! 376: CORD locate_string = CORD_EMPTY; /* Current search string. */
! 377:
! 378: char * arg_file_name;
! 379:
! 380: #ifdef WIN32
! 381: /* Change the current position to whatever is currently displayed at */
! 382: /* the given SCREEN coordinates. */
! 383: void set_position(int c, int l)
! 384: {
! 385: line = l + dis_line;
! 386: col = c + dis_col;
! 387: fix_pos();
! 388: move_cursor(col - dis_col, line - dis_line);
! 389: }
! 390: #endif /* WIN32 */
! 391:
! 392: /* Perform the command associated with character c. C may be an */
! 393: /* integer > 256 denoting a windows command, one of the above control */
! 394: /* characters, or another ASCII character to be used as either a */
! 395: /* character to be inserted, a repeat count, or a search string, */
! 396: /* depending on the current state. */
! 397: void do_command(int c)
! 398: {
! 399: int i;
! 400: int need_fix_pos;
! 401: FILE * out;
! 402:
! 403: if ( c == '\r') c = '\n';
! 404: if (locate_mode) {
! 405: size_t new_pos;
! 406:
! 407: if (c == LOCATE) {
! 408: locate_mode = 0;
! 409: locate_string = CORD_EMPTY;
! 410: return;
! 411: }
! 412: locate_string = CORD_cat_char(locate_string, (char)c);
! 413: new_pos = CORD_str(current, file_pos - CORD_len(locate_string) + 1,
! 414: locate_string);
! 415: if (new_pos != CORD_NOT_FOUND) {
! 416: need_redisplay = ALL;
! 417: new_pos += CORD_len(locate_string);
! 418: for (;;) {
! 419: file_pos = line_pos(line + 1, 0);
! 420: if (file_pos > new_pos) break;
! 421: line++;
! 422: }
! 423: col = new_pos - line_pos(line, 0);
! 424: file_pos = new_pos;
! 425: fix_cursor();
! 426: } else {
! 427: locate_string = CORD_substr(locate_string, 0,
! 428: CORD_len(locate_string) - 1);
! 429: beep();
! 430: }
! 431: return;
! 432: }
! 433: if (c == REPEAT) {
! 434: repeat_count = BARE_PREFIX; return;
! 435: } else if (c < 0x100 && isdigit(c)){
! 436: if (repeat_count == BARE_PREFIX) {
! 437: repeat_count = c - '0'; return;
! 438: } else if (repeat_count != NO_PREFIX) {
! 439: repeat_count = 10 * repeat_count + c - '0'; return;
! 440: }
! 441: }
! 442: if (repeat_count == NO_PREFIX) repeat_count = 1;
! 443: if (repeat_count == BARE_PREFIX && (c == UP || c == DOWN)) {
! 444: repeat_count = LINES - dis_granularity;
! 445: }
! 446: if (repeat_count == BARE_PREFIX) repeat_count = 8;
! 447: need_fix_pos = 0;
! 448: for (i = 0; i < repeat_count; i++) {
! 449: switch(c) {
! 450: case LOCATE:
! 451: locate_mode = 1;
! 452: break;
! 453: case TOP:
! 454: line = col = file_pos = 0;
! 455: break;
! 456: case UP:
! 457: if (line != 0) {
! 458: line--;
! 459: need_fix_pos = 1;
! 460: }
! 461: break;
! 462: case DOWN:
! 463: line++;
! 464: need_fix_pos = 1;
! 465: break;
! 466: case LEFT:
! 467: if (col != 0) {
! 468: col--; file_pos--;
! 469: }
! 470: break;
! 471: case RIGHT:
! 472: if (CORD_fetch(current, file_pos) == '\n') break;
! 473: col++; file_pos++;
! 474: break;
! 475: case UNDO:
! 476: del_hist();
! 477: need_redisplay = ALL; need_fix_pos = 1;
! 478: break;
! 479: case BS:
! 480: if (col == 0) {
! 481: beep();
! 482: break;
! 483: }
! 484: col--; file_pos--;
! 485: /* fall through: */
! 486: case DEL:
! 487: if (file_pos == current_len-1) break;
! 488: /* Can't delete trailing newline */
! 489: if (CORD_fetch(current, file_pos) == '\n') {
! 490: need_redisplay = ALL; need_fix_pos = 1;
! 491: } else {
! 492: need_redisplay = line - dis_line;
! 493: }
! 494: add_hist(CORD_cat(
! 495: CORD_substr(current, 0, file_pos),
! 496: CORD_substr(current, file_pos+1, current_len)));
! 497: invalidate_map(line);
! 498: break;
! 499: case WRITE:
! 500: {
! 501: CORD name = CORD_cat(CORD_from_char_star(arg_file_name),
! 502: ".new");
! 503:
! 504: if ((out = fopen(CORD_to_const_char_star(name), "wb")) == NULL
! 505: || CORD_put(current, out) == EOF) {
! 506: de_error("Write failed\n");
! 507: need_redisplay = ALL;
! 508: } else {
! 509: fclose(out);
! 510: }
! 511: }
! 512: break;
! 513: default:
! 514: {
! 515: CORD left_part = CORD_substr(current, 0, file_pos);
! 516: CORD right_part = CORD_substr(current, file_pos, current_len);
! 517:
! 518: add_hist(CORD_cat(CORD_cat_char(left_part, (char)c),
! 519: right_part));
! 520: invalidate_map(line);
! 521: if (c == '\n') {
! 522: col = 0; line++; file_pos++;
! 523: need_redisplay = ALL;
! 524: } else {
! 525: col++; file_pos++;
! 526: need_redisplay = line - dis_line;
! 527: }
! 528: break;
! 529: }
! 530: }
! 531: }
! 532: if (need_fix_pos) fix_pos();
! 533: fix_cursor();
! 534: repeat_count = NO_PREFIX;
! 535: }
! 536:
! 537: /* OS independent initialization */
! 538:
! 539: void generic_init(void)
! 540: {
! 541: FILE * f;
! 542: CORD initial;
! 543:
! 544: if ((f = fopen(arg_file_name, "rb")) == NULL) {
! 545: initial = "\n";
! 546: } else {
! 547: initial = CORD_from_file(f);
! 548: if (initial == CORD_EMPTY
! 549: || CORD_fetch(initial, CORD_len(initial)-1) != '\n') {
! 550: initial = CORD_cat(initial, "\n");
! 551: }
! 552: }
! 553: add_map(0,0);
! 554: add_hist(initial);
! 555: now -> map = current_map;
! 556: now -> previous = now; /* Can't back up further: beginning of the world */
! 557: need_redisplay = ALL;
! 558: fix_cursor();
! 559: }
! 560:
! 561: #ifndef WIN32
! 562:
! 563: main(argc, argv)
! 564: int argc;
! 565: char ** argv;
! 566: {
! 567: int c;
! 568:
! 569: #if defined(MACINTOSH)
! 570: console_options.title = "\pDumb Editor";
! 571: cshow(stdout);
! 572: GC_init();
! 573: argc = ccommand(&argv);
! 574: #endif
! 575:
! 576: if (argc != 2) goto usage;
! 577: arg_file_name = argv[1];
! 578: setvbuf(stdout, GC_MALLOC_ATOMIC(8192), _IOFBF, 8192);
! 579: initscr();
! 580: noecho(); nonl(); cbreak();
! 581: generic_init();
! 582: while ((c = getchar()) != QUIT) {
! 583: if (c == EOF) break;
! 584: do_command(c);
! 585: }
! 586: done:
! 587: move(LINES-1, 0);
! 588: clrtoeol();
! 589: refresh();
! 590: nl();
! 591: echo();
! 592: endwin();
! 593: exit(0);
! 594: usage:
! 595: fprintf(stderr, "Usage: %s file\n", argv[0]);
! 596: fprintf(stderr, "Cursor keys: ^B(left) ^F(right) ^P(up) ^N(down)\n");
! 597: fprintf(stderr, "Undo: ^U Write to <file>.new: ^W");
! 598: fprintf(stderr, "Quit:^D Repeat count: ^R[n]\n");
! 599: fprintf(stderr, "Top: ^T Locate (search, find): ^L text ^L\n");
! 600: exit(1);
! 601: }
! 602:
! 603: #endif /* !WIN32 */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>