nano.c 60.7 KB
Newer Older
Chris Allegretta's avatar
Chris Allegretta committed
1
/* $Id$ */
Chris Allegretta's avatar
Chris Allegretta committed
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/**************************************************************************
 *   nano.c                                                               *
 *                                                                        *
 *   Copyright (C) 1999 Chris Allegretta                                  *
 *   This program is free software; you can redistribute it and/or modify *
 *   it under the terms of the GNU General Public License as published by *
 *   the Free Software Foundation; either version 1, or (at your option)  *
 *   any later version.                                                   *
 *                                                                        *
 *   This program is distributed in the hope that it will be useful,      *
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of       *
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the        *
 *   GNU General Public License for more details.                         *
 *                                                                        *
 *   You should have received a copy of the GNU General Public License    *
 *   along with this program; if not, write to the Free Software          *
 *   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.            *
 *                                                                        *
 **************************************************************************/

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <signal.h>
26
#include <setjmp.h>
Chris Allegretta's avatar
Chris Allegretta committed
27
28
29
30
31
32
#include <unistd.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/param.h>
Chris Allegretta's avatar
Chris Allegretta committed
33
34
#include <sys/types.h>
#include <sys/wait.h>
Chris Allegretta's avatar
Chris Allegretta committed
35
36
37
38
#include <errno.h>
#include <ctype.h>
#include <locale.h>
#include <limits.h>
39
#include <assert.h>
Chris Allegretta's avatar
Chris Allegretta committed
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65

#include "config.h"
#include "proto.h"
#include "nano.h"

#ifndef NANO_SMALL
#include <libintl.h>
#define _(string) gettext(string)
#else
#define _(string) (string)
#endif

#ifdef HAVE_TERMIOS_H
#include <termios.h>
#endif

#ifdef HAVE_TERMIO_H
#include <termio.h>
#endif

#ifdef HAVE_GETOPT_H
#include <getopt.h>
#endif

/* Former globals, now static */
int fill = 0;			/* Fill - where to wrap lines, basically */
66
67

#ifndef DISABLE_SPELLER
Chris Allegretta's avatar
Chris Allegretta committed
68
static char *alt_speller;	/* Alternative spell command */
69
70
#endif

Chris Allegretta's avatar
Chris Allegretta committed
71
struct termios oldterm;		/* The user's original term settings */
72
static struct sigaction act;	/* For all out fun signal handlers */
Chris Allegretta's avatar
Chris Allegretta committed
73

74
#ifndef DISABLE_HELP
75
static char *help_text_init = "";	/* Initial message, not including shortcuts */
76
77
#endif

Chris Allegretta's avatar
Chris Allegretta committed
78
79
80
81
char *last_search = NULL;	/* Last string we searched for */
char *last_replace = NULL;	/* Last replacement string */
int search_last_line;		/* Is this the last search line? */

82
83
static sigjmp_buf jmpbuf;	/* Used to return to mainloop after SIGWINCH */

Chris Allegretta's avatar
Chris Allegretta committed
84
85
86
/* What we do when we're all set to exit */
RETSIGTYPE finish(int sigage)
{
87
88
89
90

    keypad(edit, TRUE);
    keypad(bottomwin, TRUE);

Chris Allegretta's avatar
Chris Allegretta committed
91
92
93
    if (!ISSET(NO_HELP)) {
	mvwaddstr(bottomwin, 1, 0, hblank);
	mvwaddstr(bottomwin, 2, 0, hblank);
Chris Allegretta's avatar
Chris Allegretta committed
94
    } else
Chris Allegretta's avatar
Chris Allegretta committed
95
	mvwaddstr(bottomwin, 0, 0, hblank);
96

Chris Allegretta's avatar
Chris Allegretta committed
97
98
99
100
    wrefresh(bottomwin);
    endwin();

    /* Restore the old term settings */
Chris Allegretta's avatar
Chris Allegretta committed
101
    tcsetattr(0, TCSANOW, &oldterm);
Chris Allegretta's avatar
Chris Allegretta committed
102
103
104
105
106
107
108
109

    exit(sigage);
}

/* Die (gracefully?) */
void die(char *msg, ...)
{
    va_list ap;
110
111
    char *name;
    int i;
Chris Allegretta's avatar
Chris Allegretta committed
112
113
114
115
116
117

    va_start(ap, msg);
    vfprintf(stderr, msg, ap);
    va_end(ap);

    /* if we can't save we have REAL bad problems,
118
     * but we might as well TRY. */
119
    if (filename[0] == '\0') {
120
121
	name = "nano.save";
	i = write_file(name, 1);
122
    } else {
123

124
125
126
	char *buf = nmalloc(strlen(filename) + 6);
	strcpy(buf, filename);
	strcat(buf, ".save");
127
128
	i = write_file(buf, 1);
	name = buf;
129
    }
Chris Allegretta's avatar
Chris Allegretta committed
130
    /* Restore the old term settings */
Chris Allegretta's avatar
Chris Allegretta committed
131
    tcsetattr(0, TCSANOW, &oldterm);
Chris Allegretta's avatar
Chris Allegretta committed
132
133
134
135
136
137
138

    clear();
    refresh();
    resetty();
    endwin();

    fprintf(stderr, msg);
139
140
141
    if (i != -1)
	fprintf(stderr, _("\nBuffer written to %s\n"), name);
    else
142
	fprintf(stderr, _("\nNo %s written (file exists?)\n"), name);
Chris Allegretta's avatar
Chris Allegretta committed
143
144
145
146

    exit(1);			/* We have a problem: exit w/ errorlevel(1) */
}

147
148
149
150
151
152
153
154
155
156
/* Die with an error message that the screen was too small if, well, the
   screen is too small */
void die_too_small(void)
{
    char *too_small_msg = _("Window size is too small for Nano...");

    die(too_small_msg);

}

Chris Allegretta's avatar
Chris Allegretta committed
157
158
159
160
161
void print_view_warning(void)
{
    statusbar(_("Key illegal in VIEW mode"));
}

Chris Allegretta's avatar
Chris Allegretta committed
162
163
164
165
166
167
168
void clear_filename(void)
{
    if (filename != NULL)
	free(filename);
    filename = nmalloc(1);
    filename[0] = 0;
}
Chris Allegretta's avatar
Chris Allegretta committed
169

Chris Allegretta's avatar
Chris Allegretta committed
170
171
172
173
174
/* Initialize global variables - no better way for now */
void global_init(void)
{
    current_x = 0;
    current_y = 0;
175
176
177
178

    if ((editwinrows = LINES - 5 + no_help()) < MIN_EDITOR_ROWS)
	die_too_small();

Chris Allegretta's avatar
Chris Allegretta committed
179
180
181
182
183
184
185
    fileage = NULL;
    cutbuffer = NULL;
    current = NULL;
    edittop = NULL;
    editbot = NULL;
    totlines = 0;
    placewewant = 0;
186

Chris Allegretta's avatar
Chris Allegretta committed
187
    if (!fill)
188
189
190
191
192
	fill = COLS - CHARS_FROM_EOL;

    if (fill < MIN_FILL_LENGTH)
	die_too_small();

Chris Allegretta's avatar
Chris Allegretta committed
193
    hblank = nmalloc(COLS + 1);
194
195
    memset(hblank, ' ', COLS);
    hblank[COLS] = 0;
Chris Allegretta's avatar
Chris Allegretta committed
196
197
}

198
#ifndef DISABLE_HELP
Chris Allegretta's avatar
Chris Allegretta committed
199
200
201
202
void init_help_msg(void)
{

    help_text_init =
Chris Allegretta's avatar
Chris Allegretta committed
203
204
205
206
207
208
209
210
211
212
213
	_(" nano help text\n\n "
	  "The nano editor is designed to emulate the functionality and "
	  "ease-of-use of the UW Pico text editor.  There are four main "
	  "sections of the editor: The top line shows the program "
	  "version, the current filename being edited, and whether "
	  "or not the file has been modified.  Next is the main editor "
	  "window showing the file being edited.  The status line is "
	  "the third line from the bottom and shows important messages. "
	  "The bottom two lines show the most commonly used shortcuts "
	  "in the editor.\n\n "
	  "The notation for shortcuts is as follows: Control-key "
214
	  "sequences are notated with a caret (^) symbol and are entered "
215
216
217
218
	  "with the Control (Ctrl) key.  Escape-key sequences are notated "
	  "with the Meta (M) symbol and can be entered using either the "
	  "Esc, Alt or Meta key depending on your keyboard setup.  The "
	  "following keystrokes are available in the main editor window. "
Chris Allegretta's avatar
Chris Allegretta committed
219
	  "Optional keys are shown in parentheses:\n\n");
Chris Allegretta's avatar
Chris Allegretta committed
220
221

}
222
#endif
Chris Allegretta's avatar
Chris Allegretta committed
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252

/* Make a copy of a node to a pointer (space will be malloc()ed) */
filestruct *copy_node(filestruct * src)
{
    filestruct *dst;

    dst = nmalloc(sizeof(filestruct));
    dst->data = nmalloc(strlen(src->data) + 1);

    dst->next = src->next;
    dst->prev = src->prev;

    strcpy(dst->data, src->data);
    dst->lineno = src->lineno;

    return dst;
}

/* Unlink a node from the rest of the struct */
void unlink_node(filestruct * fileptr)
{
    if (fileptr->prev != NULL)
	fileptr->prev->next = fileptr->next;

    if (fileptr->next != NULL)
	fileptr->next->prev = fileptr->prev;
}

void delete_node(filestruct * fileptr)
{
253
254
255
    if (fileptr == NULL)
	return;

Chris Allegretta's avatar
Chris Allegretta committed
256
    if (fileptr->data != NULL)
257
	free(fileptr->data);
Chris Allegretta's avatar
Chris Allegretta committed
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
    free(fileptr);
}

/* Okay, now let's duplicate a whole struct! */
filestruct *copy_filestruct(filestruct * src)
{
    filestruct *dst, *tmp, *head, *prev;

    head = copy_node(src);
    dst = head;			/* Else we barf on copying just one line */
    head->prev = NULL;
    tmp = src->next;
    prev = head;

    while (tmp != NULL) {
	dst = copy_node(tmp);
	dst->prev = prev;
	prev->next = dst;

	prev = dst;
	tmp = tmp->next;
    }

    dst->next = NULL;
    return head;
}

int free_filestruct(filestruct * src)
{
    filestruct *fileptr = src;

    if (src == NULL)
	return 0;

    while (fileptr->next != NULL) {
	fileptr = fileptr->next;
294
	delete_node(fileptr->prev);
Chris Allegretta's avatar
Chris Allegretta committed
295
296

#ifdef DEBUG
297
	fprintf(stderr, _("delete_node(): free'd a node, YAY!\n"));
Chris Allegretta's avatar
Chris Allegretta committed
298
299
#endif
    }
300
    delete_node(fileptr);
Chris Allegretta's avatar
Chris Allegretta committed
301
#ifdef DEBUG
302
    fprintf(stderr, _("delete_node(): free'd last node.\n"));
Chris Allegretta's avatar
Chris Allegretta committed
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
#endif

    return 1;
}

int renumber_all(void)
{
    filestruct *temp;
    long i = 1;

    for (temp = fileage; temp != NULL; temp = temp->next) {
	temp->lineno = i++;
    }

    return 0;
}

int renumber(filestruct * fileptr)
{
    filestruct *temp;

    if (fileptr == NULL || fileptr->prev == NULL || fileptr == fileage) {
	renumber_all();
	return 0;
    }
    for (temp = fileptr; temp != NULL; temp = temp->next) {
Chris Allegretta's avatar
Chris Allegretta committed
329
330
331
332
	if (temp->prev != NULL)
	    temp->lineno = temp->prev->lineno + 1;
	else
	    temp->lineno = 1;
Chris Allegretta's avatar
Chris Allegretta committed
333
334
335
336
337
338
339
340
341
342
343
344
345
346
    }

    return 0;
}

/* Fix the memory allocation for a string */
void align(char **strp)
{
    /* There was a serious bug here:  the new address was never
       stored anywhere... */

    *strp = nrealloc(*strp, strlen(*strp) + 1);
}

Chris Allegretta's avatar
Chris Allegretta committed
347
348
349
350
351
352
353
/* Null a string at a certain index and align it */
void null_at(char *data, int index)
{
    data[index] = 0;
    align(&data);
}

Chris Allegretta's avatar
Chris Allegretta committed
354
355
356
357
358
void usage(void)
{
#ifdef HAVE_GETOPT_LONG
    printf(_("Usage: nano [GNU long option] [option] +LINE <file>\n\n"));
    printf(_("Option		Long option		Meaning\n"));
Chris Allegretta's avatar
Chris Allegretta committed
359
    printf(_
360
	   (" -T [num]	--tabsize=[num]		Set width of a tab to num\n"));
361
#ifdef HAVE_REGEX_H
362
    printf(_
363
	   (" -R		--regexp		Use regular expressions for search\n"));
364
#endif
Chris Allegretta's avatar
Chris Allegretta committed
365
366
367
368
369
370
371
    printf
	(_
	 (" -V 		--version		Print version information and exit\n"));
    printf(_
	   (" -c 		--const			Constantly show cursor position\n"));
    printf(_
	   (" -h 		--help			Show this message\n"));
372
373
    printf(_
	   (" -i 		--autoindent		Automatically indent new lines\n"));
374
#ifndef NANO_SMALL
375
376
    printf(_
	   (" -k 		--cut			Let ^K cut from cursor to end of line\n"));
377
#endif
Chris Allegretta's avatar
Chris Allegretta committed
378
    printf(_
379
	   (" -l 		--nofollow		Don't follow symbolic links, overwrite\n"));
Chris Allegretta's avatar
Chris Allegretta committed
380
381
382
383
384
#ifndef NANO_SMALL
#ifdef NCURSES_MOUSE_VERSION
    printf(_(" -m 		--mouse			Enable mouse\n"));
#endif
#endif
385
386
    printf(_
	   (" -p	 	--pico			Emulate Pico as closely as possible\n"));
Chris Allegretta's avatar
Chris Allegretta committed
387
388
389
    printf
	(_
	 (" -r [#cols] 	--fill=[#cols]		Set fill cols to (wrap lines at) #cols\n"));
390
#ifndef DISABLE_SPELLER
Chris Allegretta's avatar
Chris Allegretta committed
391
392
    printf(_
	   (" -s [prog] 	--speller=[prog]	Enable alternate speller\n"));
393
#endif
Chris Allegretta's avatar
Chris Allegretta committed
394
395
396
397
    printf(_
	   (" -t 		--tempfile		Auto save on exit, don't prompt\n"));
    printf(_
	   (" -v 		--view			View (read only) mode\n"));
398
#ifndef DISABLE_WRAPPING
Chris Allegretta's avatar
Chris Allegretta committed
399
400
    printf(_
	   (" -w 		--nowrap		Don't wrap long lines\n"));
401
#endif
Chris Allegretta's avatar
Chris Allegretta committed
402
403
404
405
406
407
408
409
410
    printf(_
	   (" -x 		--nohelp		Don't show help window\n"));
    printf(_
	   (" -z 		--suspend		Enable suspend\n"));
    printf(_
	   (" +LINE					Start at line number LINE\n"));
#else
    printf(_("Usage: nano [option] +LINE <file>\n\n"));
    printf(_("Option		Meaning\n"));
Chris Allegretta's avatar
Chris Allegretta committed
411
    printf(_(" -T [num]	Set width of a tab to num\n"));
412
    printf(_(" -R		Use regular expressions for search\n"));
Chris Allegretta's avatar
Chris Allegretta committed
413
414
415
    printf(_(" -V 		Print version information and exit\n"));
    printf(_(" -c 		Constantly show cursor position\n"));
    printf(_(" -h 		Show this message\n"));
416
    printf(_(" -i 		Automatically indent new lines\n"));
417
#ifndef NANO_SMALL
418
    printf(_(" -k 		Let ^K cut from cursor to end of line\n"));
419
#endif
Chris Allegretta's avatar
Chris Allegretta committed
420
    printf(_
421
	   (" -l 		Don't follow symbolic links, overwrite\n"));
Chris Allegretta's avatar
Chris Allegretta committed
422
423
424
425
426
#ifndef NANO_SMALL
#ifdef NCURSES_MOUSE_VERSION
    printf(_(" -m 		Enable mouse\n"));
#endif
#endif
427
    printf(_(" -p 		Emulate Pico as closely as possible\n"));
428
429
    printf(_
	   (" -r [#cols] 	Set fill cols to (wrap lines at) #cols\n"));
430
#ifndef DISABLE_SPELLER
431
    printf(_(" -s [prog]  	Enable alternate speller\n"));
432
#endif
Chris Allegretta's avatar
Chris Allegretta committed
433
434
    printf(_(" -t 		Auto save on exit, don't prompt\n"));
    printf(_(" -v 		View (read only) mode\n"));
435
#ifndef DISABLE_WRAPPING
Chris Allegretta's avatar
Chris Allegretta committed
436
    printf(_(" -w 		Don't wrap long lines\n"));
437
#endif
Chris Allegretta's avatar
Chris Allegretta committed
438
439
440
441
442
443
444
445
446
    printf(_(" -x 		Don't show help window\n"));
    printf(_(" -z 		Enable suspend\n"));
    printf(_(" +LINE		Start at line number LINE\n"));
#endif
    exit(0);
}

void version(void)
{
Chris Allegretta's avatar
Chris Allegretta committed
447
    printf(_(" GNU nano version %s (compiled %s, %s)\n"),
Chris Allegretta's avatar
Chris Allegretta committed
448
	   VERSION, __TIME__, __DATE__);
449
    printf(_
450
451
	   (" Email: nano@nano-editor.org	Web: http://www.nano-editor.org"));
    printf(_("\n Compiled options:"));
452

453
454
455
#ifdef NANO_EXTRA
    printf(" --enable-extra");
#endif
456
457
458
459

#ifdef NANO_SMALL
    printf(" --enable-tiny");
#else
460
#ifdef DISABLE_BROWSER
461
    printf(" --disable-browser");
462
463
#endif
#ifdef DISABLE_TABCOMP
464
    printf(" --disable-tabcomp");
465
466
#endif
#ifdef DISABLE_JUSTIFY
467
    printf(" --disable-justify");
468
469
#endif
#ifdef DISABLE_SPELLER
470
    printf(" --disable-speller");
471
472
#endif
#ifdef DISABLE_HELP
473
    printf(" --disable-help");
474
#endif
475
#endif
476

477
478
479
#ifdef DISABLE_WRAPPING
    printf(" --disable-wrapping");
#endif
480
481
482
483
484
#ifdef USE_SLANG
    printf(" --with-slang");
#endif
    printf("\n");

Chris Allegretta's avatar
Chris Allegretta committed
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
}

filestruct *make_new_node(filestruct * prevnode)
{
    filestruct *newnode;

    newnode = nmalloc(sizeof(filestruct));
    newnode->data = NULL;

    newnode->prev = prevnode;
    newnode->next = NULL;

    if (prevnode != NULL)
	newnode->lineno = prevnode->lineno + 1;

    return newnode;
}

503
/* Splice a node into an existing filestruct */
504
505
void splice_node(filestruct * begin, filestruct * newnode,
		 filestruct * end)
506
{
507
508
509
    newnode->next = end;
    newnode->prev = begin;
    begin->next = newnode;
510
    if (end != NULL)
511
	end->prev = newnode;
512
513
}

514
int do_mark(void)
Chris Allegretta's avatar
Chris Allegretta committed
515
516
{
#ifdef NANO_SMALL
517
    nano_disabled_msg();
Chris Allegretta's avatar
Chris Allegretta committed
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
#else
    if (!ISSET(MARK_ISSET)) {
	statusbar(_("Mark Set"));
	SET(MARK_ISSET);
	mark_beginbuf = current;
	mark_beginx = current_x;
    } else {
	statusbar(_("Mark UNset"));
	UNSET(MARK_ISSET);
	mark_beginbuf = NULL;
	mark_beginx = 0;

	edit_refresh();
    }
#endif
    return 1;
}

int no_help(void)
{
    if ISSET
	(NO_HELP)
	    return 2;
    else
	return 0;
}

545
#if defined(DISABLE_JUSTIFY) || defined(DISABLE_SPELLER) || defined(DISABLE_HELP)
546
547
548
549
void nano_disabled_msg(void)
{
    statusbar("Sorry, support for this function has been disabled");
}
550
#endif
551

Chris Allegretta's avatar
Chris Allegretta committed
552
553
554
/* The user typed a printable character; add it to the edit buffer */
void do_char(char ch)
{
Robert Siemborski's avatar
Robert Siemborski committed
555
556
    /* magic-line: when a character is inserted on the current magic line,
     * it means we need a new one! */
557
    if (filebot == current && current->data[0] == '\0') {
Robert Siemborski's avatar
Robert Siemborski committed
558
	new_magicline();
559
	fix_editbot();
Robert Siemborski's avatar
Robert Siemborski committed
560
561
    }

Chris Allegretta's avatar
Chris Allegretta committed
562
563
564
565
566
567
568
569
    /* More dangerousness fun =) */
    current->data = nrealloc(current->data, strlen(current->data) + 2);
    memmove(&current->data[current_x + 1],
	    &current->data[current_x],
	    strlen(current->data) - current_x + 1);
    current->data[current_x] = ch;
    do_right();

570
#ifndef DISABLE_WRAPPING
571
    if (!ISSET(NO_WRAP) && (ch != '\t'))
Chris Allegretta's avatar
Chris Allegretta committed
572
	check_wrap(current, ch);
573
574
#endif

Chris Allegretta's avatar
Chris Allegretta committed
575
576
577
578
579
580
581
582
583
584
    set_modified();
    check_statblank();
    UNSET(KEEP_CUTBUFFER);
    totsize++;

}

/* Someone hits return *gasp!* */
int do_enter(filestruct * inptr)
{
585
    filestruct *newnode;
Chris Allegretta's avatar
Chris Allegretta committed
586
587
588
    char *tmp, *spc;
    int extra = 0;

589
    newnode = make_new_node(inptr);
Chris Allegretta's avatar
Chris Allegretta committed
590
591
592
593
594
595
596
597
598
599
600
    tmp = &current->data[current_x];
    current_x = 0;

    /* Do auto-indenting, like the neolithic Turbo Pascal editor */
    if (ISSET(AUTOINDENT)) {
	spc = current->data;
	if (spc) {
	    while ((*spc == ' ') || (*spc == '\t')) {
		extra++;
		spc++;
		current_x++;
601
		totsize++;
Chris Allegretta's avatar
Chris Allegretta committed
602
	    }
603
604
605
	    newnode->data = nmalloc(strlen(tmp) + extra + 1);
	    strncpy(newnode->data, current->data, extra);
	    strcpy(&newnode->data[extra], tmp);
Chris Allegretta's avatar
Chris Allegretta committed
606
607
	}
    } else {
608
609
	newnode->data = nmalloc(strlen(tmp) + 1);
	strcpy(newnode->data, tmp);
Chris Allegretta's avatar
Chris Allegretta committed
610
611
612
    }
    *tmp = 0;

613
    if (inptr->next == NULL) {
614
615
	filebot = newnode;
	editbot = newnode;
Chris Allegretta's avatar
Chris Allegretta committed
616
    }
617
    splice_node(inptr, newnode, inptr->next);
Chris Allegretta's avatar
Chris Allegretta committed
618
619
620

    totsize++;
    renumber(current);
621
    current = newnode;
Chris Allegretta's avatar
Chris Allegretta committed
622
623
    align(&current->data);

Robert Siemborski's avatar
Robert Siemborski committed
624
625
626
627
628
629
    /* The logic here is as follows:
     *    -> If we are at the bottom of the buffer, we want to recenter
     *       (read: rebuild) the screen and forcably move the cursor.
     *    -> otherwise, we want simply to redraw the screen and update
     *       where we think the cursor is.
     */
Chris Allegretta's avatar
Chris Allegretta committed
630
    if (current_y == editwinrows - 1) {
631
	edit_update(current, CENTER);
632
	reset_cursor();
Robert Siemborski's avatar
Robert Siemborski committed
633
    } else {
Chris Allegretta's avatar
Chris Allegretta committed
634
	current_y++;
Robert Siemborski's avatar
Robert Siemborski committed
635
636
637
	edit_refresh();
	update_cursor();
    }
Chris Allegretta's avatar
Chris Allegretta committed
638
639
640
641

    totlines++;
    set_modified();

642
    placewewant = xplustabs();
Chris Allegretta's avatar
Chris Allegretta committed
643
644
645
646
647
648
649
650
651
652
    return 1;
}

int do_enter_void(void)
{
    return do_enter(current);
}

void do_next_word(void)
{
653
    filestruct *fileptr, *old;
Chris Allegretta's avatar
Chris Allegretta committed
654
655
656
657
658
    int i;

    if (current == NULL)
	return;

659
    old = current;
Chris Allegretta's avatar
Chris Allegretta committed
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
    i = current_x;
    for (fileptr = current; fileptr != NULL; fileptr = fileptr->next) {
	if (fileptr == current) {
	    while (isalnum((int) fileptr->data[i])
		   && fileptr->data[i] != 0)
		i++;

	    if (fileptr->data[i] == 0) {
		i = 0;
		continue;
	    }
	}
	while (!isalnum((int) fileptr->data[i]) && fileptr->data[i] != 0)
	    i++;

	if (fileptr->data[i] != 0)
	    break;

	i = 0;
    }
    if (fileptr == NULL)
	current = filebot;
    else
	current = fileptr;

    current_x = i;
    placewewant = xplustabs();
687

Chris Allegretta's avatar
Chris Allegretta committed
688
    if (current->lineno >= editbot->lineno)
689
	edit_update(current, CENTER);
690
691
692
693
694
695
696
697
698
699
    else {
	/* If we've jumped lines, refresh the old line.  We can't just use
	 * current->prev here, because we may have skipped over some blank
	 * lines, in which case the previous line is the wrong one.
	 */
	if (current != old)
	    update_line(old, 0);

	update_line(current, current_x);
    }
Chris Allegretta's avatar
Chris Allegretta committed
700
701
702

}

703
#ifndef DISABLE_WRAPPING
Chris Allegretta's avatar
Chris Allegretta committed
704
void do_wrap(filestruct * inptr, char input_char)
Chris Allegretta's avatar
Chris Allegretta committed
705
{
Chris Allegretta's avatar
Chris Allegretta committed
706
707
708
709
710
711
712
    int i = 0;			/* Index into ->data for line. */
    int i_tabs = 0;		/* Screen position of ->data[i]. */
    int last_word_end = -1;	/* Location of end of last word found. */
    int current_word_start = -1;	/* Location of start of current word. */
    int current_word_start_t = -1;	/* Location of start of current word screen position. */
    int current_word_end = -1;	/* Location of end   of current word */
    int current_word_end_t = -1;	/* Location of end   of current word screen position. */
713
    int len = strlen(inptr->data);
Chris Allegretta's avatar
Chris Allegretta committed
714

Chris Allegretta's avatar
Chris Allegretta committed
715
    int down = 0;
716
717
    int right = 0;
    struct filestruct *temp = NULL;
Chris Allegretta's avatar
Chris Allegretta committed
718

Chris Allegretta's avatar
Chris Allegretta committed
719
    assert(strlenpt(inptr->data) > fill);
Chris Allegretta's avatar
Chris Allegretta committed
720

721
    for (i = 0, i_tabs = 0; i < len; i++, i_tabs++) {
722
	if (!isspace((int) inptr->data[i])) {
Chris Allegretta's avatar
Chris Allegretta committed
723
	    last_word_end = current_word_end;
Chris Allegretta's avatar
Chris Allegretta committed
724

Chris Allegretta's avatar
Chris Allegretta committed
725
	    current_word_start = i;
726
	    current_word_start_t = i_tabs;
Chris Allegretta's avatar
Chris Allegretta committed
727

728
	    while (!isspace((int) inptr->data[i])
729
		   && inptr->data[i]) {
730
731
		i++;
		i_tabs++;
Chris Allegretta's avatar
Chris Allegretta committed
732
733
		if (inptr->data[i] < 32)
		    i_tabs++;
734
	    }
Chris Allegretta's avatar
Chris Allegretta committed
735

736
	    if (inptr->data[i]) {
Chris Allegretta's avatar
Chris Allegretta committed
737
		current_word_end = i;
738
		current_word_end_t = i_tabs;
Chris Allegretta's avatar
Chris Allegretta committed
739
740
	    } else {
		current_word_end = i - 1;
741
742
743
		current_word_end_t = i_tabs - 1;
	    }
	}
Chris Allegretta's avatar
Chris Allegretta committed
744

Chris Allegretta's avatar
Chris Allegretta committed
745
	if (inptr->data[i] == NANO_CONTROL_I) {
746
747
	    if (i_tabs % tabsize != 0);
	    i_tabs += tabsize - (i_tabs % tabsize);
Chris Allegretta's avatar
Chris Allegretta committed
748
	}
Chris Allegretta's avatar
Chris Allegretta committed
749

750
	if (current_word_end_t > fill)
751
752
	    break;
    }
Chris Allegretta's avatar
Chris Allegretta committed
753

754
    /* There are a few (ever changing) cases of what the line could look like.
755
     * 1) only one word on the line before wrap point.
756
757
758
     *    a) one word takes up the whole line with no starting spaces.
     *         - do nothing and return.
     *    b) cursor is on word or before word at wrap point and there are spaces at beginning.
759
760
761
762
763
764
     *         - word starts new line.
     *         - keep white space on original line up to the cursor.
     *    *) cursor is after word at wrap point
     *         - either it's all white space after word, and this routine isn't called.
     *         - or we are actually in case 2 (2 words).
     * 2) Two or more words on the line before wrap point.
765
     *    a) cursor is at a word or space before wrap point
766
     *         - word at wrap point starts a new line.
767
768
     *         - white space at end of original line is cleared, unless
     *           it is all spaces between previous word and next word which appears after fill.
769
770
     *    b) cursor is at the word at the wrap point.
     *         - word at wrap point starts a new line.
771
     *         1. pressed a space and at first character of wrap point word.
772
     *            - white space on original line is kept to where cursor was.
773
     *         2. pressed non space (or space elsewhere).
774
775
776
777
778
779
     *            - white space at end of original line is cleared.
     *    c) cursor is past the word at the wrap point.
     *         - word at wrap point starts a new line.
     *            - white space at end of original line is cleared
     */

Chris Allegretta's avatar
Chris Allegretta committed
780
    temp = nmalloc(sizeof(filestruct));
781

782
    /* Category 1a: one word taking up the whole line with no beginning spaces. */
783
    if ((last_word_end == -1) && (!isspace((int) inptr->data[0]))) {
784
	for (i = current_word_end; i < len; i++) {
785
	    if (!isspace((int) inptr->data[i]) && i < len) {
Chris Allegretta's avatar
Chris Allegretta committed
786
		current_word_start = i;
787
		while (!isspace((int) inptr->data[i]) && (i < len)) {
788
789
		    i++;
		}
Chris Allegretta's avatar
Chris Allegretta committed
790
		last_word_end = current_word_end;
791
792
793
794
795
796
		current_word_end = i;
		break;
	    }
	}

	if (last_word_end == -1) {
Chris Allegretta's avatar
Chris Allegretta committed
797
	    free(temp);
798
799
800
801
802
	    return;
	}
	if (current_x >= last_word_end) {
	    right = (current_x - current_word_start) + 1;
	    current_x = last_word_end;
Chris Allegretta's avatar
Chris Allegretta committed
803
	    down = 1;
804
	}
805

806
807
808
809
	temp->data = nmalloc(strlen(&inptr->data[current_word_start]) + 1);
	strcpy(temp->data, &inptr->data[current_word_start]);
	inptr->data = nrealloc(inptr->data, last_word_end + 2);
	inptr->data[last_word_end + 1] = 0;
Chris Allegretta's avatar
Chris Allegretta committed
810
811
812
    } else
	/* Category 1b: one word on the line and word not taking up whole line
	   (i.e. there are spaces at the beginning of the line) */
813
    if (last_word_end == -1) {
814
815
816
817
818
819
820
	temp->data = nmalloc(strlen(&inptr->data[current_word_start]) + 1);
	strcpy(temp->data, &inptr->data[current_word_start]);

	/* Inside word, remove it from original, and move cursor to right spot. */
	if (current_x >= current_word_start) {
	    right = current_x - current_word_start;
	    current_x = 0;
821
822
	    if (ISSET(AUTOINDENT)) {
		int i = 0;
823
824
		while ((inptr->next->data[i] == ' '
			|| inptr->next->data[i] == '\t')) {
825
826
		    i++;
		    right++;
827
		}
828
	    }
Chris Allegretta's avatar
Chris Allegretta committed
829
	    down = 1;
Chris Allegretta's avatar
Chris Allegretta committed
830
831
	}

Chris Allegretta's avatar
Chris Allegretta committed
832
	null_at(inptr->data, current_x);
833
834
835
836
837

	if (ISSET(MARK_ISSET) && (mark_beginbuf == inptr)) {
	    mark_beginbuf = temp;
	    mark_beginx = 0;
	}
838
839
840
841
842
    }

    /* Category 2: two or more words on the line. */
    else {
	/* Case 2a: cursor before word at wrap point. */
Chris Allegretta's avatar
Chris Allegretta committed
843
844
845
846
847
	if (current_x < current_word_start) {
	    temp->data =
		nmalloc(strlen(&inptr->data[current_word_start]) + 1);
	    strcpy(temp->data, &inptr->data[current_word_start]);

848
	    if (!isspace((int) input_char)) {
Chris Allegretta's avatar
Chris Allegretta committed
849
		i = current_word_start - 1;
850
		while (isspace((int) inptr->data[i])) {
Chris Allegretta's avatar
Chris Allegretta committed
851
852
853
854
855
856
857
858
859
860
861
		    i--;
		    assert(i >= 0);
		}
	    } else if (current_x <= last_word_end)
		i = last_word_end - 1;
	    else
		i = current_x;

	    inptr->data = nrealloc(inptr->data, i + 2);
	    inptr->data[i + 1] = 0;
	}
862
863
864


	/* Case 2b: cursor at word at wrap point. */
Chris Allegretta's avatar
Chris Allegretta committed
865
866
867
868
	else if ((current_x >= current_word_start)
		 && (current_x <= (current_word_end + 1))) {
	    temp->data =
		nmalloc(strlen(&inptr->data[current_word_start]) + 1);
869
870
871
872
873
	    strcpy(temp->data, &inptr->data[current_word_start]);

	    down = 1;

	    right = current_x - current_word_start;
874
875
	    if (ISSET(AUTOINDENT)) {
		int i = 0;
876
877
		while ((inptr->next->data[i] == ' '
			|| inptr->next->data[i] == '\t')) {
878
879
		    i++;
		    right++;
880
		}
881
882
	    }

883
	    i = current_word_start - 1;
884
885
	    if (isspace((int) input_char)
		&& (current_x == current_word_start)) {
886
887
		current_x = current_word_start;

Chris Allegretta's avatar
Chris Allegretta committed
888
		null_at(inptr->data, current_word_start);
Chris Allegretta's avatar
Chris Allegretta committed
889
	    } else {
890

891
		while (isspace((int) inptr->data[i])) {
892
		    i--;
Chris Allegretta's avatar
Chris Allegretta committed
893
		    assert(i >= 0);
894
895
896
897
		}
		inptr->data = nrealloc(inptr->data, i + 2);
		inptr->data[i + 1] = 0;
	    }
Chris Allegretta's avatar
Chris Allegretta committed
898
	}
899
900
901


	/* Case 2c: cursor past word at wrap point. */
Chris Allegretta's avatar
Chris Allegretta committed
902
903
904
	else {
	    temp->data =
		nmalloc(strlen(&inptr->data[current_word_start]) + 1);
905
906
907
908
909
910
911
912
	    strcpy(temp->data, &inptr->data[current_word_start]);

	    down = 1;
	    right = current_x - current_word_start;

	    current_x = current_word_start;
	    i = current_word_start - 1;

913
	    while (isspace((int) inptr->data[i])) {
914
		i--;
Chris Allegretta's avatar
Chris Allegretta committed
915
		assert(i >= 0);
916
917
918
		inptr->data = nrealloc(inptr->data, i + 2);
		inptr->data[i + 1] = 0;
	    }
Chris Allegretta's avatar
Chris Allegretta committed
919
	}
Chris Allegretta's avatar
Chris Allegretta committed
920
921
    }

Chris Allegretta's avatar
Chris Allegretta committed
922
    /* We pre-pend wrapped part to next line. */
923
    if (ISSET(SAMELINEWRAP) && inptr->next) {
924
	int old_x = current_x, old_y = current_y;
925

926
	/* Plus one for the space which concatenates the two lines together plus 1 for \0. */
927
928
929
	char *p =
	    nmalloc((strlen(temp->data) + strlen(inptr->next->data) + 2)
		    * sizeof(char));
930
931
932
933

	if (ISSET(AUTOINDENT)) {
	    int non = 0;

934
935
936
937
938
939
	    /* Grab the beginning of the next line until it's not a 
	       space or tab, then null terminate it so we can strcat it
	       to hell */
	    while ((inptr->next->data[non] == ' '
		    || inptr->next->data[non] == '\t'))
		p[non] = inptr->next->data[non++];
940
941
942
943
944

	    p[non] = 0;
	    strcat(p, temp->data);
	    strcat(p, " ");

945
946
	    /* Now tack on the rest of the next line after the spaces and
	       tabs */
947
948
949
950
951
952
	    strcat(p, &inptr->next->data[non]);
	} else {
	    strcpy(p, temp->data);
	    strcat(p, " ");
	    strcat(p, inptr->next->data);
	}
953

Chris Allegretta's avatar
Chris Allegretta committed
954
	free(inptr->next->data);
955
956
	inptr->next->data = p;

Chris Allegretta's avatar
Chris Allegretta committed
957
958
	free(temp->data);
	free(temp);
959
960
961

	current_x = old_x;
	current_y = old_y;
962
    }
Chris Allegretta's avatar
Chris Allegretta committed
963
    /* Else we start a new line. */
964
    else {
965

966
967
968
969
970
971
972
973
974
975
976
	temp->prev = inptr;
	temp->next = inptr->next;

	if (inptr->next)
	    inptr->next->prev = temp;
	inptr->next = temp;

	if (!temp->next)
	    filebot = temp;

	SET(SAMELINEWRAP);
977
978
979
980
981
982

	if (ISSET(AUTOINDENT)) {
	    char *spc = inptr->data;
	    char *t = NULL;
	    int extra = 0;
	    if (spc) {
983
		while ((*spc == ' ') || (*spc == '\t')) {
984
985
986
987
988
989
990
991
992
993
994
		    extra++;
		    spc++;
		    totsize++;
		}
		t = nmalloc(strlen(temp->data) + extra + 1);
		strncpy(t, inptr->data, extra);
		strcpy(t + extra, temp->data);
		free(temp->data);
		temp->data = t;
	    }
	}
995
996
997
998
    }


    totlines++;
999
1000
1001
    /* Everything about it makes me want this line here but it causes
     * totsize to be high by one for some reason.  Sigh. (Rob) */
    /* totsize++; */
1002

Chris Allegretta's avatar
Chris Allegretta committed
1003
    renumber(inptr);
1004
    edit_update(edittop, TOP);
1005

1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016

    /* Move the cursor to the new line if appropriate. */
    if (down) {
	do_right();
    }

    /* Move the cursor to the correct spot in the line if appropriate. */
    while (right--) {
	do_right();
    }

1017
    edit_update(edittop, TOP);
1018
1019
    reset_cursor();
    edit_refresh();
Chris Allegretta's avatar
Chris Allegretta committed
1020
1021
1022
}

/* Check to see if we've just caused the line to wrap to a new line */
1023
void check_wrap(filestruct * inptr, char ch)
Chris Allegretta's avatar
Chris Allegretta committed
1024
{
Chris Allegretta's avatar
Chris Allegretta committed
1025
    int len = strlenpt(inptr->data);
Chris Allegretta's avatar
Chris Allegretta committed
1026
1027
1028
1029
1030
#ifdef DEBUG
    fprintf(stderr, _("check_wrap called with inptr->data=\"%s\"\n"),
	    inptr->data);
#endif

1031
    if (len <= fill)
Chris Allegretta's avatar
Chris Allegretta committed
1032
	return;
1033
    else {
Chris Allegretta's avatar
Chris Allegretta committed
1034
	int i = actual_x(inptr, fill);
1035
1036

	/* Do not wrap if there are no words on or after wrap point. */
1037
	int char_found = 0;
1038

1039
	while (isspace((int) inptr->data[i]) && inptr->data[i])
1040
	    i++;
Chris Allegretta's avatar
Chris Allegretta committed
1041

1042
1043
	if (!inptr->data[i])
	    return;
1044

1045
1046
	/* String must be at least 1 character long. */
	for (i = strlen(inptr->data) - 1; i >= 0; i--) {
1047
	    if (isspace((int) inptr->data[i])) {
1048
1049
		if (!char_found)
		    continue;
Chris Allegretta's avatar
Chris Allegretta committed
1050
		char_found = 2;	/* 2 for yes do wrap. */
1051
		break;
Chris Allegretta's avatar
Chris Allegretta committed
1052
1053
	    } else
		char_found = 1;	/* 1 for yes found a word, but must check further. */
1054
	}
1055
1056
1057

	if (char_found == 2)
	    do_wrap(inptr, ch);
1058
    }
Chris Allegretta's avatar
Chris Allegretta committed
1059
}
1060
#endif				/* DISABLE_WRAPPING */
Chris Allegretta's avatar
Chris Allegretta committed
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101

/* Stuff we do when we abort from programs and want to clean up the
 * screen.  This doesnt do much right now.
 */
void do_early_abort(void)
{
    blank_statusbar_refresh();
}

int do_backspace(void)
{
    filestruct *previous, *tmp;

    if (current_x != 0) {
	/* Let's get dangerous */
	memmove(&current->data[current_x - 1], &current->data[current_x],
		strlen(current->data) - current_x + 1);
#ifdef DEBUG
	fprintf(stderr, _("current->data now = \"%s\"\n"), current->data);
#endif
	align(&current->data);
	do_left();
    } else {
	if (current == fileage)
	    return 0;		/* Can't delete past top of file */

	previous = current->prev;
	current_x = strlen(previous->data);
	previous->data = nrealloc(previous->data,
				  strlen(previous->data) +
				  strlen(current->data) + 1);
	strcat(previous->data, current->data);

	tmp = current;
	unlink_node(current);
	delete_node(current);
	if (current == edittop) {
	    if (previous->next)
		current = previous->next;
	    else
		current = previous;
1102
	    page_up_center();
Chris Allegretta's avatar
Chris Allegretta committed
1103
1104
1105
1106
1107
1108
1109
1110
1111
	} else {
	    if (previous->next)
		current = previous->next;
	    else
		current = previous;
	    update_line(current, current_x);
	}

	/* Ooops, sanity check */
Chris Allegretta's avatar
Chris Allegretta committed
1112
	if (tmp == filebot) {
Chris Allegretta's avatar
Chris Allegretta committed
1113
1114
	    filebot = current;
	    editbot = current;
1115
1116

	    /* Recreate the magic line if we're deleting it AND if the
1117
1118
1119
	       line we're on now is NOT blank.  if it is blank we
	       can just use IT for the magic line.   This is how Pico
	       appears to do it, in any case */
1120
1121
1122
1123
	    if (strcmp(current->data, "")) {
		new_magicline();
		fix_editbot();
	    }
Chris Allegretta's avatar
Chris Allegretta committed
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
	}

	current = previous;
	renumber(current);
	previous_line();
	totlines--;
#ifdef DEBUG
	fprintf(stderr, _("After, data = \"%s\"\n"), current->data);
#endif

    }

    totsize--;
    set_modified();
    UNSET(KEEP_CUTBUFFER);
    edit_refresh();
    return 1;
}

int do_delete(void)
{
    filestruct *foo;

    if (current_x != strlen(current->data)) {
	/* Let's get dangerous */
	memmove(&current->data[current_x], &current->data[current_x + 1],
		strlen(current->data) - current_x);

	align(&current->data);

1154
1155
	/* Now that we have a magic lnie again, we can check for both being
	   on the line before filebot as well as at filebot */
1156
    } else if (current->next != NULL && current->next != filebot) {
Chris Allegretta's avatar
Chris Allegretta committed
1157
1158
1159
1160
1161
1162
	current->data = nrealloc(current->data,
				 strlen(current->data) +
				 strlen(current->next->data) + 1);
	strcat(current->data, current->next->data);

	foo = current->next;
Chris Allegretta's avatar
Chris Allegretta committed
1163
	if (filebot == foo) {
Chris Allegretta's avatar
Chris Allegretta committed
1164
1165
1166
1167
1168
1169
1170
1171
	    filebot = current;
	    editbot = current;
	}

	unlink_node(foo);
	delete_node(foo);
	update_line(current, current_x);

1172
	/* Please see the comment in do_backspace if you don't understand
1173
	   this test */
1174
	if (current == filebot && strcmp(current->data, "")) {
1175
1176
	    new_magicline();
	    fix_editbot();
1177
	    totsize++;
1178
	}
Chris Allegretta's avatar
Chris Allegretta committed
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
	renumber(current);
	totlines--;
    } else
	return 0;

    totsize--;
    set_modified();
    UNSET(KEEP_CUTBUFFER);
    edit_refresh();
    return 1;
}

void wrap_reset(void)
{
    UNSET(SAMELINEWRAP);
}

1196
#ifndef DISABLE_SPELLER
Chris Allegretta's avatar
Chris Allegretta committed
1197
1198

int do_int_spell_fix(char *word)
Chris Allegretta's avatar
Chris Allegretta committed
1199
{
Chris Allegretta's avatar
Chris Allegretta committed
1200
    char *prevanswer = NULL, *save_search = NULL, *save_replace = NULL;
1201
1202
    filestruct *begin;
    int i = 0, j = 0, beginx, beginx_top;
Chris Allegretta's avatar
Chris Allegretta committed
1203

Chris Allegretta's avatar
Chris Allegretta committed
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
    /* save where we are */
    begin = current;
    beginx = current_x + 1;

    /* save the current search/replace strings */
    search_init_globals();
    save_search = mallocstrcpy(save_search, last_search);
    save_replace = mallocstrcpy(save_replace, last_replace);

    /* set search/replace strings to mis-spelt word */
    prevanswer = mallocstrcpy(prevanswer, word);
    last_search = mallocstrcpy(last_search, word);
    last_replace = mallocstrcpy(last_replace, word);

    /* start from the top of file */
1219
1220
    current = fileage;
    current_x = beginx_top = -1;
Chris Allegretta's avatar
Chris Allegretta committed
1221
1222
1223

    search_last_line = FALSE;

1224
1225
    edit_update(fileage, TOP);

Chris Allegretta's avatar
Chris Allegretta committed
1226
    /* make sure word is still mis-spelt (i.e. when multi-errors) */
1227
    if (findnextstr(TRUE, fileage, beginx_top, prevanswer) != NULL) {
1228
1229
1230
	do_replace_highlight(TRUE, prevanswer);

	/* allow replace word to be corrected */
1231
1232
	i = statusq(0, spell_list, SPELL_LIST_LEN, last_replace,
		    _("Edit a replacement"));
1233
1234
1235

	do_replace_highlight(FALSE, prevanswer);

Chris Allegretta's avatar
Chris Allegretta committed
1236
	/* start from the start of this line again */
1237
	current = fileage;
Chris Allegretta's avatar
Chris Allegretta committed
1238
1239
1240
1241
	current_x = beginx_top;

	search_last_line = FALSE;

1242
1243
	j = i;
	do_replace_loop(prevanswer, fileage, &beginx_top, TRUE, &j);
Chris Allegretta's avatar
Chris Allegretta committed
1244
1245
    }

Chris Allegretta's avatar
Chris Allegretta committed
1246
1247
1248
    /* restore the search/replace strings */
    last_search = mallocstrcpy(last_search, save_search);
    last_replace = mallocstrcpy(last_replace, save_replace);
Chris Allegretta's avatar
Chris Allegretta committed
1249

Chris Allegretta's avatar
Chris Allegretta committed
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
    /* restore where we were */
    current = begin;
    current_x = beginx - 1;

    edit_update(current, CENTER);

    if (i == -1)
	return FALSE;

    return TRUE;
}

/* Integrated spell checking using 'spell' program */
Chris Allegretta's avatar
Chris Allegretta committed
1263
int do_int_speller(char *tempfile_name)
Chris Allegretta's avatar
Chris Allegretta committed
1264
{
Chris Allegretta's avatar
Chris Allegretta committed
1265
1266
1267
    char *read_buff, *read_buff_ptr, *read_buff_word;
    long pipe_buff_size;
    int in_fd[2], tempfile_fd;
Chris Allegretta's avatar
Chris Allegretta committed
1268
1269
1270
1271
    int spell_status;
    pid_t pid_spell;
    ssize_t bytesread;

Chris Allegretta's avatar
Chris Allegretta committed
1272
1273
    /* Create a pipe to spell program */

Chris Allegretta's avatar
Chris Allegretta committed
1274
1275
1276
    if (pipe(in_fd) == -1)
	return FALSE;

Chris Allegretta's avatar
Chris Allegretta committed
1277
1278
    /* A new process to run spell in */

1279
    if ((pid_spell = fork()) == 0) {
Chris Allegretta's avatar
Chris Allegretta committed
1280
1281

	/* Child continues, (i.e. future spell process) */
Chris Allegretta's avatar
Chris Allegretta committed
1282
1283
1284

	close(in_fd[0]);

Chris Allegretta's avatar
Chris Allegretta committed
1285
	/* replace the standard in with the tempfile */
Chris Allegretta's avatar
Chris Allegretta committed
1286

1287
	if ((tempfile_fd = open(tempfile_name, O_RDONLY)) == -1) {
Chris Allegretta's avatar
Chris Allegretta committed
1288

Chris Allegretta's avatar
Chris Allegretta committed
1289
1290
1291
	    close(in_fd[1]);
	    exit(1);
	}
Chris Allegretta's avatar
Chris Allegretta committed
1292

Chris Allegretta's avatar
Chris Allegretta committed
1293
	if (dup2(tempfile_fd, STDIN_FILENO) != STDIN_FILENO) {
Chris Allegretta's avatar
Chris Allegretta committed
1294

Chris Allegretta's avatar
Chris Allegretta committed
1295
1296
1297
	    close(tempfile_fd);
	    close(in_fd[1]);
	    exit(1);
Chris Allegretta's avatar
Chris Allegretta committed
1298
	}
Chris Allegretta's avatar
Chris Allegretta committed
1299
	close(tempfile_fd);
Chris Allegretta's avatar
Chris Allegretta committed
1300

1301

Chris Allegretta's avatar
Chris Allegretta committed
1302
1303
1304
1305
1306
1307
	/* send spell's standard out to the pipe */

	if (dup2(in_fd[1], STDOUT_FILENO) != STDOUT_FILENO) {

	    close(in_fd[1]);
	    exit(1);
Chris Allegretta's avatar
Chris Allegretta committed
1308
	}
Chris Allegretta's avatar
Chris Allegretta committed
1309
	close(in_fd[1]);
Chris Allegretta's avatar
Chris Allegretta committed
1310

Chris Allegretta's avatar
Chris Allegretta committed
1311
	/* Start spell program, we are using the PATH here!?!? */
Chris Allegretta's avatar
Chris Allegretta committed
1312
1313
	execlp("spell", "spell", NULL);

Chris Allegretta's avatar
Chris Allegretta committed
1314
	/* Should not be reached, if spell is found!!! */
Chris Allegretta's avatar
Chris Allegretta committed
1315

Chris Allegretta's avatar
Chris Allegretta committed
1316
	exit(1);
Chris Allegretta's avatar
Chris Allegretta committed
1317
    }
Chris Allegretta's avatar
Chris Allegretta committed
1318
1319
1320

    /* Parent continues here */

Chris Allegretta's avatar
Chris Allegretta committed
1321
    close(in_fd[1]);
Chris Allegretta's avatar
Chris Allegretta committed
1322

Chris Allegretta's avatar
Chris Allegretta committed
1323
    /* Child process was not forked successfully */
Chris Allegretta's avatar
Chris Allegretta committed
1324

Chris Allegretta's avatar
Chris Allegretta committed
1325
    if (pid_spell < 0) {
Chris Allegretta's avatar
Chris Allegretta committed
1326

Chris Allegretta's avatar
Chris Allegretta committed
1327
	close(in_fd[0]);
Chris Allegretta's avatar
Chris Allegretta committed
1328
1329
1330
	return FALSE;
    }

Chris Allegretta's avatar
Chris Allegretta committed
1331
    /* Get system pipe buffer size */
Chris Allegretta's avatar
Chris Allegretta committed
1332

1333
    if ((pipe_buff_size = fpathconf(in_fd[0], _PC_PIPE_BUF)) < 1) {
Chris Allegretta's avatar
Chris Allegretta committed
1334

Chris Allegretta's avatar
Chris Allegretta committed
1335
1336
	close(in_fd[0]);
	return FALSE;
Chris Allegretta's avatar
Chris Allegretta committed
1337
1338
    }

1339
    read_buff = nmalloc(pipe_buff_size + 1);
Chris Allegretta's avatar
Chris Allegretta committed
1340

Chris Allegretta's avatar
Chris Allegretta committed
1341
    /* Process the returned spelling errors */
Chris Allegretta's avatar
Chris Allegretta committed
1342

1343
    while ((bytesread = read(in_fd[0], read_buff, pipe_buff_size)) > 0) {
Chris Allegretta's avatar
Chris Allegretta committed
1344

Chris Allegretta's avatar
Chris Allegretta committed
1345
1346
	read_buff[bytesread] = (char) NULL;
	read_buff_word = read_buff_ptr = read_buff;
Chris Allegretta's avatar
Chris Allegretta committed
1347

Chris Allegretta's avatar
Chris Allegretta committed
1348
1349
1350
1351
1352
	while (*read_buff_ptr != (char) NULL) {

	    /* Windows version may need to process additional char '\r' */

	    /* Possible problem here if last word not followed by '\n' */
Chris Allegretta's avatar
Chris Allegretta committed
1353
1354

	    if (*read_buff_ptr == '\n') {
Chris Allegretta's avatar
Chris Allegretta committed
1355
		*read_buff_ptr = (char) NULL;
1356
		if (!do_int_spell_fix(read_buff_word)) {
Chris Allegretta's avatar
Chris Allegretta committed
1357
1358
1359
1360
1361
1362

		    close(in_fd[0]);
		    free(read_buff);
		    replace_abort();

		    return TRUE;
1363
		}
Chris Allegretta's avatar
Chris Allegretta committed
1364
1365
		read_buff_word = read_buff_ptr;
		read_buff_word++;
Chris Allegretta's avatar
Chris Allegretta committed
1366
1367
1368
1369
1370
	    }

	    read_buff_ptr++;
	}
    }
Chris Allegretta's avatar
Chris Allegretta committed
1371
1372
1373

    close(in_fd[0]);
    free(read_buff);
Chris Allegretta's avatar
Chris Allegretta committed
1374
1375
    replace_abort();

Chris Allegretta's avatar
Chris Allegretta committed
1376
1377
1378
1379
1380
1381
    /* Process end of spell process */

    wait(&spell_status);
    if (WIFEXITED(spell_status)) {
	if (WEXITSTATUS(spell_status) != 0)
	    return FALSE;
1382
    } else
Chris Allegretta's avatar
Chris Allegretta committed
1383
1384
	return FALSE;

Chris Allegretta's avatar
Chris Allegretta committed
1385
1386
1387
1388
    return TRUE;
}

/* External spell checking */
Chris Allegretta's avatar
Chris Allegretta committed
1389
int do_alt_speller(char *file_name)
Chris Allegretta's avatar
Chris Allegretta committed
1390
{
Chris Allegretta's avatar
Chris Allegretta committed
1391
1392
    int alt_spell_status;
    pid_t pid_spell;
Chris Allegretta's avatar
Chris Allegretta committed
1393
1394
1395
    char *ptr;
    static int arglen = 3;
    static char **spellargs = (char **) NULL;
Chris Allegretta's avatar
Chris Allegretta committed
1396
1397
1398

    endwin();

1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
    /* Set up an argument list to pass the execvp function */
    if (spellargs == NULL) {
	spellargs = nmalloc(arglen * sizeof(char *));

	spellargs[0] = strtok(alt_speller, " ");
	while ((ptr = strtok(NULL, " ")) != NULL) {
	    arglen++;
	    spellargs = nrealloc(spellargs, arglen * sizeof(char *));
	    spellargs[arglen - 3] = ptr;
	}
	spellargs[arglen - 1] = NULL;
    }
    spellargs[arglen - 2] = file_name;
Chris Allegretta's avatar
Chris Allegretta committed
1412

1413
    /* Start a new process for the alternate speller */
1414
    if ((pid_spell = fork()) == 0) {
Chris Allegretta's avatar
Chris Allegretta committed
1415
1416

	/* Start alternate spell program, we are using the PATH here!?!? */
Chris Allegretta's avatar
Chris Allegretta committed
1417
	execvp(spellargs[0], spellargs);
Chris Allegretta's avatar
Chris Allegretta committed
1418
1419
1420
1421
1422
1423
1424
1425
1426

	/* Should not be reached, if alternate speller is found!!! */

	exit(1);
    }

    /* Could not fork?? */

    if (pid_spell < 0)
Chris Allegretta's avatar
Chris Allegretta committed
1427
1428
	return FALSE;

Chris Allegretta's avatar
Chris Allegretta committed
1429
1430
1431
1432
1433
1434
    /* Wait for alternate speller to complete */

    wait(&alt_spell_status);
    if (WIFEXITED(alt_spell_status)) {
	if (WEXITSTATUS(alt_spell_status) != 0)
	    return FALSE;
1435
    } else
Chris Allegretta's avatar
Chris Allegretta committed
1436
	return FALSE;
Chris Allegretta's avatar
Chris Allegretta committed
1437

Chris Allegretta's avatar
Chris Allegretta committed
1438
    refresh();
Chris Allegretta's avatar
Chris Allegretta committed
1439
1440
    free_filestruct(fileage);
    global_init();
Chris Allegretta's avatar
Chris Allegretta committed
1441
    open_file(file_name, 0, 1);
1442
    edit_update(fileage, CENTER);
Chris Allegretta's avatar
Chris Allegretta committed
1443
    display_main_list();
Chris Allegretta's avatar
Chris Allegretta committed
1444
    set_modified();
Chris Allegretta's avatar
Chris Allegretta committed
1445
1446
1447
1448
1449
1450
1451
1452

    return TRUE;
}
#endif

int do_spell(void)
{

1453
#ifdef DISABLE_SPELLER
1454
1455
    nano_disabled_msg();
    return (TRUE);
Chris Allegretta's avatar
Chris Allegretta committed
1456
#else
Chris Allegretta's avatar
Chris Allegretta committed
1457
1458
    char *temp;
    int spell_res;
Chris Allegretta's avatar
Chris Allegretta committed
1459

Chris Allegretta's avatar
Chris Allegretta committed
1460
1461
    if ((temp = tempnam(0, "nano.")) == NULL) {
	statusbar(_("Could not create a temporary filename: %s"),
1462
		  strerror(errno));
Chris Allegretta's avatar
Chris Allegretta committed
1463
1464
	return 0;
    }
Chris Allegretta's avatar
Chris Allegretta committed
1465

1466
1467
    if (write_file(temp, 1) == -1) {
	statusbar(_("Spell checking failed: unable to write temp file!"));
Chris Allegretta's avatar
Chris Allegretta committed
1468
	return 0;
1469
    }
Chris Allegretta's avatar
Chris Allegretta committed
1470

Chris Allegretta's avatar
Chris Allegretta committed
1471
1472
1473
1474
    if (alt_speller)
	spell_res = do_alt_speller(temp);
    else
	spell_res = do_int_speller(temp);
Chris Allegretta's avatar
Chris Allegretta committed
1475

Chris Allegretta's avatar
Chris Allegretta committed
1476
    remove(temp);
Chris Allegretta's avatar
Chris Allegretta committed
1477
1478
1479
1480
1481
1482
1483
1484

    if (spell_res)
	statusbar(_("Finished checking spelling"));
    else
	statusbar(_("Spell checking failed"));

    return spell_res;

1485
#endif
Chris Allegretta's avatar
Chris Allegretta committed
1486
}
Chris Allegretta's avatar
Chris Allegretta committed
1487
1488
1489
1490
1491
1492
1493
1494

int do_exit(void)
{
    int i;

    if (!ISSET(MODIFIED))
	finish(0);

1495
    if (ISSET(TEMP_OPT)) {
Chris Allegretta's avatar
Chris Allegretta committed
1496
1497
	i = 1;
    } else {
1498
	i = do_yesno(0, 0,
Chris Allegretta's avatar
Chris Allegretta committed
1499
1500
1501
1502
1503
1504
1505
1506
1507
		     _
		     ("Save modified buffer (ANSWERING \"No\" WILL DESTROY CHANGES) ? "));
    }

#ifdef DEBUG
    dump_buffer(fileage);
#endif

    if (i == 1) {
1508
	if (do_writeout(filename, 1) > 0)
Chris Allegretta's avatar
Chris Allegretta committed
1509
1510
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
	    finish(0);
    } else if (i == 0)
	finish(0);
    else
	statusbar(_("Cancelled"));

    display_main_list();
    return 1;
}

#ifndef NANO_SMALL
#ifdef NCURSES_MOUSE_VERSION
void do_mouse(void)
{
    MEVENT mevent;
Chris Allegretta's avatar
Chris Allegretta committed
1524
    int foo = 0, tab_found = 0;
Chris Allegretta's avatar
Chris Allegretta committed
1525
1526
1527
1528

    if (getmouse(&mevent) == ERR)
	return;

1529
1530
    /* If mouse not in edit or bottom window, return */
    if (wenclose(edit, mevent.y, mevent.x)) {
Chris Allegretta's avatar
Chris Allegretta committed
1531

1532
1533
1534
	/* Don't let people screw with the marker when they're in a
	   subfunction */
	if (currshortcut != main_list)
Chris Allegretta's avatar
Chris Allegretta committed
1535
	    return;
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568

	/* Subtract out size of topwin.  Perhaps we need a constant somewhere? */
	mevent.y -= 2;

	/* Selecting where the cursor is sets the mark.
	 * Selecting beyond the line length with the cursor at the end of the
	 * line sets the mark as well. 
	 */
	if ((mevent.y == current_y) &&
	    ((mevent.x == current_x) || (current_x == strlen(current->data)
					 && (mevent.x >
					     strlen(current->data))))) {
	    if (ISSET(VIEW_MODE)) {
		print_view_warning();
		return;
	    }
	    do_mark();
	} else if (mevent.y > current_y) {
	    while (mevent.y > current_y) {
		if (current->next != NULL)
		    current = current->next;
		else
		    break;
		current_y++;
	    }
	} else if (mevent.y < current_y) {
	    while (mevent.y < current_y) {
		if (current->prev != NULL)
		    current = current->prev;
		else
		    break;
		current_y--;
	    }
Chris Allegretta's avatar
Chris Allegretta committed
1569
	}
1570
1571
1572
1573
1574
1575
1576
1577
1578
1579
	current_x = mevent.x;
	placewewant = current_x;
	while (foo < current_x) {
	    if (current->data[foo] == NANO_CONTROL_I) {
		current_x -= tabsize - (foo % tabsize);
		tab_found = 1;
	    } else if (current->data[foo] & 0x80);
	    else if (current->data[foo] < 32)
		current_x--;
	    foo++;
Chris Allegretta's avatar
Chris Allegretta committed
1580
	}
1581
1582
1583
1584
1585
	/* This is where tab_found comes in.  I can't figure out why,
	 * but without it any line with a tab will place the cursor
	 * one character behind.  Whatever, this fixes it. */
	if (tab_found == 1)
	    current_x++;
Chris Allegretta's avatar
Chris Allegretta committed
1586

1587
1588
	if (current_x > strlen(current->data))
	    current_x = strlen(current->data);
Chris Allegretta's avatar
Chris Allegretta committed
1589

1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
	update_cursor();
	edit_refresh();
    } else if (wenclose(bottomwin, mevent.y, mevent.x) && !ISSET(NO_HELP)) {
	int k = COLS / 6, val = 0;

	/* Determine what shortcut list was clicked */
	mevent.y -= (editwinrows + 3);

	if (mevent.y < 0) /* They clicked on the statusbar */
	    return;
Chris Allegretta's avatar
Chris Allegretta committed
1600

1601
1602
1603
1604
1605
1606
1607
1608
1609
	/* Don't select stuff beyond list length */
	if (mevent.x / k >= currslen)	
	    return;

	val = currshortcut[(mevent.x / k) * 2 + mevent.y].val;

	/* And ungetch that value */
	ungetch(val);
    }
Chris Allegretta's avatar
Chris Allegretta committed
1610
1611
1612
1613
1614
1615
1616
}
#endif
#endif

/* Handler for SIGHUP */
RETSIGTYPE handle_hup(int signal)
{
1617
    die(_("Received SIGHUP"));
Chris Allegretta's avatar
Chris Allegretta committed
1618
1619
}

1620
1621
1622
1623
1624
1625
1626
1627
1628
/* What do we do when we catch the suspend signal */
RETSIGTYPE do_suspend(int signal)
{

    act.sa_handler = SIG_DFL;
    sigemptyset(&act.sa_mask);
    sigaction(SIGTSTP, &act, NULL);

    endwin();
1629
    fprintf(stderr, "\n\n\n\n\nUse \"fg\" to return to nano\n");
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
    raise(SIGTSTP);
}

/* Restore the suspend handler when we come back into the prog */
RETSIGTYPE do_cont(int signal)
{

    act.sa_handler = do_suspend;
    sigemptyset(&act.sa_mask);
    sigaction(SIGTSTP, &act, NULL);
    initscr();
    total_refresh();
}
Chris Allegretta's avatar
Chris Allegretta committed
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665

void handle_sigwinch(int s)
{
#ifndef NANO_SMALL
    char *tty = NULL;
    int fd = 0;
    int result = 0;
    struct winsize win;

    tty = ttyname(0);
    if (!tty)
	return;
    fd = open(tty, O_RDWR);
    if (fd == -1)
	return;
    result = ioctl(fd, TIOCGWINSZ, &win);
    if (result == -1)
	return;


    COLS = win.ws_col;
    LINES = win.ws_row;

1666
1667
1668
1669
1670
    if ((editwinrows = LINES - 5 + no_help()) < MIN_EDITOR_ROWS)
	die_too_small();

    if ((fill = COLS - CHARS_FROM_EOL) < MIN_FILL_LENGTH)
	die_too_small();
Chris Allegretta's avatar
Chris Allegretta committed
1671

1672
1673
1674
    hblank = nrealloc(hblank, COLS + 1);
    memset(hblank, ' ', COLS);
    hblank[COLS] = 0;
Chris Allegretta's avatar
Chris Allegretta committed
1675

1676
#ifdef HAVE_RESIZETERM
Chris Allegretta's avatar
Chris Allegretta committed
1677
1678
1679
1680
1681
1682
1683
1684
1685
1686
1687
1688
1689
1690
1691
    resizeterm(LINES, COLS);
#ifdef HAVE_WRESIZE
    if (wresize(topwin, 2, COLS) == ERR)
	die(_("Cannot resize top win"));
    if (mvwin(topwin, 0, 0) == ERR)
	die(_("Cannot move top win"));
    if (wresize(edit, editwinrows, COLS) == ERR)
	die(_("Cannot resize edit win"));
    if (mvwin(edit, 2, 0) == ERR)
	die(_("Cannot move edit win"));
    if (wresize(bottomwin, 3 - no_help(), COLS) == ERR)
	die(_("Cannot resize bottom win"));
    if (mvwin(bottomwin, LINES - 3 + no_help(), 0) == ERR)
	die(_("Cannot move bottom win"));
#endif				/* HAVE_WRESIZE */
1692
#endif				/* HAVE_RESIZETERM */
Chris Allegretta's avatar
Chris Allegretta committed
1693

Robert Siemborski's avatar
Robert Siemborski committed
1694
    fix_editbot();
Chris Allegretta's avatar
Chris Allegretta committed
1695

Chris Allegretta's avatar
Chris Allegretta committed
1696
    if (current_y > editwinrows - 1) {
1697
	edit_update(editbot, CENTER);
Chris Allegretta's avatar
Chris Allegretta committed
1698
1699
    }
    erase();
1700
1701

    /* Do these b/c width may have changed... */
Chris Allegretta's avatar
Chris Allegretta committed
1702
    refresh();
Chris Allegretta's avatar
Chris Allegretta committed
1703
    titlebar(NULL);
1704
1705
    edit_refresh();
    display_main_list();
1706
    blank_statusbar();
Chris Allegretta's avatar
Chris Allegretta committed
1707
    total_refresh();
1708

1709
1710
1711
    /* Turn cursor back on for sure */
    curs_set(1);

1712
1713
1714
    /* Jump back to mainloop */
    siglongjmp(jmpbuf, 1);

Chris Allegretta's avatar
Chris Allegretta committed
1715
1716
1717
#endif
}

1718
1719
1720
1721
1722
1723
1724
1725
void signal_init(void)
{

    /* Trap SIGINT and SIGQUIT  cuz we want them to do useful things. */
    memset(&act, 0, sizeof(struct sigaction));
    act.sa_handler = SIG_IGN;
    sigaction(SIGINT, &act, NULL);

1726
1727
    if (!ISSET(SUSPEND)) {
	sigaction(SIGTSTP, &act, NULL);
1728
    } else {
1729
	act.sa_handler = do_suspend;
1730
1731
	sigaction(SIGTSTP, &act, NULL);

1732
	act.sa_handler = do_cont;
1733
	sigaction(SIGCONT, &act, NULL);
1734
1735
1736
    }


1737
1738
1739
1740
1741
1742
1743
1744
1745
    /* Trap SIGHUP  cuz we want to write the file out. */
    act.sa_handler = handle_hup;
    sigaction(SIGHUP, &act, NULL);

    act.sa_handler = handle_sigwinch;
    sigaction(SIGWINCH, &act, NULL);

}

Chris Allegretta's avatar
Chris Allegretta committed
1746
1747
void window_init(void)
{
1748
1749
    if ((editwinrows = LINES - 5 + no_help()) < MIN_EDITOR_ROWS)
	die_too_small();
1750

Chris Allegretta's avatar
Chris Allegretta committed
1751
1752
1753
1754
1755
    /* Setup up the main text window */
    edit = newwin(editwinrows, COLS, 2, 0);

    /* And the other windows */
    topwin = newwin(2, COLS, 0, 0);
1756
    bottomwin = newwin(3 - no_help(), COLS, LINES - 3 + no_help(), 0);
1757

Chris Allegretta's avatar
Chris Allegretta committed
1758
1759
1760
1761
1762
1763
1764
#ifdef PDCURSES
    /* Oops, I guess we need this again.
       Moved here so the keypad still works after a Meta-X, for example */
    keypad(edit, TRUE);
    keypad(bottomwin, TRUE);
#endif

Chris Allegretta's avatar
Chris Allegretta committed
1765
1766
}

1767
1768
1769
1770
1771
void mouse_init(void)
{
#ifndef NANO_SMALL
#ifdef NCURSES_MOUSE_VERSION
    if (ISSET(USE_MOUSE)) {
1772
	keypad_on(edit, 1);
1773
	keypad_on(bottomwin, 1);
1774

1775
1776
	mousemask(BUTTON1_RELEASED, NULL);
	mouseinterval(50);
1777

1778
    } else
1779
	mousemask(0, NULL);
1780

1781
1782
1783
1784
1785
#endif
#endif

}

Chris Allegretta's avatar
Chris Allegretta committed
1786
1787
1788
1789
1790
1791
int do_tab(void)
{
    do_char('\t');
    return 1;
}

1792
#ifndef DISABLE_JUSTIFY
Chris Allegretta's avatar
Chris Allegretta committed
1793
1794
1795
int empty_line(const char *data)
{
    while (*data) {
1796
	if (!isspace((int) *data))
Chris Allegretta's avatar
Chris Allegretta committed
1797
1798
1799
1800
1801
1802
1803
1804
1805
1806
1807
	    return 0;

	data++;
    }

    return 1;
}

int no_spaces(const char *data)
{
    while (*data) {
1808
	if (isspace((int) *data))
Chris Allegretta's avatar
Chris Allegretta committed
1809
1810
1811
1812
1813
1814
1815
1816
1817
1818
1819
1820
1821
1822
1823
	    return 0;

	data++;
    }

    return 1;
}

void justify_format(char *data)
{
    int i = 0;
    int len = strlen(data);

    /* Skip first character regardless and leading whitespace. */
    for (i = 1; i < len; i++) {
1824
	if (!isspace((int) data[i]))
Chris Allegretta's avatar
Chris Allegretta committed
1825
1826
1827
1828
1829
1830
1831
	    break;
    }

    i++;			/* (i) is now at least 2. */

    /* No double spaces allowed unless following a period.  Tabs -> space.  No double tabs. */
    for (; i < len; i++) {
1832
	if (isspace((int) data[i]) && isspace((int) data[i - 1])
Chris Allegretta's avatar
Chris Allegretta committed
1833
1834
1835
1836
1837
1838
1839
1840
1841
1842
1843
	    && (data[i - 2] != '.')) {
	    memmove(data + i, data + i + 1, len - i);
	    len--;
	    i--;
	}
    }
}
#endif

int do_justify(void)
{
1844
#ifdef DISABLE_JUSTIFY
1845
1846
1847
    nano_disabled_msg();
    return 1;
#else
Chris Allegretta's avatar
Chris Allegretta committed
1848
    int slen = 0;		/* length of combined lines on one line. */
1849
    int initial_y, kbinput = 0, totbak;
1850
    filestruct *initial = NULL, *tmpjust = NULL, *cutbak, *tmptop, *tmpbot;
Chris Allegretta's avatar
Chris Allegretta committed
1851
1852
1853
1854
1855
1856
1857
1858
1859
1860
1861
1862
1863
1864
1865
1866
1867

    if (empty_line(current->data)) {
	/* Justify starting at first non-empty line. */
	do {
	    if (!current->next)
		return 1;

	    current = current->next;
	    current_y++;
	}
	while (empty_line(current->data));
    } else {
	/* Search back for the beginning of the paragraph, where
	 *   Paragraph is  1)  A line with leading whitespace
	 *             or  2)  A line following an empty line.
	 */
	while (current->prev != NULL) {
1868
	    if (isspace((int) current->data[0]) || !current->data[0])
Chris Allegretta's avatar
Chris Allegretta committed
1869
1870
1871
1872
1873
1874
1875
1876
1877
1878
1879
1880
1881
1882
1883
1884
1885
1886
1887
		break;

	    current = current->prev;
	    current_y--;
	}

	/* First line with leading whitespace may be empty. */
	if (empty_line(current->data)) {
	    if (current->next) {
		current = current->next;
		current_y++;
	    } else
		return 1;
	}
    }
    initial = current;
    initial_y = current_y;

    set_modified();
1888
    cutbak = cutbuffer;		/* Got to like cutbak ;) */
1889
    totbak = totsize;
1890
1891
1892
1893
    cutbuffer = NULL;

    tmptop = current;
    tmpjust = copy_node(current);
Robert Siemborski's avatar
Robert Siemborski committed
1894
1895

    /* This is annoying because it mucks with totsize */
1896
    add_to_cutbuffer(tmpjust);
Robert Siemborski's avatar
Robert Siemborski committed
1897

Chris Allegretta's avatar
Chris Allegretta committed
1898
    /* Put the whole paragraph into one big line. */
1899
    while (current->next && !isspace((int) current->next->data[0])
Chris Allegretta's avatar
Chris Allegretta committed
1900
1901
1902
1903
1904
	   && current->next->data[0]) {
	filestruct *tmpnode = current->next;
	int len = strlen(current->data);
	int len2 = strlen(current->next->data);

1905
1906
1907
1908
	tmpjust = NULL;
	tmpjust = copy_node(current->next);
	add_to_cutbuffer(tmpjust);

Robert Siemborski's avatar
Robert Siemborski committed
1909
	/* Wiping out a newline */
1910
	totsize--;
Robert Siemborski's avatar
Robert Siemborski committed
1911

Chris Allegretta's avatar
Chris Allegretta committed
1912
1913
1914
1915
1916
1917
1918
1919
1920
1921
1922
1923
1924
1925
	/* length of both strings plus space between strings and ending \0. */
	current->data = nrealloc(current->data, len + len2 + 2);
	current->data[len++] = ' ';
	current->data[len] = '\0';

	strncat(current->data, current->next->data, len2);

	unlink_node(tmpnode);
	delete_node(tmpnode);
    }

    justify_format(current->data);

    slen = strlen(current->data);
1926
    totsize += slen;
Chris Allegretta's avatar
Chris Allegretta committed
1927

1928
1929
    if ((strlenpt(current->data) > (fill))
	&& !no_spaces(current->data)) {
1930
1931
1932
1933
1934
1935
1936
1937
1938
1939
1940
1941
	do {
	    int i = 0;
	    int len2 = 0;
	    filestruct *tmpline = nmalloc(sizeof(filestruct));

	    /* Start at fill , unless line isn't that long (but it 
	     * appears at least fill long with tabs.
	     */
	    if (slen > fill)
		i = fill;
	    else
		i = slen;
Robert Siemborski's avatar
Robert Siemborski committed
1942

1943
	    for (; i > 0; i--) {
1944
		if (isspace((int) current->data[i]) &&
1945
1946
1947
		    ((strlenpt(current->data) - strlen(current->data + i))
		     <= fill))
		    break;
1948
	    }
Robert Siemborski's avatar
Robert Siemborski committed
1949

1950
	    if (!i)
1951
		break;
Chris Allegretta's avatar
Chris Allegretta committed
1952

1953
	    current->data[i] = '\0';
Chris Allegretta's avatar
Chris Allegretta committed
1954

1955
1956
	    len2 = strlen(current->data + i + 1);
	    tmpline->data = nmalloc(len2 + 1);
Chris Allegretta's avatar
Chris Allegretta committed
1957

1958
1959
1960
	    /* Skip the white space in current. */
	    memcpy(tmpline->data, current->data + i + 1, len2);
	    tmpline->data[len2] = '\0';
Chris Allegretta's avatar
Chris Allegretta committed
1961

1962
	    current->data = nrealloc(current->data, i + 1);
Chris Allegretta's avatar
Chris Allegretta committed
1963

1964
1965
1966
1967
	    tmpline->prev = current;
	    tmpline->next = current->next;
	    if (current->next != NULL)
		current->next->prev = tmpline;
Chris Allegretta's avatar
Chris Allegretta committed
1968

1969
1970
1971
1972
	    current->next = tmpline;
	    current = tmpline;
	    slen -= i + 1;
	    current_y++;
1973
1974
	} while ((strlenpt(current->data) > (fill))
		 && !no_spaces(current->data));
1975
    }
1976
    tmpbot = current;
Chris Allegretta's avatar
Chris Allegretta committed
1977
1978
1979

    if (current->next)
	current = current->next;
1980
1981
    else
	filebot = current;
Chris Allegretta's avatar
Chris Allegretta committed
1982
1983
1984
    current_x = 0;
    placewewant = 0;

1985
1986
1987
1988
1989
    renumber(initial);
    totlines = filebot->lineno;

    werase(edit);

Chris Allegretta's avatar
Chris Allegretta committed
1990
1991
    if ((current_y < 0) || (current_y >= editwinrows - 1)
	|| (initial_y <= 0)) {
1992
	edit_update(current, CENTER);
Chris Allegretta's avatar
Chris Allegretta committed
1993
1994
	center_cursor();
    } else {
Robert Siemborski's avatar
Robert Siemborski committed
1995
	fix_editbot();
Chris Allegretta's avatar
Chris Allegretta committed
1996
1997
    }

1998
    edit_refresh();
1999
    statusbar(_("Can now UnJustify!"));
2000
2001
2002
    /* Change the shortcut list to display the unjustify code */
    shortcut_init(1);
    display_main_list();
2003
2004
    reset_cursor();

2005
    /* Now get a keystroke and see if it's unjustify, if not unget the keytroke 
2006
       and return */
2007
    if ((kbinput = wgetch(edit)) != NANO_UNJUSTIFY_KEY) {
2008
	ungetch(kbinput);
2009
2010
	blank_statusbar_refresh();
    } else {
2011
2012
2013
	/* Else restore the justify we just did (ungrateful user!) */
	if (tmptop->prev != NULL)
	    tmptop->prev->next = tmpbot->next;
2014
2015
	else
	    fileage = current;
2016
	tmpbot->next->prev = tmptop->prev;
2017
	current = tmpbot->next;
2018
2019
	tmpbot->next = NULL;
	do_uncut_text();
2020
2021
2022
	if (tmptop->prev == NULL)
	    edit_refresh();

2023
2024
	/* Restore totsize from befure justify */
	totsize = totbak;
2025
2026
2027
	free_filestruct(tmptop);
	blank_statusbar_refresh();
    }
2028
2029
    shortcut_init(0);
    display_main_list();
2030
2031
    free_filestruct(cutbuffer);
    cutbuffer = cutbak;
2032

Chris Allegretta's avatar
Chris Allegretta committed
2033
2034
2035
2036
    return 1;
#endif
}

2037
#ifndef DISABLE_HELP
Chris Allegretta's avatar
Chris Allegretta committed
2038
2039
2040
2041
void help_init(void)
{
    int i, sofar = 0;
    long allocsize = 1;		/* How much space we're gonna need for the help text */
2042
    char buf[BUFSIZ] = "";
Chris Allegretta's avatar
Chris Allegretta committed
2043
2044
2045

    /* Compute the space needed for the shortcut lists - we add 15 to
       have room for the shortcut abbrev and its possible alternate keys */
2046
    for (i = 0; i <= MAIN_LIST_LEN - 1; i++)
Chris Allegretta's avatar
Chris Allegretta committed
2047
2048
2049
	if (main_list[i].help != NULL)
	    allocsize += strlen(main_list[i].help) + 15;

2050
    /* And for the toggle list, we also allocate space for extra text. */
2051
    for (i = 0; i <= TOGGLE_LEN - 1; i++)
2052
2053
2054
	if (toggles[i].desc != NULL)
	    allocsize += strlen(toggles[i].desc) + 30;

Chris Allegretta's avatar
Chris Allegretta committed
2055
2056
2057
2058
2059
2060
2061
2062
2063
2064
2065
2066
    allocsize += strlen(help_text_init);

    if (help_text != NULL)
	free(help_text);

    /* Allocate space for the help text */
    help_text = nmalloc(allocsize);

    /* Now add the text we want */
    strcpy(help_text, help_text_init);

    /* Now add our shortcut info */
2067
    for (i = 0; i <= MAIN_LIST_LEN - 1; i++) {
2068
	sofar = snprintf(buf, BUFSIZ, "^%c	", main_list[i].val + 64);
Chris Allegretta's avatar
Chris Allegretta committed
2069
2070

	if (main_list[i].misc1 > KEY_F0 && main_list[i].misc1 <= KEY_F(64))
2071
	    sofar += snprintf(&buf[sofar], BUFSIZ - sofar, "(F%d)	",
2072
			      main_list[i].misc1 - KEY_F0);
Chris Allegretta's avatar
Chris Allegretta committed
2073
	else
2074
	    sofar += snprintf(&buf[sofar], BUFSIZ - sofar, "	");
Chris Allegretta's avatar
Chris Allegretta committed
2075
2076

	if (main_list[i].altval > 0)
2077
	    sofar += snprintf(&buf[sofar], BUFSIZ - sofar, "(M-%c)	",
2078
			      main_list[i].altval - 32);
Chris Allegretta's avatar
Chris Allegretta committed
2079
	else
2080
	    sofar += snprintf(&buf[sofar], BUFSIZ - sofar, "	");
Chris Allegretta's avatar
Chris Allegretta committed
2081

2082

Chris Allegretta's avatar
Chris Allegretta committed
2083
	if (main_list[i].help != NULL)
2084
	    snprintf(&buf[sofar], BUFSIZ - sofar, "%s", main_list[i].help);
Chris Allegretta's avatar
Chris Allegretta committed
2085

2086

Chris Allegretta's avatar
Chris Allegretta committed
2087
	strcat(help_text, buf);
2088
	strcat(help_text, "\n");
Chris Allegretta's avatar
Chris Allegretta committed
2089
2090
    }

2091
    /* And the toggles... */
2092
    for (i = 0; i <= TOGGLE_LEN - 1; i++) {
2093
2094
	sofar = snprintf(buf, BUFSIZ,
			 "M-%c			", toggles[i].val - 32);
2095
2096

	if (toggles[i].desc != NULL)
2097
2098
	    snprintf(&buf[sofar], BUFSIZ - sofar, _("%s enable/disable"),
		     toggles[i].desc);
2099
2100
2101
2102
2103
2104

	strcat(help_text, buf);
	strcat(help_text, "\n");
    }

}
2105
#endif
2106
2107
2108

void do_toggle(int which)
{
2109
2110
2111
#ifdef NANO_SMALL
    nano_disabled_msg();
#else
Jordi Mallach's avatar
   
Jordi Mallach committed
2112
2113
    char *enabled = _("enabled");
    char *disabled = _("disabled");
2114

Chris Allegretta's avatar
Chris Allegretta committed
2115
    if (ISSET(toggles[which].flag))
2116
	UNSET(toggles[which].flag);
Chris Allegretta's avatar
Chris Allegretta committed
2117
    else
2118
	SET(toggles[which].flag);
Chris Allegretta's avatar
Chris Allegretta committed
2119

2120
2121
    switch (toggles[which].val) {
    case TOGGLE_PICOMODE_KEY:
2122
	shortcut_init(0);
2123
2124
2125
2126
2127
2128
2129
2130
2131
	display_main_list();
	break;
    case TOGGLE_SUSPEND_KEY:
	signal_init();
	break;
    case TOGGLE_MOUSE_KEY:
	mouse_init();
	break;
    case TOGGLE_NOHELP_KEY:
Chris Allegretta's avatar
Chris Allegretta committed
2132
2133
2134
	wclear(bottomwin);
	wrefresh(bottomwin);
	window_init();
2135
	fix_editbot();
Chris Allegretta's avatar
Chris Allegretta committed
2136
2137
	edit_refresh();
	display_main_list();
2138
2139
	break;
    }
Chris Allegretta's avatar
Chris Allegretta committed
2140
2141
2142

    if (!ISSET(toggles[which].flag)) {
	if (toggles[which].val == TOGGLE_NOHELP_KEY ||
2143
	    toggles[which].val == TOGGLE_WRAP_KEY)
Chris Allegretta's avatar
Chris Allegretta committed
2144
2145
2146
2147
2148
	    statusbar("%s %s", toggles[which].desc, enabled);
	else
	    statusbar("%s %s", toggles[which].desc, disabled);
    } else {
	if (toggles[which].val == TOGGLE_NOHELP_KEY ||
2149
	    toggles[which].val == TOGGLE_WRAP_KEY)
Chris Allegretta's avatar
Chris Allegretta committed
2150
2151
2152
2153
	    statusbar("%s %s", toggles[which].desc, disabled);
	else
	    statusbar("%s %s", toggles[which].desc, enabled);
    }
2154
2155
2156
    SET(DISABLE_CURPOS);

#endif
Chris Allegretta's avatar
Chris Allegretta committed
2157
2158
}

2159
2160
2161
2162
2163
2164
/* If the NumLock key has made the keypad gone awry, print an error
   message, hopefully we can address it later. */
void print_numlock_warning(void)
{
    static int didmsg = 0;
    if (!didmsg) {
2165
2166
	statusbar(_
		  ("NumLock glitch detected.  Keypad will malfunction with NumLock off"));
2167
2168
2169
2170
	didmsg = 1;
    }
}

2171
2172
2173
2174
2175
/* This function returns the correct keystroke, given the A,B,C or D
   input key.  This is a common sequence of many terms which send
   Esc-O-[A-D] or Esc-[-[A-D]. */
int ABCD(int input)
{
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
    switch (input) {
    case 'A':
	return (KEY_UP);
    case 'B':
	return (KEY_DOWN);
    case 'C':
	return (KEY_RIGHT);
    case 'D':
	return (KEY_LEFT);
    default:
	return 0;
2187
2188
2189
    }
}

Chris Allegretta's avatar
Chris Allegretta committed
2190
2191
2192
2193
2194
int main(int argc, char *argv[])
{
    int optchr;
    int kbinput;		/* Input from keyboard */
    long startline = 0;		/* Line to try and start at */
2195
2196
    int keyhandled;		/* Have we handled the keystroke yet? */
    int i, modify_control_seq;
Chris Allegretta's avatar
Chris Allegretta committed
2197
    char *argv0;
Chris Allegretta's avatar
Chris Allegretta committed
2198
#ifdef _POSIX_VDISABLE
Chris Allegretta's avatar
Chris Allegretta committed
2199
    struct termios term;
Chris Allegretta's avatar
Chris Allegretta committed
2200
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2201
2202
2203
2204

#ifdef HAVE_GETOPT_LONG
    int option_index = 0;
    struct option long_options[] = {
2205
#ifdef HAVE_REGEX_H
2206
	{"regexp", 0, 0, 'R'},
2207
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2208
2209
2210
2211
2212
2213
	{"version", 0, 0, 'V'},
	{"const", 0, 0, 'c'},
	{"suspend", 0, 0, 'z'},
	{"nowrap", 0, 0, 'w'},
	{"nohelp", 0, 0, 'x'},
	{"help", 0, 0, 'h'},
2214
	{"view", 0, 0, 'v'},
2215
#ifndef NANO_SMALL
2216
	{"cut", 0, 0, 'k'},
2217
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2218
2219
	{"autoindent", 0, 0, 'i'},
	{"tempfile", 0, 0, 't'},
2220
#ifndef DISABLE_SPELLER
Chris Allegretta's avatar
Chris Allegretta committed
2221
	{"speller", 1, 0, 's'},
2222
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2223
2224
2225
2226
	{"fill", 1, 0, 'r'},
	{"mouse", 0, 0, 'm'},
	{"pico", 0, 0, 'p'},
	{"nofollow", 0, 0, 'l'},
2227
	{"tabsize", 1, 0, 'T'},
Chris Allegretta's avatar
Chris Allegretta committed
2228
2229
2230
2231
2232
2233
2234
2235
	{0, 0, 0, 0}
    };
#endif

    /* Flag inits... */
    SET(FOLLOW_SYMLINKS);

#ifndef NANO_SMALL
2236
#ifdef ENABLE_NLS
Chris Allegretta's avatar
Chris Allegretta committed
2237
2238
2239
2240
    setlocale(LC_ALL, "");
    bindtextdomain(PACKAGE, LOCALEDIR);
    textdomain(PACKAGE);
#endif
2241
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2242
2243

#ifdef HAVE_GETOPT_LONG
2244
    while ((optchr = getopt_long(argc, argv, "?T:RVbcefhiklmpr:s:tvwxz",
Chris Allegretta's avatar
Chris Allegretta committed
2245
2246
				 long_options, &option_index)) != EOF) {
#else
2247
2248
    while ((optchr =
	    getopt(argc, argv, "h?T:RVbcefiklmpr:s:tvwxz")) != EOF) {
Chris Allegretta's avatar
Chris Allegretta committed
2249
2250
2251
#endif

	switch (optchr) {
Chris Allegretta's avatar
Chris Allegretta committed
2252
	case 'T':
2253
2254
	    tabsize = atoi(optarg);
	    if (tabsize <= 0) {
Chris Allegretta's avatar
Chris Allegretta committed
2255
2256
2257
2258
		usage();	/* To stop bogus data for tab width */
		finish(1);
	    }
	    break;
2259
#ifdef HAVE_REGEX_H
2260
2261
2262
	case 'R':
	    SET(USE_REGEXP);
	    break;
2263
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2264
2265
2266
	case 'V':
	    version();
	    exit(0);
2267
2268
2269
	case 'b':
	case 'e':
	case 'f':
2270
2271
	    /* Pico compatibility flags */
	    break;
Chris Allegretta's avatar
Chris Allegretta committed
2272
2273
2274
2275
2276
2277
2278
2279
2280
2281
	case 'c':
	    SET(CONSTUPDATE);
	    break;
	case 'h':
	case '?':
	    usage();
	    exit(0);
	case 'i':
	    SET(AUTOINDENT);
	    break;
2282
#ifndef NANO_SMALL
2283
2284
2285
	case 'k':
	    SET(CUT_TO_END);
	    break;
2286
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2287
2288
2289
2290
2291
2292
2293
	case 'l':
	    UNSET(FOLLOW_SYMLINKS);
	    break;
	case 'm':
	    SET(USE_MOUSE);
	    break;
	case 'p':
2294
	    SET(PICO_MODE);
Chris Allegretta's avatar
Chris Allegretta committed
2295
2296
2297
2298
2299
2300
2301
2302
	    break;
	case 'r':
	    fill = atoi(optarg);
	    if (fill <= 0) {
		usage();	/* To stop bogus data (like a string) */
		finish(1);
	    }
	    break;
2303
#ifndef DISABLE_SPELLER
Chris Allegretta's avatar
Chris Allegretta committed
2304
2305
2306
2307
	case 's':
	    alt_speller = nmalloc(strlen(optarg) + 1);
	    strcpy(alt_speller, optarg);
	    break;
2308
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2309
	case 't':
2310
	    SET(TEMP_OPT);
Chris Allegretta's avatar
Chris Allegretta committed
2311
2312
2313
2314
2315
	    break;
	case 'v':
	    SET(VIEW_MODE);
	    break;
	case 'w':
2316
2317
2318
2319
#ifdef DISABLE_WRAPPING
	    usage();
	    exit(0);
#else
Chris Allegretta's avatar
Chris Allegretta committed
2320
2321
	    SET(NO_WRAP);
	    break;
2322
#endif				/* DISABLE_WRAPPING */
Chris Allegretta's avatar
Chris Allegretta committed
2323
2324
2325
2326
2327
2328
2329
2330
	case 'x':
	    SET(NO_HELP);
	    break;
	case 'z':
	    SET(SUSPEND);
	    break;
	default:
	    usage();
Chris Allegretta's avatar
Chris Allegretta committed
2331
	    exit(0);
Chris Allegretta's avatar
Chris Allegretta committed
2332
2333
2334
2335
2336
2337
2338
	}

    }

    argv0 = strrchr(argv[0], '/');
    if ((argv0 && strstr(argv0, "pico"))
	|| (!argv0 && strstr(argv[0], "pico")))
2339
	SET(PICO_MODE);
Chris Allegretta's avatar
Chris Allegretta committed
2340
2341
2342
2343

    /* See if there's a non-option in argv (first non-option is the
       filename, if +LINE is not given) */
    if (argc == 1 || argc <= optind)
Chris Allegretta's avatar
Chris Allegretta committed
2344
	clear_filename();
Chris Allegretta's avatar
Chris Allegretta committed
2345
2346
2347
2348
2349
2350
    else {
	/* Look for the +line flag... */
	if (argv[optind][0] == '+') {
	    startline = atoi(&argv[optind][1]);
	    optind++;
	    if (argc == 1 || argc <= optind)
Chris Allegretta's avatar
Chris Allegretta committed
2351
		clear_filename();
Chris Allegretta's avatar
Chris Allegretta committed
2352
	    else
Chris Allegretta's avatar
Chris Allegretta committed
2353
		filename = mallocstrcpy(filename, argv[optind]);
Chris Allegretta's avatar
Chris Allegretta committed
2354

Chris Allegretta's avatar
Chris Allegretta committed
2355
2356
	} else
	    filename = mallocstrcpy(filename, argv[optind]);
Chris Allegretta's avatar
Chris Allegretta committed
2357
2358
2359
2360
    }


    /* First back up the old settings so they can be restored, duh */
Chris Allegretta's avatar
Chris Allegretta committed
2361
    tcgetattr(0, &oldterm);
Chris Allegretta's avatar
Chris Allegretta committed
2362

2363
#ifdef _POSIX_VDISABLE
2364
2365
2366
2367
    term = oldterm;
    term.c_cc[VINTR] = _POSIX_VDISABLE;
    term.c_cc[VQUIT] = _POSIX_VDISABLE;
    term.c_lflag &= ~IEXTEN;
Chris Allegretta's avatar
Chris Allegretta committed
2368
    tcsetattr(0, TCSANOW, &term);
2369
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2370
2371
2372
2373
2374
2375
2376
2377
2378
2379

    /* now ncurses init stuff... */
    initscr();
    savetty();
    nonl();
    cbreak();
    noecho();

    /* Set up some global variables */
    global_init();
2380
    shortcut_init(0);
2381
#ifndef DISABLE_HELP
Chris Allegretta's avatar
Chris Allegretta committed
2382
2383
    init_help_msg();
    help_init();
2384
#endif
2385
    signal_init();
Chris Allegretta's avatar
Chris Allegretta committed
2386
2387
2388
2389
2390

#ifdef DEBUG
    fprintf(stderr, _("Main: set up windows\n"));
#endif

Chris Allegretta's avatar
Chris Allegretta committed
2391
    window_init();
2392
    mouse_init();
Chris Allegretta's avatar
Chris Allegretta committed
2393
2394
2395
2396
2397
2398
2399
2400
2401
2402
2403

#ifdef DEBUG
    fprintf(stderr, _("Main: bottom win\n"));
#endif
    /* Set up up bottom of window */
    display_main_list();

#ifdef DEBUG
    fprintf(stderr, _("Main: open file\n"));
#endif

Chris Allegretta's avatar
Chris Allegretta committed
2404
    titlebar(NULL);
2405
2406
2407
2408

    /* Now we check to see if argv[optind] is non-null to determine if
       we're dealing with a new file or not, not argc == 1... */
    if (argv[optind] == NULL)
Chris Allegretta's avatar
Chris Allegretta committed
2409
2410
2411
2412
2413
2414
2415
	new_file();
    else
	open_file(filename, 0, 0);

    if (startline > 0)
	do_gotoline(startline);
    else
2416
	edit_update(fileage, CENTER);
Chris Allegretta's avatar
Chris Allegretta committed
2417

2418
    /* return here after a sigwinch */
2419
    sigsetjmp(jmpbuf, 1);
2420
2421
2422
2423
2424
2425

    /* Fix clobber-age */
    kbinput = 0;
    keyhandled = 0;
    modify_control_seq = 0;

Robert Siemborski's avatar
Robert Siemborski committed
2426
2427
2428
    edit_refresh();
    reset_cursor();

Chris Allegretta's avatar
Chris Allegretta committed
2429
    while (1) {
2430

2431
2432
2433
	currshortcut = main_list;
	currslen = MAIN_VISIBLE;

2434
2435
2436
2437
2438
#ifndef _POSIX_VDISABLE
	/* We're going to have to do it the old way, i.e. on cygwin */
	raw();
#endif

Chris Allegretta's avatar
Chris Allegretta committed
2439
	kbinput = wgetch(edit);
2440
2441
2442
#ifdef DEBUG
	fprintf(stderr, "AHA!  %c (%d)\n", kbinput, kbinput);
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2443
2444
	if (kbinput == 27) {	/* Grab Alt-key stuff first */
	    switch (kbinput = wgetch(edit)) {
2445
		/* Alt-O, suddenly very important ;) */
2446
2447
	    case 79:
		kbinput = wgetch(edit);
2448
		if (kbinput <= 'D' && kbinput >= 'A')
2449
		    kbinput = ABCD(kbinput);
2450
2451
2452
		else if (kbinput <= 'z' && kbinput >= 'j')
		    print_numlock_warning();
		else if (kbinput <= 'S' && kbinput >= 'P')
2453
		    kbinput = KEY_F(kbinput - 79);
2454
2455
#ifdef DEBUG
		else {
2456
2457
2458
		    fprintf(stderr, _("I got Alt-O-%c! (%d)\n"),
			    kbinput, kbinput);
		    break;
2459
2460
2461
		}
#endif
		break;
2462
2463
2464
2465
2466
2467
	    case 27:
		/* If we get Alt-Alt, the next keystroke should be the same as a
		   control sequence */
		modify_control_seq = 1;
		keyhandled = 1;
		break;
Chris Allegretta's avatar
Chris Allegretta committed
2468
2469
	    case 91:
		switch (kbinput = wgetch(edit)) {
2470
		case '1':	/* Alt-[-1-[0-5,7-9] = F1-F8 in X at least */
2471
2472
		    kbinput = wgetch(edit);
		    if (kbinput >= '1' && kbinput <= '5') {
2473
2474
2475
2476
2477
2478
2479
			kbinput = KEY_F(kbinput - 48);
			wgetch(edit);
		    } else if (kbinput >= '7' && kbinput <= '9') {
			kbinput = KEY_F(kbinput - 49);
			wgetch(edit);
		    } else if (kbinput == 126)
			kbinput = KEY_HOME;
2480
2481
2482

#ifdef DEBUG
		    else {
2483
2484
2485
			fprintf(stderr, _("I got Alt-[-1-%c! (%d)\n"),
				kbinput, kbinput);
			break;
2486
2487
2488
2489
		    }
#endif

		    break;
2490
		case '2':	/* Alt-[-2-[0,1,3,4] = F9-F12 in many terms */
2491
2492
		    kbinput = wgetch(edit);
		    switch (kbinput) {
2493
2494
2495
2496
2497
2498
2499
2500
2501
2502
2503
2504
2505
2506
2507
2508
2509
2510
2511
2512
2513
		    case '0':
			kbinput = KEY_F(9);
			wgetch(edit);
			break;
		    case '1':
			kbinput = KEY_F(10);
			wgetch(edit);
			break;
		    case '3':
			kbinput = KEY_F(11);
			wgetch(edit);
			break;
		    case '4':
			kbinput = KEY_F(12);
			wgetch(edit);
			break;
		    case 126:	/* Hack, make insert key do something 
				   usefile, like insert file */
			do_insertfile();
			keyhandled = 1;
			break;
2514
#ifdef DEBUG
2515
2516
2517
2518
		    default:
			fprintf(stderr, _("I got Alt-[-2-%c! (%d)\n"),
				kbinput, kbinput);
			break;
2519
2520
2521
2522
#endif

		    }
		    break;
2523
		case '3':	/* Alt-[-3 = Delete? */
2524
2525
2526
		    kbinput = NANO_DELETE_KEY;
		    wgetch(edit);
		    break;
2527
		case '4':	/* Alt-[-4 = End? */
2528
2529
2530
		    kbinput = NANO_END_KEY;
		    wgetch(edit);
		    break;
2531
		case '5':	/* Alt-[-5 = Page Up */
2532
2533
2534
		    kbinput = KEY_PPAGE;
		    wgetch(edit);
		    break;
2535
		case '6':	/* Alt-[-6 = Page Down */
2536
2537
2538
		    kbinput = KEY_NPAGE;
		    wgetch(edit);
		    break;
2539
		case '[':	/* Alt-[-[-[A-E], F1-F5 in linux console */
2540
		    kbinput = wgetch(edit);
2541
2542
		    if (kbinput >= 'A' && kbinput <= 'E')
			kbinput = KEY_F(kbinput - 64);
2543
		    break;
Chris Allegretta's avatar
Chris Allegretta committed
2544
2545
2546
2547
		case 'A':
		case 'B':
		case 'C':
		case 'D':
2548
		    kbinput = ABCD(kbinput);
Chris Allegretta's avatar
Chris Allegretta committed
2549
2550
2551
2552
2553
2554
2555
2556
2557
2558
2559
2560
2561
2562
2563
2564
2565
2566
2567
2568
2569
		    break;
		case 'H':
		    kbinput = KEY_HOME;
		    break;
		case 'F':
		    kbinput = KEY_END;
		    break;
		default:
#ifdef DEBUG
		    fprintf(stderr, _("I got Alt-[-%c! (%d)\n"),
			    kbinput, kbinput);
#endif
		    break;
		}
		break;
	    default:

		/* Check for the altkey defs.... */
		for (i = 0; i <= MAIN_LIST_LEN - 1; i++)
		    if (kbinput == main_list[i].altval ||
			kbinput == main_list[i].altval - 32) {
2570
2571
			kbinput = main_list[i].val;
			break;
Chris Allegretta's avatar
Chris Allegretta committed
2572
		    }
2573
2574
2575
2576
2577
#ifndef NANO_SMALL
		/* And for toggle switches */
		for (i = 0; i <= TOGGLE_LEN - 1 && !keyhandled; i++)
		    if (kbinput == toggles[i].val ||
			kbinput == toggles[i].val - 32) {
2578
2579
2580
			do_toggle(i);
			keyhandled = 1;
			break;
2581
2582
		    }
#endif
Chris Allegretta's avatar
Chris Allegretta committed
2583
2584
2585
2586
2587
2588
2589
#ifdef DEBUG
		fprintf(stderr, _("I got Alt-%c! (%d)\n"), kbinput,
			kbinput);
#endif
		break;
	    }
	}
2590
2591
2592
2593
2594
2595
2596
2597
2598
2599
2600
2601
	/* If the modify_control_seq is set, we received an Alt-Alt 
	   sequence before this, so we make this key a control sequence 
	   by subtracting 64 or 96, depending on its value. */
	if (!keyhandled && modify_control_seq) {
	    if (kbinput >= 'A' && kbinput < 'a')
		kbinput -= 64;
	    else if (kbinput >= 'a' && kbinput <= 'z')
		kbinput -= 96;

	    modify_control_seq = 0;
	}

Chris Allegretta's avatar
Chris Allegretta committed
2602
2603
	/* Look through the main shortcut list to see if we've hit a
	   shortcut key */
2604
	for (i = 0; i < MAIN_LIST_LEN && !keyhandled; i++) {
Chris Allegretta's avatar
Chris Allegretta committed
2605
2606
2607
2608
2609
2610
2611
2612
2613
2614
	    if (kbinput == main_list[i].val ||
		(main_list[i].misc1 && kbinput == main_list[i].misc1) ||
		(main_list[i].misc2 && kbinput == main_list[i].misc2)) {
		if (ISSET(VIEW_MODE) && !main_list[i].viewok)
		    print_view_warning();
		else
		    main_list[i].func();
		keyhandled = 1;
	    }
	}
2615
2616
	/* If we're in raw mode or using Alt-Alt-x, we have to catch
	   Control-S and Control-Q */
2617
2618
2619
	if (kbinput == 17 || kbinput == 19)
	    keyhandled = 1;

2620
	/* Catch ^Z by hand when triggered also */
2621
2622
2623
2624
2625
2626
	if (kbinput == 26) {
	    if (ISSET(SUSPEND))
		do_suspend(0);
	    keyhandled = 1;
	}

Chris Allegretta's avatar
Chris Allegretta committed
2627
2628
2629
2630
2631
2632
2633
2634
2635
2636
2637
2638
2639
	/* Last gasp, stuff that's not in the main lists */
	if (!keyhandled)
	    switch (kbinput) {
#ifndef NANO_SMALL
#ifdef NCURSES_MOUSE_VERSION
	    case KEY_MOUSE:
		do_mouse();
		break;
#endif
#endif
	    case 0:		/* Erg */
		do_next_word();
		break;
2640

Chris Allegretta's avatar
Chris Allegretta committed
2641
2642
2643
	    case 331:		/* Stuff that we don't want to do squat */
	    case -1:
	    case 410:		/* Must ignore this, it gets sent when we resize */
Chris Allegretta's avatar
Chris Allegretta committed
2644
#ifdef PDCURSES
2645
2646
	    case 541:		/* ???? */
	    case 542:		/* Control and alt in Windows *shrug* */
2647
	    case 543:		/* Right ctrl key */
Chris Allegretta's avatar
Chris Allegretta committed
2648
	    case 544:
2649
	    case 545:		/* Right alt key */
Chris Allegretta's avatar
Chris Allegretta committed
2650
#endif
2651

Chris Allegretta's avatar
Chris Allegretta committed
2652
2653
2654
2655
2656
2657
2658
2659
2660
2661
2662
2663
2664
2665
		break;
	    default:
#ifdef DEBUG
		fprintf(stderr, "I got %c (%d)!\n", kbinput, kbinput);
#endif
		/* We no longer stop unhandled sequences so that people with
		   odd character sets can type... */

		if (ISSET(VIEW_MODE)) {
		    print_view_warning();
		    break;
		}
		do_char(kbinput);
	    }
2666
2667
2668
2669
2670
2671
	if (ISSET(CONSTUPDATE)) {
	    if (ISSET(DISABLE_CURPOS))
		UNSET(DISABLE_CURPOS);
	    else
		do_cursorpos();
	}
Chris Allegretta's avatar
Chris Allegretta committed
2672
2673
2674
2675
2676
2677
2678
2679
2680
2681

	reset_cursor();
	wrefresh(edit);
	keyhandled = 0;
    }

    getchar();
    finish(0);

}