]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/ee/ee.c
zfs: merge openzfs/zfs@4a1195ca5 (master) into main
[FreeBSD/FreeBSD.git] / contrib / ee / ee.c
1 /*
2  |      ee (easy editor)
3  |
4  |      An easy to use, simple screen oriented editor.
5  |
6  |      written by Hugh Mahon
7  |
8  |
9  |      Copyright (c) 2009, Hugh Mahon
10  |      All rights reserved.
11  |      
12  |      Redistribution and use in source and binary forms, with or without
13  |      modification, are permitted provided that the following conditions
14  |      are met:
15  |      
16  |          * Redistributions of source code must retain the above copyright
17  |            notice, this list of conditions and the following disclaimer.
18  |          * Redistributions in binary form must reproduce the above
19  |            copyright notice, this list of conditions and the following
20  |            disclaimer in the documentation and/or other materials provided
21  |            with the distribution.
22  |      
23  |      THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24  |      "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
25  |      LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
26  |      FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
27  |      COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
28  |      INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
29  |      BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
30  |      LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
31  |      CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32  |      LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
33  |      ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
34  |      POSSIBILITY OF SUCH DAMAGE.
35  |
36  |     -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
37  |
38  |      This editor was purposely developed to be simple, both in 
39  |      interface and implementation.  This editor was developed to 
40  |      address a specific audience: the user who is new to computers 
41  |      (especially UNIX).
42  |      
43  |      ee is not aimed at technical users; for that reason more 
44  |      complex features were intentionally left out.  In addition, 
45  |      ee is intended to be compiled by people with little computer 
46  |      experience, which means that it needs to be small, relatively 
47  |      simple in implementation, and portable.
48  |
49  |      This software and documentation contains
50  |      proprietary information which is protected by
51  |      copyright.  All rights are reserved.
52  |
53  |      $Header: /home/hugh/sources/old_ae/RCS/ee.c,v 1.104 2010/06/04 01:55:31 hugh Exp hugh $
54  |
55  */
56
57 #include <sys/cdefs.h>
58 __FBSDID("$FreeBSD$");
59
60 char *ee_copyright_message = 
61 "Copyright (c) 1986, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 2009 Hugh Mahon ";
62
63 #include "ee_version.h"
64
65 char *version = "@(#) ee, version "  EE_VERSION  " $Revision: 1.104 $";
66
67 #ifdef NCURSE
68 #include "new_curse.h"
69 #elif HAS_NCURSES
70 #include <ncurses.h>
71 #else
72 #include <curses.h>
73 #endif
74
75 #include <ctype.h>
76 #include <signal.h>
77 #include <fcntl.h>
78 #include <sys/types.h>
79 #include <sys/stat.h>
80 #include <errno.h>
81 #include <string.h>
82 #include <pwd.h>
83 #include <locale.h>
84
85 #ifdef HAS_SYS_WAIT
86 #include <sys/wait.h>
87 #endif
88
89 #ifdef HAS_STDLIB
90 #include <stdlib.h>
91 #endif
92
93 #ifdef HAS_STDARG
94 #include <stdarg.h>
95 #endif
96
97 #ifdef HAS_UNISTD
98 #include <unistd.h>
99 #endif
100
101 #ifndef NO_CATGETS
102 #include <nl_types.h>
103
104 nl_catd catalog;
105 #else
106 #define catgetlocal(a, b) (b)
107 #endif /* NO_CATGETS */
108
109 #ifndef SIGCHLD
110 #define SIGCHLD SIGCLD
111 #endif
112
113 #define TAB 9
114 #define max(a, b)       (a > b ? a : b)
115 #define min(a, b)       (a < b ? a : b)
116
117 /*
118  |      defines for type of data to show in info window
119  */
120
121 #define CONTROL_KEYS 1
122 #define COMMANDS     2
123
124 struct text {
125         unsigned char *line;            /* line of characters           */
126         int line_number;                /* line number                  */
127         int line_length;        /* actual number of characters in the line */
128         int max_length; /* maximum number of characters the line handles */
129         struct text *next_line;         /* next line of text            */
130         struct text *prev_line;         /* previous line of text        */
131         };
132
133 struct text *first_line;        /* first line of current buffer         */
134 struct text *dlt_line;          /* structure for info on deleted line   */
135 struct text *curr_line;         /* current line cursor is on            */
136 struct text *tmp_line;          /* temporary line pointer               */
137 struct text *srch_line;         /* temporary pointer for search routine */
138
139 struct files {          /* structure to store names of files to be edited*/
140         unsigned char *name;            /* name of file                         */
141         struct files *next_name;
142         };
143
144 struct files *top_of_stack = NULL;
145
146 int d_wrd_len;                  /* length of deleted word               */
147 int position;                   /* offset in bytes from begin of line   */
148 int scr_pos;                    /* horizontal position                  */
149 int scr_vert;                   /* vertical position on screen          */
150 int scr_horz;                   /* horizontal position on screen        */
151 int absolute_lin;               /* number of lines from top             */
152 int tmp_vert, tmp_horz;
153 int input_file;                 /* indicate to read input file          */
154 int recv_file;                  /* indicate reading a file              */
155 int edit;                       /* continue executing while true        */
156 int gold;                       /* 'gold' function key pressed          */
157 int fildes;                     /* file descriptor                      */
158 int case_sen;                   /* case sensitive search flag           */
159 int last_line;                  /* last line for text display           */
160 int last_col;                   /* last column for text display         */
161 int horiz_offset = 0;           /* offset from left edge of text        */
162 int clear_com_win;              /* flag to indicate com_win needs clearing */
163 int text_changes = FALSE;       /* indicate changes have been made to text */
164 int get_fd;                     /* file descriptor for reading a file   */
165 int info_window = TRUE;         /* flag to indicate if help window visible */
166 int info_type = CONTROL_KEYS;   /* flag to indicate type of info to display */
167 int expand_tabs = TRUE;         /* flag for expanding tabs              */
168 int right_margin = 0;           /* the right margin                     */
169 int observ_margins = TRUE;      /* flag for whether margins are observed */
170 int shell_fork;
171 int temp_stdin;                 /* temporary storage for stdin          */
172 int temp_stdout;                /* temp storage for stdout descriptor   */
173 int temp_stderr;                /* temp storage for stderr descriptor   */
174 int pipe_out[2];                /* pipe file desc for output            */
175 int pipe_in[2];                 /* pipe file descriptors for input      */
176 int out_pipe;                   /* flag that info is piped out          */
177 int in_pipe;                    /* flag that info is piped in           */
178 int formatted = FALSE;          /* flag indicating paragraph formatted  */
179 int auto_format = FALSE;        /* flag for auto_format mode            */
180 int restricted = FALSE;         /* flag to indicate restricted mode     */
181 int nohighlight = FALSE;        /* turns off highlighting               */
182 int eightbit = TRUE;            /* eight bit character flag             */
183 int local_LINES = 0;            /* copy of LINES, to detect when win resizes */
184 int local_COLS = 0;             /* copy of COLS, to detect when win resizes  */
185 int curses_initialized = FALSE; /* flag indicating if curses has been started*/
186 int emacs_keys_mode = FALSE;    /* mode for if emacs key binings are used    */
187 int ee_chinese = FALSE;         /* allows handling of multi-byte characters  */
188                                 /* by checking for high bit in a byte the    */
189                                 /* code recognizes a two-byte character      */
190                                 /* sequence                                  */
191
192 unsigned char *point;           /* points to current position in line   */
193 unsigned char *srch_str;        /* pointer for search string            */
194 unsigned char *u_srch_str;      /* pointer to non-case sensitive search */
195 unsigned char *srch_1;          /* pointer to start of suspect string   */
196 unsigned char *srch_2;          /* pointer to next character of string  */
197 unsigned char *srch_3;
198 unsigned char *in_file_name = NULL;     /* name of input file           */
199 char *tmp_file; /* temporary file name                  */
200 unsigned char *d_char;          /* deleted character                    */
201 unsigned char *d_word;          /* deleted word                         */
202 unsigned char *d_line;          /* deleted line                         */
203 char in_string[513];    /* buffer for reading a file            */
204 unsigned char *print_command = (unsigned char *)"lpr";  /* string to use for the print command  */
205 unsigned char *start_at_line = NULL;    /* move to this line at start of session*/
206 int in;                         /* input character                      */
207
208 FILE *temp_fp;                  /* temporary file pointer               */
209 FILE *bit_bucket;               /* file pointer to /dev/null            */
210
211 char *table[] = { 
212         "^@", "^A", "^B", "^C", "^D", "^E", "^F", "^G", "^H", "\t", "^J", 
213         "^K", "^L", "^M", "^N", "^O", "^P", "^Q", "^R", "^S", "^T", "^U", 
214         "^V", "^W", "^X", "^Y", "^Z", "^[", "^\\", "^]", "^^", "^_"
215         };
216
217 WINDOW *com_win;
218 WINDOW *text_win;
219 WINDOW *help_win;
220 WINDOW *info_win;
221
222 #if defined(__STDC__) || defined(__cplusplus)
223 #define P_(s) s
224 #else
225 #define P_(s) ()
226 #endif
227
228
229 /*
230  |      The following structure allows menu items to be flexibly declared.
231  |      The first item is the string describing the selection, the second 
232  |      is the address of the procedure to call when the item is selected,
233  |      and the third is the argument for the procedure.
234  |
235  |      For those systems with i18n, the string should be accompanied by a
236  |      catalog number.  The 'int *' should be replaced with 'void *' on 
237  |      systems with that type.
238  |
239  |      The first menu item will be the title of the menu, with NULL 
240  |      parameters for the procedure and argument, followed by the menu items.
241  |
242  |      If the procedure value is NULL, the menu item is displayed, but no 
243  |      procedure is called when the item is selected.  The number of the 
244  |      item will be returned.  If the third (argument) parameter is -1, no 
245  |      argument is given to the procedure when it is called.
246  */
247
248 struct menu_entries {
249         char *item_string;
250         int (*procedure)P_((struct menu_entries *));
251         struct menu_entries *ptr_argument;
252         int (*iprocedure)P_((int));
253         void (*nprocedure)P_((void));
254         int argument;
255         };
256
257 int main P_((int argc, char *argv[]));
258 unsigned char *resiz_line P_((int factor, struct text *rline, int rpos));
259 void insert P_((int character));
260 void delete P_((int disp));
261 void scanline P_((unsigned char *pos));
262 int tabshift P_((int temp_int));
263 int out_char P_((WINDOW *window, int character, int column));
264 int len_char P_((int character, int column));
265 void draw_line P_((int vertical, int horiz, unsigned char *ptr, int t_pos, int length));
266 void insert_line P_((int disp));
267 struct text *txtalloc P_((void));
268 struct files *name_alloc P_((void));
269 unsigned char *next_word P_((unsigned char *string));
270 void prev_word P_((void));
271 void control P_((void));
272 void emacs_control P_((void));
273 void bottom P_((void));
274 void top P_((void));
275 void nextline P_((void));
276 void prevline P_((void));
277 void left P_((int disp));
278 void right P_((int disp));
279 void find_pos P_((void));
280 void up P_((void));
281 void down P_((void));
282 void function_key P_((void));
283 void print_buffer P_((void));
284 void command_prompt P_((void));
285 void command P_((char *cmd_str1));
286 int scan P_((char *line, int offset, int column));
287 char *get_string P_((char *prompt, int advance));
288 int compare P_((char *string1, char *string2, int sensitive));
289 void goto_line P_((char *cmd_str));
290 void midscreen P_((int line, unsigned char *pnt));
291 void get_options P_((int numargs, char *arguments[]));
292 void check_fp P_((void));
293 void get_file P_((char *file_name));
294 void get_line P_((int length, unsigned char *in_string, int *append));
295 void draw_screen P_((void));
296 void finish P_((void));
297 int quit P_((int noverify));
298 void edit_abort P_((int arg));
299 void delete_text P_((void));
300 int write_file P_((char *file_name, int warn_if_exists));
301 int search P_((int display_message));
302 void search_prompt P_((void));
303 void del_char P_((void));
304 void undel_char P_((void));
305 void del_word P_((void));
306 void undel_word P_((void));
307 void del_line P_((void));
308 void undel_line P_((void));
309 void adv_word P_((void));
310 void move_rel P_((int direction, int lines));
311 void eol P_((void));
312 void bol P_((void));
313 void adv_line P_((void));
314 void sh_command P_((char *string));
315 void set_up_term P_((void));
316 void resize_check P_((void));
317 int menu_op P_((struct menu_entries *));
318 void paint_menu P_((struct menu_entries menu_list[], int max_width, int max_height, int list_size, int top_offset, WINDOW *menu_win, int off_start, int vert_size));
319 void help P_((void));
320 void paint_info_win P_((void));
321 void no_info_window P_((void));
322 void create_info_window P_((void));
323 int file_op P_((int arg));
324 void shell_op P_((void));
325 void leave_op P_((void));
326 void redraw P_((void));
327 int Blank_Line P_((struct text *test_line));
328 void Format P_((void));
329 void ee_init P_((void));
330 void dump_ee_conf P_((void));
331 void echo_string P_((char *string));
332 void spell_op P_((void));
333 void ispell_op P_((void));
334 int first_word_len P_((struct text *test_line));
335 void Auto_Format P_((void));
336 void modes_op P_((void));
337 char *is_in_string P_((char *string, char *substring));
338 char *resolve_name P_((char *name));
339 int restrict_mode P_((void));
340 int unique_test P_((char *string, char *list[]));
341 void strings_init P_((void));
342
343 #undef P_
344 /*
345  |      allocate space here for the strings that will be in the menu
346  */
347
348 struct menu_entries modes_menu[] = {
349         {"", NULL, NULL, NULL, NULL, 0},        /* title                */
350         {"", NULL, NULL, NULL, NULL, -1},       /* 1. tabs to spaces    */
351         {"", NULL, NULL, NULL, NULL, -1},       /* 2. case sensitive search*/
352         {"", NULL, NULL, NULL, NULL, -1},       /* 3. margins observed  */
353         {"", NULL, NULL, NULL, NULL, -1},       /* 4. auto-paragraph    */
354         {"", NULL, NULL, NULL, NULL, -1},       /* 5. eightbit characters*/
355         {"", NULL, NULL, NULL, NULL, -1},       /* 6. info window       */
356         {"", NULL, NULL, NULL, NULL, -1},       /* 7. emacs key bindings*/
357         {"", NULL, NULL, NULL, NULL, -1},       /* 8. right margin      */
358         {"", NULL, NULL, NULL, NULL, -1},       /* 9. chinese text      */
359         {"", NULL, NULL, NULL, dump_ee_conf, -1}, /* 10. save editor config */
360         {NULL, NULL, NULL, NULL, NULL, -1}      /* terminator           */
361         };
362
363 char *mode_strings[11]; 
364
365 #define NUM_MODES_ITEMS 10
366
367 struct menu_entries config_dump_menu[] = {
368         {"", NULL, NULL, NULL, NULL, 0}, 
369         {"", NULL, NULL, NULL, NULL, -1},
370         {"", NULL, NULL, NULL, NULL, -1},
371         {NULL, NULL, NULL, NULL, NULL, -1}
372         };
373
374 struct menu_entries leave_menu[] = {
375         {"", NULL, NULL, NULL, NULL, -1}, 
376         {"", NULL, NULL, NULL, finish, -1}, 
377         {"", NULL, NULL, quit, NULL, TRUE}, 
378         {NULL, NULL, NULL, NULL, NULL, -1}
379         };
380
381 #define READ_FILE 1
382 #define WRITE_FILE 2
383 #define SAVE_FILE 3
384
385 struct menu_entries file_menu[] = {
386         {"", NULL, NULL, NULL, NULL, -1},
387         {"", NULL, NULL, file_op, NULL, READ_FILE},
388         {"", NULL, NULL, file_op, NULL, WRITE_FILE},
389         {"", NULL, NULL, file_op, NULL, SAVE_FILE},
390         {"", NULL, NULL, NULL, print_buffer, -1},
391         {NULL, NULL, NULL, NULL, NULL, -1}
392         };
393
394 struct menu_entries search_menu[] = {
395         {"", NULL, NULL, NULL, NULL, 0}, 
396         {"", NULL, NULL, NULL, search_prompt, -1},
397         {"", NULL, NULL, search, NULL, TRUE},
398         {NULL, NULL, NULL, NULL, NULL, -1}
399         };
400
401 struct menu_entries spell_menu[] = {
402         {"", NULL, NULL, NULL, NULL, -1}, 
403         {"", NULL, NULL, NULL, spell_op, -1},
404         {"", NULL, NULL, NULL, ispell_op, -1},
405         {NULL, NULL, NULL, NULL, NULL, -1}
406         };
407
408 struct menu_entries misc_menu[] = {
409         {"", NULL, NULL, NULL, NULL, -1}, 
410         {"", NULL, NULL, NULL, Format, -1},
411         {"", NULL, NULL, NULL, shell_op, -1}, 
412         {"", menu_op, spell_menu, NULL, NULL, -1}, 
413         {NULL, NULL, NULL, NULL, NULL, -1}
414         };
415
416 struct menu_entries main_menu[] = {
417         {"", NULL, NULL, NULL, NULL, -1}, 
418         {"", NULL, NULL, NULL, leave_op, -1}, 
419         {"", NULL, NULL, NULL, help, -1},
420         {"", menu_op, file_menu, NULL, NULL, -1}, 
421         {"", NULL, NULL, NULL, redraw, -1}, 
422         {"", NULL, NULL, NULL, modes_op, -1}, 
423         {"", menu_op, search_menu, NULL, NULL, -1}, 
424         {"", menu_op, misc_menu, NULL, NULL, -1}, 
425         {NULL, NULL, NULL, NULL, NULL, -1}
426         };
427
428 char *help_text[23];
429 char *control_keys[5];
430
431 char *emacs_help_text[22];
432 char *emacs_control_keys[5];
433
434 char *command_strings[5];
435 char *commands[32];
436 char *init_strings[22];
437
438 #define MENU_WARN 1
439
440 #define max_alpha_char 36
441
442 /*
443  |      Declarations for strings for localization
444  */
445
446 char *com_win_message;          /* to be shown in com_win if no info window */
447 char *no_file_string;
448 char *ascii_code_str;
449 char *printer_msg_str;
450 char *command_str;
451 char *file_write_prompt_str;
452 char *file_read_prompt_str;
453 char *char_str;
454 char *unkn_cmd_str;
455 char *non_unique_cmd_msg;
456 char *line_num_str;
457 char *line_len_str;
458 char *current_file_str;
459 char *usage0;
460 char *usage1;
461 char *usage2;
462 char *usage3;
463 char *usage4;
464 char *file_is_dir_msg;
465 char *new_file_msg;
466 char *cant_open_msg;
467 char *open_file_msg;
468 char *file_read_fin_msg;
469 char *reading_file_msg;
470 char *read_only_msg;
471 char *file_read_lines_msg;
472 char *save_file_name_prompt;
473 char *file_not_saved_msg;
474 char *changes_made_prompt;
475 char *yes_char;
476 char *file_exists_prompt;
477 char *create_file_fail_msg;
478 char *writing_file_msg;
479 char *file_written_msg;
480 char *searching_msg;
481 char *str_not_found_msg;
482 char *search_prompt_str;
483 char *exec_err_msg;
484 char *continue_msg;
485 char *menu_cancel_msg;
486 char *menu_size_err_msg;
487 char *press_any_key_msg;
488 char *shell_prompt;
489 char *formatting_msg;
490 char *shell_echo_msg;
491 char *spell_in_prog_msg;
492 char *margin_prompt;
493 char *restricted_msg;
494 char *ON;
495 char *OFF;
496 char *HELP;
497 char *WRITE;
498 char *READ;
499 char *LINE;
500 char *FILE_str;
501 char *CHARACTER;
502 char *REDRAW;
503 char *RESEQUENCE;
504 char *AUTHOR;
505 char *VERSION;
506 char *CASE;
507 char *NOCASE;
508 char *EXPAND;
509 char *NOEXPAND;
510 char *Exit_string;
511 char *QUIT_string;
512 char *INFO;
513 char *NOINFO;
514 char *MARGINS;
515 char *NOMARGINS;
516 char *AUTOFORMAT;
517 char *NOAUTOFORMAT;
518 char *Echo;
519 char *PRINTCOMMAND;
520 char *RIGHTMARGIN;
521 char *HIGHLIGHT;
522 char *NOHIGHLIGHT;
523 char *EIGHTBIT;
524 char *NOEIGHTBIT;
525 char *EMACS_string;
526 char *NOEMACS_string;
527 char *conf_dump_err_msg;
528 char *conf_dump_success_msg;
529 char *conf_not_saved_msg;
530 char *ree_no_file_msg;
531 char *cancel_string;
532 char *menu_too_lrg_msg;
533 char *more_above_str, *more_below_str;
534 char *separator = "===============================================================================";
535
536 char *chinese_cmd, *nochinese_cmd;
537
538 #ifndef __STDC__
539 #ifndef HAS_STDLIB
540 extern char *malloc();
541 extern char *realloc();
542 extern char *getenv();
543 FILE *fopen();                  /* declaration for open function        */
544 #endif /* HAS_STDLIB */
545 #endif /* __STDC__ */
546
547 int
548 main(argc, argv)                /* beginning of main program            */
549 int argc;
550 char *argv[];
551 {
552         int counter;
553
554         for (counter = 1; counter < 24; counter++)
555                 signal(counter, SIG_IGN);
556
557         /* Always read from (and write to) a terminal. */
558         if (!isatty(STDIN_FILENO) || !isatty(STDOUT_FILENO)) {
559                 fprintf(stderr,
560                     "ee's standard input and output must be a terminal\n");
561                 exit(1);
562         }
563
564         signal(SIGCHLD, SIG_DFL);
565         signal(SIGSEGV, SIG_DFL);
566         signal(SIGINT, edit_abort);
567         d_char = malloc(3);     /* provide a buffer for multi-byte chars */
568         d_word = malloc(150);
569         *d_word = '\0';
570         d_line = NULL;
571         dlt_line = txtalloc();
572         dlt_line->line = d_line;
573         dlt_line->line_length = 0;
574         curr_line = first_line = txtalloc();
575         curr_line->line = point = malloc(10);
576         curr_line->line_length = 1;
577         curr_line->max_length = 10;
578         curr_line->prev_line = NULL;
579         curr_line->next_line = NULL;
580         curr_line->line_number  = 1;
581         srch_str = NULL;
582         u_srch_str = NULL;
583         position = 1;
584         scr_pos =0;
585         scr_vert = 0;
586         scr_horz = 0;
587         absolute_lin = 1;
588         bit_bucket = fopen("/dev/null", "w");
589         edit = TRUE;
590         gold = case_sen = FALSE;
591         shell_fork = TRUE;
592         strings_init();
593         ee_init();
594         if (argc > 0 )
595                 get_options(argc, argv);
596         set_up_term();
597         if (right_margin == 0)
598                 right_margin = COLS - 1;
599         if (top_of_stack == NULL)
600         {
601                 if (restrict_mode())
602                 {
603                         wmove(com_win, 0, 0);
604                         werase(com_win);
605                         wprintw(com_win, ree_no_file_msg);
606                         wrefresh(com_win);
607                         edit_abort(0);
608                 }
609                 wprintw(com_win, no_file_string);
610                 wrefresh(com_win);
611         }
612         else
613                 check_fp();
614
615         clear_com_win = TRUE;
616
617         counter = 0;
618
619         while(edit) 
620         {
621                 /*
622                  |  display line and column information
623                  */
624                 if (info_window)
625                 {
626                         if (!nohighlight)
627                                 wstandout(info_win);
628                         wmove(info_win, 5, 0);
629                         wprintw(info_win, separator);
630                         wmove(info_win, 5, 5);
631                         wprintw(info_win, "line %d col %d lines from top %d ", 
632                                   curr_line->line_number, scr_horz, absolute_lin);
633                         wstandend(info_win);
634                         wrefresh(info_win);
635                 }
636
637                 wrefresh(text_win);
638                 in = wgetch(text_win);
639                 if (in == -1)
640                         exit(0);  /* without this exit ee will go into an 
641                                      infinite loop if the network 
642                                      session detaches */
643
644                 resize_check();
645
646                 if (clear_com_win)
647                 {
648                         clear_com_win = FALSE;
649                         wmove(com_win, 0, 0);
650                         werase(com_win);
651                         if (!info_window)
652                         {
653                                 wprintw(com_win, "%s", com_win_message);
654                         }
655                         wrefresh(com_win);
656                 }
657
658                 if (in > 255)
659                         function_key();
660                 else if ((in == '\10') || (in == 127))
661                 {
662                         in = 8;         /* make sure key is set to backspace */
663                         delete(TRUE);
664                 }
665                 else if ((in > 31) || (in == 9))
666                         insert(in);
667                 else if ((in >= 0) && (in <= 31))
668                 {
669                         if (emacs_keys_mode)
670                                 emacs_control();
671                         else
672                                 control();
673                 }
674         }
675         return(0);
676 }
677
678 unsigned char *
679 resiz_line(factor, rline, rpos) /* resize the line to length + factor*/
680 int factor;             /* resize factor                                */
681 struct text *rline;     /* position in line                             */
682 int rpos;
683 {
684         unsigned char *rpoint;
685         int resiz_var;
686  
687         rline->max_length += factor;
688         rpoint = rline->line = realloc(rline->line, rline->max_length );
689         for (resiz_var = 1 ; (resiz_var < rpos) ; resiz_var++)
690                 rpoint++;
691         return(rpoint);
692 }
693
694 void 
695 insert(character)               /* insert character into line           */
696 int character;                  /* new character                        */
697 {
698         int counter;
699         int value;
700         unsigned char *temp;    /* temporary pointer                    */
701         unsigned char *temp2;   /* temporary pointer                    */
702
703         if ((character == '\011') && (expand_tabs))
704         {
705                 counter = len_char('\011', scr_horz);
706                 for (; counter > 0; counter--)
707                         insert(' ');
708                 if (auto_format)
709                         Auto_Format();
710                 return;
711         }
712         text_changes = TRUE;
713         if ((curr_line->max_length - curr_line->line_length) < 5)
714                 point = resiz_line(10, curr_line, position);
715         curr_line->line_length++;
716         temp = point;
717         counter = position;
718         while (counter < curr_line->line_length)        /* find end of line */
719         {
720                 counter++;
721                 temp++;
722         }
723         temp++;                 /* increase length of line by one       */
724         while (point < temp)
725         {
726                 temp2=temp - 1;
727                 *temp= *temp2;  /* shift characters over by one         */
728                 temp--;
729         }
730         *point = character;     /* insert new character                 */
731         wclrtoeol(text_win);
732         if (!isprint((unsigned char)character)) /* check for TAB character*/
733         {
734                 scr_pos = scr_horz += out_char(text_win, character, scr_horz);
735                 point++;
736                 position++;
737         }
738         else
739         {
740                 waddch(text_win, (unsigned char)character);
741                 scr_pos = ++scr_horz;
742                 point++;
743                 position ++;
744         }
745
746         if ((observ_margins) && (right_margin < scr_pos))
747         {
748                 counter = position;
749                 while (scr_pos > right_margin)
750                         prev_word();
751                 if (scr_pos == 0)
752                 {
753                         while (position < counter)
754                                 right(TRUE);
755                 }
756                 else
757                 {
758                         counter -= position;
759                         insert_line(TRUE);
760                         for (value = 0; value < counter; value++)
761                                 right(TRUE);
762                 }
763         }
764
765         if ((scr_horz - horiz_offset) > last_col)
766         {
767                 horiz_offset += 8;
768                 midscreen(scr_vert, point);
769         }
770
771         if ((auto_format) && (character == ' ') && (!formatted))
772                 Auto_Format();
773         else if ((character != ' ') && (character != '\t'))
774                 formatted = FALSE;
775
776         draw_line(scr_vert, scr_horz, point, position, curr_line->line_length);
777 }
778
779 void 
780 delete(disp)                    /* delete character             */
781 int disp;
782 {
783         unsigned char *tp;
784         unsigned char *temp2;
785         struct text *temp_buff;
786         int temp_vert;
787         int temp_pos;
788         int del_width = 1;
789
790         if (point != curr_line->line)   /* if not at beginning of line  */
791         {
792                 text_changes = TRUE;
793                 temp2 = tp = point;
794                 if ((ee_chinese) && (position >= 2) && (*(point - 2) > 127))
795                 {
796                         del_width = 2;
797                 }
798                 tp -= del_width;
799                 point -= del_width;
800                 position -= del_width;
801                 temp_pos = position;
802                 curr_line->line_length -= del_width;
803                 if ((*tp < ' ') || (*tp >= 127))        /* check for TAB */
804                         scanline(tp);
805                 else
806                         scr_horz -= del_width;
807                 scr_pos = scr_horz;
808                 if (in == 8)
809                 {
810                         if (del_width == 1)
811                                 *d_char = *point; /* save deleted character  */
812                         else
813                         {
814                                 d_char[0] = *point;
815                                 d_char[1] = *(point + 1);
816                         }
817                         d_char[del_width] = '\0';
818                 }
819                 while (temp_pos <= curr_line->line_length)
820                 {
821                         temp_pos++;
822                         *tp = *temp2;
823                         tp++;
824                         temp2++;
825                 }
826                 if ((scr_horz < horiz_offset) && (horiz_offset > 0))
827                 {
828                         horiz_offset -= 8;
829                         midscreen(scr_vert, point);
830                 }
831         }
832         else if (curr_line->prev_line != NULL)
833         {
834                 text_changes = TRUE;
835                 left(disp);                     /* go to previous line  */
836                 temp_buff = curr_line->next_line;
837                 point = resiz_line(temp_buff->line_length, curr_line, position);
838                 if (temp_buff->next_line != NULL)
839                         temp_buff->next_line->prev_line = curr_line;
840                 curr_line->next_line = temp_buff->next_line;
841                 temp2 = temp_buff->line;
842                 if (in == 8)
843                 {
844                         d_char[0] = '\n';
845                         d_char[1] = '\0';
846                 }
847                 tp = point;
848                 temp_pos = 1;
849                 while (temp_pos < temp_buff->line_length)
850                 {
851                         curr_line->line_length++;
852                         temp_pos++;
853                         *tp = *temp2;
854                         tp++;
855                         temp2++;
856                 }
857                 *tp = '\0';
858                 free(temp_buff->line);
859                 free(temp_buff);
860                 temp_buff = curr_line;
861                 temp_vert = scr_vert;
862                 scr_pos = scr_horz;
863                 if (scr_vert < last_line)
864                 {
865                         wmove(text_win, scr_vert + 1, 0);
866                         wdeleteln(text_win);
867                 }
868                 while ((temp_buff != NULL) && (temp_vert < last_line))
869                 {
870                         temp_buff = temp_buff->next_line;
871                         temp_vert++;
872                 }
873                 if ((temp_vert == last_line) && (temp_buff != NULL))
874                 {
875                         tp = temp_buff->line;
876                         wmove(text_win, last_line,0);
877                         wclrtobot(text_win);
878                         draw_line(last_line, 0, tp, 1, temp_buff->line_length);
879                         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
880                 }
881         }
882         draw_line(scr_vert, scr_horz, point, position, curr_line->line_length);
883         formatted = FALSE;
884 }
885
886 void 
887 scanline(pos)   /* find the proper horizontal position for the pointer  */
888 unsigned char *pos;
889 {
890         int temp;
891         unsigned char *ptr;
892
893         ptr = curr_line->line;
894         temp = 0;
895         while (ptr < pos)
896         {
897                 if (*ptr <= 8)
898                         temp += 2;
899                 else if (*ptr == 9)
900                         temp += tabshift(temp);
901                 else if ((*ptr >= 10) && (*ptr <= 31))
902                         temp += 2;
903                 else if ((*ptr >= 32) && (*ptr < 127))
904                         temp++;
905                 else if (*ptr == 127)
906                         temp += 2;
907                 else if (!eightbit)
908                         temp += 5;
909                 else
910                         temp++;
911                 ptr++;
912         }
913         scr_horz = temp;
914         if ((scr_horz - horiz_offset) > last_col)
915         {
916                 horiz_offset = (scr_horz - (scr_horz % 8)) - (COLS - 8);
917                 midscreen(scr_vert, point);
918         }
919         else if (scr_horz < horiz_offset)
920         {
921                 horiz_offset = max(0, (scr_horz - (scr_horz % 8)));
922                 midscreen(scr_vert, point);
923         }
924 }
925
926 int 
927 tabshift(temp_int)              /* give the number of spaces to shift   */
928 int temp_int;
929 {
930         int leftover;
931
932         leftover = ((temp_int + 1) % 8);
933         if (leftover == 0)
934                 return (1);
935         else
936                 return (9 - leftover);
937 }
938
939 int 
940 out_char(window, character, column)     /* output non-printing character */
941 WINDOW *window;
942 int character;
943 int column;
944 {
945         int i1, i2;
946         char *string;
947         char string2[8];
948
949         if (character == TAB)
950         {
951                 i1 = tabshift(column);
952                 for (i2 = 0; 
953                   (i2 < i1) && (((column+i2+1)-horiz_offset) < last_col); i2++)
954                 {
955                         waddch(window, ' ');
956                 }
957                 return(i1);
958         }
959         else if ((character >= '\0') && (character < ' '))
960         {
961                 string = table[(int) character];
962         }
963         else if ((character < 0) || (character >= 127))
964         {
965                 if (character == 127)
966                         string = "^?";
967                 else if (!eightbit)
968                 {
969                         sprintf(string2, "<%d>", (character < 0) ? (character + 256) : character);
970                         string = string2;
971                 }
972                 else
973                 {
974                         waddch(window, (unsigned char)character );
975                         return(1);
976                 }
977         }
978         else
979         {
980                 waddch(window, (unsigned char)character);
981                 return(1);
982         }
983         for (i2 = 0; (string[i2] != '\0') && (((column+i2+1)-horiz_offset) < last_col); i2++)
984                 waddch(window, (unsigned char)string[i2]);
985         return(strlen(string));
986 }
987
988 int 
989 len_char(character, column)     /* return the length of the character   */
990 int character;
991 int column;     /* the column must be known to provide spacing for tabs */
992 {
993         int length;
994
995         if (character == '\t')
996                 length = tabshift(column);
997         else if ((character >= 0) && (character < 32))
998                 length = 2;
999         else if ((character >= 32) && (character <= 126))
1000                 length = 1;
1001         else if (character == 127)
1002                 length = 2;
1003         else if (((character > 126) || (character < 0)) && (!eightbit))
1004                 length = 5;
1005         else
1006                 length = 1;
1007
1008         return(length);
1009 }
1010
1011 void 
1012 draw_line(vertical, horiz, ptr, t_pos, length)  /* redraw line from current position */
1013 int vertical;   /* current vertical position on screen          */
1014 int horiz;      /* current horizontal position on screen        */
1015 unsigned char *ptr;     /* pointer to line                              */
1016 int t_pos;      /* current position (offset in bytes) from bol  */
1017 int length;     /* length (in bytes) of line                    */
1018 {
1019         int d;          /* partial length of special or tab char to display  */
1020         unsigned char *temp;    /* temporary pointer to position in line             */
1021         int abs_column; /* offset in screen units from begin of line         */
1022         int column;     /* horizontal position on screen                     */
1023         int row;        /* vertical position on screen                       */
1024         int posit;      /* temporary position indicator within line          */
1025
1026         abs_column = horiz;
1027         column = horiz - horiz_offset;
1028         row = vertical;
1029         temp = ptr;
1030         d = 0;
1031         posit = t_pos;
1032         if (column < 0)
1033         {
1034                 wmove(text_win, row, 0);
1035                 wclrtoeol(text_win);
1036         }
1037         while (column < 0)
1038         {
1039                 d = len_char(*temp, abs_column);
1040                 abs_column += d;
1041                 column += d;
1042                 posit++;
1043                 temp++;
1044         }
1045         wmove(text_win, row, column);
1046         wclrtoeol(text_win);
1047         while ((posit < length) && (column <= last_col))
1048         {
1049                 if (!isprint(*temp))
1050                 {
1051                         column += len_char(*temp, abs_column);
1052                         abs_column += out_char(text_win, *temp, abs_column);
1053                 }
1054                 else
1055                 {
1056                         abs_column++;
1057                         column++;
1058                         waddch(text_win, *temp);
1059                 }
1060                 posit++;
1061                 temp++;
1062         }
1063         if (column < last_col)
1064                 wclrtoeol(text_win);
1065         wmove(text_win, vertical, (horiz - horiz_offset));
1066 }
1067
1068 void 
1069 insert_line(disp)                       /* insert new line              */
1070 int disp;
1071 {
1072         int temp_pos;
1073         int temp_pos2;
1074         unsigned char *temp;
1075         unsigned char *extra;
1076         struct text *temp_nod;
1077
1078         text_changes = TRUE;
1079         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1080         wclrtoeol(text_win);
1081         temp_nod= txtalloc();
1082         temp_nod->line = extra= malloc(10);
1083         temp_nod->line_length = 1;
1084         temp_nod->max_length = 10;
1085         temp_nod->line_number = curr_line->line_number + 1;
1086         temp_nod->next_line = curr_line->next_line;
1087         if (temp_nod->next_line != NULL)
1088                 temp_nod->next_line->prev_line = temp_nod;
1089         temp_nod->prev_line = curr_line;
1090         curr_line->next_line = temp_nod;
1091         temp_pos2 = position;
1092         temp = point;
1093         if (temp_pos2 < curr_line->line_length)
1094         {
1095                 temp_pos = 1;
1096                 while (temp_pos2 < curr_line->line_length)
1097                 {
1098                         if ((temp_nod->max_length - temp_nod->line_length)< 5)
1099                                 extra = resiz_line(10, temp_nod, temp_pos);
1100                         temp_nod->line_length++;
1101                         temp_pos++;
1102                         temp_pos2++;
1103                         *extra= *temp;
1104                         extra++;
1105                         temp++;
1106                 }
1107                 temp=point;
1108                 *temp = '\0';
1109                 temp = resiz_line((1 - temp_nod->line_length), curr_line, position);
1110                 curr_line->line_length = 1 + temp - curr_line->line;
1111         }
1112         curr_line->line_length = position;
1113         absolute_lin++;
1114         curr_line = temp_nod;
1115         *extra = '\0';
1116         position = 1;
1117         point= curr_line->line;
1118         if (disp)
1119         {
1120                 if (scr_vert < last_line)
1121                 {
1122                         scr_vert++;
1123                         wclrtoeol(text_win);
1124                         wmove(text_win, scr_vert, 0);
1125                         winsertln(text_win);
1126                 }
1127                 else
1128                 {
1129                         wmove(text_win, 0,0);
1130                         wdeleteln(text_win);
1131                         wmove(text_win, last_line,0);
1132                         wclrtobot(text_win);
1133                 }
1134                 scr_pos = scr_horz = 0;
1135                 if (horiz_offset)
1136                 {
1137                         horiz_offset = 0;
1138                         midscreen(scr_vert, point);
1139                 }
1140                 draw_line(scr_vert, scr_horz, point, position,
1141                         curr_line->line_length);
1142         }
1143 }
1144
1145 struct text *txtalloc()         /* allocate space for line structure    */
1146 {
1147         return((struct text *) malloc(sizeof( struct text)));
1148 }
1149
1150 struct files *name_alloc()      /* allocate space for file name list node */
1151 {
1152         return((struct files *) malloc(sizeof( struct files)));
1153 }
1154
1155 unsigned char *next_word(string)                /* move to next word in string          */
1156 unsigned char *string;
1157 {
1158         while ((*string != '\0') && ((*string != 32) && (*string != 9)))
1159                 string++;
1160         while ((*string != '\0') && ((*string == 32) || (*string == 9)))
1161                 string++;
1162         return(string);
1163 }
1164
1165 void 
1166 prev_word()     /* move to start of previous word in text       */
1167 {
1168         if (position != 1)
1169         {
1170                 if ((position != 1) && ((point[-1] == ' ') || (point[-1] == '\t')))
1171                 {       /* if at the start of a word    */
1172                         while ((position != 1) && ((*point != ' ') && (*point != '\t')))
1173                                 left(TRUE);
1174                 }
1175                 while ((position != 1) && ((*point == ' ') || (*point == '\t')))
1176                         left(TRUE);
1177                 while ((position != 1) && ((*point != ' ') && (*point != '\t')))
1178                         left(TRUE);
1179                 if ((position != 1) && ((*point == ' ') || (*point == '\t')))
1180                         right(TRUE);
1181         }
1182         else
1183                 left(TRUE);
1184 }
1185
1186 void 
1187 control()                       /* use control for commands             */
1188 {
1189         char *string;
1190
1191         if (in == 1)            /* control a    */
1192         {
1193                 string = get_string(ascii_code_str, TRUE);
1194                 if (*string != '\0')
1195                 {
1196                         in = atoi(string);
1197                         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1198                         insert(in);
1199                 }
1200                 free(string);
1201         }
1202         else if (in == 2)       /* control b    */
1203                 bottom();
1204         else if (in == 3)       /* control c    */
1205         {
1206                 command_prompt();
1207         }
1208         else if (in == 4)       /* control d    */
1209                 down();
1210         else if (in == 5)       /* control e    */
1211                 search_prompt();
1212         else if (in == 6)       /* control f    */
1213                 undel_char();
1214         else if (in == 7)       /* control g    */
1215                 bol();
1216         else if (in == 8)       /* control h    */
1217                 delete(TRUE);
1218         else if (in == 9)       /* control i    */
1219                 ;
1220         else if (in == 10)      /* control j    */
1221                 insert_line(TRUE);
1222         else if (in == 11)      /* control k    */
1223                 del_char();
1224         else if (in == 12)      /* control l    */
1225                 left(TRUE);
1226         else if (in == 13)      /* control m    */
1227                 insert_line(TRUE);
1228         else if (in == 14)      /* control n    */
1229                 move_rel('d', max(5, (last_line - 5)));
1230         else if (in == 15)      /* control o    */
1231                 eol();
1232         else if (in == 16)      /* control p    */
1233                 move_rel('u', max(5, (last_line - 5)));
1234         else if (in == 17)      /* control q    */
1235                 ;
1236         else if (in == 18)      /* control r    */
1237                 right(TRUE);
1238         else if (in == 19)      /* control s    */
1239                 ;
1240         else if (in == 20)      /* control t    */
1241                 top();
1242         else if (in == 21)      /* control u    */
1243                 up();
1244         else if (in == 22)      /* control v    */
1245                 undel_word();
1246         else if (in == 23)      /* control w    */
1247                 del_word();
1248         else if (in == 24)      /* control x    */
1249                 search(TRUE);
1250         else if (in == 25)      /* control y    */
1251                 del_line();
1252         else if (in == 26)      /* control z    */
1253                 undel_line();
1254         else if (in == 27)      /* control [ (escape)   */
1255         {
1256                 menu_op(main_menu);
1257         }       
1258 }
1259
1260 /*
1261  |      Emacs control-key bindings
1262  */
1263
1264 void 
1265 emacs_control()
1266 {
1267         char *string;
1268
1269         if (in == 1)            /* control a    */
1270                 bol();
1271         else if (in == 2)       /* control b    */
1272                 left(TRUE);
1273         else if (in == 3)       /* control c    */
1274         {
1275                 command_prompt();
1276         }
1277         else if (in == 4)       /* control d    */
1278                 del_char();
1279         else if (in == 5)       /* control e    */
1280                 eol();
1281         else if (in == 6)       /* control f    */
1282                 right(TRUE);
1283         else if (in == 7)       /* control g    */
1284                 move_rel('u', max(5, (last_line - 5)));
1285         else if (in == 8)       /* control h    */
1286                 delete(TRUE);
1287         else if (in == 9)       /* control i    */
1288                 ;
1289         else if (in == 10)      /* control j    */
1290                 undel_char();
1291         else if (in == 11)      /* control k    */
1292                 del_line();
1293         else if (in == 12)      /* control l    */
1294                 undel_line();
1295         else if (in == 13)      /* control m    */
1296                 insert_line(TRUE);
1297         else if (in == 14)      /* control n    */
1298                 down();
1299         else if (in == 15)      /* control o    */
1300         {
1301                 string = get_string(ascii_code_str, TRUE);
1302                 if (*string != '\0')
1303                 {
1304                         in = atoi(string);
1305                         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1306                         insert(in);
1307                 }
1308                 free(string);
1309         }
1310         else if (in == 16)      /* control p    */
1311                 up();
1312         else if (in == 17)      /* control q    */
1313                 ;
1314         else if (in == 18)      /* control r    */
1315                 undel_word();
1316         else if (in == 19)      /* control s    */
1317                 ;
1318         else if (in == 20)      /* control t    */
1319                 top();
1320         else if (in == 21)      /* control u    */
1321                 bottom();
1322         else if (in == 22)      /* control v    */
1323                 move_rel('d', max(5, (last_line - 5)));
1324         else if (in == 23)      /* control w    */
1325                 del_word();
1326         else if (in == 24)      /* control x    */
1327                 search(TRUE);
1328         else if (in == 25)      /* control y    */
1329                 search_prompt();
1330         else if (in == 26)      /* control z    */
1331                 adv_word();
1332         else if (in == 27)      /* control [ (escape)   */
1333         {
1334                 menu_op(main_menu);
1335         }       
1336 }
1337
1338 void 
1339 bottom()                        /* go to bottom of file                 */
1340 {
1341         while (curr_line->next_line != NULL)
1342         {
1343                 curr_line = curr_line->next_line;
1344                 absolute_lin++;
1345         }
1346         point = curr_line->line;
1347         if (horiz_offset)
1348                 horiz_offset = 0;
1349         position = 1;
1350         midscreen(last_line, point);
1351         scr_pos = scr_horz;
1352 }
1353
1354 void 
1355 top()                           /* go to top of file                    */
1356 {
1357         while (curr_line->prev_line != NULL)
1358         {
1359                 curr_line = curr_line->prev_line;
1360                 absolute_lin--;
1361         }
1362         point = curr_line->line;
1363         if (horiz_offset)
1364                 horiz_offset = 0;
1365         position = 1;
1366         midscreen(0, point);
1367         scr_pos = scr_horz;
1368 }
1369
1370 void 
1371 nextline()                      /* move pointers to start of next line  */
1372 {
1373         curr_line = curr_line->next_line;
1374         absolute_lin++;
1375         point = curr_line->line;
1376         position = 1;
1377         if (scr_vert == last_line)
1378         {
1379                 wmove(text_win, 0,0);
1380                 wdeleteln(text_win);
1381                 wmove(text_win, last_line,0);
1382                 wclrtobot(text_win);
1383                 draw_line(last_line,0,point,1,curr_line->line_length);
1384         }
1385         else
1386                 scr_vert++;
1387 }
1388
1389 void 
1390 prevline()                      /* move pointers to start of previous line*/
1391 {
1392         curr_line = curr_line->prev_line;
1393         absolute_lin--;
1394         point = curr_line->line;
1395         position = 1;
1396         if (scr_vert == 0)
1397         {
1398                 winsertln(text_win);
1399                 draw_line(0,0,point,1,curr_line->line_length);
1400         }
1401         else
1402                 scr_vert--;
1403         while (position < curr_line->line_length)
1404         {
1405                 position++;
1406                 point++;
1407         }
1408 }
1409
1410 void 
1411 left(disp)                              /* move left one character      */
1412 int disp;
1413 {
1414         if (point != curr_line->line)   /* if not at begin of line      */
1415         {
1416                 if ((ee_chinese) && (position >= 2) && (*(point - 2) > 127))
1417                 {
1418                         point--;
1419                         position--;
1420                 }
1421                 point--;
1422                 position--;
1423                 scanline(point);
1424                 wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1425                 scr_pos = scr_horz;
1426         }
1427         else if (curr_line->prev_line != NULL)
1428         {
1429                 if (!disp)
1430                 {
1431                         absolute_lin--;
1432                         curr_line = curr_line->prev_line;
1433                         point = curr_line->line + curr_line->line_length;
1434                         position = curr_line->line_length;
1435                         return;
1436                 }
1437                 position = 1;
1438                 prevline();
1439                 scanline(point);
1440                 scr_pos = scr_horz;
1441                 wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1442         }
1443 }
1444
1445 void 
1446 right(disp)                             /* move right one character     */
1447 int disp;
1448 {
1449         if (position < curr_line->line_length)
1450         {
1451                 if ((ee_chinese) && (*point > 127) && 
1452                     ((curr_line->line_length - position) >= 2))
1453                 {
1454                         point++;
1455                         position++;
1456                 }
1457                 point++;
1458                 position++;
1459                 scanline(point);
1460                 wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1461                 scr_pos = scr_horz;
1462         }
1463         else if (curr_line->next_line != NULL)
1464         {
1465                 if (!disp)
1466                 {
1467                         absolute_lin++;
1468                         curr_line = curr_line->next_line;
1469                         point = curr_line->line;
1470                         position = 1;
1471                         return;
1472                 }
1473                 nextline();
1474                 scr_pos = scr_horz = 0;
1475                 if (horiz_offset)
1476                 {
1477                         horiz_offset = 0;
1478                         midscreen(scr_vert, point);
1479                 }
1480                 wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1481                 position = 1;   
1482         }
1483 }
1484
1485 void 
1486 find_pos()              /* move to the same column as on other line     */
1487 {
1488         scr_horz = 0;
1489         position = 1;
1490         while ((scr_horz < scr_pos) && (position < curr_line->line_length))
1491         {
1492                 if (*point == 9)
1493                         scr_horz += tabshift(scr_horz);
1494                 else if (*point < ' ')
1495                         scr_horz += 2;
1496                 else if ((ee_chinese) && (*point > 127) && 
1497                     ((curr_line->line_length - position) >= 2))
1498                 {
1499                         scr_horz += 2;
1500                         point++;
1501                         position++;
1502                 }
1503                 else
1504                         scr_horz++;
1505                 position++;
1506                 point++;
1507         }
1508         if ((scr_horz - horiz_offset) > last_col)
1509         {
1510                 horiz_offset = (scr_horz - (scr_horz % 8)) - (COLS - 8);
1511                 midscreen(scr_vert, point);
1512         }
1513         else if (scr_horz < horiz_offset)
1514         {
1515                 horiz_offset = max(0, (scr_horz - (scr_horz % 8)));
1516                 midscreen(scr_vert, point);
1517         }
1518         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1519 }
1520
1521 void 
1522 up()                                    /* move up one line             */
1523 {
1524         if (curr_line->prev_line != NULL)
1525         {
1526                 prevline();
1527                 point = curr_line->line;
1528                 find_pos();
1529         }
1530 }
1531
1532 void 
1533 down()                                  /* move down one line           */
1534 {
1535         if (curr_line->next_line != NULL)
1536         {
1537                 nextline();
1538                 find_pos();
1539         }
1540 }
1541
1542 void 
1543 function_key()                          /* process function key         */
1544 {
1545         if (in == KEY_LEFT)
1546                 left(TRUE);
1547         else if (in == KEY_RIGHT)
1548                 right(TRUE);
1549         else if (in == KEY_HOME)
1550                 bol();
1551         else if (in == KEY_END)
1552                 eol();
1553         else if (in == KEY_UP)
1554                 up();
1555         else if (in == KEY_DOWN)
1556                 down();
1557         else if (in == KEY_NPAGE)
1558                 move_rel('d', max( 5, (last_line - 5)));
1559         else if (in == KEY_PPAGE)
1560                 move_rel('u', max(5, (last_line - 5)));
1561         else if (in == KEY_DL)
1562                 del_line();
1563         else if (in == KEY_DC)
1564                 del_char();
1565         else if (in == KEY_BACKSPACE)
1566                 delete(TRUE);
1567         else if (in == KEY_IL)
1568         {               /* insert a line before current line    */
1569                 insert_line(TRUE);
1570                 left(TRUE);
1571         }
1572         else if (in == KEY_F(1))
1573                 gold = !gold;
1574         else if (in == KEY_F(2))
1575         {
1576                 if (gold)
1577                 {
1578                         gold = FALSE;
1579                         undel_line();
1580                 }
1581                 else
1582                         undel_char();
1583         }
1584         else if (in == KEY_F(3))
1585         {
1586                 if (gold)
1587                 {
1588                         gold = FALSE;
1589                         undel_word();
1590                 }
1591                 else
1592                         del_word();
1593         }
1594         else if (in == KEY_F(4))
1595         {
1596                 if (gold)
1597                 {
1598                         gold = FALSE;
1599                         paint_info_win();
1600                         midscreen(scr_vert, point);
1601                 }
1602                 else
1603                         adv_word();
1604         }
1605         else if (in == KEY_F(5))
1606         {
1607                 if (gold)
1608                 {
1609                         gold = FALSE;
1610                         search_prompt();
1611                 }
1612                 else
1613                         search(TRUE);
1614         }
1615         else if (in == KEY_F(6))
1616         {
1617                 if (gold)
1618                 {
1619                         gold = FALSE;
1620                         bottom();
1621                 }
1622                 else
1623                         top();
1624         }
1625         else if (in == KEY_F(7))
1626         {
1627                 if (gold)
1628                 {
1629                         gold = FALSE;
1630                         eol();
1631                 }
1632                 else
1633                         bol();
1634         }
1635         else if (in == KEY_F(8))
1636         {
1637                 if (gold)
1638                 {
1639                         gold = FALSE;
1640                         command_prompt();
1641                 } 
1642                 else
1643                         adv_line();
1644         }
1645 }
1646
1647 void 
1648 print_buffer()
1649 {
1650         char buffer[256];
1651
1652         sprintf(buffer, ">!%s", print_command);
1653         wmove(com_win, 0, 0);
1654         wclrtoeol(com_win);
1655         wprintw(com_win, printer_msg_str, print_command);
1656         wrefresh(com_win);
1657         command(buffer);
1658 }
1659
1660 void 
1661 command_prompt()
1662 {
1663         char *cmd_str;
1664         int result;
1665
1666         info_type = COMMANDS;
1667         paint_info_win();
1668         cmd_str = get_string(command_str, TRUE);
1669         if ((result = unique_test(cmd_str, commands)) != 1)
1670         {
1671                 werase(com_win);
1672                 wmove(com_win, 0, 0);
1673                 if (result == 0)
1674                         wprintw(com_win, unkn_cmd_str, cmd_str);
1675                 else
1676                         wprintw(com_win, non_unique_cmd_msg);
1677
1678                 wrefresh(com_win);
1679
1680                 info_type = CONTROL_KEYS;
1681                 paint_info_win();
1682
1683                 if (cmd_str != NULL)
1684                         free(cmd_str);
1685                 return;
1686         }
1687         command(cmd_str);
1688         wrefresh(com_win);
1689         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
1690         info_type = CONTROL_KEYS;
1691         paint_info_win();
1692         if (cmd_str != NULL)
1693                 free(cmd_str);
1694 }
1695
1696 void 
1697 command(cmd_str1)               /* process commands from keyboard       */
1698 char *cmd_str1;
1699 {
1700         char *cmd_str2 = NULL;
1701         char *cmd_str = cmd_str1;
1702
1703         clear_com_win = TRUE;
1704         if (compare(cmd_str, HELP, FALSE))
1705                 help();
1706         else if (compare(cmd_str, WRITE, FALSE))
1707         {
1708                 if (restrict_mode())
1709                 {
1710                         return;
1711                 }
1712                 cmd_str = next_word(cmd_str);
1713                 if (*cmd_str == '\0')
1714                 {
1715                         cmd_str = cmd_str2 = get_string(file_write_prompt_str, TRUE);
1716                 }
1717                 tmp_file = resolve_name(cmd_str);
1718                 write_file(tmp_file, 1);
1719                 if (tmp_file != cmd_str)
1720                         free(tmp_file);
1721         }
1722         else if (compare(cmd_str, READ, FALSE))
1723         {
1724                 if (restrict_mode())
1725                 {
1726                         return;
1727                 }
1728                 cmd_str = next_word(cmd_str);
1729                 if (*cmd_str == '\0')
1730                 {
1731                         cmd_str = cmd_str2 = get_string(file_read_prompt_str, TRUE);
1732                 }
1733                 tmp_file = cmd_str;
1734                 recv_file = TRUE;
1735                 tmp_file = resolve_name(cmd_str);
1736                 check_fp();
1737                 if (tmp_file != cmd_str)
1738                         free(tmp_file);
1739         }
1740         else if (compare(cmd_str, LINE, FALSE))
1741         {
1742                 wmove(com_win, 0, 0);
1743                 wclrtoeol(com_win);
1744                 wprintw(com_win, line_num_str, curr_line->line_number);
1745                 wprintw(com_win, line_len_str, curr_line->line_length);
1746         }
1747         else if (compare(cmd_str, FILE_str, FALSE))
1748         {
1749                 wmove(com_win, 0, 0);
1750                 wclrtoeol(com_win);
1751                 if (in_file_name == NULL)
1752                         wprintw(com_win, no_file_string);
1753                 else
1754                         wprintw(com_win, current_file_str, in_file_name);
1755         }
1756         else if ((*cmd_str >= '0') && (*cmd_str <= '9'))
1757                 goto_line(cmd_str);
1758         else if (compare(cmd_str, CHARACTER, FALSE))
1759         {
1760                 wmove(com_win, 0, 0);
1761                 wclrtoeol(com_win);
1762                 wprintw(com_win, char_str, *point);
1763         }
1764         else if (compare(cmd_str, REDRAW, FALSE))
1765                 redraw();
1766         else if (compare(cmd_str, RESEQUENCE, FALSE))
1767         {
1768                 tmp_line = first_line->next_line;
1769                 while (tmp_line != NULL)
1770                 {
1771                 tmp_line->line_number = tmp_line->prev_line->line_number + 1;
1772                         tmp_line = tmp_line->next_line;
1773                 }
1774         }
1775         else if (compare(cmd_str, AUTHOR, FALSE))
1776         {
1777                 wmove(com_win, 0, 0);
1778                 wclrtoeol(com_win);
1779                 wprintw(com_win, "written by Hugh Mahon");
1780         }
1781         else if (compare(cmd_str, VERSION, FALSE))
1782         {
1783                 wmove(com_win, 0, 0);
1784                 wclrtoeol(com_win);
1785                 wprintw(com_win, "%s", version);
1786         }
1787         else if (compare(cmd_str, CASE, FALSE))
1788                 case_sen = TRUE;
1789         else if (compare(cmd_str, NOCASE, FALSE))
1790                 case_sen = FALSE;
1791         else if (compare(cmd_str, EXPAND, FALSE))
1792                 expand_tabs = TRUE;
1793         else if (compare(cmd_str, NOEXPAND, FALSE))
1794                 expand_tabs = FALSE;
1795         else if (compare(cmd_str, Exit_string, FALSE))
1796                 finish();
1797         else if (compare(cmd_str, chinese_cmd, FALSE))
1798         {
1799                 ee_chinese = TRUE;
1800 #ifdef NCURSE
1801                 nc_setattrib(A_NC_BIG5);
1802 #endif /* NCURSE */
1803         }
1804         else if (compare(cmd_str, nochinese_cmd, FALSE))
1805         {
1806                 ee_chinese = FALSE;
1807 #ifdef NCURSE
1808                 nc_clearattrib(A_NC_BIG5);
1809 #endif /* NCURSE */
1810         }
1811         else if (compare(cmd_str, QUIT_string, FALSE))
1812                 quit(0);
1813         else if (*cmd_str == '!')
1814         {
1815                 cmd_str++;
1816                 if ((*cmd_str == ' ') || (*cmd_str == 9))
1817                         cmd_str = next_word(cmd_str);
1818                 sh_command(cmd_str);
1819         }
1820         else if ((*cmd_str == '<') && (!in_pipe))
1821         {
1822                 in_pipe = TRUE;
1823                 shell_fork = FALSE;
1824                 cmd_str++;
1825                 if ((*cmd_str == ' ') || (*cmd_str == '\t'))
1826                         cmd_str = next_word(cmd_str);
1827                 command(cmd_str);
1828                 in_pipe = FALSE;
1829                 shell_fork = TRUE;
1830         }
1831         else if ((*cmd_str == '>') && (!out_pipe))
1832         {
1833                 out_pipe = TRUE;
1834                 cmd_str++;
1835                 if ((*cmd_str == ' ') || (*cmd_str == '\t'))
1836                         cmd_str = next_word(cmd_str);
1837                 command(cmd_str);
1838                 out_pipe = FALSE;
1839         }
1840         else
1841         {
1842                 wmove(com_win, 0, 0);
1843                 wclrtoeol(com_win);
1844                 wprintw(com_win, unkn_cmd_str, cmd_str);
1845         }
1846         if (cmd_str2 != NULL)
1847                 free(cmd_str2);
1848 }
1849
1850 int 
1851 scan(line, offset, column)      /* determine horizontal position for get_string */
1852 char *line;
1853 int offset;
1854 int column;
1855 {
1856         char *stemp;
1857         int i;
1858         int j;
1859
1860         stemp = line;
1861         i = 0;
1862         j = column;
1863         while (i < offset)
1864         {
1865                 i++;
1866                 j += len_char(*stemp, j);
1867                 stemp++;
1868         }
1869         return(j);
1870 }
1871
1872 char *
1873 get_string(prompt, advance)     /* read string from input on command line */
1874 char *prompt;           /* string containing user prompt message        */
1875 int advance;            /* if true, skip leading spaces and tabs        */
1876 {
1877         char *string;
1878         char *tmp_string;
1879         char *nam_str;
1880         char *g_point;
1881         int tmp_int;
1882         int g_horz, g_position, g_pos;
1883         int esc_flag;
1884
1885         g_point = tmp_string = malloc(512);
1886         wmove(com_win,0,0);
1887         wclrtoeol(com_win);
1888         waddstr(com_win, prompt);
1889         wrefresh(com_win);
1890         nam_str = tmp_string;
1891         clear_com_win = TRUE;
1892         g_horz = g_position = scan(prompt, strlen(prompt), 0);
1893         g_pos = 0;
1894         do
1895         {
1896                 esc_flag = FALSE;
1897                 in = wgetch(com_win);
1898                 if (in == -1)
1899                         exit(0);
1900                 if (((in == 8) || (in == 127) || (in == KEY_BACKSPACE)) && (g_pos > 0))
1901                 {
1902                         tmp_int = g_horz;
1903                         g_pos--;
1904                         g_horz = scan(g_point, g_pos, g_position);
1905                         tmp_int = tmp_int - g_horz;
1906                         for (; 0 < tmp_int; tmp_int--)
1907                         {
1908                                 if ((g_horz+tmp_int) < (last_col - 1))
1909                                 {
1910                                         waddch(com_win, '\010');
1911                                         waddch(com_win, ' ');
1912                                         waddch(com_win, '\010');
1913                                 }
1914                         }
1915                         nam_str--;
1916                 }
1917                 else if ((in != 8) && (in != 127) && (in != '\n') && (in != '\r') && (in < 256))
1918                 {
1919                         if (in == '\026')       /* control-v, accept next character verbatim    */
1920                         {                       /* allows entry of ^m, ^j, and ^h       */
1921                                 esc_flag = TRUE;
1922                                 in = wgetch(com_win);
1923                                 if (in == -1)
1924                                         exit(0);
1925                         }
1926                         *nam_str = in;
1927                         g_pos++;
1928                         if (!isprint((unsigned char)in) && (g_horz < (last_col - 1)))
1929                                 g_horz += out_char(com_win, in, g_horz);
1930                         else
1931                         {
1932                                 g_horz++;
1933                                 if (g_horz < (last_col - 1))
1934                                         waddch(com_win, (unsigned char)in);
1935                         }
1936                         nam_str++;
1937                 }
1938                 wrefresh(com_win);
1939                 if (esc_flag)
1940                         in = '\0';
1941         } while ((in != '\n') && (in != '\r'));
1942         *nam_str = '\0';
1943         nam_str = tmp_string;
1944         if (((*nam_str == ' ') || (*nam_str == 9)) && (advance))
1945                 nam_str = next_word(nam_str);
1946         string = malloc(strlen(nam_str) + 1);
1947         strcpy(string, nam_str);
1948         free(tmp_string);
1949         wrefresh(com_win);
1950         return(string);
1951 }
1952
1953 int 
1954 compare(string1, string2, sensitive)    /* compare two strings  */
1955 char *string1;
1956 char *string2;
1957 int sensitive;
1958 {
1959         char *strng1;
1960         char *strng2;
1961         int tmp;
1962         int equal;
1963
1964         strng1 = string1;
1965         strng2 = string2;
1966         tmp = 0;
1967         if ((strng1 == NULL) || (strng2 == NULL) || (*strng1 == '\0') || (*strng2 == '\0'))
1968                 return(FALSE);
1969         equal = TRUE;
1970         while (equal)
1971         {
1972                 if (sensitive)
1973                 {
1974                         if (*strng1 != *strng2)
1975                                 equal = FALSE;
1976                 }
1977                 else
1978                 {
1979                         if (toupper((unsigned char)*strng1) != toupper((unsigned char)*strng2))
1980                                 equal = FALSE;
1981                 }
1982                 strng1++;
1983                 strng2++;
1984                 if ((*strng1 == '\0') || (*strng2 == '\0') || (*strng1 == ' ') || (*strng2 == ' '))
1985                         break;
1986                 tmp++;
1987         }
1988         return(equal);
1989 }
1990
1991 void 
1992 goto_line(cmd_str)
1993 char *cmd_str;
1994 {
1995         int number;
1996         int i;
1997         char *ptr;
1998         char direction = '\0';
1999         struct text *t_line;
2000
2001         ptr = cmd_str;
2002         i= 0;
2003         while ((*ptr >='0') && (*ptr <= '9'))
2004         {
2005                 i= i * 10 + (*ptr - '0');
2006                 ptr++;
2007         }
2008         number = i;
2009         i = 0;
2010         t_line = curr_line;
2011         while ((t_line->line_number > number) && (t_line->prev_line != NULL))
2012         {
2013                 i++;
2014                 t_line = t_line->prev_line;
2015                 direction = 'u';
2016         }
2017         while ((t_line->line_number < number) && (t_line->next_line != NULL))
2018         {
2019                 i++;
2020                 direction = 'd';
2021                 t_line = t_line->next_line;
2022         }
2023         if ((i < 30) && (i > 0))
2024         {
2025                 move_rel(direction, i);
2026         }
2027         else
2028         {
2029                 if (direction != 'd')
2030                 {
2031                         absolute_lin += i;
2032                 }
2033                 else
2034                 {
2035                         absolute_lin -= i;
2036                 }
2037                 curr_line = t_line;
2038                 point = curr_line->line;
2039                 position = 1;
2040                 midscreen((last_line / 2), point);
2041                 scr_pos = scr_horz;
2042         }
2043         wmove(com_win, 0, 0);
2044         wclrtoeol(com_win);
2045         wprintw(com_win, line_num_str, curr_line->line_number);
2046         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
2047 }
2048
2049 void 
2050 midscreen(line, pnt)    /* put current line in middle of screen */
2051 int line;
2052 unsigned char *pnt;
2053 {
2054         struct text *mid_line;
2055         int i;
2056
2057         line = min(line, last_line);
2058         mid_line = curr_line;
2059         for (i = 0; ((i < line) && (curr_line->prev_line != NULL)); i++)
2060                 curr_line = curr_line->prev_line;
2061         scr_vert = scr_horz = 0;
2062         wmove(text_win, 0, 0);
2063         draw_screen();
2064         scr_vert = i;
2065         curr_line = mid_line;
2066         scanline(pnt);
2067         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
2068 }
2069
2070 void 
2071 get_options(numargs, arguments) /* get arguments from command line      */
2072 int numargs;
2073 char *arguments[];
2074 {
2075         char *buff;
2076         int count;
2077         struct files *temp_names = NULL;
2078         char *name;
2079         char *ptr;
2080         int no_more_opts = FALSE;
2081
2082         /*
2083          |      see if editor was invoked as 'ree' (restricted mode)
2084          */
2085
2086         if (!(name = strrchr(arguments[0], '/')))
2087                 name = arguments[0];
2088         else
2089                 name++;
2090         if (!strcmp(name, "ree"))
2091                 restricted = TRUE;
2092
2093         top_of_stack = NULL;
2094         input_file = FALSE;
2095         recv_file = FALSE;
2096         count = 1;
2097         while ((count < numargs)&& (!no_more_opts))
2098         {
2099                 buff = arguments[count];
2100                 if (!strcmp("-i", buff))
2101                 {
2102                         info_window = FALSE;
2103                 }
2104                 else if (!strcmp("-e", buff))
2105                 {
2106                         expand_tabs = FALSE;
2107                 }
2108                 else if (!strcmp("-h", buff))
2109                 {
2110                         nohighlight = TRUE;
2111                 }
2112                 else if (!strcmp("-?", buff))
2113                 {
2114                         fprintf(stderr, usage0, arguments[0]);
2115                         fputs(usage1, stderr);
2116                         fputs(usage2, stderr);
2117                         fputs(usage3, stderr);
2118                         fputs(usage4, stderr);
2119                         exit(1);
2120                 }
2121                 else if ((*buff == '+') && (start_at_line == NULL))
2122                 {
2123                         buff++;
2124                         start_at_line = buff;
2125                 }
2126                 else if (!(strcmp("--", buff)))
2127                         no_more_opts = TRUE;
2128                 else
2129                 {
2130                         count--;
2131                         no_more_opts = TRUE;
2132                 }
2133                 count++;
2134         }
2135         while (count < numargs)
2136         {
2137                 buff = arguments[count];
2138                 if (top_of_stack == NULL)
2139                 {
2140                         temp_names = top_of_stack = name_alloc();
2141                 }
2142                 else
2143                 {
2144                         temp_names->next_name = name_alloc();
2145                         temp_names = temp_names->next_name;
2146                 }
2147                 ptr = temp_names->name = malloc(strlen(buff) + 1);
2148                 while (*buff != '\0')
2149                 {
2150                         *ptr = *buff;
2151                         buff++;
2152                         ptr++;
2153                 }
2154                 *ptr = '\0';
2155                 temp_names->next_name = NULL;
2156                 input_file = TRUE;
2157                 recv_file = TRUE;
2158                 count++;
2159         }
2160 }
2161
2162 void 
2163 check_fp()              /* open or close files according to flags */
2164 {
2165         int line_num;
2166         int temp;
2167         struct stat buf;
2168
2169         clear_com_win = TRUE;
2170         tmp_vert = scr_vert;
2171         tmp_horz = scr_horz;
2172         tmp_line = curr_line;
2173         if (input_file)
2174         {
2175                 in_file_name = tmp_file = top_of_stack->name;
2176                 top_of_stack = top_of_stack->next_name;
2177         }
2178         temp = stat(tmp_file, &buf);
2179         buf.st_mode &= ~07777;
2180         if ((temp != -1) && (buf.st_mode != 0100000) && (buf.st_mode != 0))
2181         {
2182                 wprintw(com_win, file_is_dir_msg, tmp_file);
2183                 wrefresh(com_win);
2184                 if (input_file)
2185                 {
2186                         quit(0);
2187                         return;
2188                 }
2189                 else
2190                         return;
2191         }
2192         if ((get_fd = open(tmp_file, O_RDONLY)) == -1)
2193         {
2194                 wmove(com_win, 0, 0);
2195                 wclrtoeol(com_win);
2196                 if (input_file)
2197                         wprintw(com_win, new_file_msg, tmp_file);
2198                 else
2199                         wprintw(com_win, cant_open_msg, tmp_file);
2200                 wrefresh(com_win);
2201                 wmove(text_win, scr_vert, (scr_horz - horiz_offset));
2202                 wrefresh(text_win);
2203                 recv_file = FALSE;
2204                 input_file = FALSE;
2205                 return;
2206         }
2207         else
2208                 get_file(tmp_file);
2209
2210         recv_file = FALSE;
2211         line_num = curr_line->line_number;
2212         scr_vert = tmp_vert;
2213         scr_horz = tmp_horz;
2214         if (input_file)
2215                 curr_line= first_line;
2216         else
2217                 curr_line = tmp_line;
2218         point = curr_line->line;
2219         draw_screen();
2220         if (input_file)
2221         {
2222                 input_file = FALSE;
2223                 if (start_at_line != NULL)
2224                 {
2225                         line_num = atoi(start_at_line) - 1;
2226                         move_rel('d', line_num);
2227                         line_num = 0;
2228                         start_at_line = NULL;
2229                 }
2230         }
2231         else
2232         {
2233                 wmove(com_win, 0, 0);
2234                 wclrtoeol(com_win);
2235                 text_changes = TRUE;
2236                 if ((tmp_file != NULL) && (*tmp_file != '\0'))
2237                         wprintw(com_win, file_read_fin_msg, tmp_file);
2238         }
2239         wrefresh(com_win);
2240         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
2241         wrefresh(text_win);
2242 }
2243
2244 void 
2245 get_file(file_name)     /* read specified file into current buffer      */
2246 char *file_name;
2247 {
2248         int can_read;           /* file has at least one character      */
2249         int length;             /* length of line read by read          */
2250         int append;             /* should text be appended to current line */
2251         struct text *temp_line;
2252         char ro_flag = FALSE;
2253
2254         if (recv_file)          /* if reading a file                    */
2255         {
2256                 wmove(com_win, 0, 0);
2257                 wclrtoeol(com_win);
2258                 wprintw(com_win, reading_file_msg, file_name);
2259                 if (access(file_name, 2))       /* check permission to write */
2260                 {
2261                         if ((errno == ENOTDIR) || (errno == EACCES) || (errno == EROFS) || (errno == ETXTBSY) || (errno == EFAULT))
2262                         {
2263                                 wprintw(com_win, read_only_msg);
2264                                 ro_flag = TRUE;
2265                         }
2266                 }
2267                 wrefresh(com_win);
2268         }
2269         if (curr_line->line_length > 1) /* if current line is not blank */
2270         {
2271                 insert_line(FALSE);
2272                 left(FALSE);
2273                 append = FALSE;
2274         }
2275         else
2276                 append = TRUE;
2277         can_read = FALSE;               /* test if file has any characters  */
2278         while (((length = read(get_fd, in_string, 512)) != 0) && (length != -1))
2279         {
2280                 can_read = TRUE;  /* if set file has at least 1 character   */
2281                 get_line(length, in_string, &append);
2282         }
2283         if ((can_read) && (curr_line->line_length == 1))
2284         {
2285                 temp_line = curr_line->prev_line;
2286                 temp_line->next_line = curr_line->next_line;
2287                 if (temp_line->next_line != NULL)
2288                         temp_line->next_line->prev_line = temp_line;
2289                 if (curr_line->line != NULL)
2290                         free(curr_line->line);
2291                 free(curr_line);
2292                 curr_line = temp_line;
2293         }
2294         if (input_file) /* if this is the file to be edited display number of lines     */
2295         {
2296                 wmove(com_win, 0, 0);
2297                 wclrtoeol(com_win);
2298                 wprintw(com_win, file_read_lines_msg, in_file_name, curr_line->line_number);
2299                 if (ro_flag)
2300                         wprintw(com_win, read_only_msg);
2301                 wrefresh(com_win);
2302         }
2303         else if (can_read)      /* not input_file and file is non-zero size */
2304                 text_changes = TRUE;
2305
2306         if (recv_file)          /* if reading a file                    */
2307         {
2308                 in = EOF;
2309         }
2310 }
2311
2312 void 
2313 get_line(length, in_string, append)     /* read string and split into lines */
2314 int length;             /* length of string read by read                */
2315 unsigned char *in_string;       /* string read by read                          */
2316 int *append;    /* TRUE if must append more text to end of current line */
2317 {
2318         unsigned char *str1;
2319         unsigned char *str2;
2320         int num;                /* offset from start of string          */
2321         int char_count;         /* length of new line (or added portion */
2322         int temp_counter;       /* temporary counter value              */
2323         struct text *tline;     /* temporary pointer to new line        */
2324         int first_time;         /* if TRUE, the first time through the loop */
2325
2326         str2 = in_string;
2327         num = 0;
2328         first_time = TRUE;
2329         while (num < length)
2330         {
2331                 if (!first_time)
2332                 {
2333                         if (num < length)
2334                         {
2335                                 str2++;
2336                                 num++;
2337                         }
2338                 }
2339                 else
2340                         first_time = FALSE;
2341                 str1 = str2;
2342                 char_count = 1;
2343                 /* find end of line     */
2344                 while ((*str2 != '\n') && (num < length))
2345                 {
2346                         str2++;
2347                         num++;
2348                         char_count++;
2349                 }
2350                 if (!(*append)) /* if not append to current line, insert new one */
2351                 {
2352                         tline = txtalloc();     /* allocate data structure for next line */
2353                         tline->line_number = curr_line->line_number + 1;
2354                         tline->next_line = curr_line->next_line;
2355                         tline->prev_line = curr_line;
2356                         curr_line->next_line = tline;
2357                         if (tline->next_line != NULL)
2358                                 tline->next_line->prev_line = tline;
2359                         curr_line = tline;
2360                         curr_line->line = point = (unsigned char *) malloc(char_count);
2361                         curr_line->line_length = char_count;
2362                         curr_line->max_length = char_count;
2363                 }
2364                 else
2365                 {
2366                         point = resiz_line(char_count, curr_line, curr_line->line_length); 
2367                         curr_line->line_length += (char_count - 1);
2368                 }
2369                 for (temp_counter = 1; temp_counter < char_count; temp_counter++)
2370                 {
2371                         *point = *str1;
2372                         point++;
2373                         str1++;
2374                 }
2375                 *point = '\0';
2376                 *append = FALSE;
2377                 if ((num == length) && (*str2 != '\n'))
2378                         *append = TRUE;
2379         }
2380 }
2381
2382 void 
2383 draw_screen()           /* redraw the screen from current postion       */
2384 {
2385         struct text *temp_line;
2386         unsigned char *line_out;
2387         int temp_vert;
2388
2389         temp_line = curr_line;
2390         temp_vert = scr_vert;
2391         wclrtobot(text_win);
2392         while ((temp_line != NULL) && (temp_vert <= last_line))
2393         {
2394                 line_out = temp_line->line;
2395                 draw_line(temp_vert, 0, line_out, 1, temp_line->line_length);
2396                 temp_vert++;
2397                 temp_line = temp_line->next_line;
2398         }
2399         wmove(text_win, temp_vert, 0);
2400         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
2401 }
2402
2403 void 
2404 finish()        /* prepare to exit edit session */
2405 {
2406         char *file_name = in_file_name;
2407
2408         /*
2409          |      changes made here should be reflected in the 'save' 
2410          |      portion of file_op()
2411          */
2412
2413         if ((file_name == NULL) || (*file_name == '\0'))
2414                 file_name = get_string(save_file_name_prompt, TRUE);
2415
2416         if ((file_name == NULL) || (*file_name == '\0'))
2417         {
2418                 wmove(com_win, 0, 0);
2419                 wprintw(com_win, file_not_saved_msg);
2420                 wclrtoeol(com_win);
2421                 wrefresh(com_win);
2422                 clear_com_win = TRUE;
2423                 return;
2424         }
2425
2426         tmp_file = resolve_name(file_name);
2427         if (tmp_file != file_name)
2428         {
2429                 free(file_name);
2430                 file_name = tmp_file;
2431         }
2432
2433         if (write_file(file_name, 1))
2434         {
2435                 text_changes = FALSE;
2436                 quit(0);
2437         }
2438 }
2439
2440 int 
2441 quit(noverify)          /* exit editor                  */
2442 int noverify;
2443 {
2444         char *ans;
2445
2446         touchwin(text_win);
2447         wrefresh(text_win);
2448         if ((text_changes) && (!noverify))
2449         {
2450                 ans = get_string(changes_made_prompt, TRUE);
2451                 if (toupper((unsigned char)*ans) == toupper((unsigned char)*yes_char))
2452                         text_changes = FALSE;
2453                 else
2454                         return(0);
2455                 free(ans);
2456         }
2457         if (top_of_stack == NULL)
2458         {
2459                 if (info_window)
2460                         wrefresh(info_win);
2461                 wrefresh(com_win);
2462                 resetty();
2463                 endwin();
2464                 putchar('\n');
2465                 exit(0);
2466         }
2467         else
2468         {
2469                 delete_text();
2470                 recv_file = TRUE;
2471                 input_file = TRUE;
2472                 check_fp();
2473         }
2474         return(0);
2475 }
2476
2477 void 
2478 edit_abort(arg)
2479 int arg;
2480 {
2481         wrefresh(com_win);
2482         resetty();
2483         endwin();
2484         putchar('\n');
2485         exit(1);
2486 }
2487
2488 void 
2489 delete_text()
2490 {
2491         while (curr_line->next_line != NULL)
2492                 curr_line = curr_line->next_line;
2493         while (curr_line != first_line)
2494         {
2495                 free(curr_line->line);
2496                 curr_line = curr_line->prev_line;
2497                 absolute_lin--;
2498                 free(curr_line->next_line);
2499         }
2500         curr_line->next_line = NULL;
2501         *curr_line->line = '\0';
2502         curr_line->line_length = 1;
2503         curr_line->line_number = 1;
2504         point = curr_line->line;
2505         scr_pos = scr_vert = scr_horz = 0;
2506         position = 1;
2507 }
2508
2509 int 
2510 write_file(file_name, warn_if_exists)
2511 char *file_name;
2512 int warn_if_exists;
2513 {
2514         char cr;
2515         char *tmp_point;
2516         struct text *out_line;
2517         int lines, charac;
2518         int temp_pos;
2519         int write_flag = TRUE;
2520
2521         charac = lines = 0;
2522         if (warn_if_exists &&
2523             ((in_file_name == NULL) || strcmp(in_file_name, file_name)))
2524         {
2525                 if ((temp_fp = fopen(file_name, "r")))
2526                 {
2527                         tmp_point = get_string(file_exists_prompt, TRUE);
2528                         if (toupper((unsigned char)*tmp_point) == toupper((unsigned char)*yes_char))
2529                                 write_flag = TRUE;
2530                         else 
2531                                 write_flag = FALSE;
2532                         fclose(temp_fp);
2533                         free(tmp_point);
2534                 }
2535         }
2536
2537         clear_com_win = TRUE;
2538
2539         if (write_flag)
2540         {
2541                 if ((temp_fp = fopen(file_name, "w")) == NULL)
2542                 {
2543                         clear_com_win = TRUE;
2544                         wmove(com_win,0,0);
2545                         wclrtoeol(com_win);
2546                         wprintw(com_win, create_file_fail_msg, file_name);
2547                         wrefresh(com_win);
2548                         return(FALSE);
2549                 }
2550                 else
2551                 {
2552                         wmove(com_win,0,0);
2553                         wclrtoeol(com_win);
2554                         wprintw(com_win, writing_file_msg, file_name);
2555                         wrefresh(com_win);
2556                         cr = '\n';
2557                         out_line = first_line;
2558                         while (out_line != NULL)
2559                         {
2560                                 temp_pos = 1;
2561                                 tmp_point= out_line->line;
2562                                 while (temp_pos < out_line->line_length)
2563                                 {
2564                                         putc(*tmp_point, temp_fp);
2565                                         tmp_point++;
2566                                         temp_pos++;
2567                                 }
2568                                 charac += out_line->line_length;
2569                                 out_line = out_line->next_line;
2570                                 putc(cr, temp_fp);
2571                                 lines++;
2572                         }
2573                         fclose(temp_fp);
2574                         wmove(com_win,0,0);
2575                         wclrtoeol(com_win);
2576                         wprintw(com_win, file_written_msg, file_name, lines, charac);
2577                         wrefresh(com_win);
2578                         return(TRUE);
2579                 }
2580         }
2581         else
2582                 return(FALSE);
2583 }
2584
2585 int 
2586 search(display_message)         /* search for string in srch_str        */
2587 int display_message;
2588 {
2589         int lines_moved;
2590         int iter;
2591         int found;
2592
2593         if ((srch_str == NULL) || (*srch_str == '\0'))
2594                 return(FALSE);
2595         if (display_message)
2596         {
2597                 wmove(com_win, 0, 0);
2598                 wclrtoeol(com_win);
2599                 wprintw(com_win, searching_msg);
2600                 wrefresh(com_win);
2601                 clear_com_win = TRUE;
2602         }
2603         lines_moved = 0;
2604         found = FALSE;
2605         srch_line = curr_line;
2606         srch_1 = point;
2607         if (position < curr_line->line_length)
2608                 srch_1++;
2609         iter = position + 1;
2610         while ((!found) && (srch_line != NULL))
2611         {
2612                 while ((iter < srch_line->line_length) && (!found))
2613                 {
2614                         srch_2 = srch_1;
2615                         if (case_sen)   /* if case sensitive            */
2616                         {
2617                                 srch_3 = srch_str;
2618                         while ((*srch_2 == *srch_3) && (*srch_3 != '\0'))
2619                                 {
2620                                         found = TRUE;
2621                                         srch_2++;
2622                                         srch_3++;
2623                                 }       /* end while    */
2624                         }
2625                         else            /* if not case sensitive        */
2626                         {
2627                                 srch_3 = u_srch_str;
2628                         while ((toupper(*srch_2) == *srch_3) && (*srch_3 != '\0'))
2629                                 {
2630                                         found = TRUE;
2631                                         srch_2++;
2632                                         srch_3++;
2633                                 }
2634                         }       /* end else     */
2635                         if (!((*srch_3 == '\0') && (found)))
2636                         {
2637                                 found = FALSE;
2638                                 if (iter < srch_line->line_length)
2639                                         srch_1++;
2640                                 iter++;
2641                         }
2642                 }
2643                 if (!found)
2644                 {
2645                         srch_line = srch_line->next_line;
2646                         if (srch_line != NULL)
2647                                 srch_1 = srch_line->line;
2648                         iter = 1;
2649                         lines_moved++;
2650                 }
2651         }
2652         if (found)
2653         {
2654                 if (display_message)
2655                 {
2656                         wmove(com_win, 0, 0);
2657                         wclrtoeol(com_win);
2658                         wrefresh(com_win);
2659                 }
2660                 if (lines_moved == 0)
2661                 {
2662                         while (position < iter)
2663                                 right(TRUE);
2664                 }
2665                 else
2666                 {
2667                         if (lines_moved < 30)
2668                         {
2669                                 move_rel('d', lines_moved);
2670                                 while (position < iter)
2671                                         right(TRUE);
2672                         }
2673                         else 
2674                         {
2675                                 absolute_lin += lines_moved;
2676                                 curr_line = srch_line;
2677                                 point = srch_1;
2678                                 position = iter;
2679                                 scanline(point);
2680                                 scr_pos = scr_horz;
2681                                 midscreen((last_line / 2), point);
2682                         }
2683                 }
2684         }
2685         else
2686         {
2687                 if (display_message)
2688                 {
2689                         wmove(com_win, 0, 0);
2690                         wclrtoeol(com_win);
2691                         wprintw(com_win, str_not_found_msg, srch_str);
2692                         wrefresh(com_win);
2693                 }
2694                 wmove(text_win, scr_vert,(scr_horz - horiz_offset));
2695         }
2696         return(found);
2697 }
2698
2699 void 
2700 search_prompt()         /* prompt and read search string (srch_str)     */
2701 {
2702         if (srch_str != NULL)
2703                 free(srch_str);
2704         if ((u_srch_str != NULL) && (*u_srch_str != '\0'))
2705                 free(u_srch_str);
2706         srch_str = get_string(search_prompt_str, FALSE);
2707         gold = FALSE;
2708         srch_3 = srch_str;
2709         srch_1 = u_srch_str = malloc(strlen(srch_str) + 1);
2710         while (*srch_3 != '\0')
2711         {
2712                 *srch_1 = toupper(*srch_3);
2713                 srch_1++;
2714                 srch_3++;
2715         }
2716         *srch_1 = '\0';
2717         search(TRUE);
2718 }
2719
2720 void 
2721 del_char()                      /* delete current character     */
2722 {
2723         in = 8;  /* backspace */
2724         if (position < curr_line->line_length)  /* if not end of line   */
2725         {
2726                 if ((ee_chinese) && (*point > 127) && 
2727                     ((curr_line->line_length - position) >= 2))
2728                 {
2729                         point++;
2730                         position++;
2731                 }
2732                 position++;
2733                 point++;
2734                 scanline(point);
2735                 delete(TRUE);
2736         }
2737         else
2738         {
2739                 right(TRUE);
2740                 delete(TRUE);
2741         }
2742 }
2743
2744 void 
2745 undel_char()                    /* undelete last deleted character      */
2746 {
2747         if (d_char[0] == '\n')  /* insert line if last del_char deleted eol */
2748                 insert_line(TRUE);
2749         else
2750         {
2751                 in = d_char[0];
2752                 insert(in);
2753                 if (d_char[1] != '\0')
2754                 {
2755                         in = d_char[1];
2756                         insert(in);
2757                 }
2758         }
2759 }
2760
2761 void 
2762 del_word()                      /* delete word in front of cursor       */
2763 {
2764         int tposit;
2765         int difference;
2766         unsigned char *d_word2;
2767         unsigned char *d_word3;
2768         unsigned char tmp_char[3];
2769
2770         if (d_word != NULL)
2771                 free(d_word);
2772         d_word = malloc(curr_line->line_length);
2773         tmp_char[0] = d_char[0];
2774         tmp_char[1] = d_char[1];
2775         tmp_char[2] = d_char[2];
2776         d_word3 = point;
2777         d_word2 = d_word;
2778         tposit = position;
2779         while ((tposit < curr_line->line_length) && 
2780                                 ((*d_word3 != ' ') && (*d_word3 != '\t')))
2781         {
2782                 tposit++;
2783                 *d_word2 = *d_word3;
2784                 d_word2++;
2785                 d_word3++;
2786         }
2787         while ((tposit < curr_line->line_length) && 
2788                                 ((*d_word3 == ' ') || (*d_word3 == '\t')))
2789         {
2790                 tposit++;
2791                 *d_word2 = *d_word3;
2792                 d_word2++;
2793                 d_word3++;
2794         }
2795         *d_word2 = '\0';
2796         d_wrd_len = difference = d_word2 - d_word;
2797         d_word2 = point;
2798         while (tposit < curr_line->line_length)
2799         {
2800                 tposit++;
2801                 *d_word2 = *d_word3;
2802                 d_word2++;
2803                 d_word3++;
2804         }
2805         curr_line->line_length -= difference;
2806         *d_word2 = '\0';
2807         draw_line(scr_vert, scr_horz,point,position,curr_line->line_length);
2808         d_char[0] = tmp_char[0];
2809         d_char[1] = tmp_char[1];
2810         d_char[2] = tmp_char[2];
2811         text_changes = TRUE;
2812         formatted = FALSE;
2813 }
2814
2815 void 
2816 undel_word()            /* undelete last deleted word           */
2817 {
2818         int temp;
2819         int tposit;
2820         unsigned char *tmp_old_ptr;
2821         unsigned char *tmp_space;
2822         unsigned char *tmp_ptr;
2823         unsigned char *d_word_ptr;
2824
2825         /*
2826          |      resize line to handle undeleted word
2827          */
2828         if ((curr_line->max_length - (curr_line->line_length + d_wrd_len)) < 5)
2829                 point = resiz_line(d_wrd_len, curr_line, position);
2830         tmp_ptr = tmp_space = malloc(curr_line->line_length + d_wrd_len);
2831         d_word_ptr = d_word;
2832         temp = 1;
2833         /*
2834          |      copy d_word contents into temp space
2835          */
2836         while (temp <= d_wrd_len)
2837         {
2838                 temp++;
2839                 *tmp_ptr = *d_word_ptr;
2840                 tmp_ptr++;
2841                 d_word_ptr++;
2842         }
2843         tmp_old_ptr = point;
2844         tposit = position;
2845         /*
2846          |      copy contents of line from curent position to eol into 
2847          |      temp space
2848          */
2849         while (tposit < curr_line->line_length)
2850         {
2851                 temp++;
2852                 tposit++;
2853                 *tmp_ptr = *tmp_old_ptr;
2854                 tmp_ptr++;
2855                 tmp_old_ptr++;
2856         }
2857         curr_line->line_length += d_wrd_len;
2858         tmp_old_ptr = point;
2859         *tmp_ptr = '\0';
2860         tmp_ptr = tmp_space;
2861         tposit = 1;
2862         /*
2863          |      now copy contents from temp space back to original line
2864          */
2865         while (tposit < temp)
2866         {
2867                 tposit++;
2868                 *tmp_old_ptr = *tmp_ptr;
2869                 tmp_ptr++;
2870                 tmp_old_ptr++;
2871         }
2872         *tmp_old_ptr = '\0';
2873         free(tmp_space);
2874         draw_line(scr_vert, scr_horz, point, position, curr_line->line_length);
2875 }
2876
2877 void 
2878 del_line()                      /* delete from cursor to end of line    */
2879 {
2880         unsigned char *dl1;
2881         unsigned char *dl2;
2882         int tposit;
2883
2884         if (d_line != NULL)
2885                 free(d_line);
2886         d_line = malloc(curr_line->line_length);
2887         dl1 = d_line;
2888         dl2 = point;
2889         tposit = position;
2890         while (tposit < curr_line->line_length)
2891         {
2892                 *dl1 = *dl2;
2893                 dl1++;
2894                 dl2++;
2895                 tposit++;
2896         }
2897         dlt_line->line_length = 1 + tposit - position;
2898         *dl1 = '\0';
2899         *point = '\0';
2900         curr_line->line_length = position;
2901         wclrtoeol(text_win);
2902         if (curr_line->next_line != NULL)
2903         {
2904                 right(FALSE);
2905                 delete(FALSE);
2906         }
2907         text_changes = TRUE;
2908 }
2909
2910 void 
2911 undel_line()                    /* undelete last deleted line           */
2912 {
2913         unsigned char *ud1;
2914         unsigned char *ud2;
2915         int tposit;
2916
2917         if (dlt_line->line_length == 0)
2918                 return;
2919
2920         insert_line(TRUE);
2921         left(TRUE);
2922         point = resiz_line(dlt_line->line_length, curr_line, position);
2923         curr_line->line_length += dlt_line->line_length - 1;
2924         ud1 = point;
2925         ud2 = d_line;
2926         tposit = 1;
2927         while (tposit < dlt_line->line_length)
2928         {
2929                 tposit++;
2930                 *ud1 = *ud2;
2931                 ud1++;
2932                 ud2++;
2933         }
2934         *ud1 = '\0';
2935         draw_line(scr_vert, scr_horz,point,position,curr_line->line_length);
2936 }
2937
2938 void 
2939 adv_word()                      /* advance to next word         */
2940 {
2941 while ((position < curr_line->line_length) && ((*point != 32) && (*point != 9)))
2942                 right(TRUE);
2943 while ((position < curr_line->line_length) && ((*point == 32) || (*point == 9)))
2944                 right(TRUE);
2945 }
2946
2947 void 
2948 move_rel(direction, lines)      /* move relative to current line        */
2949 int direction;
2950 int lines;
2951 {
2952         int i;
2953         char *tmp;
2954
2955         if (direction == 'u')
2956         {
2957                 scr_pos = 0;
2958                 while (position > 1)
2959                         left(TRUE);
2960                 for (i = 0; i < lines; i++)
2961                 {
2962                         up();
2963                 }
2964                 if ((last_line > 5) && ( scr_vert < 4))
2965                 {
2966                         tmp = point;
2967                         tmp_line = curr_line;
2968                         for (i= 0;(i<5)&&(curr_line->prev_line != NULL); i++)
2969                         {
2970                                 up();
2971                         }
2972                         scr_vert = scr_vert + i;
2973                         curr_line = tmp_line;
2974                         absolute_lin += i;
2975                         point = tmp;
2976                         scanline(point);
2977                 }
2978         }
2979         else
2980         {
2981                 if ((position != 1) && (curr_line->next_line != NULL))
2982                 {
2983                         nextline();
2984                         scr_pos = scr_horz = 0;
2985                         if (horiz_offset)
2986                         {
2987                                 horiz_offset = 0;
2988                                 midscreen(scr_vert, point);
2989                         }
2990                 }
2991                 else
2992                         adv_line();
2993                 for (i = 1; i < lines; i++)
2994                 {
2995                         down();
2996                 }
2997                 if ((last_line > 10) && (scr_vert > (last_line - 5)))
2998                 {
2999                         tmp = point;
3000                         tmp_line = curr_line;
3001                         for (i=0; (i<5) && (curr_line->next_line != NULL); i++)
3002                         {
3003                                 down();
3004                         }
3005                         absolute_lin -= i;
3006                         scr_vert = scr_vert - i;
3007                         curr_line = tmp_line;
3008                         point = tmp;
3009                         scanline(point);
3010                 }
3011         }
3012         wmove(text_win, scr_vert, (scr_horz - horiz_offset));
3013 }
3014
3015 void 
3016 eol()                           /* go to end of line                    */
3017 {
3018         if (position < curr_line->line_length)
3019         {
3020                 while (position < curr_line->line_length)
3021                         right(TRUE);
3022         }
3023         else if (curr_line->next_line != NULL)
3024         {
3025                 right(TRUE);
3026                 while (position < curr_line->line_length)
3027                         right(TRUE);
3028         }
3029 }
3030
3031 void 
3032 bol()                           /* move to beginning of line    */
3033 {
3034         if (point != curr_line->line)
3035         {
3036                 while (point != curr_line->line)
3037                         left(TRUE);
3038         }
3039         else if (curr_line->prev_line != NULL)
3040         {
3041                 scr_pos = 0;
3042                 up();
3043         }
3044 }
3045
3046 void 
3047 adv_line()      /* advance to beginning of next line    */
3048 {
3049         if ((point != curr_line->line) || (scr_pos > 0))
3050         {
3051                 while (position < curr_line->line_length)
3052                         right(TRUE);
3053                 right(TRUE);
3054         }
3055         else if (curr_line->next_line != NULL)
3056         {
3057                 scr_pos = 0;
3058                 down();
3059         }
3060 }
3061
3062 void 
3063 from_top()
3064 {
3065         struct text *tmpline = first_line;
3066         int x = 1;
3067
3068         while ((tmpline != NULL) && (tmpline != curr_line))
3069         {
3070                 x++;
3071                 tmpline = tmpline->next_line;
3072         }
3073         absolute_lin = x;
3074 }
3075
3076 void 
3077 sh_command(string)      /* execute shell command                        */
3078 char *string;           /* string containing user command               */
3079 {
3080         char *temp_point;
3081         char *last_slash;
3082         char *path;             /* directory path to executable         */
3083         int parent;             /* zero if child, child's pid if parent */
3084         int value;
3085         int return_val;
3086         struct text *line_holder;
3087
3088         if (restrict_mode())
3089         {
3090                 return;
3091         }
3092
3093         if (!(path = getenv("SHELL")))
3094                 path = "/bin/sh";
3095         last_slash = temp_point = path;
3096         while (*temp_point != '\0')
3097         {
3098                 if (*temp_point == '/')
3099                         last_slash = ++temp_point;
3100                 else
3101                         temp_point++;
3102         }
3103
3104         /*
3105          |      if in_pipe is true, then output of the shell operation will be 
3106          |      read by the editor, and curses doesn't need to be turned off
3107          */
3108
3109         if (!in_pipe)
3110         {
3111                 keypad(com_win, FALSE);
3112                 keypad(text_win, FALSE);
3113                 echo();
3114                 nl();
3115                 noraw();
3116                 resetty();
3117
3118 #ifndef NCURSE
3119                 endwin();
3120 #endif
3121         }
3122
3123         if (in_pipe)
3124         {
3125                 pipe(pipe_in);          /* create a pipe        */
3126                 parent = fork();
3127                 if (!parent)            /* if the child         */
3128                 {
3129 /*
3130  |  child process which will fork and exec shell command (if shell output is
3131  |  to be read by editor)
3132  */
3133                         in_pipe = FALSE;
3134 /*
3135  |  redirect stdout to pipe
3136  */
3137                         temp_stdout = dup(1);
3138                         close(1);
3139                         dup(pipe_in[1]);
3140 /*
3141  |  redirect stderr to pipe
3142  */
3143                         temp_stderr = dup(2);
3144                         close(2);
3145                         dup(pipe_in[1]);
3146                         close(pipe_in[1]);
3147                         /*
3148                          |      child will now continue down 'if (!in_pipe)' 
3149                          |      path below
3150                          */
3151                 }
3152                 else  /* if the parent  */
3153                 {
3154 /*
3155  |  prepare editor to read from the pipe
3156  */
3157                         signal(SIGCHLD, SIG_IGN);
3158                         line_holder = curr_line;
3159                         tmp_vert = scr_vert;
3160                         close(pipe_in[1]);
3161                         get_fd = pipe_in[0];
3162                         get_file("");
3163                         close(pipe_in[0]);
3164                         scr_vert = tmp_vert;
3165                         scr_horz = scr_pos = 0;
3166                         position = 1;
3167                         curr_line = line_holder;
3168                         from_top();
3169                         point = curr_line->line;
3170                         out_pipe = FALSE;
3171                         signal(SIGCHLD, SIG_DFL);
3172 /*
3173  |  since flag "in_pipe" is still TRUE, the path which waits for the child 
3174  |  process to die will be avoided.
3175  |  (the pipe is closed, no more output can be expected)
3176  */
3177                 }
3178         }
3179         if (!in_pipe)
3180         {
3181                 signal(SIGINT, SIG_IGN);
3182                 if (out_pipe)
3183                 {
3184                         pipe(pipe_out);
3185                 }
3186 /*
3187  |  fork process which will exec command
3188  */
3189                 parent = fork();   
3190                 if (!parent)            /* if the child */
3191                 {
3192                         if (shell_fork)
3193                                 putchar('\n');
3194                         if (out_pipe)
3195                         {
3196 /*
3197  |  prepare the child process (soon to exec a shell command) to read from the 
3198  |  pipe (which will be output from the editor's buffer)
3199  */
3200                                 close(0);
3201                                 dup(pipe_out[0]);
3202                                 close(pipe_out[0]);
3203                                 close(pipe_out[1]);
3204                         }
3205                         for (value = 1; value < 24; value++)
3206                                 signal(value, SIG_DFL);
3207                         execl(path, last_slash, "-c", string, NULL);
3208                         fprintf(stderr, exec_err_msg, path);
3209                         exit(-1);
3210                 }
3211                 else    /* if the parent        */
3212                 {
3213                         if (out_pipe)
3214                         {
3215 /*
3216  |  output the contents of the buffer to the pipe (to be read by the 
3217  |  process forked and exec'd above as stdin)
3218  */
3219                                 close(pipe_out[0]);
3220                                 line_holder = first_line;
3221                                 while (line_holder != NULL)
3222                                 {
3223                                         write(pipe_out[1], line_holder->line, (line_holder->line_length-1));
3224                                         write(pipe_out[1], "\n", 1);
3225                                         line_holder = line_holder->next_line;
3226                                 }
3227                                 close(pipe_out[1]);
3228                                 out_pipe = FALSE;
3229                         }
3230                         do
3231                         {
3232                                 return_val = wait((int *) 0);
3233                         }
3234                         while ((return_val != parent) && (return_val != -1));
3235 /*
3236  |  if this process is actually the child of the editor, exit.  Here's how it 
3237  |  works:
3238  |  The editor forks a process.  If output must be sent to the command to be 
3239  |  exec'd another process is forked, and that process (the child's child) 
3240  |  will exec the command.  In this case, "shell_fork" will be FALSE.  If no 
3241  |  output is to be performed to the shell command, "shell_fork" will be TRUE.
3242  |  If this is the editor process, shell_fork will be true, otherwise this is 
3243  |  the child of the edit process.
3244  */
3245                         if (!shell_fork)
3246                                 exit(0);
3247                 }
3248                 signal(SIGINT, edit_abort);
3249         }
3250         if (shell_fork)
3251         {
3252                 fputs(continue_msg, stdout);
3253                 fflush(stdout);
3254                 while ((in = getchar()) != '\n')
3255                         ;
3256         }
3257
3258         if (!in_pipe)
3259         {
3260                 fixterm();
3261                 noecho();
3262                 nonl();
3263                 raw();
3264                 keypad(text_win, TRUE);
3265                 keypad(com_win, TRUE);
3266                 if (info_window)
3267                         clearok(info_win, TRUE);
3268         }
3269
3270         redraw();
3271 }
3272
3273 void 
3274 set_up_term()           /* set up the terminal for operating with ae    */
3275 {
3276         if (!curses_initialized)
3277         {
3278                 initscr();
3279                 savetty();
3280                 noecho();
3281                 raw();
3282                 nonl();
3283                 curses_initialized = TRUE;
3284         }
3285
3286         if (((LINES > 15) && (COLS >= 80)) && info_window)
3287                 last_line = LINES - 8;
3288         else
3289         {
3290                 info_window = FALSE;
3291                 last_line = LINES - 2;
3292         }
3293
3294         idlok(stdscr, TRUE);
3295         com_win = newwin(1, COLS, (LINES - 1), 0);
3296         keypad(com_win, TRUE);
3297         idlok(com_win, TRUE);
3298         wrefresh(com_win);
3299         if (!info_window)
3300                 text_win = newwin((LINES - 1), COLS, 0, 0);
3301         else
3302                 text_win = newwin((LINES - 7), COLS, 6, 0);
3303         keypad(text_win, TRUE);
3304         idlok(text_win, TRUE);
3305         wrefresh(text_win);
3306         help_win = newwin((LINES - 1), COLS, 0, 0);
3307         keypad(help_win, TRUE);
3308         idlok(help_win, TRUE);
3309         if (info_window)
3310         {
3311                 info_type = CONTROL_KEYS;
3312                 info_win = newwin(6, COLS, 0, 0);
3313                 werase(info_win);
3314                 paint_info_win();
3315         }
3316
3317         last_col = COLS - 1;
3318         local_LINES = LINES;
3319         local_COLS = COLS;
3320
3321 #ifdef NCURSE
3322         if (ee_chinese)
3323                 nc_setattrib(A_NC_BIG5);
3324 #endif /* NCURSE */
3325
3326 }
3327
3328 void 
3329 resize_check()
3330 {
3331         if ((LINES == local_LINES) && (COLS == local_COLS))
3332                 return;
3333
3334         if (info_window)
3335                 delwin(info_win);
3336         delwin(text_win);
3337         delwin(com_win);
3338         delwin(help_win);
3339         set_up_term();
3340         redraw();
3341         wrefresh(text_win);
3342 }
3343
3344 static char item_alpha[] = "abcdefghijklmnopqrstuvwxyz0123456789 ";
3345
3346 int 
3347 menu_op(menu_list)
3348 struct menu_entries menu_list[];
3349 {
3350         WINDOW *temp_win;
3351         int max_width, max_height;
3352         int x_off, y_off;
3353         int counter;
3354         int length;
3355         int input;
3356         int temp;
3357         int list_size;
3358         int top_offset;         /* offset from top where menu items start */
3359         int vert_pos;           /* vertical position                      */
3360         int vert_size;          /* vertical size for menu list item display */
3361         int off_start = 1;      /* offset from start of menu items to start display */
3362
3363
3364         /*
3365          |      determine number and width of menu items
3366          */
3367
3368         list_size = 1;
3369         while (menu_list[list_size + 1].item_string != NULL)
3370                 list_size++;
3371         max_width = 0;
3372         for (counter = 0; counter <= list_size; counter++)
3373         {
3374                 if ((length = strlen(menu_list[counter].item_string)) > max_width)
3375                         max_width = length;
3376         }
3377         max_width += 3;
3378         max_width = max(max_width, strlen(menu_cancel_msg));
3379         max_width = max(max_width, max(strlen(more_above_str), strlen(more_below_str)));
3380         max_width += 6;
3381
3382         /*
3383          |      make sure that window is large enough to handle menu
3384          |      if not, print error message and return to calling function
3385          */
3386
3387         if (max_width > COLS)
3388         {
3389                 wmove(com_win, 0, 0);
3390                 werase(com_win);
3391                 wprintw(com_win, menu_too_lrg_msg);
3392                 wrefresh(com_win);
3393                 clear_com_win = TRUE;
3394                 return(0);
3395         }
3396
3397         top_offset = 0;
3398
3399         if (list_size > LINES)
3400         {
3401                 max_height = LINES;
3402                 if (max_height > 11)
3403                         vert_size = max_height - 8;
3404                 else
3405                         vert_size = max_height;
3406         }
3407         else
3408         {
3409                 vert_size = list_size;
3410                 max_height = list_size;
3411         }
3412
3413         if (LINES >= (vert_size + 8))
3414         {
3415                 if (menu_list[0].argument != MENU_WARN)
3416                         max_height = vert_size + 8;
3417                 else
3418                         max_height = vert_size + 7;
3419                 top_offset = 4;
3420         }
3421         x_off = (COLS - max_width) / 2;
3422         y_off = (LINES - max_height - 1) / 2;
3423         temp_win = newwin(max_height, max_width, y_off, x_off);
3424         keypad(temp_win, TRUE);
3425
3426         paint_menu(menu_list, max_width, max_height, list_size, top_offset, temp_win, off_start, vert_size);
3427
3428         counter = 1;
3429         vert_pos = 0;
3430         do
3431         {
3432                 if (off_start > 2)
3433                         wmove(temp_win, (1 + counter + top_offset - off_start), 3);
3434                 else
3435                         wmove(temp_win, (counter + top_offset - off_start), 3);
3436
3437                 wrefresh(temp_win);
3438                 in = wgetch(temp_win);
3439                 input = in;
3440                 if (input == -1)
3441                         exit(0);
3442
3443                 if (isascii(input) && isalnum(input))
3444                 {
3445                         if (isalpha(input))
3446                         {
3447                                 temp = 1 + tolower(input) - 'a';
3448                         }
3449                         else if (isdigit(input))
3450                         {
3451                                 temp = (2 + 'z' - 'a') + (input - '0');
3452                         }
3453
3454                         if (temp <= list_size)
3455                         {
3456                                 input = '\n';
3457                                 counter = temp;
3458                         }
3459                 }
3460                 else
3461                 {               
3462                         switch (input)
3463                         {
3464                                 case ' ':       /* space        */
3465                                 case '\004':    /* ^d, down     */
3466                                 case KEY_RIGHT:
3467                                 case KEY_DOWN:
3468                                         counter++;
3469                                         if (counter > list_size)
3470                                                 counter = 1;
3471                                         break;
3472                                 case '\010':    /* ^h, backspace*/
3473                                 case '\025':    /* ^u, up       */
3474                                 case 127:       /* ^?, delete   */
3475                                 case KEY_BACKSPACE:
3476                                 case KEY_LEFT:
3477                                 case KEY_UP:
3478                                         counter--;
3479                                         if (counter == 0)
3480                                                 counter = list_size;
3481                                         break;
3482                                 case '\033':    /* escape key   */
3483                                         if (menu_list[0].argument != MENU_WARN)
3484                                                 counter = 0;
3485                                         break;
3486                                 case '\014':    /* ^l           */
3487                                 case '\022':    /* ^r, redraw   */
3488                                         paint_menu(menu_list, max_width, max_height, 
3489                                                 list_size, top_offset, temp_win, 
3490                                                 off_start, vert_size);
3491                                         break;
3492                                 default:
3493                                         break;
3494                         }
3495                 }
3496         
3497                 if (((list_size - off_start) >= (vert_size - 1)) && 
3498                         (counter > (off_start + vert_size - 3)) && 
3499                                 (off_start > 1))
3500                 {
3501                         if (counter == list_size)
3502                                 off_start = (list_size - vert_size) + 2;
3503                         else
3504                                 off_start++;
3505
3506                         paint_menu(menu_list, max_width, max_height, 
3507                                    list_size, top_offset, temp_win, off_start, 
3508                                    vert_size);
3509                 }
3510                 else if ((list_size != vert_size) && 
3511                                 (counter > (off_start + vert_size - 2)))
3512                 {
3513                         if (counter == list_size)
3514                                 off_start = 2 + (list_size - vert_size);
3515                         else if (off_start == 1)
3516                                 off_start = 3;
3517                         else
3518                                 off_start++;
3519
3520                         paint_menu(menu_list, max_width, max_height, 
3521                                    list_size, top_offset, temp_win, off_start, 
3522                                    vert_size);
3523                 }
3524                 else if (counter < off_start)
3525                 {
3526                         if (counter <= 2)
3527                                 off_start = 1;
3528                         else
3529                                 off_start = counter;
3530
3531                         paint_menu(menu_list, max_width, max_height, 
3532                                    list_size, top_offset, temp_win, off_start, 
3533                                    vert_size);
3534                 }
3535         }
3536         while ((input != '\r') && (input != '\n') && (counter != 0));
3537
3538         werase(temp_win);
3539         wrefresh(temp_win);
3540         delwin(temp_win);
3541
3542         if ((menu_list[counter].procedure != NULL) || 
3543             (menu_list[counter].iprocedure != NULL) || 
3544             (menu_list[counter].nprocedure != NULL))
3545         {
3546                 if (menu_list[counter].argument != -1)
3547                         (*menu_list[counter].iprocedure)(menu_list[counter].argument);
3548                 else if (menu_list[counter].ptr_argument != NULL)
3549                         (*menu_list[counter].procedure)(menu_list[counter].ptr_argument);
3550                 else
3551                         (*menu_list[counter].nprocedure)();
3552         }
3553
3554         if (info_window)
3555                 paint_info_win();
3556         redraw();
3557
3558         return(counter);
3559 }
3560
3561 void 
3562 paint_menu(menu_list, max_width, max_height, list_size, top_offset, menu_win, 
3563            off_start, vert_size)
3564 struct menu_entries menu_list[];
3565 int max_width, max_height, list_size, top_offset;
3566 WINDOW *menu_win;
3567 int off_start, vert_size;
3568 {
3569         int counter, temp_int;
3570
3571         werase(menu_win);
3572
3573         /*
3574          |      output top and bottom portions of menu box only if window 
3575          |      large enough 
3576          */
3577
3578         if (max_height > vert_size)
3579         {
3580                 wmove(menu_win, 1, 1);
3581                 if (!nohighlight)
3582                         wstandout(menu_win);
3583                 waddch(menu_win, '+');
3584                 for (counter = 0; counter < (max_width - 4); counter++)
3585                         waddch(menu_win, '-');
3586                 waddch(menu_win, '+');
3587
3588                 wmove(menu_win, (max_height - 2), 1);
3589                 waddch(menu_win, '+');
3590                 for (counter = 0; counter < (max_width - 4); counter++)
3591                         waddch(menu_win, '-');
3592                 waddch(menu_win, '+');
3593                 wstandend(menu_win);
3594                 wmove(menu_win, 2, 3);
3595                 waddstr(menu_win, menu_list[0].item_string);
3596                 wmove(menu_win, (max_height - 3), 3);
3597                 if (menu_list[0].argument != MENU_WARN)
3598                         waddstr(menu_win, menu_cancel_msg);
3599         }
3600         if (!nohighlight)
3601                 wstandout(menu_win);
3602
3603         for (counter = 0; counter < (vert_size + top_offset); counter++)
3604         {
3605                 if (top_offset == 4)
3606                 {
3607                         temp_int = counter + 2;
3608                 }
3609                 else
3610                         temp_int = counter;
3611
3612                 wmove(menu_win, temp_int, 1);
3613                 waddch(menu_win, '|');
3614                 wmove(menu_win, temp_int, (max_width - 2));
3615                 waddch(menu_win, '|');
3616         }
3617         wstandend(menu_win);
3618
3619         if (list_size > vert_size)
3620         {
3621                 if (off_start >= 3)
3622                 {
3623                         temp_int = 1;
3624                         wmove(menu_win, top_offset, 3);
3625                         waddstr(menu_win, more_above_str);
3626                 }
3627                 else
3628                         temp_int = 0;
3629
3630                 for (counter = off_start; 
3631                         ((temp_int + counter - off_start) < (vert_size - 1));
3632                                 counter++)
3633                 {
3634                         wmove(menu_win, (top_offset + temp_int + 
3635                                                 (counter - off_start)), 3);
3636                         if (list_size > 1)
3637                                 wprintw(menu_win, "%c) ", item_alpha[min((counter - 1), max_alpha_char)]);
3638                         waddstr(menu_win, menu_list[counter].item_string);
3639                 }
3640
3641                 wmove(menu_win, (top_offset + (vert_size - 1)), 3);
3642
3643                 if (counter == list_size)
3644                 {
3645                         if (list_size > 1)
3646                                 wprintw(menu_win, "%c) ", item_alpha[min((counter - 1), max_alpha_char)]);
3647                         wprintw(menu_win, menu_list[counter].item_string);
3648                 }
3649                 else
3650                         wprintw(menu_win, more_below_str);
3651         }
3652         else
3653         {
3654                 for (counter = 1; counter <= list_size; counter++)
3655                 {
3656                         wmove(menu_win, (top_offset + counter - 1), 3);
3657                         if (list_size > 1)
3658                                 wprintw(menu_win, "%c) ", item_alpha[min((counter - 1), max_alpha_char)]);
3659                         waddstr(menu_win, menu_list[counter].item_string);
3660                 }
3661         }
3662 }
3663
3664 void 
3665 help()
3666 {
3667         int counter;
3668
3669         werase(help_win);
3670         clearok(help_win, TRUE);
3671         for (counter = 0; counter < 22; counter++)
3672         {
3673                 wmove(help_win, counter, 0);
3674                 waddstr(help_win, (emacs_keys_mode) ? 
3675                         emacs_help_text[counter] : help_text[counter]);
3676         }
3677         wrefresh(help_win);
3678         werase(com_win);
3679         wmove(com_win, 0, 0);
3680         wprintw(com_win, press_any_key_msg);
3681         wrefresh(com_win);
3682         counter = wgetch(com_win);
3683         if (counter == -1)
3684                 exit(0);
3685         werase(com_win);
3686         wmove(com_win, 0, 0);
3687         werase(help_win);
3688         wrefresh(help_win);
3689         wrefresh(com_win);
3690         redraw();
3691 }
3692
3693 void 
3694 paint_info_win()
3695 {
3696         int counter;
3697
3698         if (!info_window)
3699                 return;
3700
3701         werase(info_win);
3702         for (counter = 0; counter < 5; counter++)
3703         {
3704                 wmove(info_win, counter, 0);
3705                 wclrtoeol(info_win);
3706                 if (info_type == CONTROL_KEYS)
3707                         waddstr(info_win, (emacs_keys_mode) ? 
3708                           emacs_control_keys[counter] : control_keys[counter]);
3709                 else if (info_type == COMMANDS)
3710                         waddstr(info_win, command_strings[counter]);
3711         }
3712         wmove(info_win, 5, 0);
3713         if (!nohighlight)
3714                 wstandout(info_win);
3715         waddstr(info_win, separator);
3716         wstandend(info_win);
3717         wrefresh(info_win);
3718 }
3719
3720 void 
3721 no_info_window()
3722 {
3723         if (!info_window)
3724                 return;
3725         delwin(info_win);
3726         delwin(text_win);
3727         info_window = FALSE;
3728         last_line = LINES - 2;
3729         text_win = newwin((LINES - 1), COLS, 0, 0);
3730         keypad(text_win, TRUE);
3731         idlok(text_win, TRUE);
3732         clearok(text_win, TRUE);
3733         midscreen(scr_vert, point);
3734         wrefresh(text_win);
3735         clear_com_win = TRUE;
3736 }
3737
3738 void 
3739 create_info_window()
3740 {
3741         if (info_window)
3742                 return;
3743         last_line = LINES - 8;
3744         delwin(text_win);
3745         text_win = newwin((LINES - 7), COLS, 6, 0);
3746         keypad(text_win, TRUE);
3747         idlok(text_win, TRUE);
3748         werase(text_win);
3749         info_window = TRUE;
3750         info_win = newwin(6, COLS, 0, 0);
3751         werase(info_win);
3752         info_type = CONTROL_KEYS;
3753         midscreen(min(scr_vert, last_line), point);
3754         clearok(info_win, TRUE);
3755         paint_info_win();
3756         wrefresh(text_win);
3757         clear_com_win = TRUE;
3758 }
3759
3760 int 
3761 file_op(arg)
3762 int arg;
3763 {
3764         char *string;
3765         int flag;
3766
3767         if (restrict_mode())
3768         {
3769                 return(0);
3770         }
3771
3772         if (arg == READ_FILE)
3773         {
3774                 string = get_string(file_read_prompt_str, TRUE);
3775                 recv_file = TRUE;
3776                 tmp_file = resolve_name(string);
3777                 check_fp();
3778                 if (tmp_file != string)
3779                         free(tmp_file);
3780                 free(string);
3781         }
3782         else if (arg == WRITE_FILE)
3783         {
3784                 string = get_string(file_write_prompt_str, TRUE);
3785                 tmp_file = resolve_name(string);
3786                 write_file(tmp_file, 1);
3787                 if (tmp_file != string)
3788                         free(tmp_file);
3789                 free(string);
3790         }
3791         else if (arg == SAVE_FILE)
3792         {
3793         /*
3794          |      changes made here should be reflected in finish()
3795          */
3796
3797                 if (in_file_name)
3798                         flag = TRUE;
3799                 else
3800                         flag = FALSE;
3801
3802                 string = in_file_name;
3803                 if ((string == NULL) || (*string == '\0'))
3804                         string = get_string(save_file_name_prompt, TRUE);
3805                 if ((string == NULL) || (*string == '\0'))
3806                 {
3807                         wmove(com_win, 0, 0);
3808                         wprintw(com_win, file_not_saved_msg);
3809                         wclrtoeol(com_win);
3810                         wrefresh(com_win);
3811                         clear_com_win = TRUE;
3812                         return(0);
3813                 }
3814                 if (!flag)
3815                 {
3816                         tmp_file = resolve_name(string);
3817                         if (tmp_file != string)
3818                         {
3819                                 free(string);
3820                                 string = tmp_file;
3821                         }
3822                 }
3823                 if (write_file(string, 1))
3824                 {
3825                         in_file_name = string;
3826                         text_changes = FALSE;
3827                 }
3828                 else if (!flag)
3829                         free(string);
3830         }
3831         return(0);
3832 }
3833
3834 void 
3835 shell_op()
3836 {
3837         char *string;
3838
3839         if (((string = get_string(shell_prompt, TRUE)) != NULL) && 
3840                         (*string != '\0'))
3841         {
3842                 sh_command(string);
3843                 free(string);
3844         }
3845 }
3846
3847 void 
3848 leave_op()
3849 {
3850         if (text_changes)
3851         {
3852                 menu_op(leave_menu);
3853         }
3854         else
3855                 quit(TRUE);
3856 }
3857
3858 void 
3859 redraw()
3860 {
3861         if (info_window)
3862         {
3863                 clearok(info_win, TRUE);
3864                 paint_info_win();
3865         }
3866         else
3867                 clearok(text_win, TRUE);
3868         midscreen(scr_vert, point);
3869 }
3870
3871 /*
3872  |      The following routines will "format" a paragraph (as defined by a 
3873  |      block of text with blank lines before and after the block).
3874  */
3875
3876 int 
3877 Blank_Line(test_line)   /* test if line has any non-space characters    */
3878 struct text *test_line;
3879 {
3880         unsigned char *line;
3881         int length;
3882         
3883         if (test_line == NULL)
3884                 return(TRUE);
3885
3886         length = 1;
3887         line = test_line->line;
3888
3889         /*
3890          |      To handle troff/nroff documents, consider a line with a 
3891          |      period ('.') in the first column to be blank.  To handle mail 
3892          |      messages with included text, consider a line with a '>' blank.
3893          */
3894
3895         if ((*line == '.') || (*line == '>'))
3896                 return(TRUE);
3897
3898         while (((*line == ' ') || (*line == '\t')) && (length < test_line->line_length))
3899         {
3900                 length++;
3901                 line++;
3902         }
3903         if (length != test_line->line_length)
3904                 return(FALSE);
3905         else
3906                 return(TRUE);
3907 }
3908
3909 void 
3910 Format()        /* format the paragraph according to set margins        */
3911 {
3912         int string_count;
3913         int offset;
3914         int temp_case;
3915         int status;
3916         int tmp_af;
3917         int counter;
3918         unsigned char *line;
3919         unsigned char *tmp_srchstr;
3920         unsigned char *temp1, *temp2;
3921         unsigned char *temp_dword;
3922         unsigned char temp_d_char[3];
3923
3924         temp_d_char[0] = d_char[0];
3925         temp_d_char[1] = d_char[1];
3926         temp_d_char[2] = d_char[2];
3927
3928 /*
3929  |      if observ_margins is not set, or the current line is blank, 
3930  |      do not format the current paragraph
3931  */
3932
3933         if ((!observ_margins) || (Blank_Line(curr_line)))
3934                 return;
3935
3936 /*
3937  |      save the currently set flags, and clear them
3938  */
3939
3940         wmove(com_win, 0, 0);
3941         wclrtoeol(com_win);
3942         wprintw(com_win, formatting_msg);
3943         wrefresh(com_win);
3944
3945 /*
3946  |      get current position in paragraph, so after formatting, the cursor 
3947  |      will be in the same relative position
3948  */
3949
3950         tmp_af = auto_format;
3951         auto_format = FALSE;
3952         offset = position;
3953         if (position != 1)
3954                 prev_word();
3955         temp_dword = d_word;
3956         d_word = NULL;
3957         temp_case = case_sen;
3958         case_sen = TRUE;
3959         tmp_srchstr = srch_str;
3960         temp2 = srch_str = (unsigned char *) malloc(1 + curr_line->line_length - position);
3961         if ((*point == ' ') || (*point == '\t'))
3962                 adv_word();
3963         offset -= position;
3964         counter = position;
3965         line = temp1 = point;
3966         while ((*temp1 != '\0') && (*temp1 != ' ') && (*temp1 != '\t') && (counter < curr_line->line_length))
3967         {
3968                 *temp2 = *temp1;
3969                 temp2++;
3970                 temp1++;
3971                 counter++;
3972         }
3973         *temp2 = '\0';
3974         if (position != 1)
3975                 bol();
3976         while (!Blank_Line(curr_line->prev_line))
3977                 bol();
3978         string_count = 0;
3979         status = TRUE;
3980         while ((line != point) && (status))
3981         {
3982                 status = search(FALSE);
3983                 string_count++;
3984         }
3985
3986         wmove(com_win, 0, 0);
3987         wclrtoeol(com_win);
3988         wprintw(com_win, formatting_msg);
3989         wrefresh(com_win);
3990
3991 /*
3992  |      now get back to the start of the paragraph to start formatting
3993  */
3994
3995         if (position != 1)
3996                 bol();
3997         while (!Blank_Line(curr_line->prev_line))
3998                 bol();
3999
4000         observ_margins = FALSE;
4001
4002 /*
4003  |      Start going through lines, putting spaces at end of lines if they do 
4004  |      not already exist.  Append lines together to get one long line, and 
4005  |      eliminate spacing at begin of lines.
4006  */
4007
4008         while (!Blank_Line(curr_line->next_line))
4009         {
4010                 eol();
4011                 left(TRUE);
4012                 if (*point != ' ')
4013                 {
4014                         right(TRUE);
4015                         insert(' ');
4016                 }
4017                 else
4018                         right(TRUE);
4019                 del_char();
4020                 if ((*point == ' ') || (*point == '\t'))
4021                         del_word();
4022         }
4023
4024 /*
4025  |      Now there is one long line.  Eliminate extra spaces within the line
4026  |      after the first word (so as not to blow away any indenting the user 
4027  |      may have put in).
4028  */
4029
4030         bol();
4031         adv_word();
4032         while (position < curr_line->line_length)
4033         {
4034                 if ((*point == ' ') && (*(point + 1) == ' '))
4035                         del_char();
4036                 else
4037                         right(TRUE);
4038         }
4039
4040 /*
4041  |      Now make sure there are two spaces after a '.'.
4042  */
4043
4044         bol();
4045         while (position < curr_line->line_length)
4046         {
4047                 if ((*point == '.') && (*(point + 1) == ' '))
4048                 {
4049                         right(TRUE);
4050                         insert(' ');
4051                         insert(' ');
4052                         while (*point == ' ')
4053                                 del_char();
4054                 }
4055                 right(TRUE);
4056         }
4057
4058         observ_margins = TRUE;
4059         bol();
4060
4061         wmove(com_win, 0, 0);
4062         wclrtoeol(com_win);
4063         wprintw(com_win, formatting_msg);
4064         wrefresh(com_win);
4065
4066 /*
4067  |      create lines between margins
4068  */
4069
4070         while (position < curr_line->line_length)
4071         {
4072                 while ((scr_pos < right_margin) && (position < curr_line->line_length))
4073                         right(TRUE);
4074                 if (position < curr_line->line_length)
4075                 {
4076                         prev_word();
4077                         if (position == 1)
4078                                 adv_word();
4079                         insert_line(TRUE);
4080                 }
4081         }
4082
4083 /*
4084  |      go back to begin of paragraph, put cursor back to original position
4085  */
4086
4087         bol();
4088         while (!Blank_Line(curr_line->prev_line))
4089                 bol();
4090
4091 /*
4092  |      find word cursor was in
4093  */
4094
4095         while ((status) && (string_count > 0))
4096         {
4097                 search(FALSE);
4098                 string_count--;
4099         }
4100
4101 /*
4102  |      offset the cursor to where it was before from the start of the word
4103  */
4104
4105         while (offset > 0)
4106         {
4107                 offset--;
4108                 right(TRUE);
4109         }
4110
4111 /*
4112  |      reset flags and strings to what they were before formatting
4113  */
4114
4115         if (d_word != NULL)
4116                 free(d_word);
4117         d_word = temp_dword;
4118         case_sen = temp_case;
4119         free(srch_str);
4120         srch_str = tmp_srchstr;
4121         d_char[0] = temp_d_char[0];
4122         d_char[1] = temp_d_char[1];
4123         d_char[2] = temp_d_char[2];
4124         auto_format = tmp_af;
4125
4126         midscreen(scr_vert, point);
4127         werase(com_win);
4128         wrefresh(com_win);
4129 }
4130
4131 unsigned char *init_name[3] = {
4132         "/usr/share/misc/init.ee", 
4133         NULL, 
4134         ".init.ee"
4135         };
4136
4137 void 
4138 ee_init()       /* check for init file and read it if it exists */
4139 {
4140         FILE *init_file;
4141         unsigned char *string;
4142         unsigned char *str1;
4143         unsigned char *str2;
4144         char *home;
4145         int counter;
4146         int temp_int;
4147
4148         string = getenv("HOME");
4149         if (string == NULL)
4150                 string = "/tmp";
4151         str1 = home = malloc(strlen(string)+10);
4152         strcpy(home, string);
4153         strcat(home, "/.init.ee");
4154         init_name[1] = home;
4155         string = malloc(512);
4156
4157         for (counter = 0; counter < 3; counter++)
4158         {
4159                 if (!(access(init_name[counter], 4)))
4160                 {
4161                         init_file = fopen(init_name[counter], "r");
4162                         while ((str2 = fgets(string, 512, init_file)) != NULL)
4163                         {
4164                                 str1 = str2 = string;
4165                                 while (*str2 != '\n')
4166                                         str2++;
4167                                 *str2 = '\0';
4168
4169                                 if (unique_test(string, init_strings) != 1)
4170                                         continue;
4171
4172                                 if (compare(str1, CASE, FALSE))
4173                                         case_sen = TRUE;
4174                                 else if (compare(str1, NOCASE, FALSE))
4175                                         case_sen = FALSE;
4176                                 else if (compare(str1, EXPAND, FALSE))
4177                                         expand_tabs = TRUE;
4178                                 else if (compare(str1, NOEXPAND, FALSE))
4179                                         expand_tabs = FALSE;
4180                                 else if (compare(str1, INFO, FALSE))
4181                                         info_window = TRUE;
4182                                 else if (compare(str1, NOINFO, FALSE))
4183                                         info_window = FALSE;   
4184                                 else if (compare(str1, MARGINS, FALSE))
4185                                         observ_margins = TRUE;
4186                                 else if (compare(str1, NOMARGINS, FALSE))
4187                                         observ_margins = FALSE;
4188                                 else if (compare(str1, AUTOFORMAT, FALSE))
4189                                 {
4190                                         auto_format = TRUE;
4191                                         observ_margins = TRUE;
4192                                 }
4193                                 else if (compare(str1, NOAUTOFORMAT, FALSE))
4194                                         auto_format = FALSE;
4195                                 else if (compare(str1, Echo, FALSE))
4196                                 {
4197                                         str1 = next_word(str1);
4198                                         if (*str1 != '\0')
4199                                                 echo_string(str1);
4200                                 }
4201                                 else if (compare(str1, PRINTCOMMAND, FALSE))
4202                                 {
4203                                         str1 = next_word(str1);
4204                                         print_command = malloc(strlen(str1)+1);
4205                                         strcpy(print_command, str1);
4206                                 }
4207                                 else if (compare(str1, RIGHTMARGIN, FALSE))
4208                                 {
4209                                         str1 = next_word(str1);
4210                                         if ((*str1 >= '0') && (*str1 <= '9'))
4211                                         {
4212                                                 temp_int = atoi(str1);
4213                                                 if (temp_int > 0)
4214                                                         right_margin = temp_int;
4215                                         }
4216                                 }
4217                                 else if (compare(str1, HIGHLIGHT, FALSE))
4218                                         nohighlight = FALSE;
4219                                 else if (compare(str1, NOHIGHLIGHT, FALSE))
4220                                         nohighlight = TRUE;
4221                                 else if (compare(str1, EIGHTBIT, FALSE))
4222                                         eightbit = TRUE;
4223                                 else if (compare(str1, NOEIGHTBIT, FALSE))
4224                                 {
4225                                         eightbit = FALSE;
4226                                         ee_chinese = FALSE;
4227                                 }
4228                                 else if (compare(str1, EMACS_string, FALSE))
4229                                         emacs_keys_mode = TRUE;
4230                                 else if (compare(str1, NOEMACS_string, FALSE))
4231                                         emacs_keys_mode = FALSE;
4232                                 else if (compare(str1, chinese_cmd, FALSE))
4233                                 {
4234                                         ee_chinese = TRUE;
4235                                         eightbit = TRUE;
4236                                 }
4237                                 else if (compare(str1, nochinese_cmd, FALSE))
4238                                         ee_chinese = FALSE;
4239                         }
4240                         fclose(init_file);
4241                 }
4242         }
4243         free(string);
4244         free(home);
4245
4246         string = getenv("LANG");
4247         if (string != NULL)
4248         {
4249                 if (strcmp(string, "zh_TW.big5") == 0)
4250                 {
4251                         ee_chinese = TRUE;
4252                         eightbit = TRUE;
4253                 }
4254         }
4255 }
4256
4257 /*
4258  |      Save current configuration to .init.ee file in the current directory.
4259  */
4260
4261 void 
4262 dump_ee_conf()  
4263 {
4264         FILE *init_file;
4265         FILE *old_init_file = NULL;
4266         char *file_name = ".init.ee";
4267         char *home_dir =  "~/.init.ee";
4268         char buffer[512];
4269         struct stat buf;
4270         char *string;
4271         int length;
4272         int option = 0;
4273
4274         if (restrict_mode())
4275         {
4276                 return;
4277         }
4278
4279         option = menu_op(config_dump_menu);
4280
4281         werase(com_win);
4282         wmove(com_win, 0, 0);
4283
4284         if (option == 0)
4285         {
4286                 wprintw(com_win, conf_not_saved_msg);
4287                 wrefresh(com_win);
4288                 return;
4289         }
4290         else if (option == 2)
4291                 file_name = resolve_name(home_dir);
4292
4293         /*
4294          |      If a .init.ee file exists, move it to .init.ee.old.
4295          */
4296
4297         if (stat(file_name, &buf) != -1)
4298         {
4299                 sprintf(buffer, "%s.old", file_name);
4300                 unlink(buffer);
4301                 link(file_name, buffer);
4302                 unlink(file_name);
4303                 old_init_file = fopen(buffer, "r");
4304         }
4305
4306         init_file = fopen(file_name, "w");
4307         if (init_file == NULL)
4308         {
4309                 wprintw(com_win, conf_dump_err_msg);
4310                 wrefresh(com_win);
4311                 return;
4312         }
4313
4314         if (old_init_file != NULL)
4315         {
4316                 /*
4317                  |      Copy non-configuration info into new .init.ee file.
4318                  */
4319                 while ((string = fgets(buffer, 512, old_init_file)) != NULL)
4320                 {
4321                         length = strlen(string);
4322                         string[length - 1] = '\0';
4323
4324                         if (unique_test(string, init_strings) == 1)
4325                         {
4326                                 if (compare(string, Echo, FALSE))
4327                                 {
4328                                         fprintf(init_file, "%s\n", string);
4329                                 }
4330                         }
4331                         else
4332                                 fprintf(init_file, "%s\n", string);
4333                 }
4334
4335                 fclose(old_init_file);
4336         }
4337
4338         fprintf(init_file, "%s\n", case_sen ? CASE : NOCASE);
4339         fprintf(init_file, "%s\n", expand_tabs ? EXPAND : NOEXPAND);
4340         fprintf(init_file, "%s\n", info_window ? INFO : NOINFO );
4341         fprintf(init_file, "%s\n", observ_margins ? MARGINS : NOMARGINS );
4342         fprintf(init_file, "%s\n", auto_format ? AUTOFORMAT : NOAUTOFORMAT );
4343         fprintf(init_file, "%s %s\n", PRINTCOMMAND, print_command);
4344         fprintf(init_file, "%s %d\n", RIGHTMARGIN, right_margin);
4345         fprintf(init_file, "%s\n", nohighlight ? NOHIGHLIGHT : HIGHLIGHT );
4346         fprintf(init_file, "%s\n", eightbit ? EIGHTBIT : NOEIGHTBIT );
4347         fprintf(init_file, "%s\n", emacs_keys_mode ? EMACS_string : NOEMACS_string );
4348         fprintf(init_file, "%s\n", ee_chinese ? chinese_cmd : nochinese_cmd );
4349
4350         fclose(init_file);
4351
4352         wprintw(com_win, conf_dump_success_msg, file_name);
4353         wrefresh(com_win);
4354
4355         if ((option == 2) && (file_name != home_dir))
4356         {
4357                 free(file_name);
4358         }
4359 }
4360
4361 void 
4362 echo_string(string)     /* echo the given string        */
4363 char *string;
4364 {
4365         char *temp;
4366         int Counter;
4367
4368                 temp = string;
4369                 while (*temp != '\0')
4370                 {
4371                         if (*temp == '\\')
4372                         {
4373                                 temp++;
4374                                 if (*temp == 'n')
4375                                         putchar('\n');
4376                                 else if (*temp == 't')
4377                                         putchar('\t');
4378                                 else if (*temp == 'b')
4379                                         putchar('\b');
4380                                 else if (*temp == 'r')
4381                                         putchar('\r');
4382                                 else if (*temp == 'f')
4383                                         putchar('\f');
4384                                 else if ((*temp == 'e') || (*temp == 'E'))
4385                                         putchar('\033');        /* escape */
4386                                 else if (*temp == '\\')
4387                                         putchar('\\');
4388                                 else if (*temp == '\'')
4389                                         putchar('\'');
4390                                 else if ((*temp >= '0') && (*temp <= '9'))
4391                                 {
4392                                         Counter = 0;
4393                                         while ((*temp >= '0') && (*temp <= '9'))
4394                                         {
4395                                                 Counter = (8 * Counter) + (*temp - '0');
4396                                                 temp++;
4397                                         }
4398                                         putchar(Counter);
4399                                         temp--;
4400                                 }
4401                                 temp++;
4402                         }
4403                         else
4404                         {
4405                                 putchar(*temp);
4406                                 temp++;
4407                         }
4408                 }
4409
4410         fflush(stdout);
4411 }
4412
4413 void 
4414 spell_op()      /* check spelling of words in the editor        */
4415 {
4416         if (restrict_mode())
4417         {
4418                 return;
4419         }
4420         top();                  /* go to top of file            */
4421         insert_line(FALSE);     /* create two blank lines       */
4422         insert_line(FALSE);
4423         top();
4424         command(shell_echo_msg);
4425         adv_line();
4426         wmove(com_win, 0, 0);
4427         wprintw(com_win, spell_in_prog_msg);
4428         wrefresh(com_win);
4429         command("<>!spell");    /* send contents of buffer to command 'spell' 
4430                                    and read the results back into the editor */
4431 }
4432
4433 void 
4434 ispell_op()
4435 {
4436         char template[128], *name;
4437         char string[256];
4438         int fd;
4439
4440         if (restrict_mode())
4441         {
4442                 return;
4443         }
4444         (void)sprintf(template, "/tmp/ee.XXXXXXXX");
4445         fd = mkstemp(template);
4446         if (fd < 0) {
4447                 wmove(com_win, 0, 0);
4448                 wprintw(com_win, create_file_fail_msg, name);
4449                 wrefresh(com_win);
4450                 return;
4451         }
4452         close(fd);
4453         if (write_file(name, 0))
4454         {
4455                 sprintf(string, "ispell %s", name);
4456                 sh_command(string);
4457                 delete_text();
4458                 tmp_file = name;
4459                 recv_file = TRUE;
4460                 check_fp();
4461                 unlink(name);
4462         }
4463 }
4464
4465 int
4466 first_word_len(test_line)
4467 struct text *test_line;
4468 {
4469         int counter;
4470         unsigned char *pnt;
4471
4472         if (test_line == NULL)
4473                 return(0);
4474
4475         pnt = test_line->line;
4476         if ((pnt == NULL) || (*pnt == '\0') || 
4477             (*pnt == '.') || (*pnt == '>'))
4478                 return(0);
4479
4480         if ((*pnt == ' ') || (*pnt == '\t'))
4481         {
4482                 pnt = next_word(pnt);
4483         }
4484
4485         if (*pnt == '\0')
4486                 return(0);
4487
4488         counter = 0;
4489         while ((*pnt != '\0') && ((*pnt != ' ') && (*pnt != '\t')))
4490         {
4491                 pnt++;
4492                 counter++;
4493         }
4494         while ((*pnt != '\0') && ((*pnt == ' ') || (*pnt == '\t')))
4495         {
4496                 pnt++;
4497                 counter++;
4498         }
4499         return(counter);
4500 }
4501
4502 void 
4503 Auto_Format()   /* format the paragraph according to set margins        */
4504 {
4505         int string_count;
4506         int offset;
4507         int temp_case;
4508         int word_len;
4509         int temp_dwl;
4510         int tmp_d_line_length;
4511         int leave_loop = FALSE;
4512         int status;
4513         int counter;
4514         char not_blank;
4515         unsigned char *line;
4516         unsigned char *tmp_srchstr;
4517         unsigned char *temp1, *temp2;
4518         unsigned char *temp_dword;
4519         unsigned char temp_d_char[3];
4520         unsigned char *tmp_d_line;
4521
4522
4523         temp_d_char[0] = d_char[0];
4524         temp_d_char[1] = d_char[1];
4525         temp_d_char[2] = d_char[2];
4526
4527 /*
4528  |      if observ_margins is not set, or the current line is blank, 
4529  |      do not format the current paragraph
4530  */
4531
4532         if ((!observ_margins) || (Blank_Line(curr_line)))
4533                 return;
4534
4535 /*
4536  |      get current position in paragraph, so after formatting, the cursor 
4537  |      will be in the same relative position
4538  */
4539
4540         tmp_d_line = d_line;
4541         tmp_d_line_length = dlt_line->line_length;
4542         d_line = NULL;
4543         auto_format = FALSE;
4544         offset = position;
4545         if ((position != 1) && ((*point == ' ') || (*point == '\t') || (position == curr_line->line_length) || (*point == '\0')))
4546                 prev_word();
4547         temp_dword = d_word;
4548         temp_dwl = d_wrd_len;
4549         d_wrd_len = 0;
4550         d_word = NULL;
4551         temp_case = case_sen;
4552         case_sen = TRUE;
4553         tmp_srchstr = srch_str;
4554         temp2 = srch_str = (unsigned char *) malloc(1 + curr_line->line_length - position);
4555         if ((*point == ' ') || (*point == '\t'))
4556                 adv_word();
4557         offset -= position;
4558         counter = position;
4559         line = temp1 = point;
4560         while ((*temp1 != '\0') && (*temp1 != ' ') && (*temp1 != '\t') && (counter < curr_line->line_length))
4561         {
4562                 *temp2 = *temp1;
4563                 temp2++;
4564                 temp1++;
4565                 counter++;
4566         }
4567         *temp2 = '\0';
4568         if (position != 1)
4569                 bol();
4570         while (!Blank_Line(curr_line->prev_line))
4571                 bol();
4572         string_count = 0;
4573         status = TRUE;
4574         while ((line != point) && (status))
4575         {
4576                 status = search(FALSE);
4577                 string_count++;
4578         }
4579
4580 /*
4581  |      now get back to the start of the paragraph to start checking
4582  */
4583
4584         if (position != 1)
4585                 bol();
4586         while (!Blank_Line(curr_line->prev_line))
4587                 bol();
4588
4589 /*
4590  |      Start going through lines, putting spaces at end of lines if they do 
4591  |      not already exist.  Check line length, and move words to the next line 
4592  |      if they cross the margin.  Then get words from the next line if they 
4593  |      will fit in before the margin.  
4594  */
4595
4596         counter = 0;
4597
4598         while (!leave_loop)
4599         {
4600                 if (position != curr_line->line_length)
4601                         eol();
4602                 left(TRUE);
4603                 if (*point != ' ')
4604                 {
4605                         right(TRUE);
4606                         insert(' ');
4607                 }
4608                 else
4609                         right(TRUE);
4610
4611                 not_blank = FALSE;
4612
4613                 /*
4614                  |      fill line if first word on next line will fit 
4615                  |      in the line without crossing the margin
4616                  */
4617
4618                 while ((curr_line->next_line != NULL) && 
4619                        ((word_len = first_word_len(curr_line->next_line)) > 0) 
4620                         && ((scr_pos + word_len) < right_margin))
4621                 {
4622                         adv_line();
4623                         if ((*point == ' ') || (*point == '\t'))
4624                                 adv_word();
4625                         del_word();
4626                         if (position != 1)
4627                                 bol();
4628
4629                         /*
4630                          |      We know this line was not blank before, so 
4631                          |      make sure that it doesn't have one of the 
4632                          |      leading characters that indicate the line 
4633                          |      should not be modified.
4634                          |
4635                          |      We also know that this character should not 
4636                          |      be left as the first character of this line.
4637                          */
4638
4639                         if ((Blank_Line(curr_line)) && 
4640                             (curr_line->line[0] != '.') && 
4641                             (curr_line->line[0] != '>'))
4642                         {
4643                                 del_line();
4644                                 not_blank = FALSE;
4645                         }
4646                         else
4647                                 not_blank = TRUE;
4648
4649                         /*
4650                          |   go to end of previous line
4651                          */
4652                         left(TRUE);
4653                         undel_word();
4654                         eol();
4655                         /*
4656                          |   make sure there's a space at the end of the line
4657                          */
4658                         left(TRUE);
4659                         if (*point != ' ')
4660                         {
4661                                 right(TRUE);
4662                                 insert(' ');
4663                         }
4664                         else
4665                                 right(TRUE);
4666                 }
4667
4668                 /*
4669                  |      make sure line does not cross right margin
4670                  */
4671
4672                 while (right_margin <= scr_pos)
4673                 {
4674                         prev_word();
4675                         if (position != 1)
4676                         {
4677                                 del_word();
4678                                 if (Blank_Line(curr_line->next_line))
4679                                         insert_line(TRUE);
4680                                 else
4681                                         adv_line();
4682                                 if ((*point == ' ') || (*point == '\t'))
4683                                         adv_word();
4684                                 undel_word();
4685                                 not_blank = TRUE;
4686                                 if (position != 1)
4687                                         bol();
4688                                 left(TRUE);
4689                         }
4690                 }
4691
4692                 if ((!Blank_Line(curr_line->next_line)) || (not_blank))
4693                 {
4694                         adv_line();
4695                         counter++;
4696                 }
4697                 else
4698                         leave_loop = TRUE;
4699         }
4700
4701 /*
4702  |      go back to begin of paragraph, put cursor back to original position
4703  */
4704
4705         if (position != 1)
4706                 bol();
4707         while ((counter-- > 0) || (!Blank_Line(curr_line->prev_line)))
4708                 bol();
4709
4710 /*
4711  |      find word cursor was in
4712  */
4713
4714         status = TRUE;
4715         while ((status) && (string_count > 0))
4716         {
4717                 status = search(FALSE);
4718                 string_count--;
4719         }
4720
4721 /*
4722  |      offset the cursor to where it was before from the start of the word
4723  */
4724
4725         while (offset > 0)
4726         {
4727                 offset--;
4728                 right(TRUE);
4729         }
4730
4731         if ((string_count > 0) && (offset < 0))
4732         {
4733                 while (offset < 0)
4734                 {
4735                         offset++;
4736                         left(TRUE);
4737                 }
4738         }
4739
4740 /*
4741  |      reset flags and strings to what they were before formatting
4742  */
4743
4744         if (d_word != NULL)
4745                 free(d_word);
4746         d_word = temp_dword;
4747         d_wrd_len = temp_dwl;
4748         case_sen = temp_case;
4749         free(srch_str);
4750         srch_str = tmp_srchstr;
4751         d_char[0] = temp_d_char[0];
4752         d_char[1] = temp_d_char[1];
4753         d_char[2] = temp_d_char[2];
4754         auto_format = TRUE;
4755         dlt_line->line_length = tmp_d_line_length;
4756         d_line = tmp_d_line;
4757
4758         formatted = TRUE;
4759         midscreen(scr_vert, point);
4760 }
4761
4762 void 
4763 modes_op()
4764 {
4765         int ret_value;
4766         int counter;
4767         char *string;
4768
4769         do
4770         {
4771                 sprintf(modes_menu[1].item_string, "%s %s", mode_strings[1], 
4772                                         (expand_tabs ? ON : OFF));
4773                 sprintf(modes_menu[2].item_string, "%s %s", mode_strings[2], 
4774                                         (case_sen ? ON : OFF));
4775                 sprintf(modes_menu[3].item_string, "%s %s", mode_strings[3], 
4776                                         (observ_margins ? ON : OFF));
4777                 sprintf(modes_menu[4].item_string, "%s %s", mode_strings[4], 
4778                                         (auto_format ? ON : OFF));
4779                 sprintf(modes_menu[5].item_string, "%s %s", mode_strings[5], 
4780                                         (eightbit ? ON : OFF));
4781                 sprintf(modes_menu[6].item_string, "%s %s", mode_strings[6], 
4782                                         (info_window ? ON : OFF));
4783                 sprintf(modes_menu[7].item_string, "%s %s", mode_strings[7], 
4784                                         (emacs_keys_mode ? ON : OFF));
4785                 sprintf(modes_menu[8].item_string, "%s %d", mode_strings[8], 
4786                                         right_margin);
4787                 sprintf(modes_menu[9].item_string, "%s %s", mode_strings[9], 
4788                                         (ee_chinese ? ON : OFF));
4789
4790                 ret_value = menu_op(modes_menu);
4791
4792                 switch (ret_value) 
4793                 {
4794                         case 1:
4795                                 expand_tabs = !expand_tabs;
4796                                 break;
4797                         case 2:
4798                                 case_sen = !case_sen;
4799                                 break;
4800                         case 3:
4801                                 observ_margins = !observ_margins;
4802                                 break;
4803                         case 4:
4804                                 auto_format = !auto_format;
4805                                 if (auto_format)
4806                                         observ_margins = TRUE;
4807                                 break;
4808                         case 5:
4809                                 eightbit = !eightbit;
4810                                 if (!eightbit)
4811                                         ee_chinese = FALSE;
4812 #ifdef NCURSE
4813                                 if (ee_chinese)
4814                                         nc_setattrib(A_NC_BIG5);
4815                                 else
4816                                         nc_clearattrib(A_NC_BIG5);
4817 #endif /* NCURSE */
4818
4819                                 redraw();
4820                                 wnoutrefresh(text_win);
4821                                 break;
4822                         case 6:
4823                                 if (info_window)
4824                                         no_info_window();
4825                                 else
4826                                         create_info_window();
4827                                 break;
4828                         case 7:
4829                                 emacs_keys_mode = !emacs_keys_mode;
4830                                 if (info_window)
4831                                         paint_info_win();
4832                                 break;
4833                         case 8:
4834                                 string = get_string(margin_prompt, TRUE);
4835                                 if (string != NULL)
4836                                 {
4837                                         counter = atoi(string);
4838                                         if (counter > 0)
4839                                                 right_margin = counter;
4840                                         free(string);
4841                                 }
4842                                 break;
4843                         case 9:
4844                                 ee_chinese = !ee_chinese;
4845                                 if (ee_chinese != FALSE)
4846                                         eightbit = TRUE;
4847 #ifdef NCURSE
4848                                 if (ee_chinese)
4849                                         nc_setattrib(A_NC_BIG5);
4850                                 else
4851                                         nc_clearattrib(A_NC_BIG5);
4852 #endif /* NCURSE */
4853                                 redraw();
4854                                 break;
4855                         default:
4856                                 break;
4857                 }
4858         }
4859         while (ret_value != 0);
4860 }
4861
4862 char *
4863 is_in_string(string, substring) /* a strchr() look-alike for systems without
4864                                    strchr() */
4865 char * string, *substring;
4866 {
4867         char *full, *sub;
4868
4869         for (sub = substring; (sub != NULL) && (*sub != '\0'); sub++)
4870         {
4871                 for (full = string; (full != NULL) && (*full != '\0'); 
4872                                 full++)
4873                 {
4874                         if (*sub == *full)
4875                                 return(full);
4876                 }
4877         }
4878         return(NULL);
4879 }
4880
4881 /*
4882  |      handle names of the form "~/file", "~user/file", 
4883  |      "$HOME/foo", "~/$FOO", etc.
4884  */
4885
4886 char *
4887 resolve_name(name)
4888 char *name;
4889 {
4890         char long_buffer[1024];
4891         char short_buffer[128];
4892         char *buffer;
4893         char *slash;
4894         char *tmp;
4895         char *start_of_var;
4896         int offset;
4897         int index;
4898         int counter;
4899         struct passwd *user;
4900
4901         if (name[0] == '~') 
4902         {
4903                 if (name[1] == '/')
4904                 {
4905                         index = getuid();
4906                         user = (struct passwd *) getpwuid(index);
4907                         slash = name + 1;
4908                 }
4909                 else
4910                 {
4911                         slash = strchr(name, '/');
4912                         if (slash == NULL) 
4913                                 return(name);
4914                         *slash = '\0';
4915                         user = (struct passwd *) getpwnam((name + 1));
4916                         *slash = '/';
4917                 }
4918                 if (user == NULL) 
4919                 {
4920                         return(name);
4921                 }
4922                 buffer = malloc(strlen(user->pw_dir) + strlen(slash) + 1);
4923                 strcpy(buffer, user->pw_dir);
4924                 strcat(buffer, slash);
4925         }
4926         else
4927                 buffer = name;
4928
4929         if (is_in_string(buffer, "$"))
4930         {
4931                 tmp = buffer;
4932                 index = 0;
4933                 
4934                 while ((*tmp != '\0') && (index < 1024))
4935                 {
4936
4937                         while ((*tmp != '\0') && (*tmp != '$') && 
4938                                 (index < 1024))
4939                         {
4940                                 long_buffer[index] = *tmp;
4941                                 tmp++;
4942                                 index++;
4943                         }
4944
4945                         if ((*tmp == '$') && (index < 1024))
4946                         {
4947                                 counter = 0;
4948                                 start_of_var = tmp;
4949                                 tmp++;
4950                                 if (*tmp == '{') /* } */        /* bracketed variable name */
4951                                 {
4952                                         tmp++;                          /* { */
4953                                         while ((*tmp != '\0') && 
4954                                                 (*tmp != '}') && 
4955                                                 (counter < 128))
4956                                         {
4957                                                 short_buffer[counter] = *tmp;
4958                                                 counter++;
4959                                                 tmp++;
4960                                         }                       /* { */
4961                                         if (*tmp == '}')
4962                                                 tmp++;
4963                                 }
4964                                 else
4965                                 {
4966                                         while ((*tmp != '\0') && 
4967                                                (*tmp != '/') && 
4968                                                (*tmp != '$') && 
4969                                                (counter < 128))
4970                                         {
4971                                                 short_buffer[counter] = *tmp;
4972                                                 counter++;
4973                                                 tmp++;
4974                                         }
4975                                 }
4976                                 short_buffer[counter] = '\0';
4977                                 if ((slash = getenv(short_buffer)) != NULL)
4978                                 {
4979                                         offset = strlen(slash);
4980                                         if ((offset + index) < 1024)
4981                                                 strcpy(&long_buffer[index], slash);
4982                                         index += offset;
4983                                 }
4984                                 else
4985                                 {
4986                                         while ((start_of_var != tmp) && (index < 1024))
4987                                         {
4988                                                 long_buffer[index] = *start_of_var;
4989                                                 start_of_var++;
4990                                                 index++;
4991                                         }
4992                                 }
4993                         }
4994                 }
4995
4996                 if (index == 1024)
4997                         return(buffer);
4998                 else
4999                         long_buffer[index] = '\0';
5000
5001                 if (name != buffer)
5002                         free(buffer);
5003                 buffer = malloc(index + 1);
5004                 strcpy(buffer, long_buffer);
5005         }
5006
5007         return(buffer);
5008 }
5009
5010 int
5011 restrict_mode()
5012 {
5013         if (!restricted)
5014                 return(FALSE);
5015
5016         wmove(com_win, 0, 0);
5017         wprintw(com_win, restricted_msg);
5018         wclrtoeol(com_win);
5019         wrefresh(com_win);
5020         clear_com_win = TRUE;
5021         return(TRUE);
5022 }
5023
5024 /*
5025  |      The following routine tests the input string against the list of 
5026  |      strings, to determine if the string is a unique match with one of the 
5027  |      valid values.
5028  */
5029
5030 int 
5031 unique_test(string, list)
5032 char *string;
5033 char *list[];
5034 {
5035         int counter;
5036         int num_match;
5037         int result;
5038
5039         num_match = 0;
5040         counter = 0;
5041         while (list[counter] != NULL)
5042         {
5043                 result = compare(string, list[counter], FALSE);
5044                 if (result)
5045                         num_match++;
5046                 counter++;
5047         }
5048         return(num_match);
5049 }
5050
5051 #ifndef NO_CATGETS
5052 /*
5053  |      Get the catalog entry, and if it got it from the catalog, 
5054  |      make a copy, since the buffer will be overwritten by the 
5055  |      next call to catgets().
5056  */
5057
5058 char *
5059 catgetlocal(number, string)
5060 int number;
5061 char *string;
5062 {
5063         char *temp1;
5064         char *temp2;
5065
5066         temp1 = catgets(catalog, 1, number, string);
5067         if (temp1 != string)
5068         {
5069                 temp2 = malloc(strlen(temp1) + 1);
5070                 strcpy(temp2, temp1);
5071                 temp1 = temp2;
5072         }
5073         return(temp1);
5074 }
5075 #endif /* NO_CATGETS */
5076
5077 /*
5078  |      The following is to allow for using message catalogs which allow 
5079  |      the software to be 'localized', that is, to use different languages 
5080  |      all with the same binary.  For more information, see your system 
5081  |      documentation, or the X/Open Internationalization Guide.
5082  */
5083
5084 void 
5085 strings_init()
5086 {
5087         int counter;
5088
5089         setlocale(LC_ALL, "");
5090 #ifndef NO_CATGETS
5091         catalog = catopen("ee", NL_CAT_LOCALE);
5092 #endif /* NO_CATGETS */
5093
5094         modes_menu[0].item_string = catgetlocal( 1, "modes menu");
5095         mode_strings[1]  = catgetlocal( 2, "tabs to spaces       "); 
5096         mode_strings[2]  = catgetlocal( 3, "case sensitive search"); 
5097         mode_strings[3]  = catgetlocal( 4, "margins observed     "); 
5098         mode_strings[4]  = catgetlocal( 5, "auto-paragraph format"); 
5099         mode_strings[5]  = catgetlocal( 6, "eightbit characters  "); 
5100         mode_strings[6]  = catgetlocal( 7, "info window          "); 
5101         mode_strings[8]  = catgetlocal( 8, "right margin         ");
5102         leave_menu[0].item_string  = catgetlocal( 9, "leave menu");
5103         leave_menu[1].item_string  = catgetlocal( 10, "save changes");
5104         leave_menu[2].item_string  = catgetlocal( 11, "no save");
5105         file_menu[0].item_string  = catgetlocal( 12, "file menu");
5106         file_menu[1].item_string  = catgetlocal( 13, "read a file");
5107         file_menu[2].item_string  = catgetlocal( 14, "write a file");
5108         file_menu[3].item_string  = catgetlocal( 15, "save file");
5109         file_menu[4].item_string  = catgetlocal( 16, "print editor contents");
5110         search_menu[0].item_string = catgetlocal( 17, "search menu");
5111         search_menu[1].item_string = catgetlocal( 18, "search for ...");
5112         search_menu[2].item_string = catgetlocal( 19, "search");
5113         spell_menu[0].item_string = catgetlocal( 20, "spell menu");
5114         spell_menu[1].item_string = catgetlocal( 21, "use 'spell'");
5115         spell_menu[2].item_string = catgetlocal( 22, "use 'ispell'");
5116         misc_menu[0].item_string = catgetlocal( 23, "miscellaneous menu");
5117         misc_menu[1].item_string = catgetlocal( 24, "format paragraph");
5118         misc_menu[2].item_string = catgetlocal( 25, "shell command");
5119         misc_menu[3].item_string = catgetlocal( 26, "check spelling");
5120         main_menu[0].item_string  = catgetlocal( 27, "main menu");
5121         main_menu[1].item_string  = catgetlocal( 28, "leave editor");
5122         main_menu[2].item_string  = catgetlocal( 29, "help");
5123         main_menu[3].item_string  = catgetlocal( 30, "file operations");
5124         main_menu[4].item_string  = catgetlocal( 31, "redraw screen");
5125         main_menu[5].item_string  = catgetlocal( 32, "settings");
5126         main_menu[6].item_string  = catgetlocal( 33, "search");
5127         main_menu[7].item_string  = catgetlocal( 34, "miscellaneous");
5128         help_text[0] = catgetlocal( 35, "Control keys:                                                              "); 
5129         help_text[1] = catgetlocal( 36, "^a ascii code           ^i tab                  ^r right                   ");
5130         help_text[2] = catgetlocal( 37, "^b bottom of text       ^j newline              ^t top of text             ");
5131         help_text[3] = catgetlocal( 38, "^c command              ^k delete char          ^u up                      ");
5132         help_text[4] = catgetlocal( 39, "^d down                 ^l left                 ^v undelete word           ");
5133         help_text[5] = catgetlocal( 40, "^e search prompt        ^m newline              ^w delete word             ");
5134         help_text[6] = catgetlocal( 41, "^f undelete char        ^n next page            ^x search                  ");
5135         help_text[7] = catgetlocal( 42, "^g begin of line        ^o end of line          ^y delete line             ");
5136         help_text[8] = catgetlocal( 43, "^h backspace            ^p prev page            ^z undelete line           ");
5137         help_text[9] = catgetlocal( 44, "^[ (escape) menu        ESC-Enter: exit ee                                 ");
5138         help_text[10] = catgetlocal( 45, "                                                                           ");
5139         help_text[11] = catgetlocal( 46, "Commands:                                                                  ");
5140         help_text[12] = catgetlocal( 47, "help    : get this info                 file    : print file name          ");
5141         help_text[13] = catgetlocal( 48, "read    : read a file                   char    : ascii code of char       ");
5142         help_text[14] = catgetlocal( 49, "write   : write a file                  case    : case sensitive search    ");
5143         help_text[15] = catgetlocal( 50, "exit    : leave and save                nocase  : case insensitive search  ");
5144         help_text[16] = catgetlocal( 51, "quit    : leave, no save                !cmd    : execute \"cmd\" in shell   ");
5145         help_text[17] = catgetlocal( 52, "line    : display line #                0-9     : go to line \"#\"           ");
5146         help_text[18] = catgetlocal( 53, "expand  : expand tabs                   noexpand: do not expand tabs         ");
5147         help_text[19] = catgetlocal( 54, "                                                                             ");
5148         help_text[20] = catgetlocal( 55, "  ee [+#] [-i] [-e] [-h] [file(s)]                                            ");
5149         help_text[21] = catgetlocal( 56, "+# :go to line #  -i :no info window  -e : don't expand tabs  -h :no highlight");
5150         control_keys[0] = catgetlocal( 57, "^[ (escape) menu  ^e search prompt  ^y delete line    ^u up     ^p prev page  ");
5151         control_keys[1] = catgetlocal( 58, "^a ascii code     ^x search         ^z undelete line  ^d down   ^n next page  ");
5152         control_keys[2] = catgetlocal( 59, "^b bottom of text ^g begin of line  ^w delete word    ^l left                 ");
5153         control_keys[3] = catgetlocal( 60, "^t top of text    ^o end of line    ^v undelete word  ^r right                ");
5154         control_keys[4] = catgetlocal( 61, "^c command        ^k delete char    ^f undelete char      ESC-Enter: exit ee  ");
5155         command_strings[0] = catgetlocal( 62, "help : get help info  |file  : print file name         |line : print line # ");
5156         command_strings[1] = catgetlocal( 63, "read : read a file    |char  : ascii code of char      |0-9 : go to line \"#\"");
5157         command_strings[2] = catgetlocal( 64, "write: write a file   |case  : case sensitive search   |exit : leave and save ");
5158         command_strings[3] = catgetlocal( 65, "!cmd : shell \"cmd\"    |nocase: ignore case in search   |quit : leave, no save");
5159         command_strings[4] = catgetlocal( 66, "expand: expand tabs   |noexpand: do not expand tabs                           ");
5160         com_win_message = catgetlocal( 67, "    press Escape (^[) for menu");
5161         no_file_string = catgetlocal( 68, "no file");
5162         ascii_code_str = catgetlocal( 69, "ascii code: ");
5163         printer_msg_str = catgetlocal( 70, "sending contents of buffer to \"%s\" ");
5164         command_str = catgetlocal( 71, "command: ");
5165         file_write_prompt_str = catgetlocal( 72, "name of file to write: ");
5166         file_read_prompt_str = catgetlocal( 73, "name of file to read: ");
5167         char_str = catgetlocal( 74, "character = %d");
5168         unkn_cmd_str = catgetlocal( 75, "unknown command \"%s\"");
5169         non_unique_cmd_msg = catgetlocal( 76, "entered command is not unique");
5170         line_num_str = catgetlocal( 77, "line %d  ");
5171         line_len_str = catgetlocal( 78, "length = %d");
5172         current_file_str = catgetlocal( 79, "current file is \"%s\" ");
5173         usage0 = catgetlocal( 80, "usage: %s [-i] [-e] [-h] [+line_number] [file(s)]\n");
5174         usage1 = catgetlocal( 81, "       -i   turn off info window\n");
5175         usage2 = catgetlocal( 82, "       -e   do not convert tabs to spaces\n");
5176         usage3 = catgetlocal( 83, "       -h   do not use highlighting\n");
5177         file_is_dir_msg = catgetlocal( 84, "file \"%s\" is a directory");
5178         new_file_msg = catgetlocal( 85, "new file \"%s\"");
5179         cant_open_msg = catgetlocal( 86, "can't open \"%s\"");
5180         open_file_msg = catgetlocal( 87, "file \"%s\", %d lines");
5181         file_read_fin_msg = catgetlocal( 88, "finished reading file \"%s\"");
5182         reading_file_msg = catgetlocal( 89, "reading file \"%s\"");
5183         read_only_msg = catgetlocal( 90, ", read only");
5184         file_read_lines_msg = catgetlocal( 91, "file \"%s\", %d lines");
5185         save_file_name_prompt = catgetlocal( 92, "enter name of file: ");
5186         file_not_saved_msg = catgetlocal( 93, "no filename entered: file not saved");
5187         changes_made_prompt = catgetlocal( 94, "changes have been made, are you sure? (y/n [n]) ");
5188         yes_char = catgetlocal( 95, "y");
5189         file_exists_prompt = catgetlocal( 96, "file already exists, overwrite? (y/n) [n] ");
5190         create_file_fail_msg = catgetlocal( 97, "unable to create file \"%s\"");
5191         writing_file_msg = catgetlocal( 98, "writing file \"%s\"");
5192         file_written_msg = catgetlocal( 99, "\"%s\" %d lines, %d characters");
5193         searching_msg = catgetlocal( 100, "           ...searching");
5194         str_not_found_msg = catgetlocal( 101, "string \"%s\" not found");
5195         search_prompt_str = catgetlocal( 102, "search for: ");
5196         exec_err_msg = catgetlocal( 103, "could not exec %s\n");
5197         continue_msg = catgetlocal( 104, "press return to continue ");
5198         menu_cancel_msg = catgetlocal( 105, "press Esc to cancel");
5199         menu_size_err_msg = catgetlocal( 106, "menu too large for window");
5200         press_any_key_msg = catgetlocal( 107, "press any key to continue ");
5201         shell_prompt = catgetlocal( 108, "shell command: ");
5202         formatting_msg = catgetlocal( 109, "...formatting paragraph...");
5203         shell_echo_msg = catgetlocal( 110, "<!echo 'list of unrecognized words'; echo -=-=-=-=-=-");
5204         spell_in_prog_msg = catgetlocal( 111, "sending contents of edit buffer to 'spell'");
5205         margin_prompt = catgetlocal( 112, "right margin is: ");
5206         restricted_msg = catgetlocal( 113, "restricted mode: unable to perform requested operation");
5207         ON = catgetlocal( 114, "ON");
5208         OFF = catgetlocal( 115, "OFF");
5209         HELP = catgetlocal( 116, "HELP");
5210         WRITE = catgetlocal( 117, "WRITE");
5211         READ = catgetlocal( 118, "READ");
5212         LINE = catgetlocal( 119, "LINE");
5213         FILE_str = catgetlocal( 120, "FILE");
5214         CHARACTER = catgetlocal( 121, "CHARACTER");
5215         REDRAW = catgetlocal( 122, "REDRAW");
5216         RESEQUENCE = catgetlocal( 123, "RESEQUENCE");
5217         AUTHOR = catgetlocal( 124, "AUTHOR");
5218         VERSION = catgetlocal( 125, "VERSION");
5219         CASE = catgetlocal( 126, "CASE");
5220         NOCASE = catgetlocal( 127, "NOCASE");
5221         EXPAND = catgetlocal( 128, "EXPAND");
5222         NOEXPAND = catgetlocal( 129, "NOEXPAND");
5223         Exit_string = catgetlocal( 130, "EXIT");
5224         QUIT_string = catgetlocal( 131, "QUIT");
5225         INFO = catgetlocal( 132, "INFO");
5226         NOINFO = catgetlocal( 133, "NOINFO");
5227         MARGINS = catgetlocal( 134, "MARGINS");
5228         NOMARGINS = catgetlocal( 135, "NOMARGINS");
5229         AUTOFORMAT = catgetlocal( 136, "AUTOFORMAT");
5230         NOAUTOFORMAT = catgetlocal( 137, "NOAUTOFORMAT");
5231         Echo = catgetlocal( 138, "ECHO");
5232         PRINTCOMMAND = catgetlocal( 139, "PRINTCOMMAND");
5233         RIGHTMARGIN = catgetlocal( 140, "RIGHTMARGIN");
5234         HIGHLIGHT = catgetlocal( 141, "HIGHLIGHT");
5235         NOHIGHLIGHT = catgetlocal( 142, "NOHIGHLIGHT");
5236         EIGHTBIT = catgetlocal( 143, "EIGHTBIT");
5237         NOEIGHTBIT = catgetlocal( 144, "NOEIGHTBIT");
5238         /*
5239          |      additions
5240          */
5241         mode_strings[7] = catgetlocal( 145, "emacs key bindings   ");
5242         emacs_help_text[0] = help_text[0];
5243         emacs_help_text[1] = catgetlocal( 146, "^a beginning of line    ^i tab                  ^r restore word            ");
5244         emacs_help_text[2] = catgetlocal( 147, "^b back 1 char          ^j undel char           ^t top of text             ");
5245         emacs_help_text[3] = catgetlocal( 148, "^c command              ^k delete line          ^u bottom of text          ");
5246         emacs_help_text[4] = catgetlocal( 149, "^d delete char          ^l undelete line        ^v next page               ");
5247         emacs_help_text[5] = catgetlocal( 150, "^e end of line          ^m newline              ^w delete word             ");
5248         emacs_help_text[6] = catgetlocal( 151, "^f forward 1 char       ^n next line            ^x search                  ");
5249         emacs_help_text[7] = catgetlocal( 152, "^g go back 1 page       ^o ascii char insert    ^y search prompt           ");
5250         emacs_help_text[8] = catgetlocal( 153, "^h backspace            ^p prev line            ^z next word               ");
5251         emacs_help_text[9] = help_text[9];
5252         emacs_help_text[10] = help_text[10];
5253         emacs_help_text[11] = help_text[11];
5254         emacs_help_text[12] = help_text[12];
5255         emacs_help_text[13] = help_text[13];
5256         emacs_help_text[14] = help_text[14];
5257         emacs_help_text[15] = help_text[15];
5258         emacs_help_text[16] = help_text[16];
5259         emacs_help_text[17] = help_text[17];
5260         emacs_help_text[18] = help_text[18];
5261         emacs_help_text[19] = help_text[19];
5262         emacs_help_text[20] = help_text[20];
5263         emacs_help_text[21] = help_text[21];
5264         emacs_control_keys[0] = catgetlocal( 154, "^[ (escape) menu ^y search prompt ^k delete line   ^p prev li     ^g prev page");
5265         emacs_control_keys[1] = catgetlocal( 155, "^o ascii code    ^x search        ^l undelete line ^n next li     ^v next page");
5266         emacs_control_keys[2] = catgetlocal( 156, "^u end of file   ^a begin of line ^w delete word   ^b back 1 char ^z next word");
5267         emacs_control_keys[3] = catgetlocal( 157, "^t top of text   ^e end of line   ^r restore word  ^f forward char            ");
5268         emacs_control_keys[4] = catgetlocal( 158, "^c command       ^d delete char   ^j undelete char              ESC-Enter: exit");
5269         EMACS_string = catgetlocal( 159, "EMACS");
5270         NOEMACS_string = catgetlocal( 160, "NOEMACS");
5271         usage4 = catgetlocal( 161, "       +#   put cursor at line #\n");
5272         conf_dump_err_msg = catgetlocal( 162, "unable to open .init.ee for writing, no configuration saved!");
5273         conf_dump_success_msg = catgetlocal( 163, "ee configuration saved in file %s");
5274         modes_menu[10].item_string = catgetlocal( 164, "save editor configuration");
5275         config_dump_menu[0].item_string = catgetlocal( 165, "save ee configuration");
5276         config_dump_menu[1].item_string = catgetlocal( 166, "save in current directory");
5277         config_dump_menu[2].item_string = catgetlocal( 167, "save in home directory");
5278         conf_not_saved_msg = catgetlocal( 168, "ee configuration not saved");
5279         ree_no_file_msg = catgetlocal( 169, "must specify a file when invoking ree");
5280         menu_too_lrg_msg = catgetlocal( 180, "menu too large for window");
5281         more_above_str = catgetlocal( 181, "^^more^^");
5282         more_below_str = catgetlocal( 182, "VVmoreVV");
5283         mode_strings[9] = catgetlocal( 183, "16 bit characters    ");
5284         chinese_cmd = catgetlocal( 184, "16BIT");
5285         nochinese_cmd = catgetlocal( 185, "NO16BIT");
5286
5287         commands[0] = HELP;
5288         commands[1] = WRITE;
5289         commands[2] = READ;
5290         commands[3] = LINE;
5291         commands[4] = FILE_str;
5292         commands[5] = REDRAW;
5293         commands[6] = RESEQUENCE;
5294         commands[7] = AUTHOR;
5295         commands[8] = VERSION;
5296         commands[9] = CASE;
5297         commands[10] = NOCASE;
5298         commands[11] = EXPAND;
5299         commands[12] = NOEXPAND;
5300         commands[13] = Exit_string;
5301         commands[14] = QUIT_string;
5302         commands[15] = "<";
5303         commands[16] = ">";
5304         commands[17] = "!";
5305         commands[18] = "0";
5306         commands[19] = "1";
5307         commands[20] = "2";
5308         commands[21] = "3";
5309         commands[22] = "4";
5310         commands[23] = "5";
5311         commands[24] = "6";
5312         commands[25] = "7";
5313         commands[26] = "8";
5314         commands[27] = "9";
5315         commands[28] = CHARACTER;
5316         commands[29] = chinese_cmd;
5317         commands[30] = nochinese_cmd;
5318         commands[31] = NULL;
5319         init_strings[0] = CASE;
5320         init_strings[1] = NOCASE;
5321         init_strings[2] = EXPAND;
5322         init_strings[3] = NOEXPAND;
5323         init_strings[4] = INFO;
5324         init_strings[5] = NOINFO;
5325         init_strings[6] = MARGINS;
5326         init_strings[7] = NOMARGINS;
5327         init_strings[8] = AUTOFORMAT;
5328         init_strings[9] = NOAUTOFORMAT;
5329         init_strings[10] = Echo;
5330         init_strings[11] = PRINTCOMMAND;
5331         init_strings[12] = RIGHTMARGIN;
5332         init_strings[13] = HIGHLIGHT;
5333         init_strings[14] = NOHIGHLIGHT;
5334         init_strings[15] = EIGHTBIT;
5335         init_strings[16] = NOEIGHTBIT;
5336         init_strings[17] = EMACS_string;
5337         init_strings[18] = NOEMACS_string;
5338         init_strings[19] = chinese_cmd;
5339         init_strings[20] = nochinese_cmd;
5340         init_strings[21] = NULL;
5341
5342         /*
5343          |      allocate space for strings here for settings menu
5344          */
5345
5346         for (counter = 1; counter < NUM_MODES_ITEMS; counter++)
5347         {
5348                 modes_menu[counter].item_string = malloc(80);
5349         }
5350
5351 #ifndef NO_CATGETS
5352         catclose(catalog);
5353 #endif /* NO_CATGETS */
5354 }
5355