about summary refs log tree commit diff stats
path: root/src/GridText.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/GridText.c')
-rw-r--r--src/GridText.c301
1 files changed, 243 insertions, 58 deletions
diff --git a/src/GridText.c b/src/GridText.c
index 4b2b763d..9ec1c6bd 100644
--- a/src/GridText.c
+++ b/src/GridText.c
@@ -1,5 +1,5 @@
 /*
- * $LynxId: GridText.c,v 1.176 2009/11/27 13:18:18 tom Exp $
+ * $LynxId: GridText.c,v 1.178 2010/04/25 20:22:51 tom Exp $
  *
  *		Character grid hypertext object
  *		===============================
@@ -6072,6 +6072,63 @@ HTChildAnchor *HText_childNextNumber(int number, void **prev)
     return a->anchor;
 }
 
+static const char *inputFieldDesc(FormInfo * input)
+{
+    const char *result = 0;
+
+    switch (input->type) {
+    case F_TEXT_TYPE:
+	result = gettext("text entry field");
+	break;
+    case F_PASSWORD_TYPE:
+	result = gettext("password entry field");
+	break;
+    case F_CHECKBOX_TYPE:
+	result = gettext("checkbox");
+	break;
+    case F_RADIO_TYPE:
+	result = gettext("radio button");
+	break;
+    case F_SUBMIT_TYPE:
+	result = gettext("submit button");
+	break;
+    case F_RESET_TYPE:
+	result = gettext("reset button");
+	break;
+    case F_BUTTON_TYPE:
+	result = gettext("script button");
+	break;
+    case F_OPTION_LIST_TYPE:
+	result = gettext("popup menu");
+	break;
+    case F_HIDDEN_TYPE:
+	result = gettext("hidden form field");
+	break;
+    case F_TEXTAREA_TYPE:
+	result = gettext("text entry area");
+	break;
+    case F_RANGE_TYPE:
+	result = gettext("range entry field");
+	break;
+    case F_FILE_TYPE:
+	result = gettext("file entry field");
+	break;
+    case F_TEXT_SUBMIT_TYPE:
+	result = gettext("text-submit field");
+	break;
+    case F_IMAGE_SUBMIT_TYPE:
+	result = gettext("image-submit button");
+	break;
+    case F_KEYGEN_TYPE:
+	result = gettext("keygen field");
+	break;
+    default:
+	result = gettext("unknown form field");
+	break;
+    }
+    return result;
+}
+
 /*
  * HText_FormDescNumber() returns a description of the form field
  * with index N.  The index corresponds to the [number] we print
@@ -6100,56 +6157,7 @@ void HText_FormDescNumber(int number,
 	}
     }
 
-    switch (a->input_field->type) {
-    case F_TEXT_TYPE:
-	*desc = gettext("text entry field");
-	return;
-    case F_PASSWORD_TYPE:
-	*desc = gettext("password entry field");
-	return;
-    case F_CHECKBOX_TYPE:
-	*desc = gettext("checkbox");
-	return;
-    case F_RADIO_TYPE:
-	*desc = gettext("radio button");
-	return;
-    case F_SUBMIT_TYPE:
-	*desc = gettext("submit button");
-	return;
-    case F_RESET_TYPE:
-	*desc = gettext("reset button");
-	return;
-    case F_BUTTON_TYPE:
-	*desc = gettext("script button");
-	return;
-    case F_OPTION_LIST_TYPE:
-	*desc = gettext("popup menu");
-	return;
-    case F_HIDDEN_TYPE:
-	*desc = gettext("hidden form field");
-	return;
-    case F_TEXTAREA_TYPE:
-	*desc = gettext("text entry area");
-	return;
-    case F_RANGE_TYPE:
-	*desc = gettext("range entry field");
-	return;
-    case F_FILE_TYPE:
-	*desc = gettext("file entry field");
-	return;
-    case F_TEXT_SUBMIT_TYPE:
-	*desc = gettext("text-submit field");
-	return;
-    case F_IMAGE_SUBMIT_TYPE:
-	*desc = gettext("image-submit button");
-	return;
-    case F_KEYGEN_TYPE:
-	*desc = gettext("keygen field");
-	return;
-    default:
-	*desc = gettext("unknown form field");
-	return;
-    }
+    *desc = inputFieldDesc(a->input_field);
 }
 
 /* HTGetRelLinkNum returns the anchor number to which follow_link_number()
@@ -7840,6 +7848,150 @@ static int TrimmedLength(char *string)
     return result;
 }
 
+typedef struct _AnchorIndex {
+    struct _AnchorIndex *next;
+    int type;			/* field type */
+    int size;			/* character-width of field */
+    int length;			/* byte-count for field's data */
+    int offset;			/* byte-offset in line's data */
+    int filler;			/* character to use for filler */
+    const char *value;		/* field's value */
+} AnchorIndex;
+
+static unsigned countHTLines(void)
+{
+    unsigned result = 0;
+    HTLine *line = FirstHTLine(HTMainText);
+
+    while (line != 0) {
+	++result;
+	if (line == HTMainText->last_line)
+	    break;
+	line = line->next;
+    }
+    CTRACE((tfp, "countHTLines %u\n", result));
+    return result;
+}
+
+/*
+ * The TextAnchor list is not organized to allow efficient dumping of a page.
+ * Make an array with one item per line of the page, and store (by byte-offset)
+ * pointers to the TextAnchor's we want to use.
+ */
+static AnchorIndex **allocAnchorIndex(unsigned *size)
+{
+    AnchorIndex **result = NULL;
+    AnchorIndex *p, *q;
+    TextAnchor *anchor = NULL;
+    FormInfo *input = NULL;
+
+    *size = countHTLines();
+    if (*size != 0) {
+	result = typecallocn(AnchorIndex *, *size + 1);
+
+	for (anchor = HTMainText->first_anchor;
+	     anchor != NULL;
+	     anchor = anchor->next) {
+
+	    if (anchor->link_type == INPUT_ANCHOR
+		&& anchor->show_anchor
+		&& anchor->line_num < (int) *size
+		&& (input = anchor->input_field) != NULL) {
+		CTRACE2(TRACE_GRIDTEXT,
+			(tfp, "line %d.%d %d %s->%s(%s)\n",
+			 anchor->line_num,
+			 anchor->line_pos,
+			 input->size,
+			 inputFieldDesc(input),
+			 input->value,
+			 input->orig_value));
+		switch (input->type) {
+		case F_TEXT_TYPE:
+		case F_PASSWORD_TYPE:
+		case F_CHECKBOX_TYPE:
+		case F_RADIO_TYPE:
+		case F_OPTION_LIST_TYPE:
+		case F_TEXTAREA_TYPE:
+		case F_RANGE_TYPE:
+		case F_FILE_TYPE:
+		    p = typecalloc(AnchorIndex);
+		    if (p == NULL)
+			outofmem(__FILE__, "allocAnchorIndex");
+
+		    p->type = input->type;
+		    p->size = input->size;
+		    p->offset = anchor->line_pos;
+		    p->value = input->value;
+
+		    switch (input->type) {
+		    case F_TEXT_TYPE:
+		    case F_PASSWORD_TYPE:
+			p->filler = '_';
+			break;
+		    case F_OPTION_LIST_TYPE:
+			p->filler = '_';
+			break;
+		    case F_CHECKBOX_TYPE:
+			p->value = (input->num_value
+				    ? checked_box
+				    : unchecked_box);
+			break;
+		    case F_RADIO_TYPE:
+			p->value = (input->num_value
+				    ? checked_radio
+				    : unchecked_radio);
+			break;
+		    default:
+			p->filler = ' ';
+			break;
+		    }
+		    p->length = strlen(input->value);
+
+		    if ((q = result[anchor->line_num]) != NULL) {
+			/* insert, ordering by offset */
+			if (q->offset < p->offset) {
+			    while (q->next != NULL
+				   && q->next->offset < p->offset) {
+				q = q->next;
+			    }
+			    p->next = q->next;
+			    q->next = p;
+			} else {
+			    p->next = q;
+			    result[anchor->line_num] = p;
+			}
+		    } else {
+			result[anchor->line_num] = p;
+		    }
+		    break;
+		}
+	    }
+	}
+    }
+    return result;
+}
+
+/*
+ * Free the data allocated in allocAnchorIndex().
+ */
+static void freeAnchorIndex(AnchorIndex ** inx, unsigned inx_size)
+{
+    AnchorIndex *cur;
+    unsigned num;
+
+    if (inx != 0) {
+	if (inx_size != 0) {
+	    for (num = 0; num < inx_size; ++num) {
+		while ((cur = inx[num]) != NULL) {
+		    inx[num] = cur->next;
+		    free(cur);
+		}
+	    }
+	}
+	free(inx);
+    }
+}
+
 /*
  * Print the contents of the file in HTMainText to
  * the file descriptor fp.
@@ -7851,10 +8003,12 @@ void print_wwwfile_to_fd(FILE *fp,
 			 BOOLEAN is_email,
 			 BOOLEAN is_reply)
 {
-    register int i;
+    int line_num, byte_num, byte_count;
     int first = TRUE;
-    int limit;
     HTLine *line;
+    AnchorIndex **inx;
+    AnchorIndex *cur;
+    unsigned inx_size;
 
 #ifndef NO_DUMP_WITH_BACKSPACES
     HText *text = HTMainText;
@@ -7870,8 +8024,16 @@ void print_wwwfile_to_fd(FILE *fp,
     if (!HTMainText)
 	return;
 
+    /*
+     * Build an index of anchors for each line, so we can override the
+     * static text which is stored in the list of HTLine's.
+     */
+    inx = allocAnchorIndex(&inx_size);
+
     line = FirstHTLine(HTMainText);
-    for (;; line = line->next) {
+    for (line_num = 0;; ++line_num, line = line->next) {
+	cur = inx[line_num];
+
 	if (first) {
 	    first = FALSE;
 	    if (is_reply) {
@@ -7896,9 +8058,31 @@ void print_wwwfile_to_fd(FILE *fp,
 	/*
 	 * Add data.
 	 */
-	limit = TrimmedLength(line->data);
-	for (i = 0; i < limit; i++) {
-	    int ch = UCH(line->data[i]);
+	byte_count = TrimmedLength(line->data);
+	for (byte_num = 0; byte_num < byte_count; byte_num++) {
+	    int offset = byte_num + line->offset;
+	    int ch = UCH(line->data[byte_num]);
+
+	    while (cur != 0 && (cur->offset + cur->size) < offset) {
+		cur = cur->next;
+	    }
+	    if (cur != 0
+		&& cur->offset <= offset
+		&& cur->offset + cur->size > offset) {
+		int off2 = offset - cur->offset;
+		int c2 = ((off2 < cur->length)
+			  ? cur->value[off2]
+			  : cur->filler);
+
+		if (ch != c2) {
+		    CTRACE2(TRACE_GRIDTEXT,
+			    (tfp, "line %d [%d..%d] map %d %c->%c\n",
+			     line_num,
+			     cur->offset, cur->offset + cur->size - 1,
+			     offset, ch, c2));
+		    ch = c2;
+		}
+	    }
 
 	    if (!IsSpecialAttrChar(ch)) {
 #ifndef NO_DUMP_WITH_BACKSPACES
@@ -7914,7 +8098,7 @@ void print_wwwfile_to_fd(FILE *fp,
 #endif
 		    fputc(ch, fp);
 	    } else if (ch == LY_SOFT_HYPHEN &&
-		       (i + 1) >= limit) {	/* last char on line */
+		       (byte_num + 1) >= byte_count) {
 		write_hyphen(fp);
 	    } else if (dump_output_immediately && use_underscore) {
 		switch (ch) {
@@ -7960,6 +8144,7 @@ void print_wwwfile_to_fd(FILE *fp,
     }
     fputc('\n', fp);
 
+    freeAnchorIndex(inx, inx_size);
 }
 
 /*