source: editwin.c @ c65d81e

barnowl_perlaimdebianowlrelease-1.10release-1.4release-1.5release-1.6release-1.7release-1.8release-1.9
Last change on this file since c65d81e was 12c35df, checked in by James M. Kretchmar <kretch@mit.edu>, 21 years ago
Fixed replies to loopback messages Fixed smartnarrow on classes/instances with spaces Don't allow duplicates in command history Added the 'loggingdirection' variable All loopback messages log to 'loopback' now Print an error message if trying an invalid color for a filter Fixed bug causing > not to go to end of editwin every time
  • Property mode set to 100644
File size: 20.6 KB
RevLine 
[7d4fbcd]1#include "owl.h"
2#include <stdlib.h>
3#include <unistd.h>
4#include <string.h>
[10b866d]5#include <ctype.h>
[7d4fbcd]6
[1aee7d9]7static const char fileIdent[] = "$Id$";
8
[7d4fbcd]9#define INCR 5000
10
[cf83b7a]11/* initialize the editwin e.
12 * 'win' is an already initialzed curses window that will be used by editwin
13 */
[c9334b1]14void owl_editwin_init(owl_editwin *e, WINDOW *win, int winlines, int wincols, int style, owl_history *hist)
15{
[10b866d]16  e->buff=owl_malloc(INCR); 
[7d4fbcd]17  e->buff[0]='\0';
18  e->bufflen=0;
[10b866d]19  e->hist=hist;
[7d4fbcd]20  e->allocated=INCR;
21  e->buffx=0;
22  e->buffy=0;
23  e->topline=0;
24  e->winlines=winlines;
25  e->wincols=wincols;
[cf83b7a]26  e->fillcol=owl_editwin_limit_maxcols(wincols-1, owl_global_get_edit_maxfillcols(&g));
27  e->wrapcol=owl_editwin_limit_maxcols(wincols-1, owl_global_get_edit_maxwrapcols(&g));
[7d4fbcd]28  e->curswin=win;
29  e->style=style;
30  if ((style!=OWL_EDITWIN_STYLE_MULTILINE) &&
31      (style!=OWL_EDITWIN_STYLE_ONELINE)) {
32    e->style=OWL_EDITWIN_STYLE_MULTILINE;
33  }
34  e->lock=0;
35  e->dotsend=0;
[c9334b1]36  e->echochar='\0';
[7d4fbcd]37  if (win) werase(win);
38}
39
[c9334b1]40void owl_editwin_set_curswin(owl_editwin *e, WINDOW *w, int winlines, int wincols)
41{
[7d4fbcd]42  e->curswin=w;
43  e->winlines=winlines;
44  e->wincols=wincols;
[cf83b7a]45  e->fillcol=owl_editwin_limit_maxcols(wincols-1, owl_global_get_edit_maxfillcols(&g));
46  e->wrapcol=owl_editwin_limit_maxcols(wincols-1, owl_global_get_edit_maxwrapcols(&g));
[7d4fbcd]47}
48
[c9334b1]49/* echo the character 'ch' for each normal character keystroke,
50 * excepting locktext.  This is useful for entering passwords etc.  If
51 * ch=='\0' characters are echo'd normally
52 */
53void owl_editwin_set_echochar(owl_editwin *e, int ch)
54{
55  e->echochar=ch;
56}
57
58WINDOW *owl_editwin_get_curswin(owl_editwin *e)
59{
[060b3b4]60  return(e->curswin);
[7d4fbcd]61}
62
[c9334b1]63void owl_editwin_set_history(owl_editwin *e, owl_history *h)
64{
[060b3b4]65  e->hist=h;
[10b866d]66}
67
[c9334b1]68owl_history *owl_editwin_get_history(owl_editwin *e)
69{
[060b3b4]70  return(e->hist);
[10b866d]71}
72
[c9334b1]73void owl_editwin_set_dotsend(owl_editwin *e)
74{
[7d4fbcd]75  e->dotsend=1;
76}
77
[c9334b1]78int owl_editwin_limit_maxcols(int v, int maxv)
79{
[d36f2cb]80  if (maxv > 5 && v > maxv) {
[060b3b4]81    return(maxv);
[d36f2cb]82  } else {
[060b3b4]83    return(v);
[d36f2cb]84  }
85}
86
[c9334b1]87/* set text to be 'locked in' at the beginning of the buffer, any
88 * previous text (including locked text) will be overwritten
89 */
90void owl_editwin_set_locktext(owl_editwin *e, char *text)
91{
[7d4fbcd]92 
93  int x, y;
94
95  x=e->buffx;
96  y=e->buffy;
97  e->buffx=0;
98  e->buffy=0;
99  owl_editwin_overwrite_string(e, text);
100  e->lock=strlen(text);
101  /* if (text[e->lock-1]=='\n') e->lock--; */
102  e->buffx=x;
103  e->buffy=y;
104  owl_editwin_adjust_for_locktext(e);
105  owl_editwin_redisplay(e, 0);
106}
107
[c9334b1]108int owl_editwin_get_style(owl_editwin *e)
109{
[7d4fbcd]110  return(e->style);
111}
112
[c9334b1]113void owl_editwin_new_style(owl_editwin *e, int newstyle, owl_history *h)
114{
[7d4fbcd]115  char *ptr;
[10b866d]116
117  owl_editwin_set_history(e, h);
[7d4fbcd]118  if (e->style==newstyle) return;
119
120  if (newstyle==OWL_EDITWIN_STYLE_MULTILINE) {
121    e->style=newstyle;
122  } else if (newstyle==OWL_EDITWIN_STYLE_ONELINE) {
123    e->style=newstyle;
124
125    /* nuke everything after the first line */
126    if (e->bufflen > 0) {
127      ptr=strchr(e->buff, '\n')-1;
128      if (ptr) {
129        e->bufflen=ptr - e->buff;
130        e->buff[e->bufflen]='\0';
131        e->buffx=0;
132        e->buffy=0;
133      }
134    }
135  }
136}
137
[c9334b1]138/* completly reinitialize the buffer */
139void owl_editwin_fullclear(owl_editwin *e)
140{
[7d4fbcd]141  owl_free(e->buff);
[10b866d]142  owl_editwin_init(e, e->curswin, e->winlines, e->wincols, e->style, e->hist);
[7d4fbcd]143}
144
[c9334b1]145/* clear all text except for locktext and put the cursor at the
146 * beginning
147 */
148void owl_editwin_clear(owl_editwin *e)
149{
150
[7d4fbcd]151  int lock;
[10b866d]152  int dotsend=e->dotsend;
[7d4fbcd]153  char *locktext=NULL;
[10b866d]154
[7d4fbcd]155  lock=0;
156  if (e->lock > 0) {
157    lock=1;
158
159    locktext=owl_malloc(e->lock+20);
160    strncpy(locktext, e->buff, e->lock);
161    locktext[e->lock]='\0';
162  }
163
164  owl_free(e->buff);
[10b866d]165  owl_editwin_init(e, e->curswin, e->winlines, e->wincols, e->style, e->hist);
[7d4fbcd]166
167  if (lock > 0) {
168    owl_editwin_set_locktext(e, locktext);
169  }
[10b866d]170  if (dotsend) {
171    owl_editwin_set_dotsend(e);
172  }
[7d4fbcd]173
174  if (locktext) owl_free(locktext);
175  owl_editwin_adjust_for_locktext(e);
176}
177
[c9334b1]178/* malloc more space for the buffer */
179void _owl_editwin_addspace(owl_editwin *e)
180{
[7d4fbcd]181  e->buff=owl_realloc(e->buff, e->allocated+INCR);
182  if (!e->buff) {
183    /* error */
184    return;
185  }
186  e->allocated+=INCR;
187}
188
[c9334b1]189void owl_editwin_recenter(owl_editwin *e)
190{
[7d4fbcd]191  e->topline=e->buffy-(e->winlines/2);
192  if (e->topline<0) e->topline=0;
193  if (e->topline>owl_editwin_get_numlines(e)) e->topline=owl_editwin_get_numlines(e);
194}
195
[c9334b1]196/* regenerate the text on the curses window */
197/* if update == 1 then do a doupdate(), otherwise do not */
198void owl_editwin_redisplay(owl_editwin *e, int update)
199{
[7d4fbcd]200 
201  char *ptr1, *ptr2, *ptr3, *buff;
[cf83b7a]202  int i;
[7d4fbcd]203
204  werase(e->curswin);
205  wmove(e->curswin, 0, 0);
206
207  /* start at topline */
208  ptr1=e->buff;
209  for (i=0; i<e->topline; i++) {
210    ptr2=strchr(ptr1, '\n');
211    if (!ptr2) {
212      /* we're already on the last line */
213      break;
214    }
215    ptr1=ptr2+1;
216  }
217  /* ptr1 now stores the starting point */
218
219  /* find the ending point and store it in ptr3 */
220  ptr2=ptr1;
221  ptr3=ptr1;
222  for (i=0; i<e->winlines; i++) {
223    ptr3=strchr(ptr2, '\n');
224    if (!ptr3) {
225      /* we've hit the last line */
226      /* print everything to the end */
227      ptr3=e->buff+e->bufflen-1;
228      ptr3--;
229      break;
230    }
231    ptr2=ptr3+1;
232  }
233  ptr3+=2;
234
235  buff=owl_malloc(ptr3-ptr1+50);
236  strncpy(buff, ptr1, ptr3-ptr1);
237  buff[ptr3-ptr1]='\0';
[c9334b1]238  if (e->echochar=='\0') {
239    waddstr(e->curswin, buff);
240  } else {
241    /* translate to echochar, *except* for the locktext */
242    int len;
243    int dolocklen=e->lock-(ptr1-e->buff);
244
245    for (i=0; i<dolocklen; i++) {
246      waddch(e->curswin, buff[i]);
247    }
248    len=strlen(buff);
249    for (i=0; i<len-dolocklen; i++) {
250      waddch(e->curswin, e->echochar);
251    }
252  }
[7d4fbcd]253  wmove(e->curswin, e->buffy-e->topline, e->buffx);
254  wnoutrefresh(e->curswin);
255  if (update==1) {
256    doupdate();
257  }
258  owl_free(buff);
259}
260
261
[c9334b1]262/* linewrap the word just before the cursor.
263 * returns 0 on success
264 * returns -1 if we could not wrap.
265 */
266int _owl_editwin_linewrap_word(owl_editwin *e)
267{
[7d4fbcd]268  int i, z;
269
270  z=_owl_editwin_get_index_from_xy(e);
271  /* move back and line wrap the previous word */
272  for (i=z-1; ; i--) {
273    /* move back until you find a space or hit the beginning of the line */
274    if (e->buff[i]==' ') {
275      /* replace the space with a newline */
276      e->buff[i]='\n';
277      e->buffy++;
278      e->buffx=z-i-1;
279      /* were we on the last line */
280      return(0);
281    } else if (e->buff[i]=='\n' || i<=e->lock) {
282      /* we hit the begginning of the line or the buffer, we cannot
283       * wrap.
284       */
285      return(-1);
286    }
287  }
288}
289
[c9334b1]290/* insert a character at the current point (shift later
291 * characters over)
292 */
293void owl_editwin_insert_char(owl_editwin *e, char c)
294{
[7d4fbcd]295 
296  int z, i, ret;
297
298  /* \r is \n */
299  if (c=='\r') {
300    c='\n';
301  }
302
303  if (c=='\n' && e->style==OWL_EDITWIN_STYLE_ONELINE) {
304    /* perhaps later this will change some state that allows the string
305       to be read */
306    return;
307  }
308
309  /* make sure there is enough memory for the new text */
310  if ((e->bufflen+1) > (e->allocated-5)) {
311    _owl_editwin_addspace(e);
312  }
313
314  /* get the insertion point */
315  z=_owl_editwin_get_index_from_xy(e);
316
317  /* If we're going to insert at the last column do word wrapping, unless it's a \n */
318  if ((e->buffx+1==e->wrapcol) && (c!='\n')) {
319    ret=_owl_editwin_linewrap_word(e);
320    if (ret==-1) {
321      /* we couldn't wrap, insert a hard newline instead */
322      owl_editwin_insert_char(e, '\n');
323    }
324  }
325
326  z=_owl_editwin_get_index_from_xy(e);
327  /* shift all the other characters right */
328  for (i=e->bufflen; i>z; i--) {
329    e->buff[i]=e->buff[i-1];
330  }
331
332  /* insert the new one */
333  e->buff[z]=c;
334
335  /* housekeeping */
336  e->bufflen++;
337  e->buff[e->bufflen]='\0';
338
339  /* advance the cursor */
340  if (c=='\n') {
341    e->buffx=0;
342    e->buffy++;
343  } else {
344    e->buffx++;
345  }
346}
347
[c9334b1]348/* overwrite the character at the current point with 'c' */
349void owl_editwin_overwrite_char(owl_editwin *e, char c)
350{
[7d4fbcd]351  int z;
352 
353  /* \r is \n */
354  if (c=='\r') {
355    c='\n';
356  }
357
358  if (c=='\n' && e->style==OWL_EDITWIN_STYLE_ONELINE) {
359    /* perhaps later this will change some state that allows the string
360       to be read */
361    return;
362  }
363
364  z=_owl_editwin_get_index_from_xy(e);
365
366  /* only if we are at the end of the buffer do we create new space */
367  if (z==e->bufflen) {
368    if ((e->bufflen+1) > (e->allocated-5)) {
369      _owl_editwin_addspace(e);
370    }
371  }
372 
373  e->buff[z]=c;
374
375  /* housekeeping if we are at the end of the buffer */
376  if (z==e->bufflen) {
377    e->bufflen++;
378    e->buff[e->bufflen]='\0';
379  }
380
381  /* advance the cursor */
382  if (c=='\n') {
383    e->buffx=0;
384    e->buffy++;
385  } else {
386    e->buffx++;
387  }
388
389}
390
[c9334b1]391/* delete the character at the current point, following chars
392 * shift left.
393 */ 
394void owl_editwin_delete_char(owl_editwin *e)
395{
[7d4fbcd]396  int z, i;
397
398  if (e->bufflen==0) return;
399 
400  /* get the deletion point */
401  z=_owl_editwin_get_index_from_xy(e);
402
403  if (z==e->bufflen) return;
404
405  for (i=z; i<e->bufflen; i++) {
406    e->buff[i]=e->buff[i+1];
407  }
408  e->bufflen--;
409  e->buff[e->bufflen]='\0';
410}
411
[c9334b1]412/* Swap the character at point with the character at point-1 and
413 * advance the pointer.  If point is at beginning of buffer do
414 * nothing.  If point is after the last character swap point-1 with
415 * point-2.  (Behaves as observed in tcsh and emacs). 
416 */
417void owl_editwin_transpose_chars(owl_editwin *e)
418{
[f2e36b5]419  int z;
420  char tmp;
421
422  if (e->bufflen==0) return;
423 
424  /* get the cursor point */
425  z=_owl_editwin_get_index_from_xy(e);
426
427  if (z==e->bufflen) {
428    /* point is after last character */
429    z--;
430  } 
431
432  if (z-1 < e->lock) {
433    /* point is at beginning of buffer, do nothing */
434    return;
435  }
436
437  tmp=e->buff[z];
438  e->buff[z]=e->buff[z-1];
439  e->buff[z-1]=tmp;
440  owl_editwin_key_right(e);
441}
442
[c9334b1]443/* insert 'string' at the current point, later text is shifted
444 * right
445 */
446void owl_editwin_insert_string(owl_editwin *e, char *string)
447{
[7d4fbcd]448  int i, j;
449
450  j=strlen(string);
451  for (i=0; i<j; i++) {
452    owl_editwin_insert_char(e, string[i]);
453  }
454}
455
[c9334b1]456/* write 'string' at the current point, overwriting text that is
457 * already there
458 */
459
460void owl_editwin_overwrite_string(owl_editwin *e, char *string)
461{
[7d4fbcd]462  int i, j;
463
464  j=strlen(string);
465  for (i=0; i<j; i++) {
466    owl_editwin_overwrite_char(e, string[i]);
467  }
468}
469
[c9334b1]470/* get the index into e->buff for the current cursor
471 * position.
472 */
473int _owl_editwin_get_index_from_xy(owl_editwin *e)
474{
[7d4fbcd]475  int i;
476  char *ptr1, *ptr2;
477
478  if (e->bufflen==0) return(0);
479 
480  /* first go to the yth line */
481  ptr1=e->buff;
482  for (i=0; i<e->buffy; i++) {
483    ptr2=strchr(ptr1, '\n');
484    if (!ptr2) {
485      /* we're already on the last line */
486      break;
487    }
488    ptr1=ptr2+1;
489  }
490
491  /* now go to the xth character */
492  ptr2=strchr(ptr1, '\n');
493  if (!ptr2) {
494    ptr2=e->buff+e->bufflen;
495  }
496
497  if ((ptr2-ptr1) < e->buffx) {
498    ptr1=ptr2-1;
499  } else {
500    ptr1+=e->buffx;
501  }
502
503  /* printf("DEBUG: index is %i\r\n", ptr1-e->buff); */
504  return(ptr1-e->buff);
505}
506
[c9334b1]507void _owl_editwin_set_xy_by_index(owl_editwin *e, int index)
508{
[7d4fbcd]509  int z, i;
510
511  z=_owl_editwin_get_index_from_xy(e);
512  if (index>z) {
513    for (i=0; i<index-z; i++) {
514      owl_editwin_key_right(e);
515    }
516  } else if (index<z) {
517    for (i=0; i<z-index; i++) {
518      owl_editwin_key_left(e);
519    }
520  }
521}
522
[c9334b1]523void owl_editwin_adjust_for_locktext(owl_editwin *e)
524{
[7d4fbcd]525  /* if we happen to have the cursor over locked text
526   * move it to be out of the locktext region */
527  if (_owl_editwin_get_index_from_xy(e)<e->lock) {
528    _owl_editwin_set_xy_by_index(e, e->lock);
529  }
530}
531
[c9334b1]532void owl_editwin_backspace(owl_editwin *e)
533{
[7d4fbcd]534  /* delete the char before the current one
535   * and shift later chars left
536   */
537  if (_owl_editwin_get_index_from_xy(e) > e->lock) {
538    owl_editwin_key_left(e);
539    owl_editwin_delete_char(e);
540  }
541  owl_editwin_adjust_for_locktext(e);
542}
543
[c9334b1]544void owl_editwin_key_up(owl_editwin *e)
545{
[7d4fbcd]546  if (e->buffy > 0) e->buffy--;
547  if (e->buffx >= owl_editwin_get_numchars_on_line(e, e->buffy)) {
548    e->buffx=owl_editwin_get_numchars_on_line(e, e->buffy);
549  }
550
551  /* do we need to scroll? */
552  if (e->buffy-e->topline < 0) {
553    e->topline-=e->winlines/2;
554  }
555
556  owl_editwin_adjust_for_locktext(e);
557}
558
[c9334b1]559void owl_editwin_key_down(owl_editwin *e)
560{
[7d4fbcd]561  /* move down if we can */
562  if (e->buffy+1 < owl_editwin_get_numlines(e)) e->buffy++;
563
564  /* if we're past the last character move back */
565  if (e->buffx >= owl_editwin_get_numchars_on_line(e, e->buffy)) {
566    e->buffx=owl_editwin_get_numchars_on_line(e, e->buffy);
567  }
568
569  /* do we need to scroll? */
570  if (e->buffy-e->topline > e->winlines) {
571    e->topline+=e->winlines/2;
572  }
573
574  /* adjust for locktext */
575  owl_editwin_adjust_for_locktext(e);
576}
577
[c9334b1]578void owl_editwin_key_left(owl_editwin *e)
579{
[7d4fbcd]580  /* move left if we can, and maybe up a line */
581  if (e->buffx>0) {
582    e->buffx--;
583  } else if (e->buffy>0) {
584    e->buffy--;
585    e->buffx=owl_editwin_get_numchars_on_line(e, e->buffy);
586  }
587
588  /* do we need to scroll up? */
589  if (e->buffy-e->topline < 0) {
590    e->topline-=e->winlines/2;
591  }
592
593  /* make sure to avoid locktext */
594  owl_editwin_adjust_for_locktext(e);
595}
596
[c9334b1]597void owl_editwin_key_right(owl_editwin *e)
598{
[7d4fbcd]599  int i;
600
601  /* move right if we can, and skip down a line if needed */
602  i=owl_editwin_get_numchars_on_line(e, e->buffy);
603  if (e->buffx < i) {
604    e->buffx++;
605    /*  } else if (e->buffy+1 < owl_editwin_get_numlines(e)) { */
606  } else if (_owl_editwin_get_index_from_xy(e) < e->bufflen) {
607    if (e->style==OWL_EDITWIN_STYLE_MULTILINE) {
608      e->buffx=0;
609      e->buffy++;
610    }
611  }
612
613  /* do we need to scroll down? */
614  if (e->buffy-e->topline >= e->winlines) {
615    e->topline+=e->winlines/2;
616  }
617}
618
[c9334b1]619void owl_editwin_move_to_nextword(owl_editwin *e)
620{
[7d4fbcd]621  int i, x;
622
623  /* if we're starting on a space, find the first non-space */
624  i=_owl_editwin_get_index_from_xy(e);
625  if (e->buff[i]==' ') {
626    for (x=i; x<e->bufflen; x++) {
627      if (e->buff[x]!=' ' && e->buff[x]!='\n') {
628        _owl_editwin_set_xy_by_index(e, x);
629        break;
630      }
631    }
632  }
633
634  /* find the next space, newline or end of line and go there, if
635     already at the end of the line, continue on to the next */
636  i=owl_editwin_get_numchars_on_line(e, e->buffy);
637  if (e->buffx < i) {
638    /* move right till end of line */
639    while (e->buffx < i) {
640      e->buffx++;
641      if (e->buff[_owl_editwin_get_index_from_xy(e)]==' ') return;
642      if (e->buffx == i) return;
643    }
644  } else if (e->buffx == i) {
645    /* try to move down */
646    if (e->style==OWL_EDITWIN_STYLE_MULTILINE) {
647      if (e->buffy+1 <  owl_editwin_get_numlines(e)) {
648        e->buffx=0;
649        e->buffy++;
650        owl_editwin_move_to_nextword(e);
651      }
652    }
653  }
654}
655
[c9334b1]656/* go backwards to the last non-space character
657 */
658void owl_editwin_move_to_previousword(owl_editwin *e)
659{
[7d4fbcd]660  int i, x;
661
662  /* are we already at the beginning of the word? */
663  i=_owl_editwin_get_index_from_xy(e);
664  if ( (e->buff[i]!=' ' && e->buff[i]!='\n' && e->buff[i]!='\0') &&
665       (e->buff[i-1]==' ' || e->buff[i-1]=='\n') ) {
666    owl_editwin_key_left(e);
667  }
668   
669  /* are we starting on a space character? */
670  i=_owl_editwin_get_index_from_xy(e);
671  if (e->buff[i]==' ' || e->buff[i]=='\n' || e->buff[i]=='\0') {
672    /* find the first non-space */
673    for (x=i; x>=e->lock; x--) {
674      if (e->buff[x]!=' ' && e->buff[x]!='\n' && e->buff[x]!='\0') {
675        _owl_editwin_set_xy_by_index(e, x);
676        break;
677      }
678    }
679  }
680
681  /* find the last non-space */
682  i=_owl_editwin_get_index_from_xy(e);
683  for (x=i; x>=e->lock; x--) {
684    if (e->buff[x-1]==' ' || e->buff[x-1]=='\n') {
685      _owl_editwin_set_xy_by_index(e, x);
686      break;
687    }
688  }
689  _owl_editwin_set_xy_by_index(e, x);
690}
691
692
[c9334b1]693void owl_editwin_delete_nextword(owl_editwin *e)
694{
[7d4fbcd]695  int z;
696
697  if (e->bufflen==0) return;
698
699  /* if we start out on a space character then gobble all the spaces
700     up first */
701  while (1) {
702    z=_owl_editwin_get_index_from_xy(e);
703    if (e->buff[z]==' ' || e->buff[z]=='\n') {
704      owl_editwin_delete_char(e);
705    } else {
706      break;
707    }
708  }
709
710  /* then nuke the next word */
711  while (1) {
712    z=_owl_editwin_get_index_from_xy(e);
713    if (e->buff[z+1]==' ' || e->buff[z+1]=='\n' || e->buff[z+1]=='\0') break;
714    owl_editwin_delete_char(e);
715  }
716  owl_editwin_delete_char(e);
717}
718
[c9334b1]719void owl_editwin_delete_previousword(owl_editwin *e)
720{
[b68f9cd]721  /* go backwards to the last non-space character, then delete chars */
722  int i, startpos, endpos;
723
724  startpos = _owl_editwin_get_index_from_xy(e);
725  owl_editwin_move_to_previousword(e);
726  endpos = _owl_editwin_get_index_from_xy(e);
727  for (i=0; i<startpos-endpos; i++) {
728    owl_editwin_delete_char(e);
729  }
730}
731
[c9334b1]732void owl_editwin_delete_to_endofline(owl_editwin *e)
733{
[7d4fbcd]734  int i;
735
736  if (owl_editwin_get_numchars_on_line(e, e->buffy)>e->buffx) {
737    /* normal line */
738    i=_owl_editwin_get_index_from_xy(e);
739    while(i < e->bufflen) {
740      if (e->buff[i]!='\n') {
741        owl_editwin_delete_char(e);
742      } else if ((e->buff[i]=='\n') && (i==e->bufflen-1)) {
743        owl_editwin_delete_char(e);
744      } else {
745        return;
746      }
747    }
748  } else if (e->buffy+1 < owl_editwin_get_numlines(e)) {
749    /* line with cursor at the end but not on very last line */
750    owl_editwin_key_right(e);
751    owl_editwin_backspace(e);
752  }
753}
754
[c9334b1]755void owl_editwin_move_to_line_end(owl_editwin *e)
756{
[7d4fbcd]757  e->buffx=owl_editwin_get_numchars_on_line(e, e->buffy);
758}
759
[c9334b1]760void owl_editwin_move_to_line_start(owl_editwin *e)
761{
[7d4fbcd]762  e->buffx=0;
763  owl_editwin_adjust_for_locktext(e);
764}
765
[c9334b1]766void owl_editwin_move_to_end(owl_editwin *e)
767{
[7d4fbcd]768  /* go to last char */
769  e->buffy=owl_editwin_get_numlines(e)-1;
770  e->buffx=owl_editwin_get_numchars_on_line(e, e->buffy);
771  owl_editwin_key_right(e);
772
773  /* do we need to scroll? */
[12c35df]774  /*
[7d4fbcd]775  if (e->buffy-e->topline > e->winlines) {
776    e->topline+=e->winlines/2;
777  }
[12c35df]778  */
779  owl_editwin_recenter(e);
[7d4fbcd]780}
781
[c9334b1]782void owl_editwin_move_to_top(owl_editwin *e)
783{
[7d4fbcd]784  _owl_editwin_set_xy_by_index(e, 0);
785
786  /* do we need to scroll? */
787  e->topline=0;
788
789  owl_editwin_adjust_for_locktext(e);
790}
791
[c9334b1]792void owl_editwin_fill_paragraph(owl_editwin *e)
793{
[7d4fbcd]794  int i, save;
795
796  /* save our starting point */
797  save=_owl_editwin_get_index_from_xy(e);
798
799  /* scan back to the beginning of this paragraph */
800  for (i=save; i>=e->lock; i--) {
801    if ( (i<=e->lock) ||
802         ((e->buff[i]=='\n') && (e->buff[i-1]=='\n'))) {
803      _owl_editwin_set_xy_by_index(e, i+1);
804      break;
805    }
806  }
807
808  /* main loop */
809  while (1) {
810    i=_owl_editwin_get_index_from_xy(e);
811
812    /* bail if we hit the end of the buffer */
813    if (i>=e->bufflen) break;
814
815    /* bail if we hit the end of the paragraph */
816    if (e->buff[i]=='\n' && e->buff[i+1]=='\n') break;
817
818    /* if we've travelled too far, linewrap */
819    if ((e->buffx) >= e->fillcol) {
820      _owl_editwin_linewrap_word(e);
821    }
822
823    /* did we hit the end of a line too soon? */
824    i=_owl_editwin_get_index_from_xy(e);
825    if (e->buff[i]=='\n' && e->buffx<e->fillcol-1) {
826      /* ********* we need to make sure we don't pull in a word that's too long ***********/
827      e->buff[i]=' ';
828    }
829   
830    /* fix spacing */
831    i=_owl_editwin_get_index_from_xy(e);
832    if (e->buff[i]==' ' && e->buff[i+1]==' ') {
833      if (e->buff[i-1]=='.' || e->buff[i-1]=='!' || e->buff[i-1]=='?') {
834        owl_editwin_key_right(e);
835      } else {
836        owl_editwin_delete_char(e);
837        /* if we did this ahead of the save point, adjust it */
838        if (i<save) save--;
839      }
840    } else {
841      owl_editwin_key_right(e);
842    }
843
844  }
845
846  /* put cursor back at starting point */
847  _owl_editwin_set_xy_by_index(e, save);
848
849  /* do we need to scroll? */
850  if (e->buffy-e->topline < 0) {
851    e->topline-=e->winlines/2;
852  }
853}
854
[cf83b7a]855/* returns true if only whitespace remains */
[c9334b1]856int owl_editwin_is_at_end(owl_editwin *e)
857{
[10b866d]858  int cur=_owl_editwin_get_index_from_xy(e);
[c9334b1]859  return (only_whitespace(e->buff+cur));
[217a43e]860}
861
[c9334b1]862int owl_editwin_check_dotsend(owl_editwin *e)
863{
[7d4fbcd]864  int i;
865
866  if (!e->dotsend) return(0);
867  for (i=e->bufflen-1; i>0; i--) {
868    if (e->buff[i] == '.' 
869        && (e->buff[i-1] == '\n' || e->buff[i-1] == '\r')
870        && (e->buff[i+1] == '\n' || e->buff[i+1] == '\r')) {
871      e->bufflen = i;
872      e->buff[i] = '\0';
873      return(1);
874    }
[601a9e0]875    if (!isspace((int) e->buff[i])) {
[7d4fbcd]876      return(0);
877    }
878  }
879  return(0);
880}
881
[c9334b1]882void owl_editwin_post_process_char(owl_editwin *e, int j)
883{
[7d4fbcd]884  /* check if we need to scroll down */
885  if (e->buffy-e->topline >= e->winlines) {
886    e->topline+=e->winlines/2;
887  }
888  if ((j==13 || j==10) && owl_editwin_check_dotsend(e)) {
889    owl_command_editmulti_done(e);
890    return;
891  }
892  owl_editwin_redisplay(e, 0); 
893}
894
[c9334b1]895void owl_editwin_process_char(owl_editwin *e, int j)
896{
[7d4fbcd]897  if (j == ERR) return;
898  if (j>127 || ((j<32) && (j!=10) && (j!=13))) {
899    return;
900  } else {
901    owl_editwin_insert_char(e, j);
902  }
903}
904
[c9334b1]905char *owl_editwin_get_text(owl_editwin *e)
906{
[7d4fbcd]907  return(e->buff+e->lock);
908}
909
[c9334b1]910int owl_editwin_get_numchars_on_line(owl_editwin *e, int line)
911{
[7d4fbcd]912  int i;
913  char *ptr1, *ptr2;
914
915  if (e->bufflen==0) return(0);
916 
917  /* first go to the yth line */
918  ptr1=e->buff;
919  for (i=0; i<line; i++) {
920    ptr2=strchr(ptr1, '\n');
921    if (!ptr2) {
922      /* we're already on the last line */
923      return(0);
924    }
925    ptr1=ptr2+1;
926  }
927
928  /* now go to the xth character */
929  ptr2=strchr(ptr1, '\n');
930  if (!ptr2) {
931    return(e->buff + e->bufflen - ptr1);
932  }
933  return(ptr2-ptr1); /* don't count the newline for now */
934}
935
[c9334b1]936int owl_editwin_get_numlines(owl_editwin *e)
937{
[7d4fbcd]938  return(owl_text_num_lines(e->buff));
939}
940
Note: See TracBrowser for help on using the repository browser.