2005-09-15 09:58:31 -04:00
|
|
|
/* Textarea form item handlers */
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE /* XXX: we want memrchr() ! */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "elinks.h"
|
|
|
|
|
|
|
|
#include "bfu/dialog.h"
|
|
|
|
#include "document/document.h"
|
|
|
|
#include "document/forms.h"
|
|
|
|
#include "document/html/parser.h"
|
|
|
|
#include "document/html/renderer.h"
|
|
|
|
#include "document/view.h"
|
|
|
|
#include "intl/gettext/libintl.h"
|
|
|
|
#include "session/session.h"
|
|
|
|
#include "terminal/draw.h"
|
|
|
|
#include "terminal/window.h"
|
|
|
|
#include "util/error.h"
|
|
|
|
#include "util/file.h"
|
|
|
|
#include "util/memory.h"
|
|
|
|
#include "util/string.h"
|
|
|
|
#include "viewer/action.h"
|
|
|
|
#include "viewer/text/form.h"
|
|
|
|
#include "viewer/text/textarea.h"
|
|
|
|
#include "viewer/text/view.h"
|
|
|
|
|
|
|
|
|
|
|
|
struct line_info {
|
|
|
|
int start;
|
|
|
|
int end;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* We add two extra entries to the table so the ending info can be added
|
|
|
|
* without reallocating. */
|
|
|
|
#define realloc_line_info(info, size) \
|
2006-02-17 12:32:59 -05:00
|
|
|
mem_align_alloc(info, size, (size) + 3, 0xFF)
|
2005-09-15 09:58:31 -04:00
|
|
|
|
2006-02-01 18:03:03 -05:00
|
|
|
/* Allocates a line_info table describing the layout of the textarea buffer.
|
|
|
|
*
|
|
|
|
* @width is max width and the offset at which text will be wrapped
|
|
|
|
* @wrap controls how the wrapping of text is performed
|
|
|
|
* @format is non zero the @text will be modified to make it suitable for
|
|
|
|
* encoding it for form posting
|
|
|
|
*/
|
|
|
|
static struct line_info *
|
|
|
|
format_textutf8(unsigned char *text, int width, enum form_wrap wrap, int format)
|
|
|
|
{
|
|
|
|
struct line_info *line = NULL;
|
|
|
|
int line_number = 0;
|
|
|
|
int begin = 0;
|
|
|
|
int pos = 0;
|
|
|
|
int skip;
|
|
|
|
unsigned char *wrappos=NULL;
|
|
|
|
int char_cnt=0; /* Number of console chars on line */
|
|
|
|
|
|
|
|
assert(text);
|
|
|
|
if_assert_failed return NULL;
|
|
|
|
|
|
|
|
/* Allocate the ending entries */
|
|
|
|
if (!realloc_line_info(&line, 0))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
while (text[pos]) {
|
|
|
|
|
|
|
|
if (text[pos] == ' ')
|
|
|
|
wrappos = &text[pos];
|
|
|
|
|
|
|
|
if (text[pos] == '\n') {
|
|
|
|
skip = 1;
|
|
|
|
|
|
|
|
} else if (wrap == FORM_WRAP_NONE || char_cnt < width) {
|
|
|
|
pos += utf8charlen(&text[pos]);
|
|
|
|
char_cnt++;
|
|
|
|
continue;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
if (wrappos) {
|
|
|
|
/* When formatting text for form submitting we
|
|
|
|
* have to apply the wrapping mode. */
|
|
|
|
if (wrap == FORM_WRAP_HARD && format)
|
|
|
|
*wrappos = '\n';
|
|
|
|
pos = wrappos - text;
|
|
|
|
}
|
|
|
|
skip = !!wrappos;
|
|
|
|
char_cnt = 0;
|
|
|
|
wrappos = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!realloc_line_info(&line, line_number)) {
|
|
|
|
mem_free_if(line);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
line[line_number].start = begin;
|
|
|
|
line[line_number++].end = pos;
|
|
|
|
begin = pos += skip;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush the last text before the loop ended */
|
|
|
|
line[line_number].start = begin;
|
|
|
|
line[line_number++].end = pos;
|
|
|
|
|
|
|
|
/* Add end marker */
|
|
|
|
line[line_number].start = line[line_number].end = -1;
|
|
|
|
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
|
2005-09-15 09:58:31 -04:00
|
|
|
/* Allocates a line_info table describing the layout of the textarea buffer.
|
|
|
|
*
|
|
|
|
* @width is max width and the offset at which text will be wrapped
|
|
|
|
* @wrap controls how the wrapping of text is performed
|
|
|
|
* @format is non zero the @text will be modified to make it suitable for
|
|
|
|
* encoding it for form posting
|
|
|
|
*/
|
|
|
|
static struct line_info *
|
|
|
|
format_text(unsigned char *text, int width, enum form_wrap wrap, int format)
|
|
|
|
{
|
|
|
|
struct line_info *line = NULL;
|
|
|
|
int line_number = 0;
|
|
|
|
int begin = 0;
|
|
|
|
int pos = 0;
|
|
|
|
int skip;
|
|
|
|
|
|
|
|
assert(text);
|
|
|
|
if_assert_failed return NULL;
|
|
|
|
|
|
|
|
/* Allocate the ending entries */
|
|
|
|
if (!realloc_line_info(&line, 0))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
while (text[pos]) {
|
|
|
|
if (text[pos] == '\n') {
|
|
|
|
skip = 1;
|
|
|
|
|
|
|
|
} else if (wrap == FORM_WRAP_NONE || pos - begin < width) {
|
|
|
|
pos++;
|
|
|
|
continue;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
unsigned char *wrappos;
|
|
|
|
|
|
|
|
/* Find a place to wrap the text */
|
|
|
|
wrappos = memrchr(&text[begin], ' ', pos - begin);
|
|
|
|
if (wrappos) {
|
|
|
|
/* When formatting text for form submitting we
|
|
|
|
* have to apply the wrapping mode. */
|
|
|
|
if (wrap == FORM_WRAP_HARD && format)
|
|
|
|
*wrappos = '\n';
|
|
|
|
pos = wrappos - text;
|
|
|
|
}
|
|
|
|
skip = !!wrappos;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!realloc_line_info(&line, line_number)) {
|
|
|
|
mem_free_if(line);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
line[line_number].start = begin;
|
|
|
|
line[line_number++].end = pos;
|
|
|
|
begin = pos += skip;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush the last text before the loop ended */
|
|
|
|
line[line_number].start = begin;
|
|
|
|
line[line_number++].end = pos;
|
|
|
|
|
|
|
|
/* Add end marker */
|
|
|
|
line[line_number].start = line[line_number].end = -1;
|
|
|
|
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Searches for @cursor_position (aka. position in the fs->value string) for
|
|
|
|
* the corresponding entry in the @line info. Returns the index or -1 if
|
|
|
|
* position is not found. */
|
|
|
|
static int
|
|
|
|
get_textarea_line_number(struct line_info *line, int cursor_position)
|
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
for (idx = 0; line[idx].start != -1; idx++) {
|
|
|
|
int wrap;
|
|
|
|
|
|
|
|
if (cursor_position < line[idx].start) continue;
|
|
|
|
|
|
|
|
wrap = (line[idx + 1].start == line[idx].end);
|
|
|
|
if (cursor_position >= line[idx].end + !wrap) continue;
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fixes up the vpos and vypos members of the form_state. Returns the
|
|
|
|
* logical position in the textarea view. */
|
|
|
|
int
|
2006-01-14 16:44:00 -05:00
|
|
|
area_cursor(struct form_control *fc, struct form_state *fs, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
struct line_info *line;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
assert(fc && fs);
|
|
|
|
if_assert_failed return 0;
|
|
|
|
|
2006-02-01 18:03:03 -05:00
|
|
|
if (utf8)
|
|
|
|
line = format_textutf8(fs->value, fc->cols, fc->wrap, 0);
|
|
|
|
else
|
|
|
|
line = format_text(fs->value, fc->cols, fc->wrap, 0);
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!line) return 0;
|
|
|
|
|
|
|
|
y = get_textarea_line_number(line, fs->state);
|
|
|
|
if (y == -1) {
|
|
|
|
mem_free(line);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-01-14 16:44:00 -05:00
|
|
|
if (utf8) {
|
|
|
|
unsigned char *text = fs->value + line[y].start;
|
|
|
|
unsigned char tmp = fs->value[fs->state];
|
|
|
|
|
|
|
|
fs->value[fs->state] = '\0';
|
|
|
|
x = strlen_utf8(&text);
|
|
|
|
fs->value[fs->state] = tmp;
|
|
|
|
} else
|
|
|
|
x = fs->state - line[y].start;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
mem_free(line);
|
|
|
|
|
|
|
|
if (fc->wrap && x == fc->cols) x--;
|
|
|
|
|
|
|
|
int_bounds(&fs->vpos, x - fc->cols + 1, x);
|
|
|
|
int_bounds(&fs->vypos, y - fc->rows + 1, y);
|
|
|
|
|
|
|
|
x -= fs->vpos;
|
|
|
|
y -= fs->vypos;
|
|
|
|
|
|
|
|
return y * fc->cols + x;
|
|
|
|
}
|
|
|
|
|
2006-01-14 16:44:00 -05:00
|
|
|
static void
|
|
|
|
draw_textarea_utf8(struct terminal *term, struct form_state *fs,
|
|
|
|
struct document_view *doc_view, struct link *link)
|
|
|
|
{
|
|
|
|
struct line_info *line, *linex;
|
|
|
|
struct form_control *fc;
|
|
|
|
struct box *box;
|
|
|
|
int vx, vy;
|
|
|
|
int sl, ye;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
assert(term && doc_view && doc_view->document && doc_view->vs && link);
|
|
|
|
if_assert_failed return;
|
|
|
|
fc = get_link_form_control(link);
|
|
|
|
assertm(fc, "link %d has no form control", (int) (link - doc_view->document->links));
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
box = &doc_view->box;
|
|
|
|
vx = doc_view->vs->x;
|
|
|
|
vy = doc_view->vs->y;
|
|
|
|
|
|
|
|
if (!link->npoints) return;
|
|
|
|
area_cursor(fc, fs, 1);
|
2006-02-01 18:03:03 -05:00
|
|
|
linex = format_textutf8(fs->value, fc->cols, fc->wrap, 0);
|
2006-01-14 16:44:00 -05:00
|
|
|
if (!linex) return;
|
|
|
|
line = linex;
|
|
|
|
sl = fs->vypos;
|
|
|
|
while (line->start != -1 && sl) sl--, line++;
|
|
|
|
|
|
|
|
x = link->points[0].x + box->x - vx;
|
|
|
|
y = link->points[0].y + box->y - vy;
|
|
|
|
ye = y + fc->rows;
|
|
|
|
|
|
|
|
for (; line->start != -1 && y < ye; line++, y++) {
|
|
|
|
int i;
|
|
|
|
unsigned char *text, *end;
|
|
|
|
|
|
|
|
text = fs->value + line->start;
|
|
|
|
end = fs->value + line->end;
|
|
|
|
|
|
|
|
for (i = 0; i < fs->vpos; i++)
|
|
|
|
utf_8_to_unicode(&text, end);
|
|
|
|
|
|
|
|
if (!row_is_in_box(box, y)) continue;
|
|
|
|
|
|
|
|
for (i = 0; i < fc->cols; i++) {
|
|
|
|
uint16_t data;
|
|
|
|
int xi = x + i;
|
|
|
|
|
|
|
|
if (!col_is_in_box(box, xi))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (i >= -fs->vpos
|
|
|
|
&& text < end)
|
|
|
|
data = utf_8_to_unicode(&text, end);
|
|
|
|
else
|
|
|
|
data = '_';
|
|
|
|
|
|
|
|
draw_char_data(term, xi, y, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; y < ye; y++) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!row_is_in_box(box, y)) continue;
|
|
|
|
|
|
|
|
for (i = 0; i < fc->cols; i++) {
|
|
|
|
int xi = x + i;
|
|
|
|
|
|
|
|
if (col_is_in_box(box, xi))
|
|
|
|
draw_char_data(term, xi, y, '_');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mem_free(linex);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-09-15 09:58:31 -04:00
|
|
|
void
|
|
|
|
draw_textarea(struct terminal *term, struct form_state *fs,
|
|
|
|
struct document_view *doc_view, struct link *link)
|
|
|
|
{
|
|
|
|
struct line_info *line, *linex;
|
|
|
|
struct form_control *fc;
|
|
|
|
struct box *box;
|
|
|
|
int vx, vy;
|
|
|
|
int sl, ye;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
assert(term && doc_view && doc_view->document && doc_view->vs && link);
|
|
|
|
if_assert_failed return;
|
2006-01-14 16:44:00 -05:00
|
|
|
|
|
|
|
if (term->utf8) {
|
|
|
|
draw_textarea_utf8(term, fs, doc_view, link);
|
|
|
|
return;
|
|
|
|
}
|
2005-09-15 09:58:31 -04:00
|
|
|
fc = get_link_form_control(link);
|
|
|
|
assertm(fc, "link %d has no form control", (int) (link - doc_view->document->links));
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
box = &doc_view->box;
|
|
|
|
vx = doc_view->vs->x;
|
|
|
|
vy = doc_view->vs->y;
|
|
|
|
|
|
|
|
if (!link->npoints) return;
|
2006-01-14 16:44:00 -05:00
|
|
|
area_cursor(fc, fs, 0);
|
2005-09-15 09:58:31 -04:00
|
|
|
linex = format_text(fs->value, fc->cols, fc->wrap, 0);
|
|
|
|
if (!linex) return;
|
|
|
|
line = linex;
|
|
|
|
sl = fs->vypos;
|
|
|
|
while (line->start != -1 && sl) sl--, line++;
|
|
|
|
|
|
|
|
x = link->points[0].x + box->x - vx;
|
|
|
|
y = link->points[0].y + box->y - vy;
|
|
|
|
ye = y + fc->rows;
|
|
|
|
|
|
|
|
for (; line->start != -1 && y < ye; line++, y++) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!row_is_in_box(box, y)) continue;
|
|
|
|
|
|
|
|
for (i = 0; i < fc->cols; i++) {
|
|
|
|
unsigned char data;
|
|
|
|
int xi = x + i;
|
|
|
|
|
|
|
|
if (!col_is_in_box(box, xi))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (i >= -fs->vpos
|
|
|
|
&& i + fs->vpos < line->end - line->start)
|
|
|
|
data = fs->value[line->start + i + fs->vpos];
|
|
|
|
else
|
|
|
|
data = '_';
|
|
|
|
|
|
|
|
draw_char_data(term, xi, y, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; y < ye; y++) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!row_is_in_box(box, y)) continue;
|
|
|
|
|
|
|
|
for (i = 0; i < fc->cols; i++) {
|
|
|
|
int xi = x + i;
|
|
|
|
|
|
|
|
if (col_is_in_box(box, xi))
|
|
|
|
draw_char_data(term, xi, y, '_');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mem_free(linex);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned char *
|
|
|
|
encode_textarea(struct submitted_value *sv)
|
|
|
|
{
|
|
|
|
struct form_control *fc;
|
|
|
|
struct string newtext;
|
|
|
|
void *blabla;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
assert(sv && sv->value);
|
|
|
|
if_assert_failed return NULL;
|
|
|
|
|
|
|
|
fc = sv->form_control;
|
|
|
|
|
|
|
|
/* We need to reformat text now if it has to be wrapped hard, just
|
|
|
|
* before encoding it. */
|
2006-02-01 18:03:03 -05:00
|
|
|
/* TODO: Do we need here UTF-8 format or not? --scrool */
|
2005-09-15 09:58:31 -04:00
|
|
|
blabla = format_text(sv->value, fc->cols, fc->wrap, 1);
|
|
|
|
mem_free_if(blabla);
|
|
|
|
|
|
|
|
if (!init_string(&newtext)) return NULL;
|
|
|
|
|
|
|
|
for (i = 0; sv->value[i]; i++) {
|
|
|
|
if (sv->value[i] != '\n')
|
|
|
|
add_char_to_string(&newtext, sv->value[i]);
|
|
|
|
else
|
|
|
|
add_crlf_to_string(&newtext);
|
|
|
|
}
|
|
|
|
|
|
|
|
return newtext.source;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* We use some evil hacking in order to make external textarea editor working.
|
|
|
|
* We need to have some way how to be notified that the editor finished and we
|
|
|
|
* should reload content of the textarea. So we use global variable
|
|
|
|
* textarea_editor as a flag whether we have one running, and if we have, we
|
|
|
|
* just call textarea_edit(1, ...). Then we recover our state from static
|
|
|
|
* variables, reload content of textarea back from file and clean up.
|
|
|
|
*
|
|
|
|
* Unfortunately, we can't support calling of editor from non-master links
|
|
|
|
* session, as it would be extremely ugly to hack (you would have to transfer
|
|
|
|
* the content of it back to master somehow, add special flags for not deleting
|
|
|
|
* of 'delete' etc) and I'm not going to do that now. Inter-links communication
|
|
|
|
* *NEEDS* rewrite, as it looks just like quick messy hack now. --pasky */
|
|
|
|
|
|
|
|
int textarea_editor = 0;
|
|
|
|
|
|
|
|
static unsigned char *
|
|
|
|
save_textarea_file(unsigned char *value)
|
|
|
|
{
|
|
|
|
unsigned char *filename;
|
|
|
|
FILE *file = NULL;
|
|
|
|
int h;
|
|
|
|
|
|
|
|
filename = get_tempdir_filename("elinks-area-XXXXXX");
|
|
|
|
if (!filename) return NULL;
|
|
|
|
|
|
|
|
h = safe_mkstemp(filename);
|
|
|
|
if (h >= 0) file = fdopen(h, "w");
|
|
|
|
|
|
|
|
if (file) {
|
|
|
|
fwrite(value, strlen(value), 1, file);
|
|
|
|
fclose(file);
|
|
|
|
} else {
|
|
|
|
mem_free(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
return filename;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
textarea_edit(int op, struct terminal *term_, struct form_state *fs_,
|
|
|
|
struct document_view *doc_view_, struct link *link_)
|
|
|
|
{
|
|
|
|
static size_t fc_maxlength;
|
|
|
|
static struct form_state *fs;
|
|
|
|
static struct terminal *term;
|
|
|
|
static struct document_view *doc_view;
|
|
|
|
static struct link *link;
|
|
|
|
static unsigned char *fn;
|
|
|
|
|
|
|
|
assert (op == 0 || op == 1);
|
|
|
|
if_assert_failed return;
|
|
|
|
assert (op == 1 || term_);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
if (op == 0 && get_cmd_opt_bool("anonymous")) {
|
|
|
|
info_box(term_, 0, N_("Error"), ALIGN_CENTER,
|
|
|
|
N_("You cannot launch an external"
|
|
|
|
" editor in the anonymous mode."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (op == 0 && !term_->master) {
|
|
|
|
info_box(term_, 0, N_("Error"), ALIGN_CENTER,
|
|
|
|
N_("You can do this only on the master terminal"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (op == 0 && !textarea_editor) {
|
|
|
|
unsigned char *ed = get_opt_str("document.browse.forms.editor");
|
|
|
|
unsigned char *ex;
|
|
|
|
|
|
|
|
fn = save_textarea_file(fs_->value);
|
|
|
|
if (!fn) return;
|
|
|
|
|
|
|
|
if (!ed || !*ed) {
|
|
|
|
ed = getenv("EDITOR");
|
|
|
|
if (!ed || !*ed) ed = "vi";
|
|
|
|
}
|
|
|
|
|
|
|
|
ex = straconcat(ed, " ", fn, NULL);
|
|
|
|
if (!ex) {
|
|
|
|
unlink(fn);
|
|
|
|
goto free_and_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fs_) fs = fs_;
|
|
|
|
if (doc_view_) doc_view = doc_view_;
|
|
|
|
if (link_) {
|
|
|
|
link = link_;
|
|
|
|
fc_maxlength = get_link_form_control(link_)->maxlength;
|
|
|
|
}
|
|
|
|
if (term_) term = term_;
|
|
|
|
|
|
|
|
exec_on_terminal(term, ex, "", 1);
|
|
|
|
mem_free(ex);
|
|
|
|
|
|
|
|
textarea_editor = 1;
|
|
|
|
|
|
|
|
} else if (op == 1 && fs) {
|
2005-12-18 11:00:48 -05:00
|
|
|
struct string file;
|
|
|
|
|
|
|
|
if (!init_string(&file)
|
|
|
|
|| !add_file_to_string(&file, fn)) {
|
|
|
|
textarea_editor = 0;
|
|
|
|
goto free_and_return;
|
|
|
|
}
|
2005-09-15 09:58:31 -04:00
|
|
|
|
2005-12-18 11:00:48 -05:00
|
|
|
if (file.length > fc_maxlength) {
|
2005-12-18 11:23:52 -05:00
|
|
|
file.source[fc_maxlength] = '\0';
|
|
|
|
info_box(term, MSGBOX_FREE_TEXT, N_("Warning"),
|
|
|
|
ALIGN_CENTER,
|
|
|
|
msg_text(term,
|
|
|
|
N_("You have exceeded the textarea's"
|
|
|
|
" size limit: your input is %d"
|
|
|
|
" bytes, but the maximum is %u"
|
|
|
|
" bytes.\n\n"
|
|
|
|
"Your input has been truncated,"
|
|
|
|
" but you can still recover the"
|
|
|
|
" text that you entered from"
|
|
|
|
" this file: %s"), file.length,
|
|
|
|
fc_maxlength, fn));
|
|
|
|
} else {
|
|
|
|
unlink(fn);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
2005-12-18 11:00:48 -05:00
|
|
|
mem_free(fs->value);
|
|
|
|
fs->value = file.source;
|
|
|
|
fs->state = file.length;
|
|
|
|
|
|
|
|
if (doc_view && link)
|
|
|
|
draw_form_entry(term, doc_view, link);
|
|
|
|
|
2005-09-15 09:58:31 -04:00
|
|
|
textarea_editor = 0;
|
|
|
|
goto free_and_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
free_and_return:
|
|
|
|
mem_free_set(&fn, NULL);
|
|
|
|
fs = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* menu_func_T */
|
|
|
|
void
|
|
|
|
menu_textarea_edit(struct terminal *term, void *xxx, void *ses_)
|
|
|
|
{
|
|
|
|
struct session *ses = ses_;
|
|
|
|
struct document_view *doc_view;
|
|
|
|
struct link *link;
|
|
|
|
struct form_state *fs;
|
|
|
|
struct form_control *fc;
|
|
|
|
|
|
|
|
assert(term && ses);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
doc_view = current_frame(ses);
|
|
|
|
|
|
|
|
assert(doc_view && doc_view->vs && doc_view->document);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
link = get_current_link(doc_view);
|
|
|
|
if (!link) return;
|
|
|
|
|
|
|
|
fc = get_link_form_control(link);
|
|
|
|
if (form_field_is_readonly(fc))
|
|
|
|
return;
|
|
|
|
|
|
|
|
fs = find_form_state(doc_view, fc);
|
|
|
|
if (!fs) return;
|
|
|
|
|
|
|
|
textarea_edit(0, term, fs, doc_view, link);
|
|
|
|
}
|
|
|
|
|
|
|
|
static enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op(struct form_state *fs, struct form_control *fc, int utf8,
|
|
|
|
int (*do_op)(struct form_state *, struct line_info *, int, int))
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
struct line_info *line;
|
|
|
|
int current, state;
|
|
|
|
|
|
|
|
assert(fs && fs->value && fc);
|
|
|
|
if_assert_failed return FRAME_EVENT_OK;
|
|
|
|
|
2006-02-01 18:03:03 -05:00
|
|
|
if (utf8)
|
|
|
|
line = format_textutf8(fs->value, fc->cols, fc->wrap, 0);
|
|
|
|
else
|
|
|
|
line = format_text(fs->value, fc->cols, fc->wrap, 0);
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!line) return FRAME_EVENT_OK;
|
|
|
|
|
|
|
|
current = get_textarea_line_number(line, fs->state);
|
|
|
|
state = fs->state;
|
2006-01-14 16:44:00 -05:00
|
|
|
if (do_op(fs, line, current, utf8)) {
|
2005-09-15 09:58:31 -04:00
|
|
|
mem_free(line);
|
|
|
|
return FRAME_EVENT_IGNORED;
|
|
|
|
}
|
|
|
|
|
|
|
|
mem_free(line);
|
|
|
|
return fs->state == state ? FRAME_EVENT_OK : FRAME_EVENT_REFRESH;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2006-01-14 16:44:00 -05:00
|
|
|
x_pos(struct form_state *fs, struct line_info *line, int current)
|
|
|
|
{
|
|
|
|
unsigned char *text = fs->value + line[current].start;
|
|
|
|
unsigned char tmp = fs->value[fs->state];
|
|
|
|
int len;
|
|
|
|
|
|
|
|
fs->value[fs->state] = '\0';
|
|
|
|
len = strlen_utf8(&text);
|
|
|
|
fs->value[fs->state] = tmp;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
new_pos(struct form_state *fs, struct line_info *line, int current, int len)
|
|
|
|
{
|
|
|
|
unsigned char *text = fs->value + line[current].start;
|
|
|
|
unsigned char *end = fs->value + line[current].end;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
unicode_val_T data = utf_8_to_unicode(&text, end);
|
|
|
|
|
|
|
|
if (data == UCS_NO_CHAR) break;
|
|
|
|
}
|
|
|
|
fs->state = (int)(text - fs->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
do_op_home(struct form_state *fs, struct line_info *line, int current, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
if (current != -1) fs->state = line[current].start;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2006-01-14 16:44:00 -05:00
|
|
|
do_op_up(struct form_state *fs, struct line_info *line, int current, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
if (current == -1) return 0;
|
|
|
|
if (!current) return 1;
|
|
|
|
|
2006-01-14 16:44:00 -05:00
|
|
|
if (utf8) {
|
|
|
|
int len = x_pos(fs, line, current);
|
|
|
|
|
|
|
|
new_pos(fs, line, current - 1, len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-09-15 09:58:31 -04:00
|
|
|
fs->state -= line[current].start - line[current-1].start;
|
|
|
|
int_upper_bound(&fs->state, line[current-1].end);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2006-01-14 16:44:00 -05:00
|
|
|
do_op_down(struct form_state *fs, struct line_info *line, int current, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
if (current == -1) return 0;
|
|
|
|
if (line[current+1].start == -1) return 1;
|
|
|
|
|
2006-01-14 16:44:00 -05:00
|
|
|
if (utf8) {
|
|
|
|
int len = x_pos(fs, line, current);
|
|
|
|
|
|
|
|
new_pos(fs, line, current + 1, len);
|
|
|
|
return 0;
|
|
|
|
}
|
2005-09-15 09:58:31 -04:00
|
|
|
fs->state += line[current+1].start - line[current].start;
|
|
|
|
int_upper_bound(&fs->state, line[current+1].end);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2006-01-14 16:44:00 -05:00
|
|
|
do_op_end(struct form_state *fs, struct line_info *line, int current, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
if (current == -1) {
|
|
|
|
fs->state = strlen(fs->value);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
int wrap = line[current + 1].start == line[current].end;
|
|
|
|
|
|
|
|
/* Don't jump to next line when wrapping. */
|
|
|
|
fs->state = int_max(0, line[current].end - wrap);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2006-01-14 16:44:00 -05:00
|
|
|
do_op_bob(struct form_state *fs, struct line_info *line, int current, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
if (current == -1) return 0;
|
|
|
|
|
|
|
|
fs->state -= line[current].start;
|
|
|
|
int_upper_bound(&fs->state, line[0].end);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2006-01-14 16:44:00 -05:00
|
|
|
do_op_eob(struct form_state *fs, struct line_info *line, int current, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
if (current == -1) {
|
|
|
|
fs->state = strlen(fs->value);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
int last = get_textarea_line_number(line, strlen(fs->value));
|
|
|
|
|
|
|
|
assertm(last != -1, "line info corrupt");
|
|
|
|
|
|
|
|
fs->state += line[last].start - line[current].start;
|
|
|
|
int_upper_bound(&fs->state, line[last].end);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_home(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
2006-01-14 16:44:00 -05:00
|
|
|
return textarea_op(fs, fc, utf8, do_op_home);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_up(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
2006-01-14 16:44:00 -05:00
|
|
|
return textarea_op(fs, fc, utf8, do_op_up);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_down(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
2006-01-14 16:44:00 -05:00
|
|
|
return textarea_op(fs, fc, utf8, do_op_down);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_end(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
2006-01-14 16:44:00 -05:00
|
|
|
return textarea_op(fs, fc, utf8, do_op_end);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the form state so the cursor is on the first line of the buffer.
|
|
|
|
* Preserve the column if possible. */
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_bob(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
2006-01-14 16:44:00 -05:00
|
|
|
return textarea_op(fs, fc, utf8, do_op_bob);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the form state so the cursor is on the last line of the buffer. Preserve
|
|
|
|
* the column if possible. This is done by getting current and last line and
|
|
|
|
* then shifting the state by the delta of both lines start position bounding
|
|
|
|
* the whole thing to the end of the last line. */
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_eob(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
2006-01-14 16:44:00 -05:00
|
|
|
return textarea_op(fs, fc, utf8, do_op_eob);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
enum frame_event_status
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_enter(struct form_state *fs, struct form_control *fc, int utf8)
|
2005-09-15 09:58:31 -04:00
|
|
|
{
|
|
|
|
assert(fs && fs->value && fc);
|
|
|
|
if_assert_failed return FRAME_EVENT_OK;
|
|
|
|
|
|
|
|
if (form_field_is_readonly(fc)
|
|
|
|
|| strlen(fs->value) >= fc->maxlength
|
|
|
|
|| !insert_in_string(&fs->value, fs->state, "\n", 1))
|
|
|
|
return FRAME_EVENT_OK;
|
|
|
|
|
|
|
|
fs->state++;
|
|
|
|
return FRAME_EVENT_REFRESH;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
set_textarea(struct document_view *doc_view, int direction)
|
|
|
|
{
|
|
|
|
struct form_control *fc;
|
|
|
|
struct form_state *fs;
|
|
|
|
struct link *link;
|
2006-01-14 16:44:00 -05:00
|
|
|
int utf8 = doc_view->document->options.utf8;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
assert(doc_view && doc_view->vs && doc_view->document);
|
|
|
|
assert(direction == 1 || direction == -1);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
link = get_current_link(doc_view);
|
|
|
|
if (!link || link->type != LINK_AREA)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fc = get_link_form_control(link);
|
|
|
|
assertm(fc, "link has no form control");
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
if (fc->mode == FORM_MODE_DISABLED) return;
|
|
|
|
|
|
|
|
fs = find_form_state(doc_view, fc);
|
|
|
|
if (!fs || !fs->value) return;
|
|
|
|
|
|
|
|
/* Depending on which way we entered the textarea move cursor so that
|
|
|
|
* it is available at end or start. */
|
|
|
|
if (direction == 1)
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_eob(fs, fc, utf8);
|
2005-09-15 09:58:31 -04:00
|
|
|
else
|
2006-01-14 16:44:00 -05:00
|
|
|
textarea_op_bob(fs, fc, utf8);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|