about summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--CHANGES4
-rw-r--r--src/LYStrings.c366
-rw-r--r--src/LYStrings.h10
3 files changed, 254 insertions, 126 deletions
diff --git a/CHANGES b/CHANGES
index b1ca9743..1a46acb4 100644
--- a/CHANGES
+++ b/CHANGES
@@ -1,9 +1,11 @@
--- $LynxId: CHANGES,v 1.293 2008/08/31 23:31:07 tom Exp $
+-- $LynxId: CHANGES,v 1.294 2008/09/04 00:27:30 tom Exp $
 ===============================================================================
 Changes since Lynx 2.8 release
 ===============================================================================
 
 2008-0?-?? (2.8.7dev.10)
+* rewrite LYRefreshEdit(), to display multibyte/multicolumn characters properly
+  when using wide-character curses -TD
 * revise introductory comment written to ".lynxrc" (Debian #461158) -TD
 * discard anchor's post_data field in HTLoadDocument() if Lynx is about to
   reload a document.  That would happen if the result of the form includes a
diff --git a/src/LYStrings.c b/src/LYStrings.c
index bb0d7a09..3ac742a1 100644
--- a/src/LYStrings.c
+++ b/src/LYStrings.c
@@ -1,4 +1,4 @@
-/* $LynxId: LYStrings.c,v 1.135 2008/08/31 18:54:07 tom Exp $ */
+/* $LynxId: LYStrings.c,v 1.146 2008/09/04 00:21:11 tom Exp $ */
 #include <HTUtils.h>
 #include <HTCJK.h>
 #include <UCAux.h>
@@ -47,6 +47,12 @@
 #define BUTTON_CTRL	0	/* Quick hack */
 #endif
 
+#ifdef DEBUG_EDIT
+#define CTRACE_EDIT(p) CTRACE(p)
+#else
+#define CTRACE_EDIT(p)		/*nothing */
+#endif
+
 /*
  * The edit_history lists allow the user to press tab when entering URL to get
  * the closest match in the closet
@@ -2808,21 +2814,27 @@ void LYTrimAllStartfile(char *buffer)
 #define EDREC	 EditFieldData
 
 /*
- * Shorthand to get rid of all most of the "edit->suchandsos".
+ * Shorthand to get rid of the "edit->suchandsos".
  */
+#define IsDirty  edit->dirty
+#define IsHidden edit->hidden
+#define StartX	 edit->sx
+#define StartY	 edit->sy
 #define Buf	 edit->buffer
-#define Pos	 edit->pos
-#define StrLen	 edit->strlen
+#define Pos	 edit->pos	/* current editing position (bytes) */
+#define StrLen	 edit->strlen	/* length (bytes) */
 #define MaxLen	 edit->maxlen
 #define DspWdth  edit->dspwdth
-#define DspStart edit->xpan
+#define DspStart edit->xpan	/* display-start (columns) */
 #define Margin	 edit->margin
+#define PanOn	 edit->panon
+#define PadChar  edit->pad
 #ifdef ENHANCED_LINEEDIT
 #define Mark	 edit->mark
 #endif
 
 #ifdef ENHANCED_LINEEDIT
-static char killbuffer[1024] = "\0";
+static char killbuffer[MAX_EDIT] = "\0";
 #endif
 
 void LYSetupEdit(EDREC * edit, char *old,
@@ -2832,10 +2844,10 @@ void LYSetupEdit(EDREC * edit, char *old,
     /*
      * Initialize edit record
      */
-    LYGetYX(edit->sy, edit->sx);
-    edit->pad = ' ';
-    edit->dirty = TRUE;
-    edit->panon = FALSE;
+    LYGetYX(StartY, StartX);
+    PadChar = ' ';
+    IsDirty = TRUE;
+    PanOn = FALSE;
     edit->current_modifiers = 0;
 
     MaxLen = maxstr;
@@ -2849,7 +2861,7 @@ void LYSetupEdit(EDREC * edit, char *old,
 
     if (maxstr > maxdsp) {	/* Need panning? */
 	if (DspWdth > 4)	/* Else "{}" take up precious screen space */
-	    edit->panon = TRUE;
+	    PanOn = TRUE;
 
 	/*
 	 * Figure out margins.  If too big, we do a lot of unnecessary
@@ -2861,8 +2873,8 @@ void LYSetupEdit(EDREC * edit, char *old,
 	    Margin = 10;
     }
 
-    LYstrncpy(edit->buffer, old, maxstr);
-    StrLen = strlen(edit->buffer);
+    LYstrncpy(Buf, old, maxstr);
+    StrLen = strlen(Buf);
 }
 
 #ifdef SUPPORT_MULTIBYTE_EDIT
@@ -2901,8 +2913,9 @@ static int mbcs_glyphs(char *s, int len)
 	for (i = 0; s[i] && i < len; i++, glyphs++)
 	    if (is8bits(s[i]))
 		i++;
-    } else
+    } else {
 	glyphs = len;
+    }
     return glyphs;
 }
 
@@ -2924,12 +2937,40 @@ static int mbcs_skip(char *s, int pos)
 	for (p = i = 0; s[i] && p < pos; p++, i++)
 	    if (is8bits(s[i]))
 		i++;
-    } else
+    } else {
 	i = pos;
+    }
 
     return i;
 }
 
+/*
+ * Given a string that would display (at least) the given number of cells,
+ * determine the number of multibyte characters that comprised those cells.
+ */
+static int cell2char(char *s, int cells)
+{
+    int result = 0;
+    int len = strlen(s);
+    int pos;
+    int have;
+
+    CTRACE_EDIT((tfp, "cell2char(%d) %d:%s\n", cells, len, s));
+    /* FIXME - make this a binary search */
+    for (pos = 0; pos <= len; ++pos) {
+	have = LYstrExtent2(s, pos);
+	CTRACE_EDIT((tfp, "  %2d:%2d:%.*s\n", pos, have, pos, s));
+	if (have >= cells) {
+	    break;
+	}
+    }
+    if (pos > len)
+	pos = len;
+    result = mbcs_glyphs(s, pos);
+    CTRACE_EDIT((tfp, "->%d\n", result));
+    return result;
+}
+
 #endif /* SUPPORT_MULTIBYTE_EDIT */
 
 #ifdef EXP_KEYBOARD_LAYOUT
@@ -3035,7 +3076,7 @@ int LYEditInsert(EDREC * edit, unsigned const char *s,
     Pos += len;
     StrLen += len;
     if (edited)
-	edit->dirty = TRUE;
+	IsDirty = TRUE;
     if (overflow && maxMessage)
 	_statusline(MAXLEN_REACHED_DEL_OR_MOV);
 #ifdef ENHANCED_LINEEDIT
@@ -3049,13 +3090,14 @@ int LYEditInsert(EDREC * edit, unsigned const char *s,
     return edited;
 }
 
+/* returns 0    character processed
+ *         -ch  if action should be performed outside of line-editing mode
+ *         ch   otherwise
+ */
 int LYEdit1(EDREC * edit, int ch,
 	    int action,
 	    BOOL maxMessage)
-{				/* returns 0    character processed
-				 *         -ch  if action should be performed outside of line-editing mode
-				 *         ch   otherwise
-				 */
+{
     int i;
     int length;
     unsigned char uch;
@@ -3443,7 +3485,7 @@ int LYEdit1(EDREC * edit, int ch,
     default:
 	return (ch);
     }
-    edit->dirty = TRUE;
+    IsDirty = TRUE;
     StrLen = strlen(&Buf[0]);
     return (0);
 }
@@ -3521,40 +3563,87 @@ int get_popup_number(const char *msg,
 #  define TmpStyleOff(s)
 #endif /* defined USE_COLOR_STYLE */
 
-void LYRefreshEdit(EDREC * edit)
+static void fill_edited_line(int prompting GCC_UNUSED, int length, int ch)
 {
-    int i;
-    int length;
-    int nrdisplayed;
-    int padsize;
-    char *str;
+    TmpStyleOn(prompting ? s_prompt_edit_pad : s_aedit_pad);
+    while (length-- > 0)
+	LYaddch(UCH(ch));
+    TmpStyleOff(prompting ? s_prompt_edit_pad : s_aedit_pad);
+}
+
+static void remember_column(EDREC * edit, int offset)
+{
+    int y0, x0;
+
+#if defined(USE_SLANG)
+    y0 = 0;
+    x0 = offset;
+#elif defined(USE_CURSES_PADS)
+    getyx(LYwin, y0, x0);
+#else
+    getyx(stdscr, y0, x0);
+#endif
+    edit->offset2col[offset] = x0;
+}
 
-#ifdef SUPPORT_MULTIBYTE_EDIT
 /*
- * Multibyte string display code.
- * EDREC fields retain their values as byte offsets,
- * while glXXXX variables hold corresponding glyph counts.
+ * Multibyte string display subroutine.
+ * EDREC fields retain their values as byte offsets.
  * All external logic still works fine with byte values.
  */
-    int glDspStart, glPos, gllength;
-    int glnrdisplayed;
+void LYRefreshEdit(EDREC * edit)
+{
+    /* bytes and characters are not the same thing */
+    int all_bytes;
+    int pos_bytes = Pos;
+    int dpy_bytes;
+    int lft_bytes;		/* base of string which is displayed */
+
+    /* cells refer to display-columns on the screen */
+    int all_cells;		/* total of display-cells in Buf */
+    int dpy_cells;		/* number of cells which are displayed */
+    int lft_cells;		/* number of cells before display (on left) */
+    int pos_cells;		/* number of display-cells up to Pos */
+
+#ifdef SUPPORT_MULTIBYTE_EDIT
+    int all_chars;
+    int dpy_chars;
+    int lft_chars;
+    int pos_chars;
 #endif
 
+    /* other data */
+    int i;
+    int padsize;
+    char *str;
+    int lft_shift = 0;
+    int rgt_shift = 0;
+
 #ifdef USE_COLOR_STYLE
-    int estyle, prompting = 0;
+    int estyle;
 #endif
+    int prompting = 0;
 
-    if (!edit->dirty || (DspWdth == 0))
+    (void) pos_bytes;
+
+    /*
+     * If we've made no changes, or if there is nothing to display, just leave.
+     */
+    if (!IsDirty || (DspWdth == 0))
 	return;
-    edit->dirty = FALSE;
 
-    length = strlen(&Buf[0]);
-    edit->strlen = length;
+    IsDirty = FALSE;
+
+    all_bytes = strlen(&Buf[0]);
+    StrLen = all_bytes;
+
+    all_cells = LYstrCells(Buf);
+    pos_cells = LYstrExtent2(Buf, Pos);
 
 #ifdef SUPPORT_MULTIBYTE_EDIT
-    glDspStart = mbcs_glyphs(Buf, DspStart);
-    glPos = mbcs_glyphs(Buf, Pos);
-    gllength = mbcs_glyphs(Buf, length);
+    lft_chars = mbcs_glyphs(Buf, DspStart);
+    pos_chars = mbcs_glyphs(Buf, Pos);
+    all_chars = mbcs_glyphs(Buf, all_bytes);
 #endif
 
     /*
@@ -3563,7 +3652,7 @@ void LYRefreshEdit(EDREC * edit)
      *      +---------+=============+-----------+
      *      |         |M           M|           |   (M=margin)
      *      +---------+=============+-----------+
-     *      0         DspStart                   length
+     *      0         DspStart                   StrLen
      *
      * Insertion point can be anywhere between 0 and stringlength.  Figure out
      * new display starting point.
@@ -3573,47 +3662,93 @@ void LYRefreshEdit(EDREC * edit)
      * data entry at low baudrates.
      */
 
-#ifndef SUPPORT_MULTIBYTE_EDIT
-    if ((DspStart + DspWdth) <= length)
-	if (Pos >= (DspStart + DspWdth) - Margin)
-	    DspStart = (Pos - DspWdth) + Margin;
+    lft_bytes = DspStart;
+    lft_cells = LYstrExtent2(Buf, DspStart);
+
+    if ((lft_cells + DspWdth) <= all_cells) {
+	if (pos_cells >= (lft_cells + DspWdth) - Margin) {
+	    lft_cells = (pos_cells - DspWdth) + Margin;
+#ifdef SUPPORT_MULTIBYTE_EDIT
+	    lft_chars = cell2char(Buf, lft_cells);
+	    lft_bytes = mbcs_skip(Buf, lft_chars);
 #else
-    if ((glDspStart + DspWdth) <= gllength)
-	if (glPos >= (glDspStart + DspWdth) - Margin) {
-	    glDspStart = (glPos - DspWdth) + Margin;
-	    DspStart = mbcs_skip(Buf, glDspStart);
-	}
+	    lft_bytes = lft_cells;
 #endif /* SUPPORT_MULTIBYTE_EDIT */
+	}
+    }
 
-#ifndef SUPPORT_MULTIBYTE_EDIT
-    if (Pos < DspStart + Margin) {
-	DspStart = Pos - Margin;
-	if (DspStart < 0)
-	    DspStart = 0;
+    if (pos_cells < lft_cells + Margin) {
+	lft_cells = pos_cells - Margin;
+	if (lft_cells < 0)
+	    lft_cells = 0;
+#ifdef SUPPORT_MULTIBYTE_EDIT
+	lft_chars = cell2char(Buf, lft_cells);
+	lft_bytes = mbcs_skip(Buf, lft_chars);
+#else
+	lft_bytes = lft_cells;
+#endif /* SUPPORT_MULTIBYTE_EDIT */
     }
-#else /* SUPPORT_MULTIBYTE_EDIT */
-    if (glPos < glDspStart + Margin) {
-	glDspStart = glPos - Margin;
-	if (glDspStart < 0)
-	    glDspStart = 0;
-	DspStart = mbcs_skip(Buf, glDspStart);
+
+    LYmove(StartY, StartX);
+
+    /*
+     * Draw the left scrolling-indicator now, to avoid the complication of
+     * overwriting part of a multicolumn character which may lie in the first
+     * position.
+     */
+    if (PanOn && lft_cells) {
+	CTRACE_EDIT((tfp, "Draw left scroll-indicator\n"));
+	TmpStyleOn(prompting ? s_prompt_edit_arr : s_aedit_arr);
+	LYmove(StartY, StartX);
+	LYaddch(ACS_LARROW);
+	TmpStyleOff(prompting ? s_prompt_edit_arr : s_aedit_arr);
+	lft_shift = 1;
     }
-#endif /* SUPPORT_MULTIBYTE_EDIT */
 
-    str = &Buf[DspStart];
+    str = &Buf[lft_bytes];
+    DspStart = lft_bytes;
 
-#ifndef SUPPORT_MULTIBYTE_EDIT
-    nrdisplayed = length - DspStart;
-    if (nrdisplayed > DspWdth)
-	nrdisplayed = DspWdth;
+    dpy_cells = all_cells - lft_cells;
+    CTRACE_EDIT((tfp, "Comparing dpy_cells %d > (%d - %d)\n",
+		 dpy_cells, DspWdth, lft_shift));
+    if (dpy_cells > (DspWdth - lft_shift)) {
+	rgt_shift = 1;
+	dpy_cells = (DspWdth - lft_shift - rgt_shift);
+    }
+    for (;;) {
+#ifdef SUPPORT_MULTIBYTE_EDIT
+	dpy_chars = cell2char(str, dpy_cells);
+	dpy_bytes = mbcs_skip(str, dpy_chars);
 #else
-    glnrdisplayed = gllength - glDspStart;
-    if (glnrdisplayed > DspWdth)
-	glnrdisplayed = DspWdth;
-    nrdisplayed = mbcs_skip(Buf + DspStart, glnrdisplayed);
+	dpy_bytes = dpy_cells;
 #endif /* SUPPORT_MULTIBYTE_EDIT */
+	/*
+	 * The last character on the display may be multicolumn, and if we take
+	 * away a single cell for the right scroll-indicator, that would force
+	 * us to display fewer characters.  Check for that, and recompute.
+	 */
+	if (rgt_shift) {
+	    int old_cells = dpy_cells;
 
-    LYmove(edit->sy, edit->sx);
+	    dpy_cells = LYstrExtent2(str, dpy_bytes);
+	    if (dpy_cells > old_cells)
+		dpy_cells = old_cells - 1;
+
+	    CTRACE_EDIT((tfp, "Comparing cells %d vs %d\n", dpy_cells, old_cells));
+	    if (dpy_cells < old_cells) {
+		CTRACE_EDIT((tfp, "Recomputing...\n"));
+		continue;
+	    }
+	}
+	break;
+    }
+
+    CTRACE_EDIT((tfp, "BYTES left %2d pos %2d dpy %2d all %2d\n",
+		 lft_bytes, pos_bytes, dpy_bytes, all_bytes));
+    CTRACE_EDIT((tfp, "CELLS left %2d pos %2d dpy %2d all %2d\n",
+		 lft_cells, pos_cells, dpy_cells, all_cells));
+    CTRACE_EDIT((tfp, "CHARS left %2d pos %2d dpy %2d all %2d\n",
+		 lft_chars, pos_chars, dpy_chars, all_chars));
 
 #ifdef USE_COLOR_STYLE
     /*
@@ -3621,35 +3756,33 @@ void LYRefreshEdit(EDREC * edit)
      * be needed for color styles.  The curses function may be used directly to
      * avoid complications.  - kw
      */
-    if (edit->sy == (LYlines - 1))
+    if (StartY == (LYlines - 1))
 	prompting = 1;
-    if (prompting)
+    if (prompting) {
 	estyle = s_prompt_edit;
-    else
+    } else {
 	estyle = s_aedit;
+    }
     CTRACE2(TRACE_STYLE,
 	    (tfp, "STYLE.getstr: switching to <edit.%s>.\n",
 	     prompting ? "prompt" : "active"));
-    if (estyle != NOSTYLE)
+    if (estyle != NOSTYLE) {
 	curses_style(estyle, STACK_ON);
-    else
+    } else {
 	wattrset(LYwin, A_NORMAL);	/* need to do something about colors? */
+    }
 #endif
-    if (edit->hidden) {
-#ifndef SUPPORT_MULTIBYTE_EDIT
-	for (i = 0; i < nrdisplayed; i++)
-	    LYaddch('*');
-#else
-	for (i = 0; i < glnrdisplayed; i++)
-	    LYaddch('*');
-#endif
+    if (IsHidden) {
+	fill_edited_line(0, dpy_cells, '*');
     } else {
 #if defined(ENHANCED_LINEEDIT) && defined(USE_COLOR_STYLE)
 	if (Mark >= 0 && DspStart > Mark)
 	    TmpStyleOn(prompting ? s_prompt_sel : s_aedit_sel);
 #endif
-	for (i = 0; i < nrdisplayed; i++) {
+	remember_column(edit, 0);
+	for (i = 0; i < dpy_bytes; i++) {
 #if defined(ENHANCED_LINEEDIT) && defined(USE_COLOR_STYLE)
+	    /* FIXME - make this work with multibyte-chars */
 	    if (Mark >= 0 && ((DspStart + i == Mark && Pos > Mark)
 			      || (DspStart + i == Pos && Pos < Mark)))
 		TmpStyleOn(prompting ? s_prompt_sel : s_aedit_sel);
@@ -3664,59 +3797,44 @@ void LYRefreshEdit(EDREC * edit)
 		    !(LYCharSet_UC[current_char_set].like8859
 		      & UCT_R_8859SPECL))))) {
 		LYaddch(' ');
-	    } else
+	    } else {
 		LYaddch(UCH(str[i]));
+	    }
+	    remember_column(edit, i + 1);
 	}
 #if defined(ENHANCED_LINEEDIT) && defined(USE_COLOR_STYLE)
+	/* FIXME - make this work with multibyte-chars */
 	if (Mark >= 0 &&
-	    ((DspStart + nrdisplayed <= Mark && DspStart + nrdisplayed > Pos)
-	     || (DspStart + nrdisplayed > Mark
-		 && DspStart + nrdisplayed <= Pos)))
+	    ((DspStart + dpy_bytes <= Mark && DspStart + dpy_bytes > Pos)
+	     || (DspStart + dpy_bytes > Mark
+		 && DspStart + dpy_bytes <= Pos))) {
 	    TmpStyleOff(prompting ? s_prompt_sel : s_aedit_sel);
+	}
 #endif
     }
 
     /*
      * Erase rest of input area.
      */
-#ifndef SUPPORT_MULTIBYTE_EDIT
-    padsize = DspWdth - nrdisplayed;
-#else
-    padsize = DspWdth - glnrdisplayed;
-#endif
-    if (padsize) {
-	TmpStyleOn(prompting ? s_prompt_edit_pad : s_aedit_pad);
-	while (padsize--)
-	    LYaddch(UCH(edit->pad));
-	TmpStyleOff(prompting ? s_prompt_edit_pad : s_aedit_pad);
-    }
+    padsize = DspWdth - (edit->offset2col[dpy_bytes] - StartX);
+    fill_edited_line(prompting, padsize, PadChar);
 
     /*
      * Scrolling indicators.
      */
-    if (edit->panon) {
-	if ((DspStart + nrdisplayed) < length) {
-	    TmpStyleOn(prompting ? s_prompt_edit_arr : s_aedit_arr);
-#ifndef SUPPORT_MULTIBYTE_EDIT
-	    LYmove(edit->sy, edit->sx + nrdisplayed - 1);
-#else
-	    LYmove(edit->sy, edit->sx + glnrdisplayed - 1);
-#endif
-	    LYaddch(ACS_RARROW);
-	    TmpStyleOff(prompting ? s_prompt_edit_arr : s_aedit_arr);
-	}
-	if (DspStart) {
-	    TmpStyleOn(prompting ? s_prompt_edit_arr : s_aedit_arr);
-	    LYmove(edit->sy, edit->sx);
-	    LYaddch(ACS_LARROW);
-	    TmpStyleOff(prompting ? s_prompt_edit_arr : s_aedit_arr);
-	}
+    if (PanOn && dpy_bytes && rgt_shift) {
+	CTRACE((tfp, "Draw right-scroller offset (%d + %d)\n",
+		dpy_cells, lft_shift));
+	TmpStyleOn(prompting ? s_prompt_edit_arr : s_aedit_arr);
+	LYmove(StartY, StartX + dpy_cells + lft_shift);
+	LYaddch(ACS_RARROW);
+	TmpStyleOff(prompting ? s_prompt_edit_arr : s_aedit_arr);
     }
-#ifndef SUPPORT_MULTIBYTE_EDIT
-    LYmove(edit->sy, edit->sx + Pos - DspStart);
-#else
-    LYmove(edit->sy, edit->sx + glPos - glDspStart);
-#endif
+
+    /*
+     * Finally, move the cursor to the point where the next edit will occur.
+     */
+    LYmove(StartY, edit->offset2col[Pos - DspStart]);
 
 #ifdef USE_COLOR_STYLE
     if (estyle != NOSTYLE)
diff --git a/src/LYStrings.h b/src/LYStrings.h
index 56cfb87a..4e821cf4 100644
--- a/src/LYStrings.h
+++ b/src/LYStrings.h
@@ -1,3 +1,6 @@
+/*
+ * $LynxId: LYStrings.h,v 1.66 2008/09/01 15:09:03 tom Exp $
+ */
 #ifndef LYSTRINGS_H
 #define LYSTRINGS_H
 
@@ -183,6 +186,8 @@ extern "C" {
 #define ENHANCED_LINEEDIT
 #endif
 
+#define MAX_EDIT 1024
+
 /* EditFieldData preserves state between calls to LYEdit1
  */
     typedef struct _EditFieldData {
@@ -207,7 +212,10 @@ extern "C" {
 				   unactive mark.  */
 #endif
 
-	char buffer[1024];	/* String buffer                          */
+	char buffer[MAX_EDIT];	/* String buffer                          */
+
+	int offset2col[MAX_EDIT * 2];
+	int col2offset[MAX_EDIT * 2];
 
     } EditFieldData;