source: editwin.c @ d7043b4

release-1.10release-1.4release-1.5release-1.6release-1.7release-1.8release-1.9
Last change on this file since d7043b4 was d7043b4, checked in by Nelson Elhage <nelhage@mit.edu>, 15 years ago
simplify line_move goal_column logic
  • Property mode set to 100644
File size: 25.5 KB
Line 
1#include "owl.h"
2#include <stdlib.h>
3#include <unistd.h>
4#include <string.h>
5#include <ctype.h>
6
7static const char fileIdent[] = "$Id$";
8
9struct _owl_editwin { /*noproto*/
10  char *buff;
11  owl_history *hist;
12  int bufflen;
13  int allocated;
14  int index;
15  int goal_column;
16  int topindex;
17  int winlines, wincols, fillcol, wrapcol;
18  WINDOW *curswin;
19  int style;
20  int lock;
21  int dotsend;
22  int echochar;
23
24  char *command;
25  void (*callback)(struct _owl_editwin*);
26  void *cbdata;
27};
28
29typedef struct { /*noproto*/
30  int index;
31  int goal_column;
32} _owl_editwin_excursion;
33
34static int owl_editwin_is_char_in(owl_editwin *e, char *set);
35static void oe_reframe(owl_editwin *e);
36
37#define INCR 5000
38
39#define WHITESPACE " \n\t"
40
41owl_editwin *owl_editwin_allocate(void) {
42  return owl_malloc(sizeof(owl_editwin));
43}
44
45/* initialize the editwin e.
46 * 'win' is an already initialzed curses window that will be used by editwin
47 */
48void owl_editwin_init(owl_editwin *e, WINDOW *win, int winlines, int wincols, int style, owl_history *hist)
49{
50  e->buff=owl_malloc(INCR);
51  e->buff[0]='\0';
52  e->bufflen=0;
53  e->hist=hist;
54  e->allocated=INCR;
55  e->index = 0;
56  e->goal_column = -1;
57  e->topindex=0;
58  e->winlines=winlines;
59  e->wincols=wincols;
60  e->fillcol=owl_editwin_limit_maxcols(wincols-7, owl_global_get_edit_maxfillcols(&g));
61  e->wrapcol=owl_editwin_limit_maxcols(wincols-7, owl_global_get_edit_maxwrapcols(&g));
62  e->curswin=win;
63  e->style=style;
64  if ((style!=OWL_EDITWIN_STYLE_MULTILINE) &&
65      (style!=OWL_EDITWIN_STYLE_ONELINE)) {
66    e->style=OWL_EDITWIN_STYLE_MULTILINE;
67  }
68  e->lock=0;
69  e->dotsend=0;
70  e->echochar='\0';
71
72  /* We get initialized multiple times, but we need to hold on to
73     the callbacks, so we can't NULL them here. */
74  /*
75    e->command = NULL;
76    e->callback = NULL;
77    e->cbdata = NULL;
78  */
79  if (win) werase(win);
80}
81
82void owl_editwin_set_curswin(owl_editwin *e, WINDOW *w, int winlines, int wincols)
83{
84  e->curswin=w;
85  e->winlines=winlines;
86  e->wincols=wincols;
87  e->fillcol=owl_editwin_limit_maxcols(wincols-7, owl_global_get_edit_maxfillcols(&g));
88  e->wrapcol=owl_editwin_limit_maxcols(wincols-7, owl_global_get_edit_maxwrapcols(&g));
89}
90
91/* echo the character 'ch' for each normal character keystroke,
92 * excepting locktext.  This is useful for entering passwords etc.  If
93 * ch=='\0' characters are echo'd normally
94 */
95void owl_editwin_set_echochar(owl_editwin *e, int ch)
96{
97  e->echochar=ch;
98}
99
100WINDOW *owl_editwin_get_curswin(owl_editwin *e)
101{
102  return(e->curswin);
103}
104
105void owl_editwin_set_history(owl_editwin *e, owl_history *h)
106{
107  e->hist=h;
108}
109
110owl_history *owl_editwin_get_history(owl_editwin *e)
111{
112  return(e->hist);
113}
114
115void owl_editwin_set_dotsend(owl_editwin *e)
116{
117  e->dotsend=1;
118}
119
120void owl_editwin_set_command(owl_editwin *e, char *command) {
121  if(e->command) owl_free(e->command);
122  e->command = owl_strdup(command);
123}
124
125char *owl_editwin_get_command(owl_editwin *e) {
126  if(e->command) return e->command;
127  return "";
128}
129
130void owl_editwin_set_callback(owl_editwin *e, void (*cb)(owl_editwin*)) {
131  e->callback = cb;
132}
133
134void (*owl_editwin_get_callback(owl_editwin *e))(owl_editwin*) {
135  return e->callback;
136}
137
138void owl_editwin_set_cbdata(owl_editwin *e, void *data) {
139  e->cbdata = data;
140}
141
142void *owl_editwin_get_cbdata(owl_editwin *e) {
143  return e->cbdata;
144}
145
146void owl_editwin_do_callback(owl_editwin *e) {
147  void (*cb)(owl_editwin*);
148  cb=owl_editwin_get_callback(e);
149  if(!cb) {
150    owl_function_error("Internal error: No editwin callback!");
151  } else {
152    /* owl_function_error("text: |%s|", owl_editwin_get_text(e)); */
153    cb(e);
154  }
155}
156
157int owl_editwin_limit_maxcols(int v, int maxv)
158{
159  if (maxv > 5 && v > maxv) {
160    return(maxv);
161  } else {
162    return(v);
163  }
164}
165
166/* set text to be 'locked in' at the beginning of the buffer, any
167 * previous text (including locked text) will be overwritten
168 */
169void owl_editwin_set_locktext(owl_editwin *e, char *text)
170{
171  e->index = 0;
172  owl_editwin_overwrite_string(e, text);
173  owl_editwin_overwrite_char(e, '\0');
174  e->lock=strlen(text);
175  e->index = e->lock;
176  owl_editwin_redisplay(e, 0);
177}
178
179int owl_editwin_get_style(owl_editwin *e)
180{
181  return(e->style);
182}
183
184void owl_editwin_new_style(owl_editwin *e, int newstyle, owl_history *h)
185{
186  char *ptr;
187
188  owl_editwin_set_history(e, h);
189  if (e->style==newstyle) return;
190
191  if (newstyle==OWL_EDITWIN_STYLE_MULTILINE) {
192    e->style=newstyle;
193  } else if (newstyle==OWL_EDITWIN_STYLE_ONELINE) {
194    e->style=newstyle;
195
196    /* nuke everything after the first line */
197    if (e->bufflen > 0) {
198      ptr=strchr(e->buff, '\n')-1;
199      if (ptr) {
200        e->bufflen=ptr - e->buff;
201        e->buff[e->bufflen]='\0';
202        e->index = 0;
203      }
204    }
205  }
206}
207
208/* completly reinitialize the buffer */
209void owl_editwin_fullclear(owl_editwin *e)
210{
211  owl_free(e->buff);
212  owl_editwin_init(e, e->curswin, e->winlines, e->wincols, e->style, e->hist);
213}
214
215/* clear all text except for locktext and put the cursor at the
216 * beginning
217 */
218void owl_editwin_clear(owl_editwin *e)
219{
220
221  int lock;
222  int dotsend=e->dotsend;
223  char *locktext=NULL;
224  char echochar=e->echochar;
225
226  lock=0;
227  if (e->lock > 0) {
228    lock=1;
229
230    locktext=owl_malloc(e->lock+20);
231    strncpy(locktext, e->buff, e->lock);
232    locktext[e->lock]='\0';
233  }
234
235  owl_free(e->buff);
236  owl_editwin_init(e, e->curswin, e->winlines, e->wincols, e->style, e->hist);
237
238  if (lock > 0) {
239    owl_editwin_set_locktext(e, locktext);
240  }
241  if (dotsend) {
242    owl_editwin_set_dotsend(e);
243  }
244  if (echochar) {
245    owl_editwin_set_echochar(e, echochar);
246  }
247
248  if (locktext) owl_free(locktext);
249  owl_editwin_adjust_for_locktext(e);
250}
251
252/* malloc more space for the buffer */
253void _owl_editwin_addspace(owl_editwin *e)
254{
255  e->buff=owl_realloc(e->buff, e->allocated+INCR);
256  if (!e->buff) {
257    /* error *//*XXXXXXXXXXXXXXXX*/
258    return;
259  }
260  e->allocated+=INCR;
261}
262
263void owl_editwin_recenter(owl_editwin *e)
264{
265  e->topindex = -1;
266}
267
268static inline char *oe_next_point(owl_editwin *e, char *p)
269{
270  char *boundary = e->buff + e->bufflen + 1;
271  char *q;
272
273  q = g_utf8_find_next_char(p, boundary);
274  while (q && g_unichar_ismark(g_utf8_get_char(q)))
275    q = g_utf8_find_next_char(q, boundary);
276
277  if (q == p)
278    return NULL;
279  return q;
280}
281
282static inline char *oe_prev_point(owl_editwin *e, char *p)
283{
284  char *boundary = e->buff + e->lock;
285
286  p = g_utf8_find_prev_char(boundary, p);
287  while (p && g_unichar_ismark(g_utf8_get_char(p)))
288    p = g_utf8_find_prev_char(boundary, p);
289
290  return p;
291}
292
293static int oe_find_display_line(owl_editwin *e, int *x, int index)
294{
295  int width = 0, cw;
296  gunichar c = -1;
297  char *p;
298
299  while(1) {
300    /* are we at the point? */
301    if (x != NULL && index == e->index)
302      *x = width;
303
304    /* get the current character */
305    c = g_utf8_get_char(e->buff + index);
306
307    /* figure out how wide it is */
308    if (c == 9) /* TAB */
309      cw = TABSIZE - width % TABSIZE;
310    else
311      cw = mk_wcwidth(c);
312    if (cw < 0) /* control characters */
313        cw = 0;
314
315    if (width + cw > e->wincols)
316      break;
317    width += cw;
318
319    if (c == '\n') {
320      ++index; /* skip the newline */
321      break;
322    }
323
324    /* find the next character */
325    p = oe_next_point(e, e->buff + index);
326    if (p == NULL) { /* we ran off the end */
327      if (x != NULL && e->index > index)
328        *x = width + 1;
329      break;
330    }
331    index = p - e->buff;
332  }
333  return index;
334}
335
336static void oe_reframe(owl_editwin *e) {
337  e->topindex = 0; /*XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX*/
338}
339
340/* regenerate the text on the curses window */
341/* if update == 1 then do a doupdate(), otherwise do not */
342void owl_editwin_redisplay(owl_editwin *e, int update)
343{
344  int x = -1, y = -1, t;
345  int line, index, lineindex;
346
347  werase(e->curswin);
348
349  do {
350    if (e->topindex == -1 || e->index < e->topindex)
351      oe_reframe(e);
352
353    line = 0;
354    index = e->topindex;
355    while(line < e->winlines) {
356      lineindex = index;
357      t = -1;
358      index = oe_find_display_line(e, &t, lineindex);
359      if (x == -1 && t != -1)
360        x = t, y = line;
361      if (index - lineindex)
362        mvwaddnstr(e->curswin, line, 0,
363                   e->buff + lineindex,
364                   index - lineindex);
365      line++;
366    }
367    if (x == -1)
368        e->topindex = -1; /* force a reframe */
369  } while(x == -1);
370
371  wmove(e->curswin, y, x);
372  wnoutrefresh(e->curswin);
373  if (update == 1)
374    doupdate();
375}
376
377/* Remove n bytes at cursor. */
378void _owl_editwin_remove_bytes(owl_editwin *e, int n) /*noproto*/
379{
380  int i = e->index + n;
381  for (; i < e->bufflen; i++) {
382    e->buff[i-n] = e->buff[i];
383  }
384
385  e->bufflen -= n;
386  e->buff[e->bufflen] = '\0';
387}
388
389/* Insert n bytes at cursor.*/
390void _owl_editwin_insert_bytes(owl_editwin *e, int n) /*noproto*/
391{
392  int i;
393
394  if ((e->bufflen + n) > (e->allocated - 5)) {
395    _owl_editwin_addspace(e);
396  }
397
398  if(e->index != e->bufflen) {
399    for (i = e->bufflen + n - 1; i > e->index; i--) {
400      e->buff[i] = e->buff[i - n];
401    }
402  }
403
404  e->bufflen += n;
405  e->buff[e->bufflen] = '\0';
406
407}
408
409
410/* linewrap the word just before the cursor.
411 * returns 0 on success
412 * returns -1 if we could not wrap.
413 */
414int _owl_editwin_linewrap_word(owl_editwin *e)
415{
416  int i;
417  char *ptr1, *start;
418  gunichar c;
419
420  start = e->buff + e->lock;
421
422  ptr1 = e->buff + e->index;
423  ptr1 = g_utf8_find_prev_char(start, ptr1);
424
425  while (ptr1) {
426    c = g_utf8_get_char(ptr1);
427    if (owl_util_can_break_after(c)) {
428      if (c != ' ') {
429        i = ptr1 - e->buff;
430        e->index = i;
431        _owl_editwin_insert_bytes(e, 1);
432        /* _owl_editwin_insert_bytes may move e->buff. */
433        ptr1 = e->buff + i;
434      }
435      *ptr1 = '\n';
436      return 0;
437    }
438    else if (c == '\n') {
439      return 0;
440    }
441    ptr1 = g_utf8_find_prev_char(start, ptr1);
442  }
443  return -1;
444}
445
446/* insert a character at the current point (shift later
447 * characters over)
448 */
449void owl_editwin_insert_char(owl_editwin *e, gunichar c)
450{
451  int i, ret, len;
452  char tmp[6];
453  memset(tmp, '\0', 6);
454
455  /* \r is \n */
456  if (c == '\r') {
457    c = '\n';
458  }
459
460  if (c == '\n' && e->style == OWL_EDITWIN_STYLE_ONELINE) {
461    /* perhaps later this will change some state that allows the string
462       to be read */
463    return;
464  }
465
466  g_unichar_to_utf8(c, tmp);
467  len = strlen(tmp);
468
469  /* make sure there is enough memory for the new text */
470  if ((e->bufflen + len) > (e->allocated - 5)) {
471    _owl_editwin_addspace(e);
472  }
473
474  /* If we're going to insert at the last column do word wrapping, unless it's a \n */
475#if 0 /* XXX */
476  if ((e->buffx + 1 == e->wrapcol) && (c != '\n')) {
477    ret = _owl_editwin_linewrap_word(e);
478    if (ret == -1) {
479      /* we couldn't wrap, insert a hard newline instead */
480      owl_editwin_insert_char(e, '\n');
481    }
482  }
483#endif
484
485  /* shift all the other characters right */
486  _owl_editwin_insert_bytes(e, len);
487
488  /* insert the new character */
489  for(i = 0; i < len; i++) {
490    e->buff[e->index + i] = tmp[i];
491  }
492
493  /* advance the cursor */
494  e->index += len;
495}
496
497/* overwrite the character at the current point with 'c' */
498void owl_editwin_overwrite_char(owl_editwin *e, gunichar c)
499{
500  int oldlen, newlen, i;
501  char tmp[6], *t;
502  memset(tmp, '\0', 6);
503
504  /* \r is \n */
505  if (c == '\r') {
506    c = '\n';
507  }
508
509  if (c == '\n' && e->style == OWL_EDITWIN_STYLE_ONELINE) {
510    /* perhaps later this will change some state that allows the string
511       to be read */
512    return;
513  }
514
515  g_unichar_to_utf8(c, tmp);
516  newlen = strlen(tmp);
517
518  t = g_utf8_find_next_char(e->buff + e->index, NULL);
519  oldlen = (t ? (t - (e->buff + e->index)) : 0);
520
521  /* only if we are at the end of the buffer do we create new space here */
522  if (e->index == e->bufflen) {
523    if ((e->bufflen+newlen) > (e->allocated-5)) {
524      _owl_editwin_addspace(e);
525    }
526  }
527  /* if not at the end of the buffer, adjust based in char size difference. */
528  else if (oldlen > newlen) {
529    _owl_editwin_remove_bytes(e, oldlen-newlen);
530  }
531  else /* oldlen < newlen */ {
532    _owl_editwin_insert_bytes(e, newlen-oldlen);
533  }
534  /* Overwrite the old char*/
535  for (i = 0; i < newlen; i++) {
536    e->buff[e->index + i] = tmp[i];
537  }
538
539  /* housekeeping */
540  if (e->index == e->bufflen) {
541    e->bufflen += newlen;
542    e->buff[e->bufflen] = '\0';
543  }
544
545  /* advance the cursor */
546  e->index += newlen;
547}
548
549/* delete the character at the current point, following chars
550 * shift left.
551 */
552void owl_editwin_delete_char(owl_editwin *e)
553{
554  char *p1, *p2;
555  gunichar c;
556
557  if (e->bufflen == 0) return;
558
559  if (e->index == e->bufflen) return;
560
561  p1 = e->buff + e->index;
562  p2 = g_utf8_next_char(p1);
563  c = g_utf8_get_char(p2);
564  while (g_unichar_ismark(c)) {
565    p2 = g_utf8_next_char(p2);
566    c = g_utf8_get_char(p2);
567  }
568  _owl_editwin_remove_bytes(e, p2-p1);
569}
570
571/* Swap the character at point with the character at point-1 and
572 * advance the pointer.  If point is at beginning of buffer do
573 * nothing.  If point is after the last character swap point-1 with
574 * point-2.  (Behaves as observed in tcsh and emacs).
575 */
576void owl_editwin_transpose_chars(owl_editwin *e)
577{
578  char *p1, *p2, *p3, *tmp;
579
580  if (e->bufflen == 0) return;
581
582  if (e->index == e->bufflen) {
583    /* point is after last character */
584    e->index--;
585  }
586
587  if (e->index - 1 < e->lock) {
588    /* point is at beginning of buffer, do nothing */
589    return;
590  }
591
592  /* Transpose two utf-8 unicode glyphs. */
593  p1 = e->buff + e->index;
594
595  p2 = oe_next_point(e, p1); /* XXX make sure we can't transpose past the end
596                                of the buffer */
597  if (p2 == NULL)
598    return;
599
600  p3 = oe_prev_point(e, p1);
601  if (p3 == NULL)
602    return;
603
604  tmp = owl_malloc(p2 - p3 + 5);
605  *tmp = '\0';
606  strncat(tmp, p1, p2 - p1);
607  strncat(tmp, p3, p1 - p3);
608  strncpy(p3, tmp, p2 - p3);
609  owl_free(tmp);
610  e->index = p3 - e->buff;
611}
612
613/* insert 'string' at the current point, later text is shifted
614 * right
615 */
616void owl_editwin_insert_string(owl_editwin *e, char *string)
617{
618  char *p;
619  gunichar c;
620  if (!g_utf8_validate(string, -1, NULL)) {
621    owl_function_debugmsg("owl_editwin_insert_string: received non-utf-8 string.");
622    return;
623  }
624  p = string;
625  c = g_utf8_get_char(p);
626  while (c) {
627    _owl_editwin_process_char(e, c);
628    p = g_utf8_next_char(p);
629    c = g_utf8_get_char(p);
630  }
631}
632
633/* write 'string' at the current point, overwriting text that is
634 * already there
635 */
636
637void owl_editwin_overwrite_string(owl_editwin *e, char *string)
638{
639  char *p;
640  gunichar c;
641
642  if (!g_utf8_validate(string, -1, NULL)) {
643    owl_function_debugmsg("owl_editwin_overwrite_string: received non-utf-8 string.");
644    return;
645  }
646  p = string;
647  c = g_utf8_get_char(p);
648  while (c) {
649    owl_editwin_overwrite_char(e, c);
650    p = g_utf8_next_char(p);
651    c = g_utf8_get_char(p);
652  }
653}
654
655/* We assume index is not set to point to a mid-char */
656gunichar _owl_editwin_get_char_at_point(owl_editwin *e)
657{
658  return g_utf8_get_char(e->buff + e->index);
659}
660
661void owl_editwin_save_excursion(owl_editwin *e, _owl_editwin_excursion *x) /*noproto*/
662{
663  x->index = e->index;
664  x->goal_column = e->goal_column;
665}
666
667void owl_editwin_restore_excursion(owl_editwin *e, _owl_editwin_excursion *x) /*noproto*/
668{
669  e->index = x->index;
670  e->goal_column = x->goal_column;
671}
672
673void owl_editwin_adjust_for_locktext(owl_editwin *e)
674{
675  /* if we happen to have the cursor over locked text
676   * move it to be out of the locktext region */
677  if (e->index < e->lock) {
678    e->index = e->lock;
679  }
680}
681
682int owl_editwin_point_move(owl_editwin *e, int delta)
683{
684  char *p;
685  int change, d = 0;
686
687  change = MAX(delta, - delta);
688  p = e->buff + e->index;
689
690  while (d < change && p != NULL) {
691    if (delta > 0)
692      p = oe_next_point(e, p);
693    else
694      p = oe_prev_point(e, p);
695    if (p != NULL) {
696      e->index = p - e->buff;
697      d++;
698    }
699  }
700
701  if (delta)
702    e->goal_column = -1;
703
704  return delta > 0 ? d : -d;
705}
706
707int owl_editwin_at_beginning_of_buffer(owl_editwin *e) {
708  if (e->index == e->lock)
709    return 1;
710
711  return 0;
712}
713
714int owl_at_end_of_buffer(owl_editwin *e) {
715  if (e->index == e->bufflen)
716    return 1;
717
718  return 0;
719}
720
721int owl_editwin_at_beginning_of_line(owl_editwin *e) /*noproto*/
722{
723  _owl_editwin_excursion x;
724  int ret;
725
726  if (owl_editwin_at_beginning_of_buffer(e))
727    return 1;
728
729  owl_editwin_save_excursion(e, &x);
730  owl_editwin_point_move(e, -1);
731  ret = (_owl_editwin_get_char_at_point(e) == '\n');
732  owl_editwin_restore_excursion(e, &x);
733
734  return ret;
735}
736
737static int owl_editwin_is_char_in(owl_editwin *e, char *set)
738{
739  char *p;
740  /* It would be awfully nice if we could do UTF-8 comparisons */
741  for (p = set; *p != 0; p++)
742    if (_owl_editwin_get_char_at_point(e) == *p)
743      return 1;
744  return 0;
745}
746
747int owl_editwin_move_if_in(owl_editwin *e, int delta, char *set)
748{
749  int change, distance = 0;
750  while (owl_editwin_is_char_in(e, set)) {
751    change = owl_editwin_point_move(e, delta);
752    distance += change;
753    if (change == 0)
754      break;
755  }
756  return distance;
757}
758
759int owl_editwin_move_if_not_in(owl_editwin *e, int delta, char *set)
760{
761  int change, distance = 0;
762  while (!owl_editwin_is_char_in(e, set)) {
763    change = owl_editwin_point_move(e, delta);
764    distance += change;
765    if (change == 0)
766      break;
767  }
768  return distance;
769}
770
771int owl_editwin_move_to_beginning_of_line(owl_editwin *e)
772{
773  int distance = 0;
774
775  if (!owl_editwin_at_beginning_of_line(e)) {
776    /* move off the \n if were at the end of a line */
777    distance += owl_editwin_point_move(e, -1);
778    distance += owl_editwin_move_if_not_in(e, -1, "\n");
779    if (distance && !owl_editwin_at_beginning_of_buffer(e))
780      distance += owl_editwin_point_move(e, 1);
781  }
782
783  return distance;
784}
785
786int owl_editwin_move_to_end_of_line(owl_editwin *e)
787{
788  return owl_editwin_move_if_not_in(e, 1, "\n");
789}
790
791int owl_editwin_line_move(owl_editwin *e, int delta)
792{
793  int goal_column, change, ll, distance;
794  int count = 0;
795
796  change = MAX(delta, -delta);
797
798  goal_column = e->goal_column;
799  distance = owl_editwin_move_to_beginning_of_line(e);
800  goal_column = goal_column == -1 ? -distance : goal_column;
801
802  while(count < change) {
803    if (delta > 0) {
804      distance += owl_editwin_move_if_not_in(e, 1, "\n");
805      distance += owl_editwin_point_move(e, 1);
806    } else {
807      /* I really want to assert delta < 0 here */
808      distance += owl_editwin_point_move(e, -1); /* to the newline on
809                                                    the previous line */
810      distance += owl_editwin_move_to_beginning_of_line(e);
811    }
812    count++;
813  }
814
815  distance += (ll = owl_editwin_move_to_end_of_line(e));
816  if (ll > goal_column)
817    distance += owl_editwin_point_move(e, goal_column - ll);
818
819  e->goal_column = goal_column;
820
821  return distance;
822}
823
824void owl_editwin_backspace(owl_editwin *e)
825{
826  /* delete the char before the current one
827   * and shift later chars left
828   */
829  if(owl_editwin_point_move(e, -1))
830    owl_editwin_delete_char(e);
831}
832
833void owl_editwin_key_up(owl_editwin *e)
834{
835  owl_editwin_line_move(e, -1);
836}
837
838void owl_editwin_key_down(owl_editwin *e)
839{
840  owl_editwin_line_move(e, 1);
841}
842
843void owl_editwin_key_left(owl_editwin *e)
844{
845  owl_editwin_point_move(e, -1);
846}
847
848void owl_editwin_key_right(owl_editwin *e)
849{
850  owl_editwin_point_move(e, 1);
851}
852
853void owl_editwin_move_to_nextword(owl_editwin *e)
854{
855  /* if we're starting on a space, find the first non-space */
856  owl_editwin_move_if_in(e, 1, WHITESPACE);
857
858  /* now find the end of this word */
859  owl_editwin_move_if_not_in(e, 1, WHITESPACE);
860}
861
862/* go backwards to the last non-space character
863 */
864void owl_editwin_move_to_previousword(owl_editwin *e)
865{
866  _owl_editwin_excursion x;
867  int beginning;
868  /* if in middle of word, beginning of word */
869
870  /* if at beginning of a word, find beginning of previous word */
871
872  if (owl_editwin_is_char_in(e, WHITESPACE)) {
873    /* if in whitespace past end of word, find a word , the find the beginning*/
874    owl_editwin_move_if_in(e, -1, WHITESPACE); /* leaves us on the last
875                                                    character of the word */
876    owl_editwin_save_excursion(e, &x);
877    /* are we at the beginning of a word? */
878    owl_editwin_point_move(e, -1);
879    beginning = owl_editwin_is_char_in(e, WHITESPACE);
880    owl_editwin_restore_excursion(e, &x);
881    if (beginning)
882      return;
883   } else {
884    /* in the middle of the word; */
885    owl_editwin_save_excursion(e, &x);
886    owl_editwin_point_move(e, -1);
887    if (owl_editwin_is_char_in(e, WHITESPACE)) { /* we were at the beginning */
888      owl_editwin_move_to_previousword(e); /* previous case */
889      return;
890    } else {
891      owl_editwin_restore_excursion(e, &x);
892    }
893  }
894  owl_editwin_move_if_not_in(e, -1, WHITESPACE);
895  /* will go past */
896  if (e->index > e->lock)
897    owl_editwin_point_move(e, 1);
898}
899
900void owl_editwin_delete_nextword(owl_editwin *e)
901{
902  _owl_editwin_excursion x;
903  int end;
904
905  owl_editwin_save_excursion(e, &x);
906  owl_editwin_move_to_nextword(e);
907  end = e->index;
908  owl_editwin_restore_excursion(e, &x);
909  _owl_editwin_remove_bytes(e, end - e->index);
910}
911
912void owl_editwin_delete_previousword(owl_editwin *e)
913{
914  /* go backwards to the last non-space character, then delete chars */
915  int startpos, endpos;
916
917  startpos = e->index;
918  owl_editwin_move_to_previousword(e);
919  endpos = e->index;
920  _owl_editwin_remove_bytes(e, startpos-endpos);
921}
922
923void owl_editwin_move_to_line_end(owl_editwin *e)
924{
925  owl_editwin_move_to_end_of_line(e);
926}
927
928void owl_editwin_delete_to_endofline(owl_editwin *e)
929{
930  _owl_editwin_excursion x;
931  int end;
932
933  owl_editwin_save_excursion(e, &x);
934  owl_editwin_move_to_line_end(e);
935  end = e->index;
936  owl_editwin_restore_excursion(e, &x);
937  _owl_editwin_remove_bytes(e, end - e->index);
938}
939
940void owl_editwin_move_to_line_start(owl_editwin *e)
941{
942  owl_editwin_move_to_beginning_of_line(e);
943}
944
945void owl_editwin_move_to_end(owl_editwin *e)
946{
947  e->index = e->bufflen;
948}
949
950void owl_editwin_move_to_top(owl_editwin *e)
951{
952  e->index = e->lock;
953}
954
955void owl_editwin_fill_paragraph(owl_editwin *e)
956{
957#if 0 /* XXX */
958  _owl_editwin_excursion x;
959  int i, save;
960
961  /* save our starting point */
962  owl_editwin_save_excursion(e, &x);
963
964  save = e->index;
965
966  /* scan back to the beginning of this paragraph */
967  for (i=save; i>=e->lock; i--) {
968    if ( (i<=e->lock) ||
969         ((e->buff[i]=='\n') && (e->buff[i-1]=='\n'))) {
970      e->index = i + 1;
971      break;
972    }
973  }
974
975  /* main loop */
976  while (1) {
977    i = _owl_editwin_get_index_from_xy(e);
978
979    /* bail if we hit the end of the buffer */
980    if (i >= e->bufflen || e->buff[i] == '\0') break;
981
982    /* bail if we hit the end of the paragraph */
983    if (e->buff[i] == '\n' && e->buff[i+1] == '\n') break;
984
985    /* bail if we hit a trailing dot on the buffer */
986    if (e->buff[i] == '\n' && e->buff[i+1] == '.'
987        && ((i+2) >= e->bufflen || e->buff[i+2] == '\0'))
988      break;
989
990    /* if we've travelled too far, linewrap */
991    if ((e->buffx) >= e->fillcol) {
992      int len = e->bufflen;
993      _owl_editwin_linewrap_word(e);
994      /* we may have added a character. */
995      if (i < save) save += e->bufflen - len;
996      e->index = i;
997    }
998
999    /* did we hit the end of a line too soon? */
1000    /* asedeno: Here we replace a newline with a space. We may want to
1001       consider removing the space if the characters to either side
1002       are CJK ideograms.*/
1003    i = _owl_editwin_get_index_from_xy(e);
1004    if (e->buff[i] == '\n' && e->buffx < e->fillcol - 1) {
1005      /* ********* we need to make sure we don't pull in a word that's too long ***********/
1006      e->buff[i]=' ';
1007    }
1008
1009    /* fix spacing */
1010    i = _owl_editwin_get_index_from_xy(e);
1011    if (e->buff[i] == ' ' && e->buff[i+1] == ' ') {
1012      if (e->buff[i-1] == '.' || e->buff[i-1] == '!' || e->buff[i-1] == '?') {
1013        owl_editwin_key_right(e);
1014      } else {
1015        owl_editwin_delete_char(e);
1016        /* if we did this ahead of the save point, adjust it. Changing
1017           by one is fine here because we're only removing an ASCII
1018           space. */
1019        if (i < save) save--;
1020      }
1021    } else {
1022      owl_editwin_key_right(e);
1023    }
1024  }
1025
1026  /* put cursor back at starting point */
1027  owl_editwin_restore_excursion(e, &x);
1028#endif
1029}
1030
1031/* returns true if only whitespace remains */
1032int owl_editwin_is_at_end(owl_editwin *e)
1033{
1034  return (only_whitespace(e->buff + e->index));
1035}
1036
1037int owl_editwin_check_dotsend(owl_editwin *e)
1038{
1039  char *p, *p_n, *p_p;
1040  gunichar c;
1041
1042  if (!e->dotsend) return(0);
1043
1044  p = g_utf8_find_prev_char(e->buff, e->buff + e->bufflen);
1045  p_n = g_utf8_find_next_char(p, NULL);
1046  p_p = g_utf8_find_prev_char(e->buff, p);
1047  c = g_utf8_get_char(p);
1048  while (p != NULL) {
1049    if (*p == '.'
1050        && p_p != NULL && (*p_p == '\n' || *p_p == '\r')
1051        && p_n != NULL && (*p_n == '\n' || *p_n == '\r')) {
1052      e->bufflen = p - e->buff;
1053      e->buff[e->bufflen] = '\0';
1054      return(1);
1055    }
1056    if (c != '\0' && !g_unichar_isspace(c)) return(0);
1057    p_n = p;
1058    p = p_p;
1059    c = g_utf8_get_char(p);
1060    p_p = g_utf8_find_prev_char(e->buff, p);
1061  }
1062  return(0);
1063}
1064
1065void owl_editwin_post_process_char(owl_editwin *e, owl_input j)
1066{
1067  /* XXX force a redisplay? */
1068  if ((j.ch==13 || j.ch==10) && owl_editwin_check_dotsend(e)) {
1069    owl_command_editmulti_done(e);
1070    return;
1071  }
1072  owl_editwin_redisplay(e, 0);
1073}
1074
1075void _owl_editwin_process_char(owl_editwin *e, gunichar j)
1076{
1077  if (!(g_unichar_iscntrl(j) && (j != 10) && (j != 13)) || j==9 ) {
1078    owl_editwin_insert_char(e, j);
1079  }
1080}
1081
1082
1083void owl_editwin_process_char(owl_editwin *e, owl_input j)
1084{
1085  if (j.ch == ERR) return;
1086  /* Ignore ncurses control characters. */
1087  if (j.ch < 0x100) {
1088    _owl_editwin_process_char(e, j.uch);
1089  }
1090}
1091
1092char *owl_editwin_get_text(owl_editwin *e)
1093{
1094  return(e->buff+e->lock);
1095}
1096
1097int owl_editwin_get_numchars_on_line(owl_editwin *e, int line)
1098{
1099  int i;
1100  char *ptr1, *ptr2;
1101
1102  if (e->bufflen==0) return(0);
1103
1104  /* first go to the yth line */
1105  ptr1=e->buff;
1106  for (i=0; i<line; i++) {
1107    ptr2=strchr(ptr1, '\n');
1108    if (!ptr2) {
1109      /* we're already on the last line */
1110      return(0);
1111    }
1112    ptr1=ptr2+1;
1113  }
1114
1115  /* now count characters */
1116  i = 0;
1117  ptr2 = ptr1;
1118  while (ptr2 - e->buff < e->bufflen
1119         && *ptr2 != '\n') {
1120    ++i;
1121    ptr2 = g_utf8_next_char(ptr2);
1122  }
1123  return i;
1124}
1125
1126int owl_editwin_get_numlines(owl_editwin *e)
1127{
1128  return(owl_text_num_lines(e->buff));
1129}
1130
1131int owl_editwin_get_echochar(owl_editwin *e) {
1132  return e->echochar;
1133}
1134
1135void owl_editwin_set_goal_column(owl_editwin *e, int column)
1136{
1137  e->goal_column = column;
1138}
1139
1140int owl_editwin_get_goal_column(owl_editwin *e)
1141{
1142  return e->goal_column;
1143}
Note: See TracBrowser for help on using the repository browser.