tweaks: convert the indentation to use only tabs
Each leading tab is converted to two tabs, and any leading four spaces is converted to one tab. The intended tab size (for keeping most lines within 80 columns) is now four.master
parent
b574f73e60
commit
87206c0607
742
configure.ac
742
configure.ac
File diff suppressed because it is too large
Load Diff
1136
src/browser.c
1136
src/browser.c
File diff suppressed because it is too large
Load Diff
762
src/chars.c
762
src/chars.c
File diff suppressed because it is too large
Load Diff
576
src/color.c
576
src/color.c
|
@ -33,7 +33,7 @@
|
||||||
/* For early versions of ncurses-6.0, use an additional A_PROTECT attribute
|
/* For early versions of ncurses-6.0, use an additional A_PROTECT attribute
|
||||||
* for all colors, in order to work around an ncurses miscoloring bug. */
|
* for all colors, in order to work around an ncurses miscoloring bug. */
|
||||||
#if defined(NCURSES_VERSION_MAJOR) && (NCURSES_VERSION_MAJOR == 6) && \
|
#if defined(NCURSES_VERSION_MAJOR) && (NCURSES_VERSION_MAJOR == 6) && \
|
||||||
(NCURSES_VERSION_MINOR == 0) && (NCURSES_VERSION_PATCH < 20151017)
|
(NCURSES_VERSION_MINOR == 0) && (NCURSES_VERSION_PATCH < 20151017)
|
||||||
#define A_BANDAID A_PROTECT
|
#define A_BANDAID A_PROTECT
|
||||||
#else
|
#else
|
||||||
#define A_BANDAID A_NORMAL
|
#define A_BANDAID A_NORMAL
|
||||||
|
@ -43,394 +43,394 @@
|
||||||
* for the colors in each syntax. */
|
* for the colors in each syntax. */
|
||||||
void set_colorpairs(void)
|
void set_colorpairs(void)
|
||||||
{
|
{
|
||||||
const syntaxtype *sint;
|
const syntaxtype *sint;
|
||||||
bool using_defaults = FALSE;
|
bool using_defaults = FALSE;
|
||||||
short foreground, background;
|
short foreground, background;
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
/* Tell ncurses to enable colors. */
|
/* Tell ncurses to enable colors. */
|
||||||
start_color();
|
start_color();
|
||||||
|
|
||||||
#ifdef HAVE_USE_DEFAULT_COLORS
|
#ifdef HAVE_USE_DEFAULT_COLORS
|
||||||
/* Allow using the default colors, if available. */
|
/* Allow using the default colors, if available. */
|
||||||
using_defaults = (use_default_colors() != ERR);
|
using_defaults = (use_default_colors() != ERR);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Initialize the color pairs for nano's interface elements. */
|
/* Initialize the color pairs for nano's interface elements. */
|
||||||
for (i = 0; i < NUMBER_OF_ELEMENTS; i++) {
|
for (i = 0; i < NUMBER_OF_ELEMENTS; i++) {
|
||||||
bool bright = FALSE;
|
bool bright = FALSE;
|
||||||
|
|
||||||
if (specified_color_combo[i] != NULL &&
|
if (specified_color_combo[i] != NULL &&
|
||||||
parse_color_names(specified_color_combo[i],
|
parse_color_names(specified_color_combo[i],
|
||||||
&foreground, &background, &bright)) {
|
&foreground, &background, &bright)) {
|
||||||
if (foreground == -1 && !using_defaults)
|
if (foreground == -1 && !using_defaults)
|
||||||
foreground = COLOR_WHITE;
|
foreground = COLOR_WHITE;
|
||||||
if (background == -1 && !using_defaults)
|
if (background == -1 && !using_defaults)
|
||||||
background = COLOR_BLACK;
|
background = COLOR_BLACK;
|
||||||
init_pair(i + 1, foreground, background);
|
init_pair(i + 1, foreground, background);
|
||||||
interface_color_pair[i] = COLOR_PAIR(i + 1) | A_BANDAID |
|
interface_color_pair[i] = COLOR_PAIR(i + 1) | A_BANDAID |
|
||||||
(bright ? A_BOLD : A_NORMAL);
|
(bright ? A_BOLD : A_NORMAL);
|
||||||
} else {
|
} else {
|
||||||
if (i != FUNCTION_TAG)
|
if (i != FUNCTION_TAG)
|
||||||
interface_color_pair[i] = hilite_attribute;
|
interface_color_pair[i] = hilite_attribute;
|
||||||
else
|
else
|
||||||
interface_color_pair[i] = A_NORMAL;
|
interface_color_pair[i] = A_NORMAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(specified_color_combo[i]);
|
||||||
|
specified_color_combo[i] = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
free(specified_color_combo[i]);
|
/* For each syntax, go through its list of colors and assign each
|
||||||
specified_color_combo[i] = NULL;
|
* its pair number, giving identical color pairs the same number. */
|
||||||
}
|
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
||||||
|
colortype *ink;
|
||||||
|
int new_number = NUMBER_OF_ELEMENTS + 1;
|
||||||
|
|
||||||
/* For each syntax, go through its list of colors and assign each
|
for (ink = sint->color; ink != NULL; ink = ink->next) {
|
||||||
* its pair number, giving identical color pairs the same number. */
|
const colortype *beforenow = sint->color;
|
||||||
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
|
||||||
colortype *ink;
|
|
||||||
int new_number = NUMBER_OF_ELEMENTS + 1;
|
|
||||||
|
|
||||||
for (ink = sint->color; ink != NULL; ink = ink->next) {
|
while (beforenow != ink && (beforenow->fg != ink->fg ||
|
||||||
const colortype *beforenow = sint->color;
|
beforenow->bg != ink->bg ||
|
||||||
|
beforenow->bright != ink->bright))
|
||||||
|
beforenow = beforenow->next;
|
||||||
|
|
||||||
while (beforenow != ink && (beforenow->fg != ink->fg ||
|
if (beforenow != ink)
|
||||||
beforenow->bg != ink->bg ||
|
ink->pairnum = beforenow->pairnum;
|
||||||
beforenow->bright != ink->bright))
|
else
|
||||||
beforenow = beforenow->next;
|
ink->pairnum = new_number++;
|
||||||
|
|
||||||
if (beforenow != ink)
|
ink->attributes = COLOR_PAIR(ink->pairnum) | A_BANDAID |
|
||||||
ink->pairnum = beforenow->pairnum;
|
(ink->bright ? A_BOLD : A_NORMAL);
|
||||||
else
|
}
|
||||||
ink->pairnum = new_number++;
|
|
||||||
|
|
||||||
ink->attributes = COLOR_PAIR(ink->pairnum) | A_BANDAID |
|
|
||||||
(ink->bright ? A_BOLD : A_NORMAL);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Initialize the color information. */
|
/* Initialize the color information. */
|
||||||
void color_init(void)
|
void color_init(void)
|
||||||
{
|
{
|
||||||
const colortype *ink;
|
const colortype *ink;
|
||||||
bool using_defaults = FALSE;
|
bool using_defaults = FALSE;
|
||||||
short foreground, background;
|
short foreground, background;
|
||||||
|
|
||||||
/* If the terminal is not capable of colors, forget it. */
|
/* If the terminal is not capable of colors, forget it. */
|
||||||
if (!has_colors())
|
if (!has_colors())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
#ifdef HAVE_USE_DEFAULT_COLORS
|
#ifdef HAVE_USE_DEFAULT_COLORS
|
||||||
/* Allow using the default colors, if available. */
|
/* Allow using the default colors, if available. */
|
||||||
using_defaults = (use_default_colors() != ERR);
|
using_defaults = (use_default_colors() != ERR);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* For each coloring expression, initialize the color pair. */
|
/* For each coloring expression, initialize the color pair. */
|
||||||
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
||||||
foreground = ink->fg;
|
foreground = ink->fg;
|
||||||
background = ink->bg;
|
background = ink->bg;
|
||||||
|
|
||||||
if (foreground == -1 && !using_defaults)
|
if (foreground == -1 && !using_defaults)
|
||||||
foreground = COLOR_WHITE;
|
foreground = COLOR_WHITE;
|
||||||
|
|
||||||
if (background == -1 && !using_defaults)
|
if (background == -1 && !using_defaults)
|
||||||
background = COLOR_BLACK;
|
background = COLOR_BLACK;
|
||||||
|
|
||||||
init_pair(ink->pairnum, foreground, background);
|
init_pair(ink->pairnum, foreground, background);
|
||||||
}
|
}
|
||||||
|
|
||||||
have_palette = TRUE;
|
have_palette = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Try to match the given shibboleth string with one of the regexes in
|
/* Try to match the given shibboleth string with one of the regexes in
|
||||||
* the list starting at head. Return TRUE upon success. */
|
* the list starting at head. Return TRUE upon success. */
|
||||||
bool found_in_list(regexlisttype *head, const char *shibboleth)
|
bool found_in_list(regexlisttype *head, const char *shibboleth)
|
||||||
{
|
{
|
||||||
regexlisttype *item;
|
regexlisttype *item;
|
||||||
regex_t rgx;
|
regex_t rgx;
|
||||||
|
|
||||||
for (item = head; item != NULL; item = item->next) {
|
for (item = head; item != NULL; item = item->next) {
|
||||||
regcomp(&rgx, fixbounds(item->full_regex), NANO_REG_EXTENDED);
|
regcomp(&rgx, fixbounds(item->full_regex), NANO_REG_EXTENDED);
|
||||||
|
|
||||||
if (regexec(&rgx, shibboleth, 0, NULL, 0) == 0) {
|
if (regexec(&rgx, shibboleth, 0, NULL, 0) == 0) {
|
||||||
regfree(&rgx);
|
regfree(&rgx);
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
regfree(&rgx);
|
||||||
}
|
}
|
||||||
|
|
||||||
regfree(&rgx);
|
return FALSE;
|
||||||
}
|
|
||||||
|
|
||||||
return FALSE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update the color information based on the current filename and content. */
|
/* Update the color information based on the current filename and content. */
|
||||||
void color_update(void)
|
void color_update(void)
|
||||||
{
|
{
|
||||||
syntaxtype *sint = NULL;
|
syntaxtype *sint = NULL;
|
||||||
colortype *ink;
|
colortype *ink;
|
||||||
|
|
||||||
/* If the rcfiles were not read, or contained no syntaxes, get out. */
|
/* If the rcfiles were not read, or contained no syntaxes, get out. */
|
||||||
if (syntaxes == NULL)
|
if (syntaxes == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* If we specified a syntax-override string, use it. */
|
/* If we specified a syntax-override string, use it. */
|
||||||
if (syntaxstr != NULL) {
|
if (syntaxstr != NULL) {
|
||||||
/* An override of "none" is like having no syntax at all. */
|
/* An override of "none" is like having no syntax at all. */
|
||||||
if (strcmp(syntaxstr, "none") == 0)
|
if (strcmp(syntaxstr, "none") == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
||||||
if (strcmp(sint->name, syntaxstr) == 0)
|
if (strcmp(sint->name, syntaxstr) == 0)
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sint == NULL && !inhelp)
|
||||||
|
statusline(ALERT, _("Unknown syntax name: %s"), syntaxstr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sint == NULL && !inhelp)
|
/* If no syntax-override string was specified, or it didn't match,
|
||||||
statusline(ALERT, _("Unknown syntax name: %s"), syntaxstr);
|
* try finding a syntax based on the filename (extension). */
|
||||||
}
|
if (sint == NULL && !inhelp) {
|
||||||
|
char *reserved = charalloc(PATH_MAX + 1);
|
||||||
|
char *currentdir = getcwd(reserved, PATH_MAX + 1);
|
||||||
|
char *joinednames = charalloc(PATH_MAX + 1);
|
||||||
|
char *fullname = NULL;
|
||||||
|
|
||||||
/* If no syntax-override string was specified, or it didn't match,
|
if (currentdir == NULL)
|
||||||
* try finding a syntax based on the filename (extension). */
|
free(reserved);
|
||||||
if (sint == NULL && !inhelp) {
|
else {
|
||||||
char *reserved = charalloc(PATH_MAX + 1);
|
/* Concatenate the current working directory with the
|
||||||
char *currentdir = getcwd(reserved, PATH_MAX + 1);
|
* specified filename, and canonicalize the result. */
|
||||||
char *joinednames = charalloc(PATH_MAX + 1);
|
sprintf(joinednames, "%s/%s", currentdir, openfile->filename);
|
||||||
char *fullname = NULL;
|
fullname = get_full_path(joinednames);
|
||||||
|
free(currentdir);
|
||||||
|
}
|
||||||
|
|
||||||
if (currentdir == NULL)
|
if (fullname == NULL)
|
||||||
free(reserved);
|
fullname = mallocstrcpy(fullname, openfile->filename);
|
||||||
else {
|
|
||||||
/* Concatenate the current working directory with the
|
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
||||||
* specified filename, and canonicalize the result. */
|
if (found_in_list(sint->extensions, fullname))
|
||||||
sprintf(joinednames, "%s/%s", currentdir, openfile->filename);
|
break;
|
||||||
fullname = get_full_path(joinednames);
|
}
|
||||||
free(currentdir);
|
|
||||||
|
free(joinednames);
|
||||||
|
free(fullname);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fullname == NULL)
|
/* If the filename didn't match anything, try the first line. */
|
||||||
fullname = mallocstrcpy(fullname, openfile->filename);
|
if (sint == NULL && !inhelp) {
|
||||||
|
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
||||||
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
if (found_in_list(sint->headers, openfile->fileage->data))
|
||||||
if (found_in_list(sint->extensions, fullname))
|
break;
|
||||||
break;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
free(joinednames);
|
|
||||||
free(fullname);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* If the filename didn't match anything, try the first line. */
|
|
||||||
if (sint == NULL && !inhelp) {
|
|
||||||
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
|
||||||
if (found_in_list(sint->headers, openfile->fileage->data))
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef HAVE_LIBMAGIC
|
#ifdef HAVE_LIBMAGIC
|
||||||
/* If we still don't have an answer, try using magic. */
|
/* If we still don't have an answer, try using magic. */
|
||||||
if (sint == NULL && !inhelp) {
|
if (sint == NULL && !inhelp) {
|
||||||
struct stat fileinfo;
|
struct stat fileinfo;
|
||||||
magic_t cookie = NULL;
|
magic_t cookie = NULL;
|
||||||
const char *magicstring = NULL;
|
const char *magicstring = NULL;
|
||||||
|
|
||||||
if (stat(openfile->filename, &fileinfo) == 0) {
|
if (stat(openfile->filename, &fileinfo) == 0) {
|
||||||
/* Open the magic database and get a diagnosis of the file. */
|
/* Open the magic database and get a diagnosis of the file. */
|
||||||
cookie = magic_open(MAGIC_SYMLINK |
|
cookie = magic_open(MAGIC_SYMLINK |
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
MAGIC_DEBUG | MAGIC_CHECK |
|
MAGIC_DEBUG | MAGIC_CHECK |
|
||||||
#endif
|
#endif
|
||||||
MAGIC_ERROR);
|
MAGIC_ERROR);
|
||||||
if (cookie == NULL || magic_load(cookie, NULL) < 0)
|
if (cookie == NULL || magic_load(cookie, NULL) < 0)
|
||||||
statusline(ALERT, _("magic_load() failed: %s"), strerror(errno));
|
statusline(ALERT, _("magic_load() failed: %s"), strerror(errno));
|
||||||
else {
|
else {
|
||||||
magicstring = magic_file(cookie, openfile->filename);
|
magicstring = magic_file(cookie, openfile->filename);
|
||||||
if (magicstring == NULL)
|
if (magicstring == NULL)
|
||||||
statusline(ALERT, _("magic_file(%s) failed: %s"),
|
statusline(ALERT, _("magic_file(%s) failed: %s"),
|
||||||
openfile->filename, magic_error(cookie));
|
openfile->filename, magic_error(cookie));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Now try and find a syntax that matches the magic string. */
|
/* Now try and find a syntax that matches the magic string. */
|
||||||
if (magicstring != NULL) {
|
if (magicstring != NULL) {
|
||||||
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
||||||
if (found_in_list(sint->magics, magicstring))
|
if (found_in_list(sint->magics, magicstring))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stat(openfile->filename, &fileinfo) == 0)
|
if (stat(openfile->filename, &fileinfo) == 0)
|
||||||
magic_close(cookie);
|
magic_close(cookie);
|
||||||
}
|
}
|
||||||
#endif /* HAVE_LIBMAGIC */
|
#endif /* HAVE_LIBMAGIC */
|
||||||
|
|
||||||
/* If nothing at all matched, see if there is a default syntax. */
|
/* If nothing at all matched, see if there is a default syntax. */
|
||||||
if (sint == NULL && !inhelp) {
|
if (sint == NULL && !inhelp) {
|
||||||
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
for (sint = syntaxes; sint != NULL; sint = sint->next) {
|
||||||
if (strcmp(sint->name, "default") == 0)
|
if (strcmp(sint->name, "default") == 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
openfile->syntax = sint;
|
|
||||||
openfile->colorstrings = (sint == NULL ? NULL : sint->color);
|
|
||||||
|
|
||||||
/* If a syntax was found, compile its specified regexes (which have
|
|
||||||
* already been checked for validity when they were read in). */
|
|
||||||
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
|
||||||
if (ink->start == NULL) {
|
|
||||||
ink->start = (regex_t *)nmalloc(sizeof(regex_t));
|
|
||||||
regcomp(ink->start, fixbounds(ink->start_regex), ink->rex_flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ink->end_regex != NULL && ink->end == NULL) {
|
openfile->syntax = sint;
|
||||||
ink->end = (regex_t *)nmalloc(sizeof(regex_t));
|
openfile->colorstrings = (sint == NULL ? NULL : sint->color);
|
||||||
regcomp(ink->end, fixbounds(ink->end_regex), ink->rex_flags);
|
|
||||||
|
/* If a syntax was found, compile its specified regexes (which have
|
||||||
|
* already been checked for validity when they were read in). */
|
||||||
|
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
||||||
|
if (ink->start == NULL) {
|
||||||
|
ink->start = (regex_t *)nmalloc(sizeof(regex_t));
|
||||||
|
regcomp(ink->start, fixbounds(ink->start_regex), ink->rex_flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ink->end_regex != NULL && ink->end == NULL) {
|
||||||
|
ink->end = (regex_t *)nmalloc(sizeof(regex_t));
|
||||||
|
regcomp(ink->end, fixbounds(ink->end_regex), ink->rex_flags);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Determine whether the matches of multiline regexes are still the same,
|
/* Determine whether the matches of multiline regexes are still the same,
|
||||||
* and if not, schedule a screen refresh, so things will be repainted. */
|
* and if not, schedule a screen refresh, so things will be repainted. */
|
||||||
void check_the_multis(filestruct *line)
|
void check_the_multis(filestruct *line)
|
||||||
{
|
{
|
||||||
const colortype *ink;
|
const colortype *ink;
|
||||||
bool astart, anend;
|
bool astart, anend;
|
||||||
regmatch_t startmatch, endmatch;
|
regmatch_t startmatch, endmatch;
|
||||||
|
|
||||||
/* If there is no syntax or no multiline regex, there is nothing to do. */
|
/* If there is no syntax or no multiline regex, there is nothing to do. */
|
||||||
if (openfile->syntax == NULL || openfile->syntax->nmultis == 0)
|
if (openfile->syntax == NULL || openfile->syntax->nmultis == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
||||||
/* If it's not a multiline regex, skip. */
|
/* If it's not a multiline regex, skip. */
|
||||||
if (ink->end == NULL)
|
if (ink->end == NULL)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
alloc_multidata_if_needed(line);
|
alloc_multidata_if_needed(line);
|
||||||
|
|
||||||
astart = (regexec(ink->start, line->data, 1, &startmatch, 0) == 0);
|
astart = (regexec(ink->start, line->data, 1, &startmatch, 0) == 0);
|
||||||
anend = (regexec(ink->end, line->data, 1, &endmatch, 0) == 0);
|
anend = (regexec(ink->end, line->data, 1, &endmatch, 0) == 0);
|
||||||
|
|
||||||
/* Check whether the multidata still matches the current situation. */
|
/* Check whether the multidata still matches the current situation. */
|
||||||
if (line->multidata[ink->id] == CNONE ||
|
if (line->multidata[ink->id] == CNONE ||
|
||||||
line->multidata[ink->id] == CWHOLELINE) {
|
line->multidata[ink->id] == CWHOLELINE) {
|
||||||
if (!astart && !anend)
|
if (!astart && !anend)
|
||||||
continue;
|
continue;
|
||||||
} else if (line->multidata[ink->id] == CSTARTENDHERE) {
|
} else if (line->multidata[ink->id] == CSTARTENDHERE) {
|
||||||
if (astart && anend && startmatch.rm_so < endmatch.rm_so)
|
if (astart && anend && startmatch.rm_so < endmatch.rm_so)
|
||||||
continue;
|
continue;
|
||||||
} else if (line->multidata[ink->id] == CBEGINBEFORE) {
|
} else if (line->multidata[ink->id] == CBEGINBEFORE) {
|
||||||
if (!astart && anend)
|
if (!astart && anend)
|
||||||
continue;
|
continue;
|
||||||
} else if (line->multidata[ink->id] == CENDAFTER) {
|
} else if (line->multidata[ink->id] == CENDAFTER) {
|
||||||
if (astart && !anend)
|
if (astart && !anend)
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* There is a mismatch, so something changed: repaint. */
|
||||||
|
refresh_needed = TRUE;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* There is a mismatch, so something changed: repaint. */
|
|
||||||
refresh_needed = TRUE;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate (for one line) the cache space for multiline color regexes. */
|
/* Allocate (for one line) the cache space for multiline color regexes. */
|
||||||
void alloc_multidata_if_needed(filestruct *fileptr)
|
void alloc_multidata_if_needed(filestruct *fileptr)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (fileptr->multidata == NULL) {
|
if (fileptr->multidata == NULL) {
|
||||||
fileptr->multidata = (short *)nmalloc(openfile->syntax->nmultis * sizeof(short));
|
fileptr->multidata = (short *)nmalloc(openfile->syntax->nmultis * sizeof(short));
|
||||||
|
|
||||||
for (i = 0; i < openfile->syntax->nmultis; i++)
|
for (i = 0; i < openfile->syntax->nmultis; i++)
|
||||||
fileptr->multidata[i] = -1;
|
fileptr->multidata[i] = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Precalculate the multi-line start and end regex info so we can
|
/* Precalculate the multi-line start and end regex info so we can
|
||||||
* speed up rendering (with any hope at all...). */
|
* speed up rendering (with any hope at all...). */
|
||||||
void precalc_multicolorinfo(void)
|
void precalc_multicolorinfo(void)
|
||||||
{
|
{
|
||||||
const colortype *ink;
|
const colortype *ink;
|
||||||
regmatch_t startmatch, endmatch;
|
regmatch_t startmatch, endmatch;
|
||||||
filestruct *line, *tailline;
|
filestruct *line, *tailline;
|
||||||
|
|
||||||
if (openfile->colorstrings == NULL || ISSET(NO_COLOR_SYNTAX))
|
if (openfile->colorstrings == NULL || ISSET(NO_COLOR_SYNTAX))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
fprintf(stderr, "Precalculating the multiline color info...\n");
|
fprintf(stderr, "Precalculating the multiline color info...\n");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) {
|
||||||
/* If this is not a multi-line regex, skip it. */
|
/* If this is not a multi-line regex, skip it. */
|
||||||
if (ink->end == NULL)
|
if (ink->end == NULL)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
for (line = openfile->fileage; line != NULL; line = line->next) {
|
for (line = openfile->fileage; line != NULL; line = line->next) {
|
||||||
int index = 0;
|
int index = 0;
|
||||||
|
|
||||||
alloc_multidata_if_needed(line);
|
alloc_multidata_if_needed(line);
|
||||||
/* Assume nothing applies until proven otherwise below. */
|
/* Assume nothing applies until proven otherwise below. */
|
||||||
line->multidata[ink->id] = CNONE;
|
line->multidata[ink->id] = CNONE;
|
||||||
|
|
||||||
/* For an unpaired start match, mark all remaining lines. */
|
/* For an unpaired start match, mark all remaining lines. */
|
||||||
if (line->prev && line->prev->multidata[ink->id] == CWOULDBE) {
|
if (line->prev && line->prev->multidata[ink->id] == CWOULDBE) {
|
||||||
line->multidata[ink->id] = CWOULDBE;
|
line->multidata[ink->id] = CWOULDBE;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* When the line contains a start match, look for an end, and if
|
/* When the line contains a start match, look for an end, and if
|
||||||
* found, mark all the lines that are affected. */
|
* found, mark all the lines that are affected. */
|
||||||
while (regexec(ink->start, line->data + index, 1,
|
while (regexec(ink->start, line->data + index, 1,
|
||||||
&startmatch, (index == 0) ? 0 : REG_NOTBOL) == 0) {
|
&startmatch, (index == 0) ? 0 : REG_NOTBOL) == 0) {
|
||||||
/* Begin looking for an end match after the start match. */
|
/* Begin looking for an end match after the start match. */
|
||||||
index += startmatch.rm_eo;
|
index += startmatch.rm_eo;
|
||||||
|
|
||||||
/* If there is an end match on this line, mark the line, but
|
/* If there is an end match on this line, mark the line, but
|
||||||
* continue looking for other starts after it. */
|
* continue looking for other starts after it. */
|
||||||
if (regexec(ink->end, line->data + index, 1,
|
if (regexec(ink->end, line->data + index, 1,
|
||||||
&endmatch, (index == 0) ? 0 : REG_NOTBOL) == 0) {
|
&endmatch, (index == 0) ? 0 : REG_NOTBOL) == 0) {
|
||||||
line->multidata[ink->id] = CSTARTENDHERE;
|
line->multidata[ink->id] = CSTARTENDHERE;
|
||||||
index += endmatch.rm_eo;
|
index += endmatch.rm_eo;
|
||||||
/* If both start and end are mere anchors, step ahead. */
|
/* If both start and end are mere anchors, step ahead. */
|
||||||
if (startmatch.rm_so == startmatch.rm_eo &&
|
if (startmatch.rm_so == startmatch.rm_eo &&
|
||||||
endmatch.rm_so == endmatch.rm_eo) {
|
endmatch.rm_so == endmatch.rm_eo) {
|
||||||
/* When at end-of-line, we're done. */
|
/* When at end-of-line, we're done. */
|
||||||
if (line->data[index] == '\0')
|
if (line->data[index] == '\0')
|
||||||
break;
|
break;
|
||||||
index = move_mbright(line->data, index);
|
index = move_mbright(line->data, index);
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Look for an end match on later lines. */
|
||||||
|
tailline = line->next;
|
||||||
|
|
||||||
|
while (tailline != NULL) {
|
||||||
|
if (regexec(ink->end, tailline->data, 1, &endmatch, 0) == 0)
|
||||||
|
break;
|
||||||
|
tailline = tailline->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tailline == NULL) {
|
||||||
|
line->multidata[ink->id] = CWOULDBE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We found it, we found it, la la la la la. Mark all
|
||||||
|
* the lines in between and the end properly. */
|
||||||
|
line->multidata[ink->id] = CENDAFTER;
|
||||||
|
|
||||||
|
for (line = line->next; line != tailline; line = line->next) {
|
||||||
|
alloc_multidata_if_needed(line);
|
||||||
|
line->multidata[ink->id] = CWHOLELINE;
|
||||||
|
}
|
||||||
|
|
||||||
|
alloc_multidata_if_needed(tailline);
|
||||||
|
tailline->multidata[ink->id] = CBEGINBEFORE;
|
||||||
|
|
||||||
|
/* Begin looking for a new start after the end match. */
|
||||||
|
index = endmatch.rm_eo;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Look for an end match on later lines. */
|
|
||||||
tailline = line->next;
|
|
||||||
|
|
||||||
while (tailline != NULL) {
|
|
||||||
if (regexec(ink->end, tailline->data, 1, &endmatch, 0) == 0)
|
|
||||||
break;
|
|
||||||
tailline = tailline->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tailline == NULL) {
|
|
||||||
line->multidata[ink->id] = CWOULDBE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* We found it, we found it, la la la la la. Mark all
|
|
||||||
* the lines in between and the end properly. */
|
|
||||||
line->multidata[ink->id] = CENDAFTER;
|
|
||||||
|
|
||||||
for (line = line->next; line != tailline; line = line->next) {
|
|
||||||
alloc_multidata_if_needed(line);
|
|
||||||
line->multidata[ink->id] = CWHOLELINE;
|
|
||||||
}
|
|
||||||
|
|
||||||
alloc_multidata_if_needed(tailline);
|
|
||||||
tailline->multidata[ink->id] = CBEGINBEFORE;
|
|
||||||
|
|
||||||
/* Begin looking for a new start after the end match. */
|
|
||||||
index = endmatch.rm_eo;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* ENABLE_COLOR */
|
#endif /* ENABLE_COLOR */
|
||||||
|
|
290
src/cut.c
290
src/cut.c
|
@ -25,18 +25,18 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
static bool keep_cutbuffer = FALSE;
|
static bool keep_cutbuffer = FALSE;
|
||||||
/* Should we keep the contents of the cutbuffer? */
|
/* Should we keep the contents of the cutbuffer? */
|
||||||
|
|
||||||
/* Indicate that we should no longer keep the contents of the cutbuffer. */
|
/* Indicate that we should no longer keep the contents of the cutbuffer. */
|
||||||
void cutbuffer_reset(void)
|
void cutbuffer_reset(void)
|
||||||
{
|
{
|
||||||
keep_cutbuffer = FALSE;
|
keep_cutbuffer = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the status of cutbuffer preservation. */
|
/* Return the status of cutbuffer preservation. */
|
||||||
inline bool keeping_cutbuffer(void)
|
inline bool keeping_cutbuffer(void)
|
||||||
{
|
{
|
||||||
return keep_cutbuffer;
|
return keep_cutbuffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If we aren't on the last line of the file, move all the text of the
|
/* If we aren't on the last line of the file, move all the text of the
|
||||||
|
@ -46,13 +46,13 @@ inline bool keeping_cutbuffer(void)
|
||||||
* current line. */
|
* current line. */
|
||||||
void cut_line(void)
|
void cut_line(void)
|
||||||
{
|
{
|
||||||
if (openfile->current != openfile->filebot)
|
if (openfile->current != openfile->filebot)
|
||||||
extract_buffer(&cutbuffer, &cutbottom, openfile->current, 0,
|
extract_buffer(&cutbuffer, &cutbottom, openfile->current, 0,
|
||||||
openfile->current->next, 0);
|
openfile->current->next, 0);
|
||||||
else
|
else
|
||||||
extract_buffer(&cutbuffer, &cutbottom, openfile->current, 0,
|
extract_buffer(&cutbuffer, &cutbottom, openfile->current, 0,
|
||||||
openfile->current, strlen(openfile->current->data));
|
openfile->current, strlen(openfile->current->data));
|
||||||
openfile->placewewant = 0;
|
openfile->placewewant = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
|
@ -60,14 +60,14 @@ void cut_line(void)
|
||||||
* current place we want to where the text used to start. */
|
* current place we want to where the text used to start. */
|
||||||
void cut_marked(bool *right_side_up)
|
void cut_marked(bool *right_side_up)
|
||||||
{
|
{
|
||||||
filestruct *top, *bot;
|
filestruct *top, *bot;
|
||||||
size_t top_x, bot_x;
|
size_t top_x, bot_x;
|
||||||
|
|
||||||
mark_order((const filestruct **)&top, &top_x,
|
mark_order((const filestruct **)&top, &top_x,
|
||||||
(const filestruct **)&bot, &bot_x, right_side_up);
|
(const filestruct **)&bot, &bot_x, right_side_up);
|
||||||
|
|
||||||
extract_buffer(&cutbuffer, &cutbottom, top, top_x, bot, bot_x);
|
extract_buffer(&cutbuffer, &cutbottom, top, top_x, bot, bot_x);
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If we aren't at the end of the current line, move all the text from
|
/* If we aren't at the end of the current line, move all the text from
|
||||||
|
@ -78,32 +78,32 @@ void cut_marked(bool *right_side_up)
|
||||||
* newline used to be. */
|
* newline used to be. */
|
||||||
void cut_to_eol(void)
|
void cut_to_eol(void)
|
||||||
{
|
{
|
||||||
size_t data_len = strlen(openfile->current->data);
|
size_t data_len = strlen(openfile->current->data);
|
||||||
|
|
||||||
if (openfile->current_x < data_len)
|
if (openfile->current_x < data_len)
|
||||||
/* If we're not at the end of the line, move all the text from
|
/* If we're not at the end of the line, move all the text from
|
||||||
* the current position up to it, not counting the newline at
|
* the current position up to it, not counting the newline at
|
||||||
* the end, into the cutbuffer. */
|
* the end, into the cutbuffer. */
|
||||||
extract_buffer(&cutbuffer, &cutbottom, openfile->current,
|
extract_buffer(&cutbuffer, &cutbottom, openfile->current,
|
||||||
openfile->current_x, openfile->current, data_len);
|
openfile->current_x, openfile->current, data_len);
|
||||||
else if (openfile->current != openfile->filebot) {
|
else if (openfile->current != openfile->filebot) {
|
||||||
/* If we're at the end of the line, and it isn't the last line
|
/* If we're at the end of the line, and it isn't the last line
|
||||||
* of the file, move all the text from the current position up
|
* of the file, move all the text from the current position up
|
||||||
* to the beginning of the next line, i.e. the newline at the
|
* to the beginning of the next line, i.e. the newline at the
|
||||||
* end, into the cutbuffer. */
|
* end, into the cutbuffer. */
|
||||||
extract_buffer(&cutbuffer, &cutbottom, openfile->current,
|
extract_buffer(&cutbuffer, &cutbottom, openfile->current,
|
||||||
openfile->current_x, openfile->current->next, 0);
|
openfile->current_x, openfile->current->next, 0);
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move all the text from the current cursor position to the end of the
|
/* Move all the text from the current cursor position to the end of the
|
||||||
* file into the cutbuffer. */
|
* file into the cutbuffer. */
|
||||||
void cut_to_eof(void)
|
void cut_to_eof(void)
|
||||||
{
|
{
|
||||||
extract_buffer(&cutbuffer, &cutbottom,
|
extract_buffer(&cutbuffer, &cutbottom,
|
||||||
openfile->current, openfile->current_x,
|
openfile->current, openfile->current_x,
|
||||||
openfile->filebot, strlen(openfile->filebot->data));
|
openfile->filebot, strlen(openfile->filebot->data));
|
||||||
}
|
}
|
||||||
#endif /* !NANO_TINY */
|
#endif /* !NANO_TINY */
|
||||||
|
|
||||||
|
@ -114,86 +114,86 @@ void cut_to_eof(void)
|
||||||
void do_cut_text(bool copy_text, bool cut_till_eof)
|
void do_cut_text(bool copy_text, bool cut_till_eof)
|
||||||
{
|
{
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
filestruct *cb_save = NULL;
|
filestruct *cb_save = NULL;
|
||||||
/* The current end of the cutbuffer, before we add text to it. */
|
/* The current end of the cutbuffer, before we add text to it. */
|
||||||
size_t cb_save_len = 0;
|
size_t cb_save_len = 0;
|
||||||
/* The length of the string at the current end of the cutbuffer,
|
/* The length of the string at the current end of the cutbuffer,
|
||||||
* before we add text to it. */
|
* before we add text to it. */
|
||||||
bool old_no_newlines = ISSET(NO_NEWLINES);
|
bool old_no_newlines = ISSET(NO_NEWLINES);
|
||||||
bool right_side_up = TRUE;
|
bool right_side_up = TRUE;
|
||||||
/* There *is* no region, *or* it is marked forward. */
|
/* There *is* no region, *or* it is marked forward. */
|
||||||
#endif
|
#endif
|
||||||
size_t was_totsize = openfile->totsize;
|
size_t was_totsize = openfile->totsize;
|
||||||
|
|
||||||
/* If a chain of cuts was broken, empty the cutbuffer. */
|
/* If a chain of cuts was broken, empty the cutbuffer. */
|
||||||
if (!keep_cutbuffer) {
|
if (!keep_cutbuffer) {
|
||||||
free_filestruct(cutbuffer);
|
free_filestruct(cutbuffer);
|
||||||
cutbuffer = NULL;
|
cutbuffer = NULL;
|
||||||
/* Indicate that future cuts should add to the cutbuffer. */
|
/* Indicate that future cuts should add to the cutbuffer. */
|
||||||
keep_cutbuffer = TRUE;
|
keep_cutbuffer = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
if (copy_text) {
|
if (copy_text) {
|
||||||
/* If the cutbuffer isn't empty, remember where it currently ends. */
|
/* If the cutbuffer isn't empty, remember where it currently ends. */
|
||||||
if (cutbuffer != NULL) {
|
if (cutbuffer != NULL) {
|
||||||
cb_save = cutbottom;
|
cb_save = cutbottom;
|
||||||
cb_save_len = strlen(cutbottom->data);
|
cb_save_len = strlen(cutbottom->data);
|
||||||
|
}
|
||||||
|
/* Don't add a magicline when moving text to the cutbuffer. */
|
||||||
|
SET(NO_NEWLINES);
|
||||||
}
|
}
|
||||||
/* Don't add a magicline when moving text to the cutbuffer. */
|
|
||||||
SET(NO_NEWLINES);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cut_till_eof) {
|
if (cut_till_eof) {
|
||||||
/* Move all text up to the end of the file into the cutbuffer. */
|
/* Move all text up to the end of the file into the cutbuffer. */
|
||||||
cut_to_eof();
|
cut_to_eof();
|
||||||
} else if (openfile->mark) {
|
} else if (openfile->mark) {
|
||||||
/* Move the marked text to the cutbuffer, and turn the mark off. */
|
/* Move the marked text to the cutbuffer, and turn the mark off. */
|
||||||
cut_marked(&right_side_up);
|
cut_marked(&right_side_up);
|
||||||
openfile->mark = NULL;
|
openfile->mark = NULL;
|
||||||
} else if (ISSET(CUT_FROM_CURSOR))
|
} else if (ISSET(CUT_FROM_CURSOR))
|
||||||
/* Move all text up to the end of the line into the cutbuffer. */
|
/* Move all text up to the end of the line into the cutbuffer. */
|
||||||
cut_to_eol();
|
cut_to_eol();
|
||||||
else
|
else
|
||||||
#endif
|
#endif
|
||||||
/* Move the entire line into the cutbuffer. */
|
/* Move the entire line into the cutbuffer. */
|
||||||
cut_line();
|
cut_line();
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
if (copy_text) {
|
if (copy_text) {
|
||||||
/* Copy the text that is in the cutbuffer (starting at its saved end,
|
/* Copy the text that is in the cutbuffer (starting at its saved end,
|
||||||
* if there is one) back into the current buffer. This effectively
|
* if there is one) back into the current buffer. This effectively
|
||||||
* uncuts the text we just cut. */
|
* uncuts the text we just cut. */
|
||||||
if (cutbuffer != NULL) {
|
if (cutbuffer != NULL) {
|
||||||
if (cb_save != NULL) {
|
if (cb_save != NULL) {
|
||||||
cb_save->data += cb_save_len;
|
cb_save->data += cb_save_len;
|
||||||
copy_from_buffer(cb_save);
|
copy_from_buffer(cb_save);
|
||||||
cb_save->data -= cb_save_len;
|
cb_save->data -= cb_save_len;
|
||||||
} else
|
} else
|
||||||
copy_from_buffer(cutbuffer);
|
copy_from_buffer(cutbuffer);
|
||||||
|
|
||||||
/* If the copied region was marked forward, put the new desired
|
/* If the copied region was marked forward, put the new desired
|
||||||
* x position at its end; otherwise, leave it at its beginning. */
|
* x position at its end; otherwise, leave it at its beginning. */
|
||||||
if (right_side_up)
|
if (right_side_up)
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
}
|
}
|
||||||
/* Restore the magicline behavior now that we're done fiddling. */
|
/* Restore the magicline behavior now that we're done fiddling. */
|
||||||
if (!old_no_newlines)
|
if (!old_no_newlines)
|
||||||
UNSET(NO_NEWLINES);
|
UNSET(NO_NEWLINES);
|
||||||
} else
|
} else
|
||||||
#endif /* !NANO_TINY */
|
#endif /* !NANO_TINY */
|
||||||
/* Only set the modification flag if actually something was cut. */
|
/* Only set the modification flag if actually something was cut. */
|
||||||
if (openfile->totsize != was_totsize)
|
if (openfile->totsize != was_totsize)
|
||||||
set_modified();
|
set_modified();
|
||||||
|
|
||||||
refresh_needed = TRUE;
|
refresh_needed = TRUE;
|
||||||
|
|
||||||
#ifdef ENABLE_COLOR
|
#ifdef ENABLE_COLOR
|
||||||
check_the_multis(openfile->current);
|
check_the_multis(openfile->current);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
dump_filestruct(cutbuffer);
|
dump_filestruct(cutbuffer);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -201,11 +201,11 @@ void do_cut_text(bool copy_text, bool cut_till_eof)
|
||||||
void do_cut_text_void(void)
|
void do_cut_text_void(void)
|
||||||
{
|
{
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
add_undo(CUT);
|
add_undo(CUT);
|
||||||
#endif
|
#endif
|
||||||
do_cut_text(FALSE, FALSE);
|
do_cut_text(FALSE, FALSE);
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
update_undo(CUT);
|
update_undo(CUT);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -215,79 +215,79 @@ void do_cut_text_void(void)
|
||||||
* was moved, blow away previous contents of the cutbuffer. */
|
* was moved, blow away previous contents of the cutbuffer. */
|
||||||
void do_copy_text(void)
|
void do_copy_text(void)
|
||||||
{
|
{
|
||||||
static struct filestruct *next_contiguous_line = NULL;
|
static struct filestruct *next_contiguous_line = NULL;
|
||||||
bool mark_is_set = (openfile->mark != NULL);
|
bool mark_is_set = (openfile->mark != NULL);
|
||||||
|
|
||||||
/* Remember the current viewport and cursor position. */
|
/* Remember the current viewport and cursor position. */
|
||||||
ssize_t is_edittop_lineno = openfile->edittop->lineno;
|
ssize_t is_edittop_lineno = openfile->edittop->lineno;
|
||||||
size_t is_firstcolumn = openfile->firstcolumn;
|
size_t is_firstcolumn = openfile->firstcolumn;
|
||||||
ssize_t is_current_lineno = openfile->current->lineno;
|
ssize_t is_current_lineno = openfile->current->lineno;
|
||||||
size_t is_current_x = openfile->current_x;
|
size_t is_current_x = openfile->current_x;
|
||||||
|
|
||||||
if (mark_is_set || openfile->current != next_contiguous_line)
|
if (mark_is_set || openfile->current != next_contiguous_line)
|
||||||
cutbuffer_reset();
|
cutbuffer_reset();
|
||||||
|
|
||||||
do_cut_text(TRUE, FALSE);
|
do_cut_text(TRUE, FALSE);
|
||||||
|
|
||||||
/* If the mark was set, blow away the cutbuffer on the next copy. */
|
/* If the mark was set, blow away the cutbuffer on the next copy. */
|
||||||
next_contiguous_line = (mark_is_set ? NULL : openfile->current);
|
next_contiguous_line = (mark_is_set ? NULL : openfile->current);
|
||||||
|
|
||||||
/* If the mark was set, restore the viewport and cursor position. */
|
/* If the mark was set, restore the viewport and cursor position. */
|
||||||
if (mark_is_set) {
|
if (mark_is_set) {
|
||||||
openfile->edittop = fsfromline(is_edittop_lineno);
|
openfile->edittop = fsfromline(is_edittop_lineno);
|
||||||
openfile->firstcolumn = is_firstcolumn;
|
openfile->firstcolumn = is_firstcolumn;
|
||||||
openfile->current = fsfromline(is_current_lineno);
|
openfile->current = fsfromline(is_current_lineno);
|
||||||
openfile->current_x = is_current_x;
|
openfile->current_x = is_current_x;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Cut from the current cursor position to the end of the file. */
|
/* Cut from the current cursor position to the end of the file. */
|
||||||
void do_cut_till_eof(void)
|
void do_cut_till_eof(void)
|
||||||
{
|
{
|
||||||
add_undo(CUT_TO_EOF);
|
add_undo(CUT_TO_EOF);
|
||||||
do_cut_text(FALSE, TRUE);
|
do_cut_text(FALSE, TRUE);
|
||||||
update_undo(CUT_TO_EOF);
|
update_undo(CUT_TO_EOF);
|
||||||
}
|
}
|
||||||
#endif /* !NANO_TINY */
|
#endif /* !NANO_TINY */
|
||||||
|
|
||||||
/* Copy text from the cutbuffer into the current buffer. */
|
/* Copy text from the cutbuffer into the current buffer. */
|
||||||
void do_uncut_text(void)
|
void do_uncut_text(void)
|
||||||
{
|
{
|
||||||
ssize_t was_lineno = openfile->current->lineno;
|
ssize_t was_lineno = openfile->current->lineno;
|
||||||
/* The line number where we started the paste. */
|
/* The line number where we started the paste. */
|
||||||
size_t was_leftedge = 0;
|
size_t was_leftedge = 0;
|
||||||
/* The leftedge where we started the paste. */
|
/* The leftedge where we started the paste. */
|
||||||
|
|
||||||
/* If the cutbuffer is empty, there is nothing to do. */
|
/* If the cutbuffer is empty, there is nothing to do. */
|
||||||
if (cutbuffer == NULL)
|
if (cutbuffer == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
add_undo(PASTE);
|
add_undo(PASTE);
|
||||||
|
|
||||||
if (ISSET(SOFTWRAP))
|
if (ISSET(SOFTWRAP))
|
||||||
was_leftedge = leftedge_for(xplustabs(), openfile->current);
|
was_leftedge = leftedge_for(xplustabs(), openfile->current);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Add a copy of the text in the cutbuffer to the current buffer
|
/* Add a copy of the text in the cutbuffer to the current buffer
|
||||||
* at the current cursor position. */
|
* at the current cursor position. */
|
||||||
copy_from_buffer(cutbuffer);
|
copy_from_buffer(cutbuffer);
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
update_undo(PASTE);
|
update_undo(PASTE);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* If we pasted less than a screenful, don't center the cursor. */
|
/* If we pasted less than a screenful, don't center the cursor. */
|
||||||
if (less_than_a_screenful(was_lineno, was_leftedge))
|
if (less_than_a_screenful(was_lineno, was_leftedge))
|
||||||
focusing = FALSE;
|
focusing = FALSE;
|
||||||
|
|
||||||
/* Set the desired x position to where the pasted text ends. */
|
/* Set the desired x position to where the pasted text ends. */
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
|
|
||||||
set_modified();
|
set_modified();
|
||||||
refresh_needed = TRUE;
|
refresh_needed = TRUE;
|
||||||
|
|
||||||
#ifdef ENABLE_COLOR
|
#ifdef ENABLE_COLOR
|
||||||
check_the_multis(openfile->current);
|
check_the_multis(openfile->current);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
3764
src/files.c
3764
src/files.c
File diff suppressed because it is too large
Load Diff
2340
src/global.c
2340
src/global.c
File diff suppressed because it is too large
Load Diff
946
src/help.c
946
src/help.c
File diff suppressed because it is too large
Load Diff
798
src/history.c
798
src/history.c
|
@ -35,146 +35,146 @@
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static bool history_changed = FALSE;
|
static bool history_changed = FALSE;
|
||||||
/* Whether any of the history lists has changed. */
|
/* Whether any of the history lists has changed. */
|
||||||
static struct stat stat_of_positions_file;
|
static struct stat stat_of_positions_file;
|
||||||
/* The last-obtained stat information of the positions file. */
|
/* The last-obtained stat information of the positions file. */
|
||||||
static char *poshistname = NULL;
|
static char *poshistname = NULL;
|
||||||
/* The name of the positions-history file. */
|
/* The name of the positions-history file. */
|
||||||
|
|
||||||
/* Initialize the lists of historical search and replace strings
|
/* Initialize the lists of historical search and replace strings
|
||||||
* and the list of historical executed commands. */
|
* and the list of historical executed commands. */
|
||||||
void history_init(void)
|
void history_init(void)
|
||||||
{
|
{
|
||||||
search_history = make_new_node(NULL);
|
search_history = make_new_node(NULL);
|
||||||
search_history->data = mallocstrcpy(NULL, "");
|
search_history->data = mallocstrcpy(NULL, "");
|
||||||
searchtop = search_history;
|
searchtop = search_history;
|
||||||
searchbot = search_history;
|
searchbot = search_history;
|
||||||
|
|
||||||
replace_history = make_new_node(NULL);
|
replace_history = make_new_node(NULL);
|
||||||
replace_history->data = mallocstrcpy(NULL, "");
|
replace_history->data = mallocstrcpy(NULL, "");
|
||||||
replacetop = replace_history;
|
replacetop = replace_history;
|
||||||
replacebot = replace_history;
|
replacebot = replace_history;
|
||||||
|
|
||||||
execute_history = make_new_node(NULL);
|
execute_history = make_new_node(NULL);
|
||||||
execute_history->data = mallocstrcpy(NULL, "");
|
execute_history->data = mallocstrcpy(NULL, "");
|
||||||
executetop = execute_history;
|
executetop = execute_history;
|
||||||
executebot = execute_history;
|
executebot = execute_history;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set the current position in the given history list to the bottom. */
|
/* Set the current position in the given history list to the bottom. */
|
||||||
void history_reset(const filestruct *list)
|
void history_reset(const filestruct *list)
|
||||||
{
|
{
|
||||||
if (list == search_history)
|
if (list == search_history)
|
||||||
search_history = searchbot;
|
search_history = searchbot;
|
||||||
else if (list == replace_history)
|
else if (list == replace_history)
|
||||||
replace_history = replacebot;
|
replace_history = replacebot;
|
||||||
else if (list == execute_history)
|
else if (list == execute_history)
|
||||||
execute_history = executebot;
|
execute_history = executebot;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return from the history list that starts at start and ends at end
|
/* Return from the history list that starts at start and ends at end
|
||||||
* the first node that contains the first len characters of the given
|
* the first node that contains the first len characters of the given
|
||||||
* text, or NULL if there is no such node. */
|
* text, or NULL if there is no such node. */
|
||||||
filestruct *find_history(const filestruct *start, const filestruct *end,
|
filestruct *find_history(const filestruct *start, const filestruct *end,
|
||||||
const char *text, size_t len)
|
const char *text, size_t len)
|
||||||
{
|
{
|
||||||
const filestruct *item;
|
const filestruct *item;
|
||||||
|
|
||||||
for (item = start; item != end->prev && item != NULL; item = item->prev) {
|
for (item = start; item != end->prev && item != NULL; item = item->prev) {
|
||||||
if (strncmp(item->data, text, len) == 0)
|
if (strncmp(item->data, text, len) == 0)
|
||||||
return (filestruct *)item;
|
return (filestruct *)item;
|
||||||
}
|
}
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update a history list (the one in which item is the current position)
|
/* Update a history list (the one in which item is the current position)
|
||||||
* with a fresh string text. That is: add text, or move it to the end. */
|
* with a fresh string text. That is: add text, or move it to the end. */
|
||||||
void update_history(filestruct **item, const char *text)
|
void update_history(filestruct **item, const char *text)
|
||||||
{
|
{
|
||||||
filestruct **htop = NULL, **hbot = NULL, *thesame;
|
filestruct **htop = NULL, **hbot = NULL, *thesame;
|
||||||
|
|
||||||
if (*item == search_history) {
|
if (*item == search_history) {
|
||||||
htop = &searchtop;
|
htop = &searchtop;
|
||||||
hbot = &searchbot;
|
hbot = &searchbot;
|
||||||
} else if (*item == replace_history) {
|
} else if (*item == replace_history) {
|
||||||
htop = &replacetop;
|
htop = &replacetop;
|
||||||
hbot = &replacebot;
|
hbot = &replacebot;
|
||||||
} else if (*item == execute_history) {
|
} else if (*item == execute_history) {
|
||||||
htop = &executetop;
|
htop = &executetop;
|
||||||
hbot = &executebot;
|
hbot = &executebot;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* See if the string is already in the history. */
|
/* See if the string is already in the history. */
|
||||||
thesame = find_history(*hbot, *htop, text, HIGHEST_POSITIVE);
|
thesame = find_history(*hbot, *htop, text, HIGHEST_POSITIVE);
|
||||||
|
|
||||||
/* If an identical string was found, delete that item. */
|
/* If an identical string was found, delete that item. */
|
||||||
if (thesame != NULL) {
|
if (thesame != NULL) {
|
||||||
filestruct *after = thesame->next;
|
filestruct *after = thesame->next;
|
||||||
|
|
||||||
/* If the string is at the head of the list, move the head. */
|
/* If the string is at the head of the list, move the head. */
|
||||||
if (thesame == *htop)
|
if (thesame == *htop)
|
||||||
*htop = after;
|
*htop = after;
|
||||||
|
|
||||||
unlink_node(thesame);
|
unlink_node(thesame);
|
||||||
renumber(after);
|
renumber(after);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If the history is full, delete the oldest item (the one at the
|
/* If the history is full, delete the oldest item (the one at the
|
||||||
* head of the list), to make room for a new item at the end. */
|
* head of the list), to make room for a new item at the end. */
|
||||||
if ((*hbot)->lineno == MAX_SEARCH_HISTORY + 1) {
|
if ((*hbot)->lineno == MAX_SEARCH_HISTORY + 1) {
|
||||||
filestruct *oldest = *htop;
|
filestruct *oldest = *htop;
|
||||||
|
|
||||||
*htop = (*htop)->next;
|
*htop = (*htop)->next;
|
||||||
unlink_node(oldest);
|
unlink_node(oldest);
|
||||||
renumber(*htop);
|
renumber(*htop);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Store the fresh string in the last item, then create a new item. */
|
/* Store the fresh string in the last item, then create a new item. */
|
||||||
(*hbot)->data = mallocstrcpy((*hbot)->data, text);
|
(*hbot)->data = mallocstrcpy((*hbot)->data, text);
|
||||||
splice_node(*hbot, make_new_node(*hbot));
|
splice_node(*hbot, make_new_node(*hbot));
|
||||||
*hbot = (*hbot)->next;
|
*hbot = (*hbot)->next;
|
||||||
(*hbot)->data = mallocstrcpy(NULL, "");
|
(*hbot)->data = mallocstrcpy(NULL, "");
|
||||||
|
|
||||||
/* Indicate that the history needs to be saved on exit. */
|
/* Indicate that the history needs to be saved on exit. */
|
||||||
history_changed = TRUE;
|
history_changed = TRUE;
|
||||||
|
|
||||||
/* Set the current position in the list to the bottom. */
|
/* Set the current position in the list to the bottom. */
|
||||||
*item = *hbot;
|
*item = *hbot;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move h to the string in the history list just before it, and return
|
/* Move h to the string in the history list just before it, and return
|
||||||
* that string. If there isn't one, don't move h and return NULL. */
|
* that string. If there isn't one, don't move h and return NULL. */
|
||||||
char *get_history_older(filestruct **h)
|
char *get_history_older(filestruct **h)
|
||||||
{
|
{
|
||||||
if ((*h)->prev == NULL)
|
if ((*h)->prev == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
*h = (*h)->prev;
|
*h = (*h)->prev;
|
||||||
|
|
||||||
return (*h)->data;
|
return (*h)->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move h to the string in the history list just after it, and return
|
/* Move h to the string in the history list just after it, and return
|
||||||
* that string. If there isn't one, don't move h and return NULL. */
|
* that string. If there isn't one, don't move h and return NULL. */
|
||||||
char *get_history_newer(filestruct **h)
|
char *get_history_newer(filestruct **h)
|
||||||
{
|
{
|
||||||
if ((*h)->next == NULL)
|
if ((*h)->next == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
*h = (*h)->next;
|
*h = (*h)->next;
|
||||||
|
|
||||||
return (*h)->data;
|
return (*h)->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* More placeholders. */
|
/* More placeholders. */
|
||||||
void get_history_newer_void(void)
|
void get_history_newer_void(void)
|
||||||
{
|
{
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
void get_history_older_void(void)
|
void get_history_older_void(void)
|
||||||
{
|
{
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_TABCOMP
|
#ifdef ENABLE_TABCOMP
|
||||||
|
@ -184,408 +184,408 @@ void get_history_older_void(void)
|
||||||
* s. */
|
* s. */
|
||||||
char *get_history_completion(filestruct **h, char *s, size_t len)
|
char *get_history_completion(filestruct **h, char *s, size_t len)
|
||||||
{
|
{
|
||||||
if (len > 0) {
|
if (len > 0) {
|
||||||
filestruct *htop = NULL, *hbot = NULL, *p;
|
filestruct *htop = NULL, *hbot = NULL, *p;
|
||||||
|
|
||||||
if (*h == search_history) {
|
if (*h == search_history) {
|
||||||
htop = searchtop;
|
htop = searchtop;
|
||||||
hbot = searchbot;
|
hbot = searchbot;
|
||||||
} else if (*h == replace_history) {
|
} else if (*h == replace_history) {
|
||||||
htop = replacetop;
|
htop = replacetop;
|
||||||
hbot = replacebot;
|
hbot = replacebot;
|
||||||
} else if (*h == execute_history) {
|
} else if (*h == execute_history) {
|
||||||
htop = executetop;
|
htop = executetop;
|
||||||
hbot = executebot;
|
hbot = executebot;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Search the history list from the current position to the top
|
||||||
|
* for a match of len characters. Skip over an exact match. */
|
||||||
|
p = find_history((*h)->prev, htop, s, len);
|
||||||
|
|
||||||
|
while (p != NULL && strcmp(p->data, s) == 0)
|
||||||
|
p = find_history(p->prev, htop, s, len);
|
||||||
|
|
||||||
|
if (p != NULL) {
|
||||||
|
*h = p;
|
||||||
|
return mallocstrcpy(s, (*h)->data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Search the history list from the bottom to the current position
|
||||||
|
* for a match of len characters. Skip over an exact match. */
|
||||||
|
p = find_history(hbot, *h, s, len);
|
||||||
|
|
||||||
|
while (p != NULL && strcmp(p->data, s) == 0)
|
||||||
|
p = find_history(p->prev, *h, s, len);
|
||||||
|
|
||||||
|
if (p != NULL) {
|
||||||
|
*h = p;
|
||||||
|
return mallocstrcpy(s, (*h)->data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Search the history list from the current position to the top
|
/* If we're here, we didn't find a match, we didn't find an inexact
|
||||||
* for a match of len characters. Skip over an exact match. */
|
* match, or len is 0. Return s. */
|
||||||
p = find_history((*h)->prev, htop, s, len);
|
return (char *)s;
|
||||||
|
|
||||||
while (p != NULL && strcmp(p->data, s) == 0)
|
|
||||||
p = find_history(p->prev, htop, s, len);
|
|
||||||
|
|
||||||
if (p != NULL) {
|
|
||||||
*h = p;
|
|
||||||
return mallocstrcpy(s, (*h)->data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Search the history list from the bottom to the current position
|
|
||||||
* for a match of len characters. Skip over an exact match. */
|
|
||||||
p = find_history(hbot, *h, s, len);
|
|
||||||
|
|
||||||
while (p != NULL && strcmp(p->data, s) == 0)
|
|
||||||
p = find_history(p->prev, *h, s, len);
|
|
||||||
|
|
||||||
if (p != NULL) {
|
|
||||||
*h = p;
|
|
||||||
return mallocstrcpy(s, (*h)->data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* If we're here, we didn't find a match, we didn't find an inexact
|
|
||||||
* match, or len is 0. Return s. */
|
|
||||||
return (char *)s;
|
|
||||||
}
|
}
|
||||||
#endif /* ENSABLE_TABCOMP */
|
#endif /* ENSABLE_TABCOMP */
|
||||||
|
|
||||||
void history_error(const char *msg, ...)
|
void history_error(const char *msg, ...)
|
||||||
{
|
{
|
||||||
va_list ap;
|
va_list ap;
|
||||||
|
|
||||||
va_start(ap, msg);
|
va_start(ap, msg);
|
||||||
vfprintf(stderr, _(msg), ap);
|
vfprintf(stderr, _(msg), ap);
|
||||||
va_end(ap);
|
va_end(ap);
|
||||||
|
|
||||||
fprintf(stderr, _("\nPress Enter to continue\n"));
|
fprintf(stderr, _("\nPress Enter to continue\n"));
|
||||||
while (getchar() != '\n')
|
while (getchar() != '\n')
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check whether we have or could make a directory for history files. */
|
/* Check whether we have or could make a directory for history files. */
|
||||||
bool have_statedir(void)
|
bool have_statedir(void)
|
||||||
{
|
{
|
||||||
struct stat dirstat;
|
struct stat dirstat;
|
||||||
const char *xdgdatadir;
|
const char *xdgdatadir;
|
||||||
|
|
||||||
get_homedir();
|
get_homedir();
|
||||||
|
|
||||||
if (homedir != NULL) {
|
if (homedir != NULL) {
|
||||||
statedir = concatenate(homedir, "/.nano/");
|
statedir = concatenate(homedir, "/.nano/");
|
||||||
|
|
||||||
if (stat(statedir, &dirstat) == 0 && S_ISDIR(dirstat.st_mode)) {
|
if (stat(statedir, &dirstat) == 0 && S_ISDIR(dirstat.st_mode)) {
|
||||||
poshistname = concatenate(statedir, POSITION_HISTORY);
|
poshistname = concatenate(statedir, POSITION_HISTORY);
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
free(statedir);
|
free(statedir);
|
||||||
xdgdatadir = getenv("XDG_DATA_HOME");
|
xdgdatadir = getenv("XDG_DATA_HOME");
|
||||||
|
|
||||||
if (homedir == NULL && xdgdatadir == NULL)
|
if (homedir == NULL && xdgdatadir == NULL)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
if (xdgdatadir != NULL)
|
if (xdgdatadir != NULL)
|
||||||
statedir = concatenate(xdgdatadir, "/nano/");
|
statedir = concatenate(xdgdatadir, "/nano/");
|
||||||
else
|
else
|
||||||
statedir = concatenate(homedir, "/.local/share/nano/");
|
statedir = concatenate(homedir, "/.local/share/nano/");
|
||||||
|
|
||||||
if (stat(statedir, &dirstat) == -1) {
|
if (stat(statedir, &dirstat) == -1) {
|
||||||
if (xdgdatadir == NULL) {
|
if (xdgdatadir == NULL) {
|
||||||
char *statepath = concatenate(homedir, "/.local");
|
char *statepath = concatenate(homedir, "/.local");
|
||||||
mkdir(statepath, S_IRWXU | S_IRWXG | S_IRWXO);
|
mkdir(statepath, S_IRWXU | S_IRWXG | S_IRWXO);
|
||||||
free(statepath);
|
free(statepath);
|
||||||
statepath = concatenate(homedir, "/.local/share");
|
statepath = concatenate(homedir, "/.local/share");
|
||||||
mkdir(statepath, S_IRWXU);
|
mkdir(statepath, S_IRWXU);
|
||||||
free(statepath);
|
free(statepath);
|
||||||
|
}
|
||||||
|
if (mkdir(statedir, S_IRWXU) == -1) {
|
||||||
|
history_error(N_("Unable to create directory %s: %s\n"
|
||||||
|
"It is required for saving/loading "
|
||||||
|
"search history or cursor positions.\n"),
|
||||||
|
statedir, strerror(errno));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
} else if (!S_ISDIR(dirstat.st_mode)) {
|
||||||
|
history_error(N_("Path %s is not a directory and needs to be.\n"
|
||||||
|
"Nano will be unable to load or save "
|
||||||
|
"search history or cursor positions.\n"),
|
||||||
|
statedir);
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
if (mkdir(statedir, S_IRWXU) == -1) {
|
|
||||||
history_error(N_("Unable to create directory %s: %s\n"
|
|
||||||
"It is required for saving/loading "
|
|
||||||
"search history or cursor positions.\n"),
|
|
||||||
statedir, strerror(errno));
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
} else if (!S_ISDIR(dirstat.st_mode)) {
|
|
||||||
history_error(N_("Path %s is not a directory and needs to be.\n"
|
|
||||||
"Nano will be unable to load or save "
|
|
||||||
"search history or cursor positions.\n"),
|
|
||||||
statedir);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
poshistname = concatenate(statedir, POSITION_HISTORY);
|
poshistname = concatenate(statedir, POSITION_HISTORY);
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Load the histories for Search and Replace and Execute Command. */
|
/* Load the histories for Search and Replace and Execute Command. */
|
||||||
void load_history(void)
|
void load_history(void)
|
||||||
{
|
{
|
||||||
char *histname = concatenate(statedir, SEARCH_HISTORY);
|
char *histname = concatenate(statedir, SEARCH_HISTORY);
|
||||||
FILE *hisfile = fopen(histname, "rb");
|
FILE *hisfile = fopen(histname, "rb");
|
||||||
|
|
||||||
if (hisfile == NULL) {
|
if (hisfile == NULL) {
|
||||||
if (errno != ENOENT) {
|
if (errno != ENOENT) {
|
||||||
/* When reading failed, don't save history when we quit. */
|
/* When reading failed, don't save history when we quit. */
|
||||||
UNSET(HISTORYLOG);
|
UNSET(HISTORYLOG);
|
||||||
history_error(N_("Error reading %s: %s"), histname,
|
history_error(N_("Error reading %s: %s"), histname,
|
||||||
strerror(errno));
|
strerror(errno));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Load the three history lists -- first search, then replace,
|
/* Load the three history lists -- first search, then replace,
|
||||||
* then execute -- from oldest entry to newest. Between two
|
* then execute -- from oldest entry to newest. Between two
|
||||||
* lists there is an empty line. */
|
* lists there is an empty line. */
|
||||||
filestruct **history = &search_history;
|
filestruct **history = &search_history;
|
||||||
char *line = NULL;
|
char *line = NULL;
|
||||||
size_t buf_len = 0;
|
size_t buf_len = 0;
|
||||||
ssize_t read;
|
ssize_t read;
|
||||||
|
|
||||||
while ((read = getline(&line, &buf_len, hisfile)) > 0) {
|
while ((read = getline(&line, &buf_len, hisfile)) > 0) {
|
||||||
line[--read] = '\0';
|
line[--read] = '\0';
|
||||||
if (read > 0) {
|
if (read > 0) {
|
||||||
/* Encode any embedded NUL as 0x0A. */
|
/* Encode any embedded NUL as 0x0A. */
|
||||||
unsunder(line, read);
|
unsunder(line, read);
|
||||||
update_history(history, line);
|
update_history(history, line);
|
||||||
} else if (history == &search_history)
|
} else if (history == &search_history)
|
||||||
history = &replace_history;
|
history = &replace_history;
|
||||||
else
|
else
|
||||||
history = &execute_history;
|
history = &execute_history;
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(hisfile);
|
||||||
|
free(line);
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose(hisfile);
|
/* After reading them in, set the status of the lists to "unchanged". */
|
||||||
free(line);
|
history_changed = FALSE;
|
||||||
}
|
|
||||||
|
|
||||||
/* After reading them in, set the status of the lists to "unchanged". */
|
free(histname);
|
||||||
history_changed = FALSE;
|
|
||||||
|
|
||||||
free(histname);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Write the lines of a history list, starting at head, from oldest to newest,
|
/* Write the lines of a history list, starting at head, from oldest to newest,
|
||||||
* to the given file. Return TRUE if writing succeeded, and FALSE otherwise. */
|
* to the given file. Return TRUE if writing succeeded, and FALSE otherwise. */
|
||||||
bool write_list(const filestruct *head, FILE *hisfile)
|
bool write_list(const filestruct *head, FILE *hisfile)
|
||||||
{
|
{
|
||||||
const filestruct *item;
|
const filestruct *item;
|
||||||
|
|
||||||
for (item = head; item != NULL; item = item->next) {
|
for (item = head; item != NULL; item = item->next) {
|
||||||
size_t length = strlen(item->data);
|
size_t length = strlen(item->data);
|
||||||
|
|
||||||
/* Decode 0x0A bytes as embedded NULs. */
|
/* Decode 0x0A bytes as embedded NULs. */
|
||||||
sunder(item->data);
|
sunder(item->data);
|
||||||
|
|
||||||
if (fwrite(item->data, sizeof(char), length, hisfile) < length)
|
if (fwrite(item->data, sizeof(char), length, hisfile) < length)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
if (putc('\n', hisfile) == EOF)
|
if (putc('\n', hisfile) == EOF)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Save the histories for Search and Replace and Execute Command. */
|
/* Save the histories for Search and Replace and Execute Command. */
|
||||||
void save_history(void)
|
void save_history(void)
|
||||||
{
|
{
|
||||||
char *histname;
|
char *histname;
|
||||||
FILE *hisfile;
|
FILE *hisfile;
|
||||||
|
|
||||||
/* If the histories are unchanged, don't bother saving them. */
|
/* If the histories are unchanged, don't bother saving them. */
|
||||||
if (!history_changed)
|
if (!history_changed)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
histname = concatenate(statedir, SEARCH_HISTORY);
|
histname = concatenate(statedir, SEARCH_HISTORY);
|
||||||
hisfile = fopen(histname, "wb");
|
hisfile = fopen(histname, "wb");
|
||||||
|
|
||||||
if (hisfile == NULL)
|
if (hisfile == NULL)
|
||||||
fprintf(stderr, _("Error writing %s: %s\n"), histname,
|
fprintf(stderr, _("Error writing %s: %s\n"), histname,
|
||||||
strerror(errno));
|
strerror(errno));
|
||||||
else {
|
else {
|
||||||
/* Don't allow others to read or write the history file. */
|
/* Don't allow others to read or write the history file. */
|
||||||
chmod(histname, S_IRUSR | S_IWUSR);
|
chmod(histname, S_IRUSR | S_IWUSR);
|
||||||
|
|
||||||
if (!write_list(searchtop, hisfile) ||
|
if (!write_list(searchtop, hisfile) ||
|
||||||
!write_list(replacetop, hisfile) ||
|
!write_list(replacetop, hisfile) ||
|
||||||
!write_list(executetop, hisfile))
|
!write_list(executetop, hisfile))
|
||||||
fprintf(stderr, _("Error writing %s: %s\n"), histname,
|
fprintf(stderr, _("Error writing %s: %s\n"), histname,
|
||||||
strerror(errno));
|
strerror(errno));
|
||||||
|
|
||||||
fclose(hisfile);
|
fclose(hisfile);
|
||||||
}
|
}
|
||||||
|
|
||||||
free(histname);
|
free(histname);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Load the recorded cursor positions for files that were edited. */
|
/* Load the recorded cursor positions for files that were edited. */
|
||||||
void load_poshistory(void)
|
void load_poshistory(void)
|
||||||
{
|
{
|
||||||
FILE *hisfile = fopen(poshistname, "rb");
|
FILE *hisfile = fopen(poshistname, "rb");
|
||||||
|
|
||||||
if (hisfile == NULL) {
|
if (hisfile == NULL) {
|
||||||
if (errno != ENOENT) {
|
if (errno != ENOENT) {
|
||||||
/* When reading failed, don't save history when we quit. */
|
/* When reading failed, don't save history when we quit. */
|
||||||
UNSET(POS_HISTORY);
|
UNSET(POS_HISTORY);
|
||||||
history_error(N_("Error reading %s: %s"), poshistname, strerror(errno));
|
history_error(N_("Error reading %s: %s"), poshistname, strerror(errno));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
char *line = NULL, *lineptr, *xptr;
|
||||||
|
size_t buf_len = 0;
|
||||||
|
ssize_t read, count = 0;
|
||||||
|
poshiststruct *record_ptr = NULL, *newrecord;
|
||||||
|
|
||||||
|
/* Read and parse each line, and store the extracted data. */
|
||||||
|
while ((read = getline(&line, &buf_len, hisfile)) > 5) {
|
||||||
|
/* Decode nulls as embedded newlines. */
|
||||||
|
unsunder(line, read);
|
||||||
|
|
||||||
|
/* Find where the x index and line number are in the line. */
|
||||||
|
xptr = revstrstr(line, " ", line + read - 3);
|
||||||
|
if (xptr == NULL)
|
||||||
|
continue;
|
||||||
|
lineptr = revstrstr(line, " ", xptr - 2);
|
||||||
|
if (lineptr == NULL)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* Now separate the three elements of the line. */
|
||||||
|
*(xptr++) = '\0';
|
||||||
|
*(lineptr++) = '\0';
|
||||||
|
|
||||||
|
/* Create a new position record. */
|
||||||
|
newrecord = (poshiststruct *)nmalloc(sizeof(poshiststruct));
|
||||||
|
newrecord->filename = mallocstrcpy(NULL, line);
|
||||||
|
newrecord->lineno = atoi(lineptr);
|
||||||
|
newrecord->xno = atoi(xptr);
|
||||||
|
newrecord->next = NULL;
|
||||||
|
|
||||||
|
/* Add the record to the list. */
|
||||||
|
if (position_history == NULL)
|
||||||
|
position_history = newrecord;
|
||||||
|
else
|
||||||
|
record_ptr->next = newrecord;
|
||||||
|
|
||||||
|
record_ptr = newrecord;
|
||||||
|
|
||||||
|
/* Impose a limit, so the file will not grow indefinitely. */
|
||||||
|
if (++count > 200) {
|
||||||
|
poshiststruct *drop_record = position_history;
|
||||||
|
|
||||||
|
position_history = position_history->next;
|
||||||
|
|
||||||
|
free(drop_record->filename);
|
||||||
|
free(drop_record);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fclose(hisfile);
|
||||||
|
free(line);
|
||||||
|
|
||||||
|
stat(poshistname, &stat_of_positions_file);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
char *line = NULL, *lineptr, *xptr;
|
|
||||||
size_t buf_len = 0;
|
|
||||||
ssize_t read, count = 0;
|
|
||||||
poshiststruct *record_ptr = NULL, *newrecord;
|
|
||||||
|
|
||||||
/* Read and parse each line, and store the extracted data. */
|
|
||||||
while ((read = getline(&line, &buf_len, hisfile)) > 5) {
|
|
||||||
/* Decode nulls as embedded newlines. */
|
|
||||||
unsunder(line, read);
|
|
||||||
|
|
||||||
/* Find where the x index and line number are in the line. */
|
|
||||||
xptr = revstrstr(line, " ", line + read - 3);
|
|
||||||
if (xptr == NULL)
|
|
||||||
continue;
|
|
||||||
lineptr = revstrstr(line, " ", xptr - 2);
|
|
||||||
if (lineptr == NULL)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* Now separate the three elements of the line. */
|
|
||||||
*(xptr++) = '\0';
|
|
||||||
*(lineptr++) = '\0';
|
|
||||||
|
|
||||||
/* Create a new position record. */
|
|
||||||
newrecord = (poshiststruct *)nmalloc(sizeof(poshiststruct));
|
|
||||||
newrecord->filename = mallocstrcpy(NULL, line);
|
|
||||||
newrecord->lineno = atoi(lineptr);
|
|
||||||
newrecord->xno = atoi(xptr);
|
|
||||||
newrecord->next = NULL;
|
|
||||||
|
|
||||||
/* Add the record to the list. */
|
|
||||||
if (position_history == NULL)
|
|
||||||
position_history = newrecord;
|
|
||||||
else
|
|
||||||
record_ptr->next = newrecord;
|
|
||||||
|
|
||||||
record_ptr = newrecord;
|
|
||||||
|
|
||||||
/* Impose a limit, so the file will not grow indefinitely. */
|
|
||||||
if (++count > 200) {
|
|
||||||
poshiststruct *drop_record = position_history;
|
|
||||||
|
|
||||||
position_history = position_history->next;
|
|
||||||
|
|
||||||
free(drop_record->filename);
|
|
||||||
free(drop_record);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fclose(hisfile);
|
|
||||||
free(line);
|
|
||||||
|
|
||||||
stat(poshistname, &stat_of_positions_file);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Save the recorded cursor positions for files that were edited. */
|
/* Save the recorded cursor positions for files that were edited. */
|
||||||
void save_poshistory(void)
|
void save_poshistory(void)
|
||||||
{
|
{
|
||||||
poshiststruct *posptr;
|
poshiststruct *posptr;
|
||||||
FILE *hisfile = fopen(poshistname, "wb");
|
FILE *hisfile = fopen(poshistname, "wb");
|
||||||
|
|
||||||
if (hisfile == NULL)
|
if (hisfile == NULL)
|
||||||
fprintf(stderr, _("Error writing %s: %s\n"), poshistname, strerror(errno));
|
fprintf(stderr, _("Error writing %s: %s\n"), poshistname, strerror(errno));
|
||||||
else {
|
else {
|
||||||
/* Don't allow others to read or write the history file. */
|
/* Don't allow others to read or write the history file. */
|
||||||
chmod(poshistname, S_IRUSR | S_IWUSR);
|
chmod(poshistname, S_IRUSR | S_IWUSR);
|
||||||
|
|
||||||
for (posptr = position_history; posptr != NULL; posptr = posptr->next) {
|
for (posptr = position_history; posptr != NULL; posptr = posptr->next) {
|
||||||
char *path_and_place;
|
char *path_and_place;
|
||||||
size_t length;
|
size_t length;
|
||||||
|
|
||||||
/* Assume 20 decimal positions each for line and column number,
|
/* Assume 20 decimal positions each for line and column number,
|
||||||
* plus two spaces, plus the line feed, plus the null byte. */
|
* plus two spaces, plus the line feed, plus the null byte. */
|
||||||
path_and_place = charalloc(strlen(posptr->filename) + 44);
|
path_and_place = charalloc(strlen(posptr->filename) + 44);
|
||||||
sprintf(path_and_place, "%s %zd %zd\n",
|
sprintf(path_and_place, "%s %zd %zd\n",
|
||||||
posptr->filename, posptr->lineno, posptr->xno);
|
posptr->filename, posptr->lineno, posptr->xno);
|
||||||
length = strlen(path_and_place);
|
length = strlen(path_and_place);
|
||||||
|
|
||||||
/* Encode newlines in filenames as nulls. */
|
/* Encode newlines in filenames as nulls. */
|
||||||
sunder(path_and_place);
|
sunder(path_and_place);
|
||||||
/* Restore the terminating newline. */
|
/* Restore the terminating newline. */
|
||||||
path_and_place[length - 1] = '\n';
|
path_and_place[length - 1] = '\n';
|
||||||
|
|
||||||
if (fwrite(path_and_place, sizeof(char), length, hisfile) < length)
|
if (fwrite(path_and_place, sizeof(char), length, hisfile) < length)
|
||||||
fprintf(stderr, _("Error writing %s: %s\n"),
|
fprintf(stderr, _("Error writing %s: %s\n"),
|
||||||
poshistname, strerror(errno));
|
poshistname, strerror(errno));
|
||||||
free(path_and_place);
|
free(path_and_place);
|
||||||
|
}
|
||||||
|
fclose(hisfile);
|
||||||
|
|
||||||
|
stat(poshistname, &stat_of_positions_file);
|
||||||
}
|
}
|
||||||
fclose(hisfile);
|
|
||||||
|
|
||||||
stat(poshistname, &stat_of_positions_file);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Reload the position history file if it has been modified since last load. */
|
/* Reload the position history file if it has been modified since last load. */
|
||||||
void reload_positions_if_needed(void)
|
void reload_positions_if_needed(void)
|
||||||
{
|
{
|
||||||
struct stat newstat;
|
struct stat newstat;
|
||||||
|
|
||||||
stat(poshistname, &newstat);
|
stat(poshistname, &newstat);
|
||||||
|
|
||||||
if (newstat.st_mtime != stat_of_positions_file.st_mtime) {
|
if (newstat.st_mtime != stat_of_positions_file.st_mtime) {
|
||||||
poshiststruct *ptr, *nextone;
|
poshiststruct *ptr, *nextone;
|
||||||
|
|
||||||
for (ptr = position_history; ptr != NULL; ptr = nextone) {
|
for (ptr = position_history; ptr != NULL; ptr = nextone) {
|
||||||
nextone = ptr->next;
|
nextone = ptr->next;
|
||||||
free(ptr->filename);
|
free(ptr->filename);
|
||||||
free(ptr);
|
free(ptr);
|
||||||
|
}
|
||||||
|
position_history = NULL;
|
||||||
|
|
||||||
|
load_poshistory();
|
||||||
}
|
}
|
||||||
position_history = NULL;
|
|
||||||
|
|
||||||
load_poshistory();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update the recorded last file positions, given a filename, a line
|
/* Update the recorded last file positions, given a filename, a line
|
||||||
* and a column. If no entry is found, add a new one at the end. */
|
* and a column. If no entry is found, add a new one at the end. */
|
||||||
void update_poshistory(char *filename, ssize_t lineno, ssize_t xpos)
|
void update_poshistory(char *filename, ssize_t lineno, ssize_t xpos)
|
||||||
{
|
{
|
||||||
poshiststruct *posptr, *theone, *posprev = NULL;
|
poshiststruct *posptr, *theone, *posprev = NULL;
|
||||||
char *fullpath = get_full_path(filename);
|
char *fullpath = get_full_path(filename);
|
||||||
|
|
||||||
if (fullpath == NULL || fullpath[strlen(fullpath) - 1] == '/' || inhelp) {
|
if (fullpath == NULL || fullpath[strlen(fullpath) - 1] == '/' || inhelp) {
|
||||||
free(fullpath);
|
free(fullpath);
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
reload_positions_if_needed();
|
|
||||||
|
|
||||||
/* Look for a matching filename in the list. */
|
|
||||||
for (posptr = position_history; posptr != NULL; posptr = posptr->next) {
|
|
||||||
if (!strcmp(posptr->filename, fullpath))
|
|
||||||
break;
|
|
||||||
posprev = posptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Don't record files that have the default cursor position. */
|
|
||||||
if (lineno == 1 && xpos == 1) {
|
|
||||||
if (posptr != NULL) {
|
|
||||||
if (posprev == NULL)
|
|
||||||
position_history = posptr->next;
|
|
||||||
else
|
|
||||||
posprev->next = posptr->next;
|
|
||||||
free(posptr->filename);
|
|
||||||
free(posptr);
|
|
||||||
save_poshistory();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
reload_positions_if_needed();
|
||||||
|
|
||||||
|
/* Look for a matching filename in the list. */
|
||||||
|
for (posptr = position_history; posptr != NULL; posptr = posptr->next) {
|
||||||
|
if (!strcmp(posptr->filename, fullpath))
|
||||||
|
break;
|
||||||
|
posprev = posptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Don't record files that have the default cursor position. */
|
||||||
|
if (lineno == 1 && xpos == 1) {
|
||||||
|
if (posptr != NULL) {
|
||||||
|
if (posprev == NULL)
|
||||||
|
position_history = posptr->next;
|
||||||
|
else
|
||||||
|
posprev->next = posptr->next;
|
||||||
|
free(posptr->filename);
|
||||||
|
free(posptr);
|
||||||
|
save_poshistory();
|
||||||
|
}
|
||||||
|
free(fullpath);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
theone = posptr;
|
||||||
|
|
||||||
|
/* If we didn't find it, make a new node; otherwise, if we're
|
||||||
|
* not at the end, move the matching one to the end. */
|
||||||
|
if (theone == NULL) {
|
||||||
|
theone = (poshiststruct *)nmalloc(sizeof(poshiststruct));
|
||||||
|
theone->filename = mallocstrcpy(NULL, fullpath);
|
||||||
|
if (position_history == NULL)
|
||||||
|
position_history = theone;
|
||||||
|
else
|
||||||
|
posprev->next = theone;
|
||||||
|
} else if (posptr->next != NULL) {
|
||||||
|
if (posprev == NULL)
|
||||||
|
position_history = posptr->next;
|
||||||
|
else
|
||||||
|
posprev->next = posptr->next;
|
||||||
|
while (posptr->next != NULL)
|
||||||
|
posptr = posptr->next;
|
||||||
|
posptr->next = theone;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Store the last cursor position. */
|
||||||
|
theone->lineno = lineno;
|
||||||
|
theone->xno = xpos;
|
||||||
|
theone->next = NULL;
|
||||||
|
|
||||||
free(fullpath);
|
free(fullpath);
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
theone = posptr;
|
save_poshistory();
|
||||||
|
|
||||||
/* If we didn't find it, make a new node; otherwise, if we're
|
|
||||||
* not at the end, move the matching one to the end. */
|
|
||||||
if (theone == NULL) {
|
|
||||||
theone = (poshiststruct *)nmalloc(sizeof(poshiststruct));
|
|
||||||
theone->filename = mallocstrcpy(NULL, fullpath);
|
|
||||||
if (position_history == NULL)
|
|
||||||
position_history = theone;
|
|
||||||
else
|
|
||||||
posprev->next = theone;
|
|
||||||
} else if (posptr->next != NULL) {
|
|
||||||
if (posprev == NULL)
|
|
||||||
position_history = posptr->next;
|
|
||||||
else
|
|
||||||
posprev->next = posptr->next;
|
|
||||||
while (posptr->next != NULL)
|
|
||||||
posptr = posptr->next;
|
|
||||||
posptr->next = theone;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Store the last cursor position. */
|
|
||||||
theone->lineno = lineno;
|
|
||||||
theone->xno = xpos;
|
|
||||||
theone->next = NULL;
|
|
||||||
|
|
||||||
free(fullpath);
|
|
||||||
|
|
||||||
save_poshistory();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check whether the given file matches an existing entry in the recorded
|
/* Check whether the given file matches an existing entry in the recorded
|
||||||
|
@ -593,24 +593,24 @@ void update_poshistory(char *filename, ssize_t lineno, ssize_t xpos)
|
||||||
* set line and column to the retrieved values. */
|
* set line and column to the retrieved values. */
|
||||||
bool has_old_position(const char *file, ssize_t *line, ssize_t *column)
|
bool has_old_position(const char *file, ssize_t *line, ssize_t *column)
|
||||||
{
|
{
|
||||||
poshiststruct *posptr = position_history;
|
poshiststruct *posptr = position_history;
|
||||||
char *fullpath = get_full_path(file);
|
char *fullpath = get_full_path(file);
|
||||||
|
|
||||||
if (fullpath == NULL)
|
if (fullpath == NULL)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
reload_positions_if_needed();
|
reload_positions_if_needed();
|
||||||
|
|
||||||
while (posptr != NULL && strcmp(posptr->filename, fullpath) != 0)
|
while (posptr != NULL && strcmp(posptr->filename, fullpath) != 0)
|
||||||
posptr = posptr->next;
|
posptr = posptr->next;
|
||||||
|
|
||||||
free(fullpath);
|
free(fullpath);
|
||||||
|
|
||||||
if (posptr == NULL)
|
if (posptr == NULL)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
*line = posptr->lineno;
|
*line = posptr->lineno;
|
||||||
*column = posptr->xno;
|
*column = posptr->xno;
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
#endif /* ENABLE_HISTORIES */
|
#endif /* ENABLE_HISTORIES */
|
||||||
|
|
664
src/move.c
664
src/move.c
|
@ -26,42 +26,42 @@
|
||||||
/* Move to the first line of the file. */
|
/* Move to the first line of the file. */
|
||||||
void to_first_line(void)
|
void to_first_line(void)
|
||||||
{
|
{
|
||||||
openfile->current = openfile->fileage;
|
openfile->current = openfile->fileage;
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
openfile->placewewant = 0;
|
openfile->placewewant = 0;
|
||||||
|
|
||||||
refresh_needed = TRUE;
|
refresh_needed = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the last line of the file. */
|
/* Move to the last line of the file. */
|
||||||
void to_last_line(void)
|
void to_last_line(void)
|
||||||
{
|
{
|
||||||
openfile->current = openfile->filebot;
|
openfile->current = openfile->filebot;
|
||||||
openfile->current_x = strlen(openfile->filebot->data);
|
openfile->current_x = strlen(openfile->filebot->data);
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
|
|
||||||
/* Set the last line of the screen as the target for the cursor. */
|
/* Set the last line of the screen as the target for the cursor. */
|
||||||
openfile->current_y = editwinrows - 1;
|
openfile->current_y = editwinrows - 1;
|
||||||
|
|
||||||
refresh_needed = TRUE;
|
refresh_needed = TRUE;
|
||||||
focusing = FALSE;
|
focusing = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Determine the actual current chunk and the target column. */
|
/* Determine the actual current chunk and the target column. */
|
||||||
void get_edge_and_target(size_t *leftedge, size_t *target_column)
|
void get_edge_and_target(size_t *leftedge, size_t *target_column)
|
||||||
{
|
{
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
if (ISSET(SOFTWRAP)) {
|
if (ISSET(SOFTWRAP)) {
|
||||||
size_t shim = editwincols * (1 + (tabsize / editwincols));
|
size_t shim = editwincols * (1 + (tabsize / editwincols));
|
||||||
|
|
||||||
*leftedge = leftedge_for(xplustabs(), openfile->current);
|
*leftedge = leftedge_for(xplustabs(), openfile->current);
|
||||||
*target_column = (openfile->placewewant + shim - *leftedge) % editwincols;
|
*target_column = (openfile->placewewant + shim - *leftedge) % editwincols;
|
||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
*leftedge = 0;
|
*leftedge = 0;
|
||||||
*target_column = openfile->placewewant;
|
*target_column = openfile->placewewant;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the index in line->data that corresponds to the given column on the
|
/* Return the index in line->data that corresponds to the given column on the
|
||||||
|
@ -69,104 +69,104 @@ void get_edge_and_target(size_t *leftedge, size_t *target_column)
|
||||||
* on a tab, prevent the cursor from falling back a row when moving forward,
|
* on a tab, prevent the cursor from falling back a row when moving forward,
|
||||||
* or from skipping a row when moving backward, by incrementing the index. */
|
* or from skipping a row when moving backward, by incrementing the index. */
|
||||||
size_t proper_x(filestruct *line, size_t *leftedge, bool forward,
|
size_t proper_x(filestruct *line, size_t *leftedge, bool forward,
|
||||||
size_t column, bool *shifted)
|
size_t column, bool *shifted)
|
||||||
{
|
{
|
||||||
size_t index = actual_x(line->data, column);
|
size_t index = actual_x(line->data, column);
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
if (ISSET(SOFTWRAP) && line->data[index] == '\t' &&
|
if (ISSET(SOFTWRAP) && line->data[index] == '\t' &&
|
||||||
((forward && strnlenpt(line->data, index) < *leftedge) ||
|
((forward && strnlenpt(line->data, index) < *leftedge) ||
|
||||||
(!forward && column / tabsize == (*leftedge - 1) / tabsize &&
|
(!forward && column / tabsize == (*leftedge - 1) / tabsize &&
|
||||||
column / tabsize < (*leftedge + editwincols - 1) / tabsize))) {
|
column / tabsize < (*leftedge + editwincols - 1) / tabsize))) {
|
||||||
index++;
|
index++;
|
||||||
|
|
||||||
if (shifted != NULL)
|
if (shifted != NULL)
|
||||||
*shifted = TRUE;
|
*shifted = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ISSET(SOFTWRAP))
|
if (ISSET(SOFTWRAP))
|
||||||
*leftedge = leftedge_for(strnlenpt(line->data, index), line);
|
*leftedge = leftedge_for(strnlenpt(line->data, index), line);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return index;
|
return index;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Adjust the values for current_x and placewewant in case we have landed in
|
/* Adjust the values for current_x and placewewant in case we have landed in
|
||||||
* the middle of a tab that crosses a row boundary. */
|
* the middle of a tab that crosses a row boundary. */
|
||||||
void set_proper_index_and_pww(size_t *leftedge, size_t target, bool forward)
|
void set_proper_index_and_pww(size_t *leftedge, size_t target, bool forward)
|
||||||
{
|
{
|
||||||
bool shifted = FALSE;
|
bool shifted = FALSE;
|
||||||
size_t was_edge = *leftedge;
|
size_t was_edge = *leftedge;
|
||||||
|
|
||||||
openfile->current_x = proper_x(openfile->current, leftedge, forward,
|
|
||||||
actual_last_column(*leftedge, target), &shifted);
|
|
||||||
|
|
||||||
/* If the index was incremented, try going to the target column. */
|
|
||||||
if (shifted || *leftedge < was_edge)
|
|
||||||
openfile->current_x = proper_x(openfile->current, leftedge, forward,
|
openfile->current_x = proper_x(openfile->current, leftedge, forward,
|
||||||
actual_last_column(*leftedge, target), &shifted);
|
actual_last_column(*leftedge, target), &shifted);
|
||||||
|
|
||||||
openfile->placewewant = *leftedge + target;
|
/* If the index was incremented, try going to the target column. */
|
||||||
|
if (shifted || *leftedge < was_edge)
|
||||||
|
openfile->current_x = proper_x(openfile->current, leftedge, forward,
|
||||||
|
actual_last_column(*leftedge, target), &shifted);
|
||||||
|
|
||||||
|
openfile->placewewant = *leftedge + target;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move up nearly one screenful. */
|
/* Move up nearly one screenful. */
|
||||||
void do_page_up(void)
|
void do_page_up(void)
|
||||||
{
|
{
|
||||||
int mustmove = (editwinrows < 3) ? 1 : editwinrows - 2;
|
int mustmove = (editwinrows < 3) ? 1 : editwinrows - 2;
|
||||||
size_t leftedge, target_column;
|
size_t leftedge, target_column;
|
||||||
|
|
||||||
/* If we're not in smooth scrolling mode, put the cursor at the
|
/* If we're not in smooth scrolling mode, put the cursor at the
|
||||||
* beginning of the top line of the edit window, as Pico does. */
|
* beginning of the top line of the edit window, as Pico does. */
|
||||||
if (!ISSET(SMOOTH_SCROLL)) {
|
if (!ISSET(SMOOTH_SCROLL)) {
|
||||||
openfile->current = openfile->edittop;
|
openfile->current = openfile->edittop;
|
||||||
leftedge = openfile->firstcolumn;
|
leftedge = openfile->firstcolumn;
|
||||||
openfile->current_y = 0;
|
openfile->current_y = 0;
|
||||||
target_column = 0;
|
target_column = 0;
|
||||||
} else
|
} else
|
||||||
get_edge_and_target(&leftedge, &target_column);
|
get_edge_and_target(&leftedge, &target_column);
|
||||||
|
|
||||||
/* Move up the required number of lines or chunks. If we can't, we're
|
/* Move up the required number of lines or chunks. If we can't, we're
|
||||||
* at the top of the file, so put the cursor there and get out. */
|
* at the top of the file, so put the cursor there and get out. */
|
||||||
if (go_back_chunks(mustmove, &openfile->current, &leftedge) > 0) {
|
if (go_back_chunks(mustmove, &openfile->current, &leftedge) > 0) {
|
||||||
to_first_line();
|
to_first_line();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_proper_index_and_pww(&leftedge, target_column, FALSE);
|
set_proper_index_and_pww(&leftedge, target_column, FALSE);
|
||||||
|
|
||||||
/* Move the viewport so that the cursor stays immobile, if possible. */
|
/* Move the viewport so that the cursor stays immobile, if possible. */
|
||||||
adjust_viewport(STATIONARY);
|
adjust_viewport(STATIONARY);
|
||||||
refresh_needed = TRUE;
|
refresh_needed = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move down nearly one screenful. */
|
/* Move down nearly one screenful. */
|
||||||
void do_page_down(void)
|
void do_page_down(void)
|
||||||
{
|
{
|
||||||
int mustmove = (editwinrows < 3) ? 1 : editwinrows - 2;
|
int mustmove = (editwinrows < 3) ? 1 : editwinrows - 2;
|
||||||
size_t leftedge, target_column;
|
size_t leftedge, target_column;
|
||||||
|
|
||||||
/* If we're not in smooth scrolling mode, put the cursor at the
|
/* If we're not in smooth scrolling mode, put the cursor at the
|
||||||
* beginning of the top line of the edit window, as Pico does. */
|
* beginning of the top line of the edit window, as Pico does. */
|
||||||
if (!ISSET(SMOOTH_SCROLL)) {
|
if (!ISSET(SMOOTH_SCROLL)) {
|
||||||
openfile->current = openfile->edittop;
|
openfile->current = openfile->edittop;
|
||||||
leftedge = openfile->firstcolumn;
|
leftedge = openfile->firstcolumn;
|
||||||
openfile->current_y = 0;
|
openfile->current_y = 0;
|
||||||
target_column = 0;
|
target_column = 0;
|
||||||
} else
|
} else
|
||||||
get_edge_and_target(&leftedge, &target_column);
|
get_edge_and_target(&leftedge, &target_column);
|
||||||
|
|
||||||
/* Move down the required number of lines or chunks. If we can't, we're
|
/* Move down the required number of lines or chunks. If we can't, we're
|
||||||
* at the bottom of the file, so put the cursor there and get out. */
|
* at the bottom of the file, so put the cursor there and get out. */
|
||||||
if (go_forward_chunks(mustmove, &openfile->current, &leftedge) > 0) {
|
if (go_forward_chunks(mustmove, &openfile->current, &leftedge) > 0) {
|
||||||
to_last_line();
|
to_last_line();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_proper_index_and_pww(&leftedge, target_column, TRUE);
|
set_proper_index_and_pww(&leftedge, target_column, TRUE);
|
||||||
|
|
||||||
/* Move the viewport so that the cursor stays immobile, if possible. */
|
/* Move the viewport so that the cursor stays immobile, if possible. */
|
||||||
adjust_viewport(STATIONARY);
|
adjust_viewport(STATIONARY);
|
||||||
refresh_needed = TRUE;
|
refresh_needed = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_JUSTIFY
|
#ifdef ENABLE_JUSTIFY
|
||||||
|
@ -174,18 +174,18 @@ void do_page_down(void)
|
||||||
* current line. If update_screen is TRUE, update the screen afterwards. */
|
* current line. If update_screen is TRUE, update the screen afterwards. */
|
||||||
void do_para_begin(bool update_screen)
|
void do_para_begin(bool update_screen)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
|
|
||||||
if (openfile->current != openfile->fileage)
|
if (openfile->current != openfile->fileage)
|
||||||
openfile->current = openfile->current->prev;
|
openfile->current = openfile->current->prev;
|
||||||
|
|
||||||
while (!begpar(openfile->current))
|
while (!begpar(openfile->current))
|
||||||
openfile->current = openfile->current->prev;
|
openfile->current = openfile->current->prev;
|
||||||
|
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
|
|
||||||
if (update_screen)
|
if (update_screen)
|
||||||
edit_redraw(was_current, CENTERING);
|
edit_redraw(was_current, CENTERING);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move down to the beginning of the last line of the current paragraph.
|
/* Move down to the beginning of the last line of the current paragraph.
|
||||||
|
@ -196,122 +196,122 @@ void do_para_begin(bool update_screen)
|
||||||
* paragraph or isn't in a paragraph. */
|
* paragraph or isn't in a paragraph. */
|
||||||
void do_para_end(bool update_screen)
|
void do_para_end(bool update_screen)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
|
|
||||||
while (openfile->current != openfile->filebot &&
|
while (openfile->current != openfile->filebot &&
|
||||||
!inpar(openfile->current))
|
!inpar(openfile->current))
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
|
|
||||||
while (openfile->current != openfile->filebot &&
|
while (openfile->current != openfile->filebot &&
|
||||||
inpar(openfile->current->next) &&
|
inpar(openfile->current->next) &&
|
||||||
!begpar(openfile->current->next)) {
|
!begpar(openfile->current->next)) {
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (openfile->current != openfile->filebot) {
|
if (openfile->current != openfile->filebot) {
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
} else
|
} else
|
||||||
openfile->current_x = strlen(openfile->current->data);
|
openfile->current_x = strlen(openfile->current->data);
|
||||||
|
|
||||||
if (update_screen)
|
if (update_screen)
|
||||||
edit_redraw(was_current, CENTERING);
|
edit_redraw(was_current, CENTERING);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move up to first start of a paragraph before the current line. */
|
/* Move up to first start of a paragraph before the current line. */
|
||||||
void do_para_begin_void(void)
|
void do_para_begin_void(void)
|
||||||
{
|
{
|
||||||
do_para_begin(TRUE);
|
do_para_begin(TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move down to just after the first end of a paragraph. */
|
/* Move down to just after the first end of a paragraph. */
|
||||||
void do_para_end_void(void)
|
void do_para_end_void(void)
|
||||||
{
|
{
|
||||||
do_para_end(TRUE);
|
do_para_end(TRUE);
|
||||||
}
|
}
|
||||||
#endif /* ENABLE_JUSTIFY */
|
#endif /* ENABLE_JUSTIFY */
|
||||||
|
|
||||||
/* Move to the preceding block of text. */
|
/* Move to the preceding block of text. */
|
||||||
void do_prev_block(void)
|
void do_prev_block(void)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
bool is_text = FALSE, seen_text = FALSE;
|
bool is_text = FALSE, seen_text = FALSE;
|
||||||
|
|
||||||
/* Skip backward until first blank line after some nonblank line(s). */
|
/* Skip backward until first blank line after some nonblank line(s). */
|
||||||
while (openfile->current->prev != NULL && (!seen_text || is_text)) {
|
while (openfile->current->prev != NULL && (!seen_text || is_text)) {
|
||||||
openfile->current = openfile->current->prev;
|
openfile->current = openfile->current->prev;
|
||||||
is_text = !white_string(openfile->current->data);
|
is_text = !white_string(openfile->current->data);
|
||||||
seen_text = seen_text || is_text;
|
seen_text = seen_text || is_text;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Step forward one line again if this one is blank. */
|
/* Step forward one line again if this one is blank. */
|
||||||
if (openfile->current->next != NULL &&
|
if (openfile->current->next != NULL &&
|
||||||
white_string(openfile->current->data))
|
white_string(openfile->current->data))
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
|
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
edit_redraw(was_current, CENTERING);
|
edit_redraw(was_current, CENTERING);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the next block of text. */
|
/* Move to the next block of text. */
|
||||||
void do_next_block(void)
|
void do_next_block(void)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
bool is_white = white_string(openfile->current->data);
|
bool is_white = white_string(openfile->current->data);
|
||||||
bool seen_white = is_white;
|
bool seen_white = is_white;
|
||||||
|
|
||||||
/* Skip forward until first nonblank line after some blank line(s). */
|
/* Skip forward until first nonblank line after some blank line(s). */
|
||||||
while (openfile->current->next != NULL && (!seen_white || is_white)) {
|
while (openfile->current->next != NULL && (!seen_white || is_white)) {
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
is_white = white_string(openfile->current->data);
|
is_white = white_string(openfile->current->data);
|
||||||
seen_white = seen_white || is_white;
|
seen_white = seen_white || is_white;
|
||||||
}
|
}
|
||||||
|
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
edit_redraw(was_current, CENTERING);
|
edit_redraw(was_current, CENTERING);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the previous word. If allow_punct is TRUE, treat punctuation
|
/* Move to the previous word. If allow_punct is TRUE, treat punctuation
|
||||||
* as part of a word. When requested, update the screen afterwards. */
|
* as part of a word. When requested, update the screen afterwards. */
|
||||||
void do_prev_word(bool allow_punct, bool update_screen)
|
void do_prev_word(bool allow_punct, bool update_screen)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
bool seen_a_word = FALSE, step_forward = FALSE;
|
bool seen_a_word = FALSE, step_forward = FALSE;
|
||||||
|
|
||||||
/* Move backward until we pass over the start of a word. */
|
/* Move backward until we pass over the start of a word. */
|
||||||
while (TRUE) {
|
while (TRUE) {
|
||||||
/* If at the head of a line, move to the end of the preceding one. */
|
/* If at the head of a line, move to the end of the preceding one. */
|
||||||
if (openfile->current_x == 0) {
|
if (openfile->current_x == 0) {
|
||||||
if (openfile->current->prev == NULL)
|
if (openfile->current->prev == NULL)
|
||||||
break;
|
break;
|
||||||
openfile->current = openfile->current->prev;
|
openfile->current = openfile->current->prev;
|
||||||
openfile->current_x = strlen(openfile->current->data);
|
openfile->current_x = strlen(openfile->current->data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Step back one character. */
|
||||||
|
openfile->current_x = move_mbleft(openfile->current->data,
|
||||||
|
openfile->current_x);
|
||||||
|
|
||||||
|
if (is_word_mbchar(openfile->current->data + openfile->current_x,
|
||||||
|
allow_punct)) {
|
||||||
|
seen_a_word = TRUE;
|
||||||
|
/* If at the head of a line now, this surely is a word start. */
|
||||||
|
if (openfile->current_x == 0)
|
||||||
|
break;
|
||||||
|
} else if (seen_a_word) {
|
||||||
|
/* This is space now: we've overshot the start of the word. */
|
||||||
|
step_forward = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Step back one character. */
|
if (step_forward)
|
||||||
openfile->current_x = move_mbleft(openfile->current->data,
|
/* Move one character forward again to sit on the start of the word. */
|
||||||
openfile->current_x);
|
openfile->current_x = move_mbright(openfile->current->data,
|
||||||
|
openfile->current_x);
|
||||||
|
|
||||||
if (is_word_mbchar(openfile->current->data + openfile->current_x,
|
if (update_screen)
|
||||||
allow_punct)) {
|
edit_redraw(was_current, FLOWING);
|
||||||
seen_a_word = TRUE;
|
|
||||||
/* If at the head of a line now, this surely is a word start. */
|
|
||||||
if (openfile->current_x == 0)
|
|
||||||
break;
|
|
||||||
} else if (seen_a_word) {
|
|
||||||
/* This is space now: we've overshot the start of the word. */
|
|
||||||
step_forward = TRUE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (step_forward)
|
|
||||||
/* Move one character forward again to sit on the start of the word. */
|
|
||||||
openfile->current_x = move_mbright(openfile->current->data,
|
|
||||||
openfile->current_x);
|
|
||||||
|
|
||||||
if (update_screen)
|
|
||||||
edit_redraw(was_current, FLOWING);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the next word. If allow_punct is TRUE, treat punctuation
|
/* Move to the next word. If allow_punct is TRUE, treat punctuation
|
||||||
|
@ -319,55 +319,55 @@ void do_prev_word(bool allow_punct, bool update_screen)
|
||||||
* Return TRUE if we started on a word, and FALSE otherwise. */
|
* Return TRUE if we started on a word, and FALSE otherwise. */
|
||||||
bool do_next_word(bool allow_punct, bool update_screen)
|
bool do_next_word(bool allow_punct, bool update_screen)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
bool started_on_word = is_word_mbchar(openfile->current->data +
|
bool started_on_word = is_word_mbchar(openfile->current->data +
|
||||||
openfile->current_x, allow_punct);
|
openfile->current_x, allow_punct);
|
||||||
bool seen_space = !started_on_word;
|
bool seen_space = !started_on_word;
|
||||||
|
|
||||||
/* Move forward until we reach the start of a word. */
|
/* Move forward until we reach the start of a word. */
|
||||||
while (TRUE) {
|
while (TRUE) {
|
||||||
/* If at the end of a line, move to the beginning of the next one. */
|
/* If at the end of a line, move to the beginning of the next one. */
|
||||||
if (openfile->current->data[openfile->current_x] == '\0') {
|
if (openfile->current->data[openfile->current_x] == '\0') {
|
||||||
/* When at end of file, stop. */
|
/* When at end of file, stop. */
|
||||||
if (openfile->current->next == NULL)
|
if (openfile->current->next == NULL)
|
||||||
break;
|
break;
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
seen_space = TRUE;
|
seen_space = TRUE;
|
||||||
} else {
|
} else {
|
||||||
/* Step forward one character. */
|
/* Step forward one character. */
|
||||||
openfile->current_x = move_mbright(openfile->current->data,
|
openfile->current_x = move_mbright(openfile->current->data,
|
||||||
openfile->current_x);
|
openfile->current_x);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If this is not a word character, then it's a separator; else
|
||||||
|
* if we've already seen a separator, then it's a word start. */
|
||||||
|
if (!is_word_mbchar(openfile->current->data + openfile->current_x,
|
||||||
|
allow_punct))
|
||||||
|
seen_space = TRUE;
|
||||||
|
else if (seen_space)
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If this is not a word character, then it's a separator; else
|
if (update_screen)
|
||||||
* if we've already seen a separator, then it's a word start. */
|
edit_redraw(was_current, FLOWING);
|
||||||
if (!is_word_mbchar(openfile->current->data + openfile->current_x,
|
|
||||||
allow_punct))
|
|
||||||
seen_space = TRUE;
|
|
||||||
else if (seen_space)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (update_screen)
|
/* Return whether we started on a word. */
|
||||||
edit_redraw(was_current, FLOWING);
|
return started_on_word;
|
||||||
|
|
||||||
/* Return whether we started on a word. */
|
|
||||||
return started_on_word;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the previous word in the file, treating punctuation as part of a
|
/* Move to the previous word in the file, treating punctuation as part of a
|
||||||
* word if the WORD_BOUNDS flag is set, and update the screen afterwards. */
|
* word if the WORD_BOUNDS flag is set, and update the screen afterwards. */
|
||||||
void do_prev_word_void(void)
|
void do_prev_word_void(void)
|
||||||
{
|
{
|
||||||
do_prev_word(ISSET(WORD_BOUNDS), TRUE);
|
do_prev_word(ISSET(WORD_BOUNDS), TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the next word in the file, treating punctuation as part of a word
|
/* Move to the next word in the file, treating punctuation as part of a word
|
||||||
* if the WORD_BOUNDS flag is set, and update the screen afterwards. */
|
* if the WORD_BOUNDS flag is set, and update the screen afterwards. */
|
||||||
void do_next_word_void(void)
|
void do_next_word_void(void)
|
||||||
{
|
{
|
||||||
do_next_word(ISSET(WORD_BOUNDS), TRUE);
|
do_next_word(ISSET(WORD_BOUNDS), TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the beginning of the current line (or softwrapped chunk).
|
/* Move to the beginning of the current line (or softwrapped chunk).
|
||||||
|
@ -375,59 +375,59 @@ void do_next_word_void(void)
|
||||||
* of the full line when already at the start of a chunk. */
|
* of the full line when already at the start of a chunk. */
|
||||||
void do_home(void)
|
void do_home(void)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
size_t was_column = xplustabs();
|
size_t was_column = xplustabs();
|
||||||
bool moved_off_chunk = TRUE;
|
bool moved_off_chunk = TRUE;
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
bool moved = FALSE;
|
bool moved = FALSE;
|
||||||
size_t leftedge = 0, leftedge_x = 0;
|
size_t leftedge = 0, leftedge_x = 0;
|
||||||
|
|
||||||
if (ISSET(SOFTWRAP)) {
|
if (ISSET(SOFTWRAP)) {
|
||||||
leftedge = leftedge_for(was_column, openfile->current);
|
leftedge = leftedge_for(was_column, openfile->current);
|
||||||
leftedge_x = proper_x(openfile->current, &leftedge, FALSE, leftedge,
|
leftedge_x = proper_x(openfile->current, &leftedge, FALSE, leftedge,
|
||||||
NULL);
|
NULL);
|
||||||
}
|
|
||||||
|
|
||||||
if (ISSET(SMART_HOME)) {
|
|
||||||
size_t indent_x = indent_length(openfile->current->data);
|
|
||||||
|
|
||||||
if (openfile->current->data[indent_x] != '\0') {
|
|
||||||
/* If we're exactly on the indent, move fully home. Otherwise,
|
|
||||||
* when not softwrapping or not after the first nonblank chunk,
|
|
||||||
* move to the first nonblank character. */
|
|
||||||
if (openfile->current_x == indent_x) {
|
|
||||||
openfile->current_x = 0;
|
|
||||||
moved = TRUE;
|
|
||||||
} else if (!ISSET(SOFTWRAP) || leftedge_x <= indent_x) {
|
|
||||||
openfile->current_x = indent_x;
|
|
||||||
moved = TRUE;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (!moved && ISSET(SOFTWRAP)) {
|
if (ISSET(SMART_HOME)) {
|
||||||
/* If already at the left edge of the screen, move fully home.
|
size_t indent_x = indent_length(openfile->current->data);
|
||||||
* Otherwise, move to the left edge. */
|
|
||||||
if (openfile->current_x == leftedge_x)
|
if (openfile->current->data[indent_x] != '\0') {
|
||||||
openfile->current_x = 0;
|
/* If we're exactly on the indent, move fully home. Otherwise,
|
||||||
else {
|
* when not softwrapping or not after the first nonblank chunk,
|
||||||
openfile->current_x = leftedge_x;
|
* move to the first nonblank character. */
|
||||||
openfile->placewewant = leftedge;
|
if (openfile->current_x == indent_x) {
|
||||||
moved_off_chunk = FALSE;
|
openfile->current_x = 0;
|
||||||
|
moved = TRUE;
|
||||||
|
} else if (!ISSET(SOFTWRAP) || leftedge_x <= indent_x) {
|
||||||
|
openfile->current_x = indent_x;
|
||||||
|
moved = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else if (!moved)
|
|
||||||
|
if (!moved && ISSET(SOFTWRAP)) {
|
||||||
|
/* If already at the left edge of the screen, move fully home.
|
||||||
|
* Otherwise, move to the left edge. */
|
||||||
|
if (openfile->current_x == leftedge_x)
|
||||||
|
openfile->current_x = 0;
|
||||||
|
else {
|
||||||
|
openfile->current_x = leftedge_x;
|
||||||
|
openfile->placewewant = leftedge;
|
||||||
|
moved_off_chunk = FALSE;
|
||||||
|
}
|
||||||
|
} else if (!moved)
|
||||||
#endif
|
#endif
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
|
|
||||||
if (moved_off_chunk)
|
if (moved_off_chunk)
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
|
|
||||||
/* If we changed chunk, we might be offscreen. Otherwise,
|
/* If we changed chunk, we might be offscreen. Otherwise,
|
||||||
* update current if the mark is on or we changed "page". */
|
* update current if the mark is on or we changed "page". */
|
||||||
if (ISSET(SOFTWRAP) && moved_off_chunk)
|
if (ISSET(SOFTWRAP) && moved_off_chunk)
|
||||||
edit_redraw(was_current, FLOWING);
|
edit_redraw(was_current, FLOWING);
|
||||||
else if (line_needs_update(was_column, openfile->placewewant))
|
else if (line_needs_update(was_column, openfile->placewewant))
|
||||||
update_line(openfile->current, openfile->current_x);
|
update_line(openfile->current, openfile->current_x);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the end of the current line (or softwrapped chunk).
|
/* Move to the end of the current line (or softwrapped chunk).
|
||||||
|
@ -435,159 +435,159 @@ void do_home(void)
|
||||||
* end of the full line. */
|
* end of the full line. */
|
||||||
void do_end(void)
|
void do_end(void)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
size_t was_column = xplustabs();
|
size_t was_column = xplustabs();
|
||||||
size_t line_len = strlen(openfile->current->data);
|
size_t line_len = strlen(openfile->current->data);
|
||||||
bool moved_off_chunk = TRUE;
|
bool moved_off_chunk = TRUE;
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
if (ISSET(SOFTWRAP)) {
|
if (ISSET(SOFTWRAP)) {
|
||||||
bool last_chunk = FALSE;
|
bool last_chunk = FALSE;
|
||||||
size_t leftedge = leftedge_for(was_column, openfile->current);
|
size_t leftedge = leftedge_for(was_column, openfile->current);
|
||||||
size_t rightedge = get_softwrap_breakpoint(openfile->current->data,
|
size_t rightedge = get_softwrap_breakpoint(openfile->current->data,
|
||||||
leftedge, &last_chunk);
|
leftedge, &last_chunk);
|
||||||
size_t rightedge_x;
|
size_t rightedge_x;
|
||||||
|
|
||||||
/* If we're on the last chunk, we're already at the end of the line.
|
/* If we're on the last chunk, we're already at the end of the line.
|
||||||
* Otherwise, we're one column past the end of the line. Shifting
|
* Otherwise, we're one column past the end of the line. Shifting
|
||||||
* backwards one column might put us in the middle of a multi-column
|
* backwards one column might put us in the middle of a multi-column
|
||||||
* character, but actual_x() will fix that. */
|
* character, but actual_x() will fix that. */
|
||||||
if (!last_chunk)
|
if (!last_chunk)
|
||||||
rightedge--;
|
rightedge--;
|
||||||
|
|
||||||
rightedge_x = actual_x(openfile->current->data, rightedge);
|
rightedge_x = actual_x(openfile->current->data, rightedge);
|
||||||
|
|
||||||
/* If already at the right edge of the screen, move fully to
|
/* If already at the right edge of the screen, move fully to
|
||||||
* the end of the line. Otherwise, move to the right edge. */
|
* the end of the line. Otherwise, move to the right edge. */
|
||||||
if (openfile->current_x == rightedge_x)
|
if (openfile->current_x == rightedge_x)
|
||||||
openfile->current_x = line_len;
|
openfile->current_x = line_len;
|
||||||
else {
|
else {
|
||||||
openfile->current_x = rightedge_x;
|
openfile->current_x = rightedge_x;
|
||||||
openfile->placewewant = rightedge;
|
openfile->placewewant = rightedge;
|
||||||
moved_off_chunk = FALSE;
|
moved_off_chunk = FALSE;
|
||||||
}
|
}
|
||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
openfile->current_x = line_len;
|
openfile->current_x = line_len;
|
||||||
|
|
||||||
if (moved_off_chunk)
|
if (moved_off_chunk)
|
||||||
openfile->placewewant = xplustabs();
|
openfile->placewewant = xplustabs();
|
||||||
|
|
||||||
/* If we changed chunk, we might be offscreen. Otherwise,
|
/* If we changed chunk, we might be offscreen. Otherwise,
|
||||||
* update current if the mark is on or we changed "page". */
|
* update current if the mark is on or we changed "page". */
|
||||||
if (ISSET(SOFTWRAP) && moved_off_chunk)
|
if (ISSET(SOFTWRAP) && moved_off_chunk)
|
||||||
edit_redraw(was_current, FLOWING);
|
edit_redraw(was_current, FLOWING);
|
||||||
else if (line_needs_update(was_column, openfile->placewewant))
|
else if (line_needs_update(was_column, openfile->placewewant))
|
||||||
update_line(openfile->current, openfile->current_x);
|
update_line(openfile->current, openfile->current_x);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move the cursor to the preceding line or chunk. If scroll_only is TRUE,
|
/* Move the cursor to the preceding line or chunk. If scroll_only is TRUE,
|
||||||
* also scroll the screen one row, so the cursor stays in the same spot. */
|
* also scroll the screen one row, so the cursor stays in the same spot. */
|
||||||
void do_up(bool scroll_only)
|
void do_up(bool scroll_only)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
size_t leftedge, target_column;
|
size_t leftedge, target_column;
|
||||||
|
|
||||||
/* When just scrolling and the top of the file is onscreen, get out. */
|
/* When just scrolling and the top of the file is onscreen, get out. */
|
||||||
if (scroll_only && openfile->edittop == openfile->fileage &&
|
if (scroll_only && openfile->edittop == openfile->fileage &&
|
||||||
openfile->firstcolumn == 0)
|
openfile->firstcolumn == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
get_edge_and_target(&leftedge, &target_column);
|
get_edge_and_target(&leftedge, &target_column);
|
||||||
|
|
||||||
/* If we can't move up one line or chunk, we're at top of file. */
|
/* If we can't move up one line or chunk, we're at top of file. */
|
||||||
if (go_back_chunks(1, &openfile->current, &leftedge) > 0)
|
if (go_back_chunks(1, &openfile->current, &leftedge) > 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
set_proper_index_and_pww(&leftedge, target_column, FALSE);
|
set_proper_index_and_pww(&leftedge, target_column, FALSE);
|
||||||
|
|
||||||
if (scroll_only)
|
if (scroll_only)
|
||||||
edit_scroll(BACKWARD, 1);
|
edit_scroll(BACKWARD, 1);
|
||||||
|
|
||||||
edit_redraw(was_current, FLOWING);
|
edit_redraw(was_current, FLOWING);
|
||||||
|
|
||||||
/* <Up> should not change placewewant, so restore it. */
|
/* <Up> should not change placewewant, so restore it. */
|
||||||
openfile->placewewant = leftedge + target_column;
|
openfile->placewewant = leftedge + target_column;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move the cursor to next line or chunk. If scroll_only is TRUE, also
|
/* Move the cursor to next line or chunk. If scroll_only is TRUE, also
|
||||||
* scroll the screen one row, so the cursor stays in the same spot. */
|
* scroll the screen one row, so the cursor stays in the same spot. */
|
||||||
void do_down(bool scroll_only)
|
void do_down(bool scroll_only)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
size_t leftedge, target_column;
|
size_t leftedge, target_column;
|
||||||
|
|
||||||
get_edge_and_target(&leftedge, &target_column);
|
get_edge_and_target(&leftedge, &target_column);
|
||||||
|
|
||||||
/* If we can't move down one line or chunk, we're at bottom of file. */
|
/* If we can't move down one line or chunk, we're at bottom of file. */
|
||||||
if (go_forward_chunks(1, &openfile->current, &leftedge) > 0)
|
if (go_forward_chunks(1, &openfile->current, &leftedge) > 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
set_proper_index_and_pww(&leftedge, target_column, TRUE);
|
set_proper_index_and_pww(&leftedge, target_column, TRUE);
|
||||||
|
|
||||||
if (scroll_only)
|
if (scroll_only)
|
||||||
edit_scroll(FORWARD, 1);
|
edit_scroll(FORWARD, 1);
|
||||||
|
|
||||||
edit_redraw(was_current, FLOWING);
|
edit_redraw(was_current, FLOWING);
|
||||||
|
|
||||||
/* <Down> should not change placewewant, so restore it. */
|
/* <Down> should not change placewewant, so restore it. */
|
||||||
openfile->placewewant = leftedge + target_column;
|
openfile->placewewant = leftedge + target_column;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move up one line or chunk. */
|
/* Move up one line or chunk. */
|
||||||
void do_up_void(void)
|
void do_up_void(void)
|
||||||
{
|
{
|
||||||
do_up(FALSE);
|
do_up(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move down one line or chunk. */
|
/* Move down one line or chunk. */
|
||||||
void do_down_void(void)
|
void do_down_void(void)
|
||||||
{
|
{
|
||||||
do_down(FALSE);
|
do_down(FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
/* Scroll up one line or chunk without scrolling the cursor. */
|
/* Scroll up one line or chunk without scrolling the cursor. */
|
||||||
void do_scroll_up(void)
|
void do_scroll_up(void)
|
||||||
{
|
{
|
||||||
do_up(TRUE);
|
do_up(TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Scroll down one line or chunk without scrolling the cursor. */
|
/* Scroll down one line or chunk without scrolling the cursor. */
|
||||||
void do_scroll_down(void)
|
void do_scroll_down(void)
|
||||||
{
|
{
|
||||||
do_down(TRUE);
|
do_down(TRUE);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Move left one character. */
|
/* Move left one character. */
|
||||||
void do_left(void)
|
void do_left(void)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
|
|
||||||
if (openfile->current_x > 0)
|
if (openfile->current_x > 0)
|
||||||
openfile->current_x = move_mbleft(openfile->current->data,
|
openfile->current_x = move_mbleft(openfile->current->data,
|
||||||
openfile->current_x);
|
openfile->current_x);
|
||||||
else if (openfile->current != openfile->fileage) {
|
else if (openfile->current != openfile->fileage) {
|
||||||
openfile->current = openfile->current->prev;
|
openfile->current = openfile->current->prev;
|
||||||
openfile->current_x = strlen(openfile->current->data);
|
openfile->current_x = strlen(openfile->current->data);
|
||||||
}
|
}
|
||||||
|
|
||||||
edit_redraw(was_current, FLOWING);
|
edit_redraw(was_current, FLOWING);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move right one character. */
|
/* Move right one character. */
|
||||||
void do_right(void)
|
void do_right(void)
|
||||||
{
|
{
|
||||||
filestruct *was_current = openfile->current;
|
filestruct *was_current = openfile->current;
|
||||||
|
|
||||||
if (openfile->current->data[openfile->current_x] != '\0')
|
if (openfile->current->data[openfile->current_x] != '\0')
|
||||||
openfile->current_x = move_mbright(openfile->current->data,
|
openfile->current_x = move_mbright(openfile->current->data,
|
||||||
openfile->current_x);
|
openfile->current_x);
|
||||||
else if (openfile->current != openfile->filebot) {
|
else if (openfile->current != openfile->filebot) {
|
||||||
openfile->current = openfile->current->next;
|
openfile->current = openfile->current->next;
|
||||||
openfile->current_x = 0;
|
openfile->current_x = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
edit_redraw(was_current, FLOWING);
|
edit_redraw(was_current, FLOWING);
|
||||||
}
|
}
|
||||||
|
|
3260
src/nano.c
3260
src/nano.c
File diff suppressed because it is too large
Load Diff
570
src/nano.h
570
src/nano.h
|
@ -111,7 +111,7 @@
|
||||||
#endif
|
#endif
|
||||||
#define gettext_noop(string) (string)
|
#define gettext_noop(string) (string)
|
||||||
#define N_(string) gettext_noop(string)
|
#define N_(string) gettext_noop(string)
|
||||||
/* Mark a string that will be sent to gettext() later. */
|
/* Mark a string that will be sent to gettext() later. */
|
||||||
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
@ -144,393 +144,393 @@
|
||||||
|
|
||||||
/* Enumeration types. */
|
/* Enumeration types. */
|
||||||
typedef enum {
|
typedef enum {
|
||||||
NIX_FILE, DOS_FILE, MAC_FILE
|
NIX_FILE, DOS_FILE, MAC_FILE
|
||||||
} file_format;
|
} file_format;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
HUSH, MILD, ALERT
|
HUSH, MILD, ALERT
|
||||||
} message_type;
|
} message_type;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
OVERWRITE, APPEND, PREPEND
|
OVERWRITE, APPEND, PREPEND
|
||||||
} kind_of_writing_type;
|
} kind_of_writing_type;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
SOFTMARK, HARDMARK
|
SOFTMARK, HARDMARK
|
||||||
} mark_type;
|
} mark_type;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
CENTERING, FLOWING, STATIONARY
|
CENTERING, FLOWING, STATIONARY
|
||||||
} update_type;
|
} update_type;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
ADD, DEL, BACK, CUT, CUT_TO_EOF, REPLACE,
|
ADD, DEL, BACK, CUT, CUT_TO_EOF, REPLACE,
|
||||||
#ifdef ENABLE_WRAPPING
|
#ifdef ENABLE_WRAPPING
|
||||||
SPLIT_BEGIN, SPLIT_END,
|
SPLIT_BEGIN, SPLIT_END,
|
||||||
#endif
|
#endif
|
||||||
INDENT, UNINDENT,
|
INDENT, UNINDENT,
|
||||||
#ifdef ENABLE_COMMENT
|
#ifdef ENABLE_COMMENT
|
||||||
COMMENT, UNCOMMENT, PREFLIGHT,
|
COMMENT, UNCOMMENT, PREFLIGHT,
|
||||||
#endif
|
#endif
|
||||||
JOIN, PASTE, INSERT, ENTER, OTHER
|
JOIN, PASTE, INSERT, ENTER, OTHER
|
||||||
} undo_type;
|
} undo_type;
|
||||||
|
|
||||||
/* Structure types. */
|
/* Structure types. */
|
||||||
#ifdef ENABLE_COLOR
|
#ifdef ENABLE_COLOR
|
||||||
typedef struct colortype {
|
typedef struct colortype {
|
||||||
short fg;
|
short fg;
|
||||||
/* This syntax's foreground color. */
|
/* This syntax's foreground color. */
|
||||||
short bg;
|
short bg;
|
||||||
/* This syntax's background color. */
|
/* This syntax's background color. */
|
||||||
bool bright;
|
bool bright;
|
||||||
/* Is this color A_BOLD? */
|
/* Is this color A_BOLD? */
|
||||||
int pairnum;
|
int pairnum;
|
||||||
/* The color pair number used for this foreground color and
|
/* The color pair number used for this foreground color and
|
||||||
* background color. */
|
* background color. */
|
||||||
int attributes;
|
int attributes;
|
||||||
/* Pair number and brightness composed into ready-to-use attributes. */
|
/* Pair number and brightness composed into ready-to-use attributes. */
|
||||||
int rex_flags;
|
int rex_flags;
|
||||||
/* The regex compilation flags (with or without REG_ICASE). */
|
/* The regex compilation flags (with or without REG_ICASE). */
|
||||||
char *start_regex;
|
char *start_regex;
|
||||||
/* The start (or all) of the regex string. */
|
/* The start (or all) of the regex string. */
|
||||||
regex_t *start;
|
regex_t *start;
|
||||||
/* The compiled start (or all) of the regex string. */
|
/* The compiled start (or all) of the regex string. */
|
||||||
char *end_regex;
|
char *end_regex;
|
||||||
/* The end (if any) of the regex string. */
|
/* The end (if any) of the regex string. */
|
||||||
regex_t *end;
|
regex_t *end;
|
||||||
/* The compiled end (if any) of the regex string. */
|
/* The compiled end (if any) of the regex string. */
|
||||||
struct colortype *next;
|
struct colortype *next;
|
||||||
/* Next set of colors. */
|
/* Next set of colors. */
|
||||||
int id;
|
int id;
|
||||||
/* Basic id for assigning to lines later. */
|
/* Basic id for assigning to lines later. */
|
||||||
} colortype;
|
} colortype;
|
||||||
|
|
||||||
typedef struct regexlisttype {
|
typedef struct regexlisttype {
|
||||||
char *full_regex;
|
char *full_regex;
|
||||||
/* A regex string to match things that imply a certain syntax. */
|
/* A regex string to match things that imply a certain syntax. */
|
||||||
struct regexlisttype *next;
|
struct regexlisttype *next;
|
||||||
/* The next regex. */
|
/* The next regex. */
|
||||||
} regexlisttype;
|
} regexlisttype;
|
||||||
|
|
||||||
typedef struct syntaxtype {
|
typedef struct syntaxtype {
|
||||||
char *name;
|
char *name;
|
||||||
/* The name of this syntax. */
|
/* The name of this syntax. */
|
||||||
regexlisttype *extensions;
|
regexlisttype *extensions;
|
||||||
/* The list of extensions that this syntax applies to. */
|
/* The list of extensions that this syntax applies to. */
|
||||||
regexlisttype *headers;
|
regexlisttype *headers;
|
||||||
/* The list of headerlines that this syntax applies to. */
|
/* The list of headerlines that this syntax applies to. */
|
||||||
regexlisttype *magics;
|
regexlisttype *magics;
|
||||||
/* The list of libmagic results that this syntax applies to. */
|
/* The list of libmagic results that this syntax applies to. */
|
||||||
char *linter;
|
char *linter;
|
||||||
/* The command with which to lint this type of file. */
|
/* The command with which to lint this type of file. */
|
||||||
char *formatter;
|
char *formatter;
|
||||||
/* The formatting command (for programming languages mainly). */
|
/* The formatting command (for programming languages mainly). */
|
||||||
#ifdef ENABLE_COMMENT
|
#ifdef ENABLE_COMMENT
|
||||||
char *comment;
|
char *comment;
|
||||||
/* The line comment prefix (and postfix) for this type of file. */
|
/* The line comment prefix (and postfix) for this type of file. */
|
||||||
#endif
|
#endif
|
||||||
colortype *color;
|
colortype *color;
|
||||||
/* The colors and their regexes used in this syntax. */
|
/* The colors and their regexes used in this syntax. */
|
||||||
int nmultis;
|
int nmultis;
|
||||||
/* How many multiline regex strings this syntax has. */
|
/* How many multiline regex strings this syntax has. */
|
||||||
struct syntaxtype *next;
|
struct syntaxtype *next;
|
||||||
/* Next syntax. */
|
/* Next syntax. */
|
||||||
} syntaxtype;
|
} syntaxtype;
|
||||||
|
|
||||||
typedef struct lintstruct {
|
typedef struct lintstruct {
|
||||||
ssize_t lineno;
|
ssize_t lineno;
|
||||||
/* Line number of the error. */
|
/* Line number of the error. */
|
||||||
ssize_t colno;
|
ssize_t colno;
|
||||||
/* Column # of the error. */
|
/* Column # of the error. */
|
||||||
char *msg;
|
char *msg;
|
||||||
/* Error message text. */
|
/* Error message text. */
|
||||||
char *filename;
|
char *filename;
|
||||||
/* Filename. */
|
/* Filename. */
|
||||||
struct lintstruct *next;
|
struct lintstruct *next;
|
||||||
/* Next error. */
|
/* Next error. */
|
||||||
struct lintstruct *prev;
|
struct lintstruct *prev;
|
||||||
/* Previous error. */
|
/* Previous error. */
|
||||||
} lintstruct;
|
} lintstruct;
|
||||||
|
|
||||||
/* Flags that indicate how a multiline regex applies to a line. */
|
/* Flags that indicate how a multiline regex applies to a line. */
|
||||||
#define CNONE (1<<1)
|
#define CNONE (1<<1)
|
||||||
/* Yay, regex doesn't apply to this line at all! */
|
/* Yay, regex doesn't apply to this line at all! */
|
||||||
#define CBEGINBEFORE (1<<2)
|
#define CBEGINBEFORE (1<<2)
|
||||||
/* Regex starts on an earlier line, ends on this one. */
|
/* Regex starts on an earlier line, ends on this one. */
|
||||||
#define CENDAFTER (1<<3)
|
#define CENDAFTER (1<<3)
|
||||||
/* Regex starts on this line and ends on a later one. */
|
/* Regex starts on this line and ends on a later one. */
|
||||||
#define CWHOLELINE (1<<4)
|
#define CWHOLELINE (1<<4)
|
||||||
/* Whole line engulfed by the regex, start < me, end > me. */
|
/* Whole line engulfed by the regex, start < me, end > me. */
|
||||||
#define CSTARTENDHERE (1<<5)
|
#define CSTARTENDHERE (1<<5)
|
||||||
/* Regex starts and ends within this line. */
|
/* Regex starts and ends within this line. */
|
||||||
#define CWOULDBE (1<<6)
|
#define CWOULDBE (1<<6)
|
||||||
/* An unpaired start match on or before this line. */
|
/* An unpaired start match on or before this line. */
|
||||||
#endif /* ENABLE_COLOR */
|
#endif /* ENABLE_COLOR */
|
||||||
|
|
||||||
/* More structure types. */
|
/* More structure types. */
|
||||||
typedef struct filestruct {
|
typedef struct filestruct {
|
||||||
char *data;
|
char *data;
|
||||||
/* The text of this line. */
|
/* The text of this line. */
|
||||||
ssize_t lineno;
|
ssize_t lineno;
|
||||||
/* The number of this line. */
|
/* The number of this line. */
|
||||||
struct filestruct *next;
|
struct filestruct *next;
|
||||||
/* Next node. */
|
/* Next node. */
|
||||||
struct filestruct *prev;
|
struct filestruct *prev;
|
||||||
/* Previous node. */
|
/* Previous node. */
|
||||||
#ifdef ENABLE_COLOR
|
#ifdef ENABLE_COLOR
|
||||||
short *multidata;
|
short *multidata;
|
||||||
/* Array of which multi-line regexes apply to this line. */
|
/* Array of which multi-line regexes apply to this line. */
|
||||||
#endif
|
#endif
|
||||||
} filestruct;
|
} filestruct;
|
||||||
|
|
||||||
typedef struct partition {
|
typedef struct partition {
|
||||||
filestruct *fileage;
|
filestruct *fileage;
|
||||||
/* The top line of this portion of the file. */
|
/* The top line of this portion of the file. */
|
||||||
filestruct *top_prev;
|
filestruct *top_prev;
|
||||||
/* The line before the top line of this portion of the file. */
|
/* The line before the top line of this portion of the file. */
|
||||||
char *top_data;
|
char *top_data;
|
||||||
/* The text before the beginning of the top line of this portion
|
/* The text before the beginning of the top line of this portion
|
||||||
* of the file. */
|
* of the file. */
|
||||||
filestruct *filebot;
|
filestruct *filebot;
|
||||||
/* The bottom line of this portion of the file. */
|
/* The bottom line of this portion of the file. */
|
||||||
filestruct *bot_next;
|
filestruct *bot_next;
|
||||||
/* The line after the bottom line of this portion of the
|
/* The line after the bottom line of this portion of the
|
||||||
* file. */
|
* file. */
|
||||||
char *bot_data;
|
char *bot_data;
|
||||||
/* The text after the end of the bottom line of this portion of
|
/* The text after the end of the bottom line of this portion of
|
||||||
* the file. */
|
* the file. */
|
||||||
} partition;
|
} partition;
|
||||||
|
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
typedef struct undo_group {
|
typedef struct undo_group {
|
||||||
ssize_t top_line;
|
ssize_t top_line;
|
||||||
/* First line of group. */
|
/* First line of group. */
|
||||||
ssize_t bottom_line;
|
ssize_t bottom_line;
|
||||||
/* Last line of group. */
|
/* Last line of group. */
|
||||||
char **indentations;
|
char **indentations;
|
||||||
/* String data used to restore the affected lines; one per line. */
|
/* String data used to restore the affected lines; one per line. */
|
||||||
struct undo_group *next;
|
struct undo_group *next;
|
||||||
} undo_group;
|
} undo_group;
|
||||||
|
|
||||||
typedef struct undo {
|
typedef struct undo {
|
||||||
ssize_t lineno;
|
ssize_t lineno;
|
||||||
undo_type type;
|
undo_type type;
|
||||||
/* What type of undo this was. */
|
/* What type of undo this was. */
|
||||||
size_t begin;
|
size_t begin;
|
||||||
/* Where did this action begin or end. */
|
/* Where did this action begin or end. */
|
||||||
char *strdata;
|
char *strdata;
|
||||||
/* String type data we will use for copying the affected line back. */
|
/* String type data we will use for copying the affected line back. */
|
||||||
size_t wassize;
|
size_t wassize;
|
||||||
/* The file size before the action. */
|
/* The file size before the action. */
|
||||||
size_t newsize;
|
size_t newsize;
|
||||||
/* The file size after the action. */
|
/* The file size after the action. */
|
||||||
int xflags;
|
int xflags;
|
||||||
/* Some flag data we need. */
|
/* Some flag data we need. */
|
||||||
undo_group *grouping;
|
undo_group *grouping;
|
||||||
/* Undo info specific to groups of lines. */
|
/* Undo info specific to groups of lines. */
|
||||||
|
|
||||||
/* Cut-specific stuff we need. */
|
/* Cut-specific stuff we need. */
|
||||||
filestruct *cutbuffer;
|
filestruct *cutbuffer;
|
||||||
/* Copy of the cutbuffer. */
|
/* Copy of the cutbuffer. */
|
||||||
filestruct *cutbottom;
|
filestruct *cutbottom;
|
||||||
/* Copy of cutbottom. */
|
/* Copy of cutbottom. */
|
||||||
ssize_t mark_begin_lineno;
|
ssize_t mark_begin_lineno;
|
||||||
/* Mostly the line number of the current line; sometimes something else. */
|
/* Mostly the line number of the current line; sometimes something else. */
|
||||||
size_t mark_begin_x;
|
size_t mark_begin_x;
|
||||||
/* The x position corresponding to the above line number. */
|
/* The x position corresponding to the above line number. */
|
||||||
struct undo *next;
|
struct undo *next;
|
||||||
/* A pointer to the undo item of the preceding action. */
|
/* A pointer to the undo item of the preceding action. */
|
||||||
} undo;
|
} undo;
|
||||||
#endif /* !NANO_TINY */
|
#endif /* !NANO_TINY */
|
||||||
|
|
||||||
#ifdef ENABLE_HISTORIES
|
#ifdef ENABLE_HISTORIES
|
||||||
typedef struct poshiststruct {
|
typedef struct poshiststruct {
|
||||||
char *filename;
|
char *filename;
|
||||||
/* The file. */
|
/* The file. */
|
||||||
ssize_t lineno;
|
ssize_t lineno;
|
||||||
/* Line number we left off on. */
|
/* Line number we left off on. */
|
||||||
ssize_t xno;
|
ssize_t xno;
|
||||||
/* x position in the file we left off on. */
|
/* x position in the file we left off on. */
|
||||||
struct poshiststruct *next;
|
struct poshiststruct *next;
|
||||||
} poshiststruct;
|
} poshiststruct;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
typedef struct openfilestruct {
|
typedef struct openfilestruct {
|
||||||
char *filename;
|
char *filename;
|
||||||
/* The file's name. */
|
/* The file's name. */
|
||||||
filestruct *fileage;
|
filestruct *fileage;
|
||||||
/* The file's first line. */
|
/* The file's first line. */
|
||||||
filestruct *filebot;
|
filestruct *filebot;
|
||||||
/* The file's last line. */
|
/* The file's last line. */
|
||||||
filestruct *edittop;
|
filestruct *edittop;
|
||||||
/* The current top of the edit window for this file. */
|
/* The current top of the edit window for this file. */
|
||||||
filestruct *current;
|
filestruct *current;
|
||||||
/* The current line for this file. */
|
/* The current line for this file. */
|
||||||
size_t totsize;
|
size_t totsize;
|
||||||
/* The file's total number of characters. */
|
/* The file's total number of characters. */
|
||||||
size_t firstcolumn;
|
size_t firstcolumn;
|
||||||
/* The starting column of the top line of the edit window.
|
/* The starting column of the top line of the edit window.
|
||||||
* When not in softwrap mode, it's always zero. */
|
* When not in softwrap mode, it's always zero. */
|
||||||
size_t current_x;
|
size_t current_x;
|
||||||
/* The file's x-coordinate position. */
|
/* The file's x-coordinate position. */
|
||||||
size_t placewewant;
|
size_t placewewant;
|
||||||
/* The file's x position we would like. */
|
/* The file's x position we would like. */
|
||||||
ssize_t current_y;
|
ssize_t current_y;
|
||||||
/* The file's y-coordinate position. */
|
/* The file's y-coordinate position. */
|
||||||
bool modified;
|
bool modified;
|
||||||
/* Whether the file has been modified. */
|
/* Whether the file has been modified. */
|
||||||
struct stat *current_stat;
|
struct stat *current_stat;
|
||||||
/* The file's current stat information. */
|
/* The file's current stat information. */
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
filestruct *mark;
|
filestruct *mark;
|
||||||
/* The line in the file where the mark is set; NULL if not set. */
|
/* The line in the file where the mark is set; NULL if not set. */
|
||||||
size_t mark_x;
|
size_t mark_x;
|
||||||
/* The mark's x position in the above line. */
|
/* The mark's x position in the above line. */
|
||||||
mark_type kind_of_mark;
|
mark_type kind_of_mark;
|
||||||
/* Whether it is a soft (with Shift) or a hard mark. */
|
/* Whether it is a soft (with Shift) or a hard mark. */
|
||||||
file_format fmt;
|
file_format fmt;
|
||||||
/* The file's format -- Unix or DOS or Mac or mixed. */
|
/* The file's format -- Unix or DOS or Mac or mixed. */
|
||||||
undo *undotop;
|
undo *undotop;
|
||||||
/* The top of the undo list. */
|
/* The top of the undo list. */
|
||||||
undo *current_undo;
|
undo *current_undo;
|
||||||
/* The current (i.e. next) level of undo. */
|
/* The current (i.e. next) level of undo. */
|
||||||
undo *last_saved;
|
undo *last_saved;
|
||||||
/* The undo item at which the file was last saved. */
|
/* The undo item at which the file was last saved. */
|
||||||
undo_type last_action;
|
undo_type last_action;
|
||||||
/* The type of the last action the user performed. */
|
/* The type of the last action the user performed. */
|
||||||
char *lock_filename;
|
char *lock_filename;
|
||||||
/* The path of the lockfile, if we created one. */
|
/* The path of the lockfile, if we created one. */
|
||||||
#endif
|
#endif
|
||||||
#ifdef ENABLE_COLOR
|
#ifdef ENABLE_COLOR
|
||||||
syntaxtype *syntax;
|
syntaxtype *syntax;
|
||||||
/* The syntax struct for this file, if any. */
|
/* The syntax struct for this file, if any. */
|
||||||
colortype *colorstrings;
|
colortype *colorstrings;
|
||||||
/* The file's associated colors. */
|
/* The file's associated colors. */
|
||||||
#endif
|
#endif
|
||||||
struct openfilestruct *next;
|
struct openfilestruct *next;
|
||||||
/* The next open file, if any. */
|
/* The next open file, if any. */
|
||||||
struct openfilestruct *prev;
|
struct openfilestruct *prev;
|
||||||
/* The preceding open file, if any. */
|
/* The preceding open file, if any. */
|
||||||
} openfilestruct;
|
} openfilestruct;
|
||||||
|
|
||||||
#ifdef ENABLE_NANORC
|
#ifdef ENABLE_NANORC
|
||||||
typedef struct rcoption {
|
typedef struct rcoption {
|
||||||
const char *name;
|
const char *name;
|
||||||
/* The name of the rcfile option. */
|
/* The name of the rcfile option. */
|
||||||
long flag;
|
long flag;
|
||||||
/* The flag associated with it, if any. */
|
/* The flag associated with it, if any. */
|
||||||
} rcoption;
|
} rcoption;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
typedef struct sc {
|
typedef struct sc {
|
||||||
const char *keystr;
|
const char *keystr;
|
||||||
/* The string that describes a keystroke, like "^C" or "M-R". */
|
/* The string that describes a keystroke, like "^C" or "M-R". */
|
||||||
bool meta;
|
bool meta;
|
||||||
/* Whether this is a Meta keystroke. */
|
/* Whether this is a Meta keystroke. */
|
||||||
int keycode;
|
int keycode;
|
||||||
/* The integer that, together with meta, identifies the keystroke. */
|
/* The integer that, together with meta, identifies the keystroke. */
|
||||||
int menus;
|
int menus;
|
||||||
/* Which menus this applies to. */
|
/* Which menus this applies to. */
|
||||||
void (*scfunc)(void);
|
void (*scfunc)(void);
|
||||||
/* The function we're going to run. */
|
/* The function we're going to run. */
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
int toggle;
|
int toggle;
|
||||||
/* If a toggle, what we're toggling. */
|
/* If a toggle, what we're toggling. */
|
||||||
int ordinal;
|
int ordinal;
|
||||||
/* The how-manieth toggle this is, in order to be able to
|
/* The how-manieth toggle this is, in order to be able to
|
||||||
* keep them in sequence. */
|
* keep them in sequence. */
|
||||||
#endif
|
#endif
|
||||||
struct sc *next;
|
struct sc *next;
|
||||||
/* Next in the list. */
|
/* Next in the list. */
|
||||||
} sc;
|
} sc;
|
||||||
|
|
||||||
typedef struct subnfunc {
|
typedef struct subnfunc {
|
||||||
void (*scfunc)(void);
|
void (*scfunc)(void);
|
||||||
/* The actual function to call. */
|
/* The actual function to call. */
|
||||||
int menus;
|
int menus;
|
||||||
/* In what menus this function applies. */
|
/* In what menus this function applies. */
|
||||||
const char *desc;
|
const char *desc;
|
||||||
/* The function's short description, for example "Where Is". */
|
/* The function's short description, for example "Where Is". */
|
||||||
#ifdef ENABLE_HELP
|
#ifdef ENABLE_HELP
|
||||||
const char *help;
|
const char *help;
|
||||||
/* The help-screen text for this function. */
|
/* The help-screen text for this function. */
|
||||||
bool blank_after;
|
bool blank_after;
|
||||||
/* Whether there should be a blank line after the help text
|
/* Whether there should be a blank line after the help text
|
||||||
* for this function. */
|
* for this function. */
|
||||||
#endif
|
#endif
|
||||||
bool viewok;
|
bool viewok;
|
||||||
/* Is this function allowed when in view mode? */
|
/* Is this function allowed when in view mode? */
|
||||||
long toggle;
|
long toggle;
|
||||||
/* If this is a toggle, which toggle to affect. */
|
/* If this is a toggle, which toggle to affect. */
|
||||||
struct subnfunc *next;
|
struct subnfunc *next;
|
||||||
/* Next item in the list. */
|
/* Next item in the list. */
|
||||||
} subnfunc;
|
} subnfunc;
|
||||||
|
|
||||||
#ifdef ENABLE_WORDCOMPLETION
|
#ifdef ENABLE_WORDCOMPLETION
|
||||||
typedef struct completion_word {
|
typedef struct completion_word {
|
||||||
char *word;
|
char *word;
|
||||||
struct completion_word *next;
|
struct completion_word *next;
|
||||||
} completion_word;
|
} completion_word;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* The elements of the interface that can be colored differently. */
|
/* The elements of the interface that can be colored differently. */
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
TITLE_BAR = 0,
|
TITLE_BAR = 0,
|
||||||
LINE_NUMBER,
|
LINE_NUMBER,
|
||||||
SELECTED_TEXT,
|
SELECTED_TEXT,
|
||||||
STATUS_BAR,
|
STATUS_BAR,
|
||||||
KEY_COMBO,
|
KEY_COMBO,
|
||||||
FUNCTION_TAG,
|
FUNCTION_TAG,
|
||||||
NUMBER_OF_ELEMENTS
|
NUMBER_OF_ELEMENTS
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Enumeration used in the flags array. See the definition of FLAGMASK. */
|
/* Enumeration used in the flags array. See the definition of FLAGMASK. */
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
DONTUSE,
|
DONTUSE,
|
||||||
CASE_SENSITIVE,
|
CASE_SENSITIVE,
|
||||||
CONSTANT_SHOW,
|
CONSTANT_SHOW,
|
||||||
NO_HELP,
|
NO_HELP,
|
||||||
SUSPEND,
|
SUSPEND,
|
||||||
NO_WRAP,
|
NO_WRAP,
|
||||||
AUTOINDENT,
|
AUTOINDENT,
|
||||||
VIEW_MODE,
|
VIEW_MODE,
|
||||||
USE_MOUSE,
|
USE_MOUSE,
|
||||||
USE_REGEXP,
|
USE_REGEXP,
|
||||||
TEMP_FILE,
|
TEMP_FILE,
|
||||||
CUT_FROM_CURSOR,
|
CUT_FROM_CURSOR,
|
||||||
BACKWARDS_SEARCH,
|
BACKWARDS_SEARCH,
|
||||||
MULTIBUFFER,
|
MULTIBUFFER,
|
||||||
SMOOTH_SCROLL,
|
SMOOTH_SCROLL,
|
||||||
REBIND_DELETE,
|
REBIND_DELETE,
|
||||||
REBIND_KEYPAD,
|
REBIND_KEYPAD,
|
||||||
NO_CONVERT,
|
NO_CONVERT,
|
||||||
BACKUP_FILE,
|
BACKUP_FILE,
|
||||||
INSECURE_BACKUP,
|
INSECURE_BACKUP,
|
||||||
NO_COLOR_SYNTAX,
|
NO_COLOR_SYNTAX,
|
||||||
PRESERVE,
|
PRESERVE,
|
||||||
HISTORYLOG,
|
HISTORYLOG,
|
||||||
RESTRICTED,
|
RESTRICTED,
|
||||||
SMART_HOME,
|
SMART_HOME,
|
||||||
WHITESPACE_DISPLAY,
|
WHITESPACE_DISPLAY,
|
||||||
MORE_SPACE,
|
MORE_SPACE,
|
||||||
TABS_TO_SPACES,
|
TABS_TO_SPACES,
|
||||||
QUICK_BLANK,
|
QUICK_BLANK,
|
||||||
WORD_BOUNDS,
|
WORD_BOUNDS,
|
||||||
NO_NEWLINES,
|
NO_NEWLINES,
|
||||||
BOLD_TEXT,
|
BOLD_TEXT,
|
||||||
QUIET,
|
QUIET,
|
||||||
SOFTWRAP,
|
SOFTWRAP,
|
||||||
POS_HISTORY,
|
POS_HISTORY,
|
||||||
LOCKING,
|
LOCKING,
|
||||||
NOREAD_MODE,
|
NOREAD_MODE,
|
||||||
MAKE_IT_UNIX,
|
MAKE_IT_UNIX,
|
||||||
TRIM_BLANKS,
|
TRIM_BLANKS,
|
||||||
SHOW_CURSOR,
|
SHOW_CURSOR,
|
||||||
LINE_NUMBERS,
|
LINE_NUMBERS,
|
||||||
NO_PAUSES,
|
NO_PAUSES,
|
||||||
AT_BLANKS
|
AT_BLANKS
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Flags for the menus in which a given function should be present. */
|
/* Flags for the menus in which a given function should be present. */
|
||||||
|
@ -552,7 +552,7 @@ enum
|
||||||
#define MFINDINHELP (1<<15)
|
#define MFINDINHELP (1<<15)
|
||||||
/* This is an abbreviation for all menus except Help and YesNo. */
|
/* This is an abbreviation for all menus except Help and YesNo. */
|
||||||
#define MMOST (MMAIN|MWHEREIS|MREPLACE|MREPLACEWITH|MGOTOLINE|MWRITEFILE|MINSERTFILE|\
|
#define MMOST (MMAIN|MWHEREIS|MREPLACE|MREPLACEWITH|MGOTOLINE|MWRITEFILE|MINSERTFILE|\
|
||||||
MEXTCMD|MBROWSER|MWHEREISFILE|MGOTODIR|MFINDINHELP|MSPELL|MLINTER)
|
MEXTCMD|MBROWSER|MWHEREISFILE|MGOTODIR|MFINDINHELP|MSPELL|MLINTER)
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
#define MSOME MMOST
|
#define MSOME MMOST
|
||||||
#else
|
#else
|
||||||
|
|
986
src/prompt.c
986
src/prompt.c
File diff suppressed because it is too large
Load Diff
32
src/proto.h
32
src/proto.h
|
@ -272,7 +272,7 @@ bool open_buffer(const char *filename, bool undoable);
|
||||||
void replace_buffer(const char *filename);
|
void replace_buffer(const char *filename);
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
void replace_marked_buffer(const char *filename, filestruct *top, size_t top_x,
|
void replace_marked_buffer(const char *filename, filestruct *top, size_t top_x,
|
||||||
filestruct *bot, size_t bot_x);
|
filestruct *bot, size_t bot_x);
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
void prepare_for_display(void);
|
void prepare_for_display(void);
|
||||||
|
@ -282,7 +282,7 @@ void switch_to_next_buffer(void);
|
||||||
bool close_buffer(void);
|
bool close_buffer(void);
|
||||||
#endif
|
#endif
|
||||||
void read_file(FILE *f, int fd, const char *filename, bool undoable,
|
void read_file(FILE *f, int fd, const char *filename, bool undoable,
|
||||||
bool checkwritable);
|
bool checkwritable);
|
||||||
int open_file(const char *filename, bool newfie, bool quiet, FILE **f);
|
int open_file(const char *filename, bool newfie, bool quiet, FILE **f);
|
||||||
char *get_next_filename(const char *name, const char *suffix);
|
char *get_next_filename(const char *name, const char *suffix);
|
||||||
void do_insertfile_void(void);
|
void do_insertfile_void(void);
|
||||||
|
@ -299,10 +299,10 @@ int write_lockfile(const char *lockfilename, const char *origfilename, bool modi
|
||||||
#endif
|
#endif
|
||||||
int copy_file(FILE *inn, FILE *out, bool close_out);
|
int copy_file(FILE *inn, FILE *out, bool close_out);
|
||||||
bool write_file(const char *name, FILE *f_open, bool tmp,
|
bool write_file(const char *name, FILE *f_open, bool tmp,
|
||||||
kind_of_writing_type method, bool fullbuffer);
|
kind_of_writing_type method, bool fullbuffer);
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
bool write_marked_file(const char *name, FILE *f_open, bool tmp,
|
bool write_marked_file(const char *name, FILE *f_open, bool tmp,
|
||||||
kind_of_writing_type method);
|
kind_of_writing_type method);
|
||||||
#endif
|
#endif
|
||||||
int do_writeout(bool exiting, bool withprompt);
|
int do_writeout(bool exiting, bool withprompt);
|
||||||
void do_writeout_void(void);
|
void do_writeout_void(void);
|
||||||
|
@ -313,7 +313,7 @@ int diralphasort(const void *va, const void *vb);
|
||||||
#endif
|
#endif
|
||||||
#ifdef ENABLE_TABCOMP
|
#ifdef ENABLE_TABCOMP
|
||||||
char *input_tab(char *buf, bool allow_files, size_t *place,
|
char *input_tab(char *buf, bool allow_files, size_t *place,
|
||||||
bool *lastwastab, void (*refresh_func)(void), bool *listed);
|
bool *lastwastab, void (*refresh_func)(void), bool *listed);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Some functions in global.c. */
|
/* Some functions in global.c. */
|
||||||
|
@ -406,10 +406,10 @@ filestruct *copy_filestruct(const filestruct *src);
|
||||||
void free_filestruct(filestruct *src);
|
void free_filestruct(filestruct *src);
|
||||||
void renumber(filestruct *fileptr);
|
void renumber(filestruct *fileptr);
|
||||||
partition *partition_filestruct(filestruct *top, size_t top_x,
|
partition *partition_filestruct(filestruct *top, size_t top_x,
|
||||||
filestruct *bot, size_t bot_x);
|
filestruct *bot, size_t bot_x);
|
||||||
void unpartition_filestruct(partition **p);
|
void unpartition_filestruct(partition **p);
|
||||||
void extract_buffer(filestruct **file_top, filestruct **file_bot,
|
void extract_buffer(filestruct **file_top, filestruct **file_bot,
|
||||||
filestruct *top, size_t top_x, filestruct *bot, size_t bot_x);
|
filestruct *top, size_t top_x, filestruct *bot, size_t bot_x);
|
||||||
void ingraft_buffer(filestruct *somebuffer);
|
void ingraft_buffer(filestruct *somebuffer);
|
||||||
void copy_from_buffer(filestruct *somebuffer);
|
void copy_from_buffer(filestruct *somebuffer);
|
||||||
openfilestruct *make_new_opennode(void);
|
openfilestruct *make_new_opennode(void);
|
||||||
|
@ -470,8 +470,8 @@ size_t get_statusbar_page_start(size_t start_col, size_t column);
|
||||||
void reinit_statusbar_x(void);
|
void reinit_statusbar_x(void);
|
||||||
void update_the_statusbar(void);
|
void update_the_statusbar(void);
|
||||||
int do_prompt(bool allow_tabs, bool allow_files,
|
int do_prompt(bool allow_tabs, bool allow_files,
|
||||||
int menu, const char *curranswer, filestruct **history_list,
|
int menu, const char *curranswer, filestruct **history_list,
|
||||||
void (*refresh_func)(void), const char *msg, ...);
|
void (*refresh_func)(void), const char *msg, ...);
|
||||||
int do_yesno_prompt(bool all, const char *msg);
|
int do_yesno_prompt(bool all, const char *msg);
|
||||||
|
|
||||||
/* Most functions in rcfile.c. */
|
/* Most functions in rcfile.c. */
|
||||||
|
@ -488,7 +488,7 @@ void do_rcfiles(void);
|
||||||
void not_found_msg(const char *str);
|
void not_found_msg(const char *str);
|
||||||
void search_replace_abort(void);
|
void search_replace_abort(void);
|
||||||
int findnextstr(const char *needle, bool whole_word_only, int modus,
|
int findnextstr(const char *needle, bool whole_word_only, int modus,
|
||||||
size_t *match_len, bool skipone, const filestruct *begin, size_t begin_x);
|
size_t *match_len, bool skipone, const filestruct *begin, size_t begin_x);
|
||||||
void do_search(void);
|
void do_search(void);
|
||||||
void do_search_forward(void);
|
void do_search_forward(void);
|
||||||
void do_search_backward(void);
|
void do_search_backward(void);
|
||||||
|
@ -499,11 +499,11 @@ void do_findnext(void);
|
||||||
void do_research(void);
|
void do_research(void);
|
||||||
void go_looking(void);
|
void go_looking(void);
|
||||||
ssize_t do_replace_loop(const char *needle, bool whole_word_only,
|
ssize_t do_replace_loop(const char *needle, bool whole_word_only,
|
||||||
const filestruct *real_current, size_t *real_current_x);
|
const filestruct *real_current, size_t *real_current_x);
|
||||||
void do_replace(void);
|
void do_replace(void);
|
||||||
void goto_line_posx(ssize_t line, size_t pos_x);
|
void goto_line_posx(ssize_t line, size_t pos_x);
|
||||||
void do_gotolinecolumn(ssize_t line, ssize_t column, bool use_answer,
|
void do_gotolinecolumn(ssize_t line, ssize_t column, bool use_answer,
|
||||||
bool interactive);
|
bool interactive);
|
||||||
void do_gotolinecolumn_void(void);
|
void do_gotolinecolumn_void(void);
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
void do_find_bracket(void);
|
void do_find_bracket(void);
|
||||||
|
@ -592,7 +592,7 @@ const char *fixbounds(const char *r);
|
||||||
bool is_separate_word(size_t position, size_t length, const char *buf);
|
bool is_separate_word(size_t position, size_t length, const char *buf);
|
||||||
#endif
|
#endif
|
||||||
const char *strstrwrapper(const char *haystack, const char *needle,
|
const char *strstrwrapper(const char *haystack, const char *needle,
|
||||||
const char *start);
|
const char *start);
|
||||||
void nperror(const char *s);
|
void nperror(const char *s);
|
||||||
void *nmalloc(size_t howmuch);
|
void *nmalloc(size_t howmuch);
|
||||||
void *nrealloc(void *ptr, size_t howmuch);
|
void *nrealloc(void *ptr, size_t howmuch);
|
||||||
|
@ -610,7 +610,7 @@ void remove_magicline(void);
|
||||||
#endif
|
#endif
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
void mark_order(const filestruct **top, size_t *top_x,
|
void mark_order(const filestruct **top, size_t *top_x,
|
||||||
const filestruct **bot, size_t *bot_x, bool *right_side_up);
|
const filestruct **bot, size_t *bot_x, bool *right_side_up);
|
||||||
void get_range(const filestruct **top, const filestruct **bot);
|
void get_range(const filestruct **top, const filestruct **bot);
|
||||||
#endif
|
#endif
|
||||||
size_t get_totsize(const filestruct *begin, const filestruct *end);
|
size_t get_totsize(const filestruct *begin, const filestruct *end);
|
||||||
|
@ -654,7 +654,7 @@ void bottombars(int menu);
|
||||||
void post_one_key(const char *keystroke, const char *tag, int width);
|
void post_one_key(const char *keystroke, const char *tag, int width);
|
||||||
void place_the_cursor(void);
|
void place_the_cursor(void);
|
||||||
void edit_draw(filestruct *fileptr, const char *converted,
|
void edit_draw(filestruct *fileptr, const char *converted,
|
||||||
int line, size_t from_col);
|
int line, size_t from_col);
|
||||||
int update_line(filestruct *fileptr, size_t index);
|
int update_line(filestruct *fileptr, size_t index);
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
int update_softwrapped_line(filestruct *fileptr);
|
int update_softwrapped_line(filestruct *fileptr);
|
||||||
|
@ -666,7 +666,7 @@ bool less_than_a_screenful(size_t was_lineno, size_t was_leftedge);
|
||||||
void edit_scroll(bool direction, int nrows);
|
void edit_scroll(bool direction, int nrows);
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
size_t get_softwrap_breakpoint(const char *text, size_t leftedge,
|
size_t get_softwrap_breakpoint(const char *text, size_t leftedge,
|
||||||
bool *end_of_line);
|
bool *end_of_line);
|
||||||
size_t get_chunk_and_edge(size_t column, filestruct *line, size_t *leftedge);
|
size_t get_chunk_and_edge(size_t column, filestruct *line, size_t *leftedge);
|
||||||
size_t chunk_for(size_t column, filestruct *line);
|
size_t chunk_for(size_t column, filestruct *line);
|
||||||
size_t leftedge_for(size_t column, filestruct *line);
|
size_t leftedge_for(size_t column, filestruct *line);
|
||||||
|
|
1744
src/rcfile.c
1744
src/rcfile.c
File diff suppressed because it is too large
Load Diff
1542
src/search.c
1542
src/search.c
File diff suppressed because it is too large
Load Diff
5576
src/text.c
5576
src/text.c
File diff suppressed because it is too large
Load Diff
624
src/utils.c
624
src/utils.c
|
@ -32,79 +32,79 @@
|
||||||
* we fall back on the home directory of the effective user ID. */
|
* we fall back on the home directory of the effective user ID. */
|
||||||
void get_homedir(void)
|
void get_homedir(void)
|
||||||
{
|
{
|
||||||
if (homedir == NULL) {
|
if (homedir == NULL) {
|
||||||
const char *homenv = getenv("HOME");
|
const char *homenv = getenv("HOME");
|
||||||
|
|
||||||
#ifdef HAVE_PWD_H
|
#ifdef HAVE_PWD_H
|
||||||
/* When HOME isn't set, or when we're root, get the home directory
|
/* When HOME isn't set, or when we're root, get the home directory
|
||||||
* from the password file instead. */
|
* from the password file instead. */
|
||||||
if (homenv == NULL || geteuid() == 0) {
|
if (homenv == NULL || geteuid() == 0) {
|
||||||
const struct passwd *userage = getpwuid(geteuid());
|
const struct passwd *userage = getpwuid(geteuid());
|
||||||
|
|
||||||
if (userage != NULL)
|
if (userage != NULL)
|
||||||
homenv = userage->pw_dir;
|
homenv = userage->pw_dir;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Only set homedir if some home directory could be determined,
|
/* Only set homedir if some home directory could be determined,
|
||||||
* otherwise keep homedir NULL. */
|
* otherwise keep homedir NULL. */
|
||||||
if (homenv != NULL && strcmp(homenv, "") != 0)
|
if (homenv != NULL && strcmp(homenv, "") != 0)
|
||||||
homedir = mallocstrcpy(NULL, homenv);
|
homedir = mallocstrcpy(NULL, homenv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the filename part of the given path. */
|
/* Return the filename part of the given path. */
|
||||||
const char *tail(const char *path)
|
const char *tail(const char *path)
|
||||||
{
|
{
|
||||||
const char *slash = strrchr(path, '/');
|
const char *slash = strrchr(path, '/');
|
||||||
|
|
||||||
if (slash == NULL)
|
if (slash == NULL)
|
||||||
return path;
|
return path;
|
||||||
else
|
else
|
||||||
return ++slash;
|
return ++slash;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return a copy of the two given strings, welded together. */
|
/* Return a copy of the two given strings, welded together. */
|
||||||
char *concatenate(const char *path, const char *name)
|
char *concatenate(const char *path, const char *name)
|
||||||
{
|
{
|
||||||
size_t pathlen = strlen(path);
|
size_t pathlen = strlen(path);
|
||||||
char *joined = charalloc(pathlen + strlen(name) + 1);
|
char *joined = charalloc(pathlen + strlen(name) + 1);
|
||||||
|
|
||||||
strcpy(joined, path);
|
strcpy(joined, path);
|
||||||
strcpy(joined + pathlen, name);
|
strcpy(joined + pathlen, name);
|
||||||
|
|
||||||
return joined;
|
return joined;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_LINENUMBERS
|
#ifdef ENABLE_LINENUMBERS
|
||||||
/* Return the number of digits that the given integer n takes up. */
|
/* Return the number of digits that the given integer n takes up. */
|
||||||
int digits(ssize_t n)
|
int digits(ssize_t n)
|
||||||
{
|
{
|
||||||
if (n < 100000) {
|
if (n < 100000) {
|
||||||
if (n < 1000) {
|
if (n < 1000) {
|
||||||
if (n < 100)
|
if (n < 100)
|
||||||
return 2;
|
return 2;
|
||||||
else
|
else
|
||||||
return 3;
|
return 3;
|
||||||
|
} else {
|
||||||
|
if (n < 10000)
|
||||||
|
return 4;
|
||||||
|
else
|
||||||
|
return 5;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
if (n < 10000)
|
if (n < 10000000) {
|
||||||
return 4;
|
if (n < 1000000)
|
||||||
else
|
return 6;
|
||||||
return 5;
|
else
|
||||||
|
return 7;
|
||||||
|
} else {
|
||||||
|
if (n < 100000000)
|
||||||
|
return 8;
|
||||||
|
else
|
||||||
|
return 9;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
if (n < 10000000) {
|
|
||||||
if (n < 1000000)
|
|
||||||
return 6;
|
|
||||||
else
|
|
||||||
return 7;
|
|
||||||
} else {
|
|
||||||
if (n < 100000000)
|
|
||||||
return 8;
|
|
||||||
else
|
|
||||||
return 9;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -112,98 +112,98 @@ int digits(ssize_t n)
|
||||||
* and return TRUE; otherwise, return FALSE. */
|
* and return TRUE; otherwise, return FALSE. */
|
||||||
bool parse_num(const char *str, ssize_t *result)
|
bool parse_num(const char *str, ssize_t *result)
|
||||||
{
|
{
|
||||||
char *first_error;
|
char *first_error;
|
||||||
ssize_t value;
|
ssize_t value;
|
||||||
|
|
||||||
/* The manual page for strtol() says this is required. */
|
/* The manual page for strtol() says this is required. */
|
||||||
errno = 0;
|
errno = 0;
|
||||||
|
|
||||||
value = (ssize_t)strtol(str, &first_error, 10);
|
value = (ssize_t)strtol(str, &first_error, 10);
|
||||||
|
|
||||||
if (errno == ERANGE || *str == '\0' || *first_error != '\0')
|
if (errno == ERANGE || *str == '\0' || *first_error != '\0')
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
*result = value;
|
*result = value;
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Read two numbers, separated by a comma, from str, and store them in
|
/* Read two numbers, separated by a comma, from str, and store them in
|
||||||
* *line and *column. Return FALSE on error, and TRUE otherwise. */
|
* *line and *column. Return FALSE on error, and TRUE otherwise. */
|
||||||
bool parse_line_column(const char *str, ssize_t *line, ssize_t *column)
|
bool parse_line_column(const char *str, ssize_t *line, ssize_t *column)
|
||||||
{
|
{
|
||||||
bool retval;
|
bool retval;
|
||||||
char *firstpart;
|
char *firstpart;
|
||||||
const char *comma;
|
const char *comma;
|
||||||
|
|
||||||
while (*str == ' ')
|
while (*str == ' ')
|
||||||
str++;
|
str++;
|
||||||
|
|
||||||
comma = strpbrk(str, "m,. /;");
|
comma = strpbrk(str, "m,. /;");
|
||||||
|
|
||||||
if (comma == NULL)
|
if (comma == NULL)
|
||||||
return parse_num(str, line);
|
return parse_num(str, line);
|
||||||
|
|
||||||
retval = parse_num(comma + 1, column);
|
retval = parse_num(comma + 1, column);
|
||||||
|
|
||||||
|
if (comma == str)
|
||||||
|
return retval;
|
||||||
|
|
||||||
|
firstpart = mallocstrcpy(NULL, str);
|
||||||
|
firstpart[comma - str] = '\0';
|
||||||
|
|
||||||
|
retval = parse_num(firstpart, line) && retval;
|
||||||
|
|
||||||
|
free(firstpart);
|
||||||
|
|
||||||
if (comma == str)
|
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
firstpart = mallocstrcpy(NULL, str);
|
|
||||||
firstpart[comma - str] = '\0';
|
|
||||||
|
|
||||||
retval = parse_num(firstpart, line) && retval;
|
|
||||||
|
|
||||||
free(firstpart);
|
|
||||||
|
|
||||||
return retval;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Reduce the memory allocation of a string to what is needed. */
|
/* Reduce the memory allocation of a string to what is needed. */
|
||||||
void snuggly_fit(char **str)
|
void snuggly_fit(char **str)
|
||||||
{
|
{
|
||||||
if (*str != NULL)
|
if (*str != NULL)
|
||||||
*str = charealloc(*str, strlen(*str) + 1);
|
*str = charealloc(*str, strlen(*str) + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Null a string at a certain index and align it. */
|
/* Null a string at a certain index and align it. */
|
||||||
void null_at(char **data, size_t index)
|
void null_at(char **data, size_t index)
|
||||||
{
|
{
|
||||||
*data = charealloc(*data, index + 1);
|
*data = charealloc(*data, index + 1);
|
||||||
(*data)[index] = '\0';
|
(*data)[index] = '\0';
|
||||||
}
|
}
|
||||||
|
|
||||||
/* For non-null-terminated lines. A line, by definition, shouldn't
|
/* For non-null-terminated lines. A line, by definition, shouldn't
|
||||||
* normally have newlines in it, so encode its nulls as newlines. */
|
* normally have newlines in it, so encode its nulls as newlines. */
|
||||||
void unsunder(char *str, size_t true_len)
|
void unsunder(char *str, size_t true_len)
|
||||||
{
|
{
|
||||||
for (; true_len > 0; true_len--, str++) {
|
for (; true_len > 0; true_len--, str++) {
|
||||||
if (*str == '\0')
|
if (*str == '\0')
|
||||||
*str = '\n';
|
*str = '\n';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* For non-null-terminated lines. A line, by definition, shouldn't
|
/* For non-null-terminated lines. A line, by definition, shouldn't
|
||||||
* normally have newlines in it, so decode its newlines as nulls. */
|
* normally have newlines in it, so decode its newlines as nulls. */
|
||||||
void sunder(char *str)
|
void sunder(char *str)
|
||||||
{
|
{
|
||||||
for (; *str != '\0'; str++) {
|
for (; *str != '\0'; str++) {
|
||||||
if (*str == '\n')
|
if (*str == '\n')
|
||||||
*str = '\0';
|
*str = '\0';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if !defined(ENABLE_TINY) || defined(ENABLE_TABCOMP) || defined(ENABLE_BROWSER)
|
#if !defined(ENABLE_TINY) || defined(ENABLE_TABCOMP) || defined(ENABLE_BROWSER)
|
||||||
/* Free the memory of the given array, which should contain len elements. */
|
/* Free the memory of the given array, which should contain len elements. */
|
||||||
void free_chararray(char **array, size_t len)
|
void free_chararray(char **array, size_t len)
|
||||||
{
|
{
|
||||||
if (array == NULL)
|
if (array == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
while (len > 0)
|
while (len > 0)
|
||||||
free(array[--len]);
|
free(array[--len]);
|
||||||
|
|
||||||
free(array);
|
free(array);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -212,30 +212,30 @@ void free_chararray(char **array, size_t len)
|
||||||
const char *fixbounds(const char *r)
|
const char *fixbounds(const char *r)
|
||||||
{
|
{
|
||||||
#ifndef GNU_WORDBOUNDS
|
#ifndef GNU_WORDBOUNDS
|
||||||
int i, j = 0;
|
int i, j = 0;
|
||||||
char *r2 = charalloc(strlen(r) * 5);
|
char *r2 = charalloc(strlen(r) * 5);
|
||||||
char *r3;
|
char *r3;
|
||||||
|
|
||||||
for (i = 0; i < strlen(r); i++) {
|
for (i = 0; i < strlen(r); i++) {
|
||||||
if (r[i] != '\0' && r[i] == '\\' && (r[i + 1] == '>' || r[i + 1] == '<')) {
|
if (r[i] != '\0' && r[i] == '\\' && (r[i + 1] == '>' || r[i + 1] == '<')) {
|
||||||
strcpy(&r2[j], "[[:");
|
strcpy(&r2[j], "[[:");
|
||||||
r2[j + 3] = r[i + 1];
|
r2[j + 3] = r[i + 1];
|
||||||
strcpy(&r2[j + 4], ":]]");
|
strcpy(&r2[j + 4], ":]]");
|
||||||
i++;
|
i++;
|
||||||
j += 6;
|
j += 6;
|
||||||
} else
|
} else
|
||||||
r2[j] = r[i];
|
r2[j] = r[i];
|
||||||
j++;
|
j++;
|
||||||
}
|
}
|
||||||
|
|
||||||
r2[j] = '\0';
|
r2[j] = '\0';
|
||||||
r3 = mallocstrcpy(NULL, r2);
|
r3 = mallocstrcpy(NULL, r2);
|
||||||
free(r2);
|
free(r2);
|
||||||
|
|
||||||
return (const char *) r3;
|
return (const char *) r3;
|
||||||
#endif /* !GNU_WORDBOUNDS */
|
#endif /* !GNU_WORDBOUNDS */
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_SPELLER
|
#ifdef ENABLE_SPELLER
|
||||||
|
@ -243,18 +243,18 @@ const char *fixbounds(const char *r)
|
||||||
* a separate word? That is: is it not part of a longer word?*/
|
* a separate word? That is: is it not part of a longer word?*/
|
||||||
bool is_separate_word(size_t position, size_t length, const char *buf)
|
bool is_separate_word(size_t position, size_t length, const char *buf)
|
||||||
{
|
{
|
||||||
char before[MAXCHARLEN], after[MAXCHARLEN];
|
char before[MAXCHARLEN], after[MAXCHARLEN];
|
||||||
size_t word_end = position + length;
|
size_t word_end = position + length;
|
||||||
|
|
||||||
/* Get the characters before and after the word, if any. */
|
/* Get the characters before and after the word, if any. */
|
||||||
parse_mbchar(buf + move_mbleft(buf, position), before, NULL);
|
parse_mbchar(buf + move_mbleft(buf, position), before, NULL);
|
||||||
parse_mbchar(buf + word_end, after, NULL);
|
parse_mbchar(buf + word_end, after, NULL);
|
||||||
|
|
||||||
/* If the word starts at the beginning of the line OR the character before
|
/* If the word starts at the beginning of the line OR the character before
|
||||||
* the word isn't a letter, and if the word ends at the end of the line OR
|
* the word isn't a letter, and if the word ends at the end of the line OR
|
||||||
* the character after the word isn't a letter, we have a whole word. */
|
* the character after the word isn't a letter, we have a whole word. */
|
||||||
return ((position == 0 || !is_alpha_mbchar(before)) &&
|
return ((position == 0 || !is_alpha_mbchar(before)) &&
|
||||||
(buf[word_end] == '\0' || !is_alpha_mbchar(after)));
|
(buf[word_end] == '\0' || !is_alpha_mbchar(after)));
|
||||||
}
|
}
|
||||||
#endif /* ENABLE_SPELLER */
|
#endif /* ENABLE_SPELLER */
|
||||||
|
|
||||||
|
@ -264,76 +264,76 @@ bool is_separate_word(size_t position, size_t length, const char *buf)
|
||||||
* than start. If we are doing a regexp search, and we find a match, we fill
|
* than start. If we are doing a regexp search, and we find a match, we fill
|
||||||
* in the global variable regmatches with at most 9 subexpression matches. */
|
* in the global variable regmatches with at most 9 subexpression matches. */
|
||||||
const char *strstrwrapper(const char *haystack, const char *needle,
|
const char *strstrwrapper(const char *haystack, const char *needle,
|
||||||
const char *start)
|
const char *start)
|
||||||
{
|
{
|
||||||
if (*needle == '\0') {
|
if (*needle == '\0') {
|
||||||
#ifndef NANO_TINY
|
#ifndef NANO_TINY
|
||||||
statusline(ALERT, "Searching for nothing -- please report a bug");
|
statusline(ALERT, "Searching for nothing -- please report a bug");
|
||||||
#endif
|
#endif
|
||||||
return (char *)start;
|
return (char *)start;
|
||||||
}
|
|
||||||
|
|
||||||
if (ISSET(USE_REGEXP)) {
|
|
||||||
if (ISSET(BACKWARDS_SEARCH)) {
|
|
||||||
size_t last_find, ceiling, far_end;
|
|
||||||
size_t floor = 0, next_rung = 0;
|
|
||||||
/* The start of the search range, and the next start. */
|
|
||||||
|
|
||||||
if (regexec(&search_regexp, haystack, 1, regmatches, 0) != 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
far_end = strlen(haystack);
|
|
||||||
ceiling = start - haystack;
|
|
||||||
last_find = regmatches[0].rm_so;
|
|
||||||
|
|
||||||
/* A result beyond the search range also means: no match. */
|
|
||||||
if (last_find > ceiling)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
/* Move the start-of-range forward until there is no more match;
|
|
||||||
* then the last match found is the first match backwards. */
|
|
||||||
while (regmatches[0].rm_so <= ceiling) {
|
|
||||||
floor = next_rung;
|
|
||||||
last_find = regmatches[0].rm_so;
|
|
||||||
/* If this is the last possible match, don't try to advance. */
|
|
||||||
if (last_find == ceiling)
|
|
||||||
break;
|
|
||||||
next_rung = move_mbright(haystack, last_find);
|
|
||||||
regmatches[0].rm_so = next_rung;
|
|
||||||
regmatches[0].rm_eo = far_end;
|
|
||||||
if (regexec(&search_regexp, haystack, 1, regmatches,
|
|
||||||
REG_STARTEND) != 0)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Find the last match again, to get possible submatches. */
|
|
||||||
regmatches[0].rm_so = floor;
|
|
||||||
regmatches[0].rm_eo = far_end;
|
|
||||||
if (regexec(&search_regexp, haystack, 10, regmatches,
|
|
||||||
REG_STARTEND) != 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
return haystack + regmatches[0].rm_so;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Do a forward regex search from the starting point. */
|
if (ISSET(USE_REGEXP)) {
|
||||||
regmatches[0].rm_so = start - haystack;
|
if (ISSET(BACKWARDS_SEARCH)) {
|
||||||
regmatches[0].rm_eo = strlen(haystack);
|
size_t last_find, ceiling, far_end;
|
||||||
if (regexec(&search_regexp, haystack, 10, regmatches,
|
size_t floor = 0, next_rung = 0;
|
||||||
REG_STARTEND) != 0)
|
/* The start of the search range, and the next start. */
|
||||||
return NULL;
|
|
||||||
else
|
|
||||||
return haystack + regmatches[0].rm_so;
|
|
||||||
}
|
|
||||||
if (ISSET(CASE_SENSITIVE)) {
|
|
||||||
if (ISSET(BACKWARDS_SEARCH))
|
|
||||||
return revstrstr(haystack, needle, start);
|
|
||||||
else
|
|
||||||
return strstr(start, needle);
|
|
||||||
} else if (ISSET(BACKWARDS_SEARCH))
|
|
||||||
return mbrevstrcasestr(haystack, needle, start);
|
|
||||||
|
|
||||||
return mbstrcasestr(start, needle);
|
if (regexec(&search_regexp, haystack, 1, regmatches, 0) != 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
far_end = strlen(haystack);
|
||||||
|
ceiling = start - haystack;
|
||||||
|
last_find = regmatches[0].rm_so;
|
||||||
|
|
||||||
|
/* A result beyond the search range also means: no match. */
|
||||||
|
if (last_find > ceiling)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
/* Move the start-of-range forward until there is no more match;
|
||||||
|
* then the last match found is the first match backwards. */
|
||||||
|
while (regmatches[0].rm_so <= ceiling) {
|
||||||
|
floor = next_rung;
|
||||||
|
last_find = regmatches[0].rm_so;
|
||||||
|
/* If this is the last possible match, don't try to advance. */
|
||||||
|
if (last_find == ceiling)
|
||||||
|
break;
|
||||||
|
next_rung = move_mbright(haystack, last_find);
|
||||||
|
regmatches[0].rm_so = next_rung;
|
||||||
|
regmatches[0].rm_eo = far_end;
|
||||||
|
if (regexec(&search_regexp, haystack, 1, regmatches,
|
||||||
|
REG_STARTEND) != 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Find the last match again, to get possible submatches. */
|
||||||
|
regmatches[0].rm_so = floor;
|
||||||
|
regmatches[0].rm_eo = far_end;
|
||||||
|
if (regexec(&search_regexp, haystack, 10, regmatches,
|
||||||
|
REG_STARTEND) != 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
return haystack + regmatches[0].rm_so;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Do a forward regex search from the starting point. */
|
||||||
|
regmatches[0].rm_so = start - haystack;
|
||||||
|
regmatches[0].rm_eo = strlen(haystack);
|
||||||
|
if (regexec(&search_regexp, haystack, 10, regmatches,
|
||||||
|
REG_STARTEND) != 0)
|
||||||
|
return NULL;
|
||||||
|
else
|
||||||
|
return haystack + regmatches[0].rm_so;
|
||||||
|
}
|
||||||
|
if (ISSET(CASE_SENSITIVE)) {
|
||||||
|
if (ISSET(BACKWARDS_SEARCH))
|
||||||
|
return revstrstr(haystack, needle, start);
|
||||||
|
else
|
||||||
|
return strstr(start, needle);
|
||||||
|
} else if (ISSET(BACKWARDS_SEARCH))
|
||||||
|
return mbrevstrcasestr(haystack, needle, start);
|
||||||
|
|
||||||
|
return mbstrcasestr(start, needle);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is a wrapper for the perror() function. The wrapper temporarily
|
/* This is a wrapper for the perror() function. The wrapper temporarily
|
||||||
|
@ -342,63 +342,63 @@ const char *strstrwrapper(const char *haystack, const char *needle,
|
||||||
* nperror() causes the window to flicker once. */
|
* nperror() causes the window to flicker once. */
|
||||||
void nperror(const char *s)
|
void nperror(const char *s)
|
||||||
{
|
{
|
||||||
endwin();
|
endwin();
|
||||||
perror(s);
|
perror(s);
|
||||||
doupdate();
|
doupdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is a wrapper for the malloc() function that properly handles
|
/* This is a wrapper for the malloc() function that properly handles
|
||||||
* things when we run out of memory. */
|
* things when we run out of memory. */
|
||||||
void *nmalloc(size_t howmuch)
|
void *nmalloc(size_t howmuch)
|
||||||
{
|
{
|
||||||
void *r = malloc(howmuch);
|
void *r = malloc(howmuch);
|
||||||
|
|
||||||
if (r == NULL && howmuch != 0)
|
if (r == NULL && howmuch != 0)
|
||||||
die(_("nano is out of memory!"));
|
die(_("nano is out of memory!"));
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is a wrapper for the realloc() function that properly handles
|
/* This is a wrapper for the realloc() function that properly handles
|
||||||
* things when we run out of memory. */
|
* things when we run out of memory. */
|
||||||
void *nrealloc(void *ptr, size_t howmuch)
|
void *nrealloc(void *ptr, size_t howmuch)
|
||||||
{
|
{
|
||||||
void *r = realloc(ptr, howmuch);
|
void *r = realloc(ptr, howmuch);
|
||||||
|
|
||||||
if (r == NULL && howmuch != 0)
|
if (r == NULL && howmuch != 0)
|
||||||
die(_("nano is out of memory!"));
|
die(_("nano is out of memory!"));
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate and copy the first n characters of the given src string, after
|
/* Allocate and copy the first n characters of the given src string, after
|
||||||
* freeing the destination. Usage: "dest = mallocstrncpy(dest, src, n);". */
|
* freeing the destination. Usage: "dest = mallocstrncpy(dest, src, n);". */
|
||||||
char *mallocstrncpy(char *dest, const char *src, size_t n)
|
char *mallocstrncpy(char *dest, const char *src, size_t n)
|
||||||
{
|
{
|
||||||
if (src == NULL)
|
if (src == NULL)
|
||||||
src = "";
|
src = "";
|
||||||
|
|
||||||
if (src != dest)
|
if (src != dest)
|
||||||
free(dest);
|
free(dest);
|
||||||
|
|
||||||
dest = charalloc(n);
|
dest = charalloc(n);
|
||||||
strncpy(dest, src, n);
|
strncpy(dest, src, n);
|
||||||
|
|
||||||
return dest;
|
return dest;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free the dest string and return a malloc'ed copy of src. Should be used as:
|
/* Free the dest string and return a malloc'ed copy of src. Should be used as:
|
||||||
* "dest = mallocstrcpy(dest, src);". */
|
* "dest = mallocstrcpy(dest, src);". */
|
||||||
char *mallocstrcpy(char *dest, const char *src)
|
char *mallocstrcpy(char *dest, const char *src)
|
||||||
{
|
{
|
||||||
return mallocstrncpy(dest, src, (src == NULL) ? 1 : strlen(src) + 1);
|
return mallocstrncpy(dest, src, (src == NULL) ? 1 : strlen(src) + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free the string at dest and return the string at src. */
|
/* Free the string at dest and return the string at src. */
|
||||||
char *free_and_assign(char *dest, char *src)
|
char *free_and_assign(char *dest, char *src)
|
||||||
{
|
{
|
||||||
free(dest);
|
free(dest);
|
||||||
return src;
|
return src;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* When not in softwrap mode, nano scrolls horizontally within a line in
|
/* When not in softwrap mode, nano scrolls horizontally within a line in
|
||||||
|
@ -408,83 +408,83 @@ char *free_and_assign(char *dest, char *src)
|
||||||
* get_page_start(column) < COLS). */
|
* get_page_start(column) < COLS). */
|
||||||
size_t get_page_start(size_t column)
|
size_t get_page_start(size_t column)
|
||||||
{
|
{
|
||||||
if (column < editwincols - 1 || ISSET(SOFTWRAP) || column == 0)
|
if (column < editwincols - 1 || ISSET(SOFTWRAP) || column == 0)
|
||||||
return 0;
|
return 0;
|
||||||
else if (editwincols > 8)
|
else if (editwincols > 8)
|
||||||
return column - 7 - (column - 7) % (editwincols - 8);
|
return column - 7 - (column - 7) % (editwincols - 8);
|
||||||
else
|
else
|
||||||
return column - (editwincols - 2);
|
return column - (editwincols - 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the placewewant associated with current_x, i.e. the zero-based
|
/* Return the placewewant associated with current_x, i.e. the zero-based
|
||||||
* column position of the cursor. */
|
* column position of the cursor. */
|
||||||
size_t xplustabs(void)
|
size_t xplustabs(void)
|
||||||
{
|
{
|
||||||
return strnlenpt(openfile->current->data, openfile->current_x);
|
return strnlenpt(openfile->current->data, openfile->current_x);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the index in text of the character that (when displayed) will
|
/* Return the index in text of the character that (when displayed) will
|
||||||
* not overshoot the given column. */
|
* not overshoot the given column. */
|
||||||
size_t actual_x(const char *text, size_t column)
|
size_t actual_x(const char *text, size_t column)
|
||||||
{
|
{
|
||||||
const char *start = text;
|
const char *start = text;
|
||||||
/* From where we start walking through the text. */
|
/* From where we start walking through the text. */
|
||||||
size_t width = 0;
|
size_t width = 0;
|
||||||
/* The current accumulated span, in columns. */
|
/* The current accumulated span, in columns. */
|
||||||
|
|
||||||
while (*text != '\0') {
|
while (*text != '\0') {
|
||||||
int charlen = parse_mbchar(text, NULL, &width);
|
int charlen = parse_mbchar(text, NULL, &width);
|
||||||
|
|
||||||
if (width > column)
|
if (width > column)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
text += charlen;
|
text += charlen;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (text - start);
|
return (text - start);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* A strnlen() with tabs and multicolumn characters factored in:
|
/* A strnlen() with tabs and multicolumn characters factored in:
|
||||||
* how many columns wide are the first maxlen bytes of text? */
|
* how many columns wide are the first maxlen bytes of text? */
|
||||||
size_t strnlenpt(const char *text, size_t maxlen)
|
size_t strnlenpt(const char *text, size_t maxlen)
|
||||||
{
|
{
|
||||||
size_t width = 0;
|
size_t width = 0;
|
||||||
/* The screen display width to text[maxlen]. */
|
/* The screen display width to text[maxlen]. */
|
||||||
|
|
||||||
if (maxlen == 0)
|
if (maxlen == 0)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
while (*text != '\0') {
|
while (*text != '\0') {
|
||||||
int charlen = parse_mbchar(text, NULL, &width);
|
int charlen = parse_mbchar(text, NULL, &width);
|
||||||
|
|
||||||
if (maxlen <= charlen)
|
if (maxlen <= charlen)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
maxlen -= charlen;
|
maxlen -= charlen;
|
||||||
text += charlen;
|
text += charlen;
|
||||||
}
|
}
|
||||||
|
|
||||||
return width;
|
return width;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the number of columns that the given text occupies. */
|
/* Return the number of columns that the given text occupies. */
|
||||||
size_t strlenpt(const char *text)
|
size_t strlenpt(const char *text)
|
||||||
{
|
{
|
||||||
size_t span = 0;
|
size_t span = 0;
|
||||||
|
|
||||||
while (*text != '\0')
|
while (*text != '\0')
|
||||||
text += parse_mbchar(text, NULL, &span);
|
text += parse_mbchar(text, NULL, &span);
|
||||||
|
|
||||||
return span;
|
return span;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Append a new magicline to the end of the buffer. */
|
/* Append a new magicline to the end of the buffer. */
|
||||||
void new_magicline(void)
|
void new_magicline(void)
|
||||||
{
|
{
|
||||||
openfile->filebot->next = make_new_node(openfile->filebot);
|
openfile->filebot->next = make_new_node(openfile->filebot);
|
||||||
openfile->filebot->next->data = mallocstrcpy(NULL, "");
|
openfile->filebot->next->data = mallocstrcpy(NULL, "");
|
||||||
openfile->filebot = openfile->filebot->next;
|
openfile->filebot = openfile->filebot->next;
|
||||||
openfile->totsize++;
|
openfile->totsize++;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if !defined(NANO_TINY) || defined(ENABLE_HELP)
|
#if !defined(NANO_TINY) || defined(ENABLE_HELP)
|
||||||
|
@ -492,13 +492,13 @@ void new_magicline(void)
|
||||||
* it isn't the only line in the file. */
|
* it isn't the only line in the file. */
|
||||||
void remove_magicline(void)
|
void remove_magicline(void)
|
||||||
{
|
{
|
||||||
if (openfile->filebot->data[0] == '\0' &&
|
if (openfile->filebot->data[0] == '\0' &&
|
||||||
openfile->filebot != openfile->fileage) {
|
openfile->filebot != openfile->fileage) {
|
||||||
openfile->filebot = openfile->filebot->prev;
|
openfile->filebot = openfile->filebot->prev;
|
||||||
free_filestruct(openfile->filebot->next);
|
free_filestruct(openfile->filebot->next);
|
||||||
openfile->filebot->next = NULL;
|
openfile->filebot->next = NULL;
|
||||||
openfile->totsize--;
|
openfile->totsize--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -507,25 +507,25 @@ void remove_magicline(void)
|
||||||
* the marked region. If right_side_up isn't NULL, set it to TRUE when the
|
* the marked region. If right_side_up isn't NULL, set it to TRUE when the
|
||||||
* mark is at the top of the marked region, and to FALSE otherwise. */
|
* mark is at the top of the marked region, and to FALSE otherwise. */
|
||||||
void mark_order(const filestruct **top, size_t *top_x,
|
void mark_order(const filestruct **top, size_t *top_x,
|
||||||
const filestruct **bot, size_t *bot_x, bool *right_side_up)
|
const filestruct **bot, size_t *bot_x, bool *right_side_up)
|
||||||
{
|
{
|
||||||
if ((openfile->current->lineno == openfile->mark->lineno &&
|
if ((openfile->current->lineno == openfile->mark->lineno &&
|
||||||
openfile->current_x > openfile->mark_x) ||
|
openfile->current_x > openfile->mark_x) ||
|
||||||
openfile->current->lineno > openfile->mark->lineno) {
|
openfile->current->lineno > openfile->mark->lineno) {
|
||||||
*top = openfile->mark;
|
*top = openfile->mark;
|
||||||
*top_x = openfile->mark_x;
|
*top_x = openfile->mark_x;
|
||||||
*bot = openfile->current;
|
*bot = openfile->current;
|
||||||
*bot_x = openfile->current_x;
|
*bot_x = openfile->current_x;
|
||||||
if (right_side_up != NULL)
|
if (right_side_up != NULL)
|
||||||
*right_side_up = TRUE;
|
*right_side_up = TRUE;
|
||||||
} else {
|
} else {
|
||||||
*bot = openfile->mark;
|
*bot = openfile->mark;
|
||||||
*bot_x = openfile->mark_x;
|
*bot_x = openfile->mark_x;
|
||||||
*top = openfile->current;
|
*top = openfile->current;
|
||||||
*top_x = openfile->current_x;
|
*top_x = openfile->current_x;
|
||||||
if (right_side_up != NULL)
|
if (right_side_up != NULL)
|
||||||
*right_side_up = FALSE;
|
*right_side_up = FALSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the set of lines to work on -- either just the current line, or the
|
/* Get the set of lines to work on -- either just the current line, or the
|
||||||
|
@ -533,74 +533,74 @@ void mark_order(const filestruct **top, size_t *top_x,
|
||||||
* at the start of the last line of the region, exclude that line. */
|
* at the start of the last line of the region, exclude that line. */
|
||||||
void get_range(const filestruct **top, const filestruct **bot)
|
void get_range(const filestruct **top, const filestruct **bot)
|
||||||
{
|
{
|
||||||
if (!openfile->mark) {
|
if (!openfile->mark) {
|
||||||
*top = openfile->current;
|
*top = openfile->current;
|
||||||
*bot = openfile->current;
|
*bot = openfile->current;
|
||||||
} else {
|
} else {
|
||||||
size_t top_x, bot_x;
|
size_t top_x, bot_x;
|
||||||
|
|
||||||
mark_order(top, &top_x, bot, &bot_x, NULL);
|
mark_order(top, &top_x, bot, &bot_x, NULL);
|
||||||
|
|
||||||
if (bot_x == 0 && *bot != *top && !also_the_last)
|
if (bot_x == 0 && *bot != *top && !also_the_last)
|
||||||
*bot = (*bot)->prev;
|
*bot = (*bot)->prev;
|
||||||
else
|
else
|
||||||
also_the_last = TRUE;
|
also_the_last = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Given a line number, return a pointer to the corresponding struct. */
|
/* Given a line number, return a pointer to the corresponding struct. */
|
||||||
filestruct *fsfromline(ssize_t lineno)
|
filestruct *fsfromline(ssize_t lineno)
|
||||||
{
|
{
|
||||||
filestruct *f = openfile->current;
|
filestruct *f = openfile->current;
|
||||||
|
|
||||||
if (lineno <= openfile->current->lineno)
|
if (lineno <= openfile->current->lineno)
|
||||||
while (f->lineno != lineno && f->prev != NULL)
|
while (f->lineno != lineno && f->prev != NULL)
|
||||||
f = f->prev;
|
f = f->prev;
|
||||||
else
|
else
|
||||||
while (f->lineno != lineno && f->next != NULL)
|
while (f->lineno != lineno && f->next != NULL)
|
||||||
f = f->next;
|
f = f->next;
|
||||||
|
|
||||||
if (f->lineno != lineno) {
|
if (f->lineno != lineno) {
|
||||||
statusline(ALERT, "Gone undo line -- please report a bug");
|
statusline(ALERT, "Gone undo line -- please report a bug");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
return f;
|
return f;
|
||||||
}
|
}
|
||||||
#endif /* !NANO_TINY */
|
#endif /* !NANO_TINY */
|
||||||
|
|
||||||
/* Count the number of characters from begin to end, and return it. */
|
/* Count the number of characters from begin to end, and return it. */
|
||||||
size_t get_totsize(const filestruct *begin, const filestruct *end)
|
size_t get_totsize(const filestruct *begin, const filestruct *end)
|
||||||
{
|
{
|
||||||
const filestruct *line;
|
const filestruct *line;
|
||||||
size_t totsize = 0;
|
size_t totsize = 0;
|
||||||
|
|
||||||
/* Sum the number of characters (plus a newline) in each line. */
|
/* Sum the number of characters (plus a newline) in each line. */
|
||||||
for (line = begin; line != end->next; line = line->next)
|
for (line = begin; line != end->next; line = line->next)
|
||||||
totsize += mbstrlen(line->data) + 1;
|
totsize += mbstrlen(line->data) + 1;
|
||||||
|
|
||||||
/* The last line of a file doesn't have a newline -- otherwise it
|
/* The last line of a file doesn't have a newline -- otherwise it
|
||||||
* wouldn't be the last line -- so subtract 1 when at EOF. */
|
* wouldn't be the last line -- so subtract 1 when at EOF. */
|
||||||
if (line == NULL)
|
if (line == NULL)
|
||||||
totsize--;
|
totsize--;
|
||||||
|
|
||||||
return totsize;
|
return totsize;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
/* Dump the given buffer to stderr. */
|
/* Dump the given buffer to stderr. */
|
||||||
void dump_filestruct(const filestruct *inptr)
|
void dump_filestruct(const filestruct *inptr)
|
||||||
{
|
{
|
||||||
if (inptr == openfile->fileage)
|
if (inptr == openfile->fileage)
|
||||||
fprintf(stderr, "Dumping file buffer to stderr...\n");
|
fprintf(stderr, "Dumping file buffer to stderr...\n");
|
||||||
else if (inptr == cutbuffer)
|
else if (inptr == cutbuffer)
|
||||||
fprintf(stderr, "Dumping cutbuffer to stderr...\n");
|
fprintf(stderr, "Dumping cutbuffer to stderr...\n");
|
||||||
else
|
else
|
||||||
fprintf(stderr, "Dumping a buffer to stderr...\n");
|
fprintf(stderr, "Dumping a buffer to stderr...\n");
|
||||||
|
|
||||||
while (inptr != NULL) {
|
while (inptr != NULL) {
|
||||||
fprintf(stderr, "(%zd) %s\n", inptr->lineno, inptr->data);
|
fprintf(stderr, "(%zd) %s\n", inptr->lineno, inptr->data);
|
||||||
inptr = inptr->next;
|
inptr = inptr->next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif /* DEBUG */
|
#endif /* DEBUG */
|
||||||
|
|
5158
src/winio.c
5158
src/winio.c
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue