mirror of
https://github.com/troglobit/editline.git
synced 2025-09-18 19:10:09 +08:00
Compare commits
45 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
7355f56d25 | ||
![]() |
5496152d51 | ||
![]() |
13f8d5f69c | ||
![]() |
c2b65646ea | ||
![]() |
e2c3b41d9a | ||
![]() |
4a17180d37 | ||
![]() |
3e046cafd2 | ||
![]() |
0e493addab | ||
![]() |
842bd67dd4 | ||
![]() |
0eb8f228f7 | ||
![]() |
8f6d724e30 | ||
![]() |
ee176c80d7 | ||
![]() |
979f05a5eb | ||
![]() |
c95d25731d | ||
![]() |
61d40f406f | ||
![]() |
385cd77e9b | ||
![]() |
448a3dac08 | ||
![]() |
7506826ac7 | ||
![]() |
ec7f752095 | ||
![]() |
959a9e7199 | ||
![]() |
397ac9dd56 | ||
![]() |
e58868bec1 | ||
![]() |
650d0f15c6 | ||
![]() |
2aee394b44 | ||
![]() |
dd1af360c3 | ||
![]() |
8660aef4b7 | ||
![]() |
6a4d77ca69 | ||
![]() |
0b554cf8c8 | ||
![]() |
8f5a5da754 | ||
![]() |
a237e39181 | ||
![]() |
3cb74b6d87 | ||
![]() |
5c8429dcee | ||
![]() |
8566eb8384 | ||
![]() |
8e34e4e417 | ||
![]() |
a4b67d2268 | ||
![]() |
0b7142eb8e | ||
![]() |
2137b9df9f | ||
![]() |
66d8ae84e2 | ||
![]() |
fbb1f8800a | ||
![]() |
cddd8d8de0 | ||
![]() |
1db83fe6c6 | ||
![]() |
3cd9894747 | ||
![]() |
4784fe2491 | ||
![]() |
4ec7d26a9d | ||
![]() |
ebefa8b890 |
20
ChangeLog.md
20
ChangeLog.md
@@ -4,6 +4,25 @@ Change Log
|
|||||||
All notable changes to the project are documented in this file.
|
All notable changes to the project are documented in this file.
|
||||||
|
|
||||||
|
|
||||||
|
[1.16.1] - 2019-06-07
|
||||||
|
---------------------
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
- Major updates to the `editline.3` man page
|
||||||
|
- Cleanup of examples `cli.c` and `fileman.c`
|
||||||
|
- Add example of hidden input prompt to `cli.c`
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Fix #20: `configure --disable-eof` does not bite
|
||||||
|
- Fix #23: Make Ctrl-L clear the screan instead of starting a new line
|
||||||
|
Like Ctrl-D, which exits, Ctrl-L only clears the screen when the line
|
||||||
|
is empty and the cursor is at the start of the line, otherwise Ctrl-L
|
||||||
|
will redraw/refresh the current line.
|
||||||
|
- Fix #24: Fix behavior when TTY is narrower than column width, by Will Dietz
|
||||||
|
- Fix #25: Avoid continuously duplicate commands in history
|
||||||
|
- Fix #31: Aborting i-search with Ctrl-C should not generate signal
|
||||||
|
|
||||||
|
|
||||||
[1.16.0][] - 2018-09-16
|
[1.16.0][] - 2018-09-16
|
||||||
-----------------------
|
-----------------------
|
||||||
|
|
||||||
@@ -204,6 +223,7 @@ Adaptations to Debian editline package.
|
|||||||
|
|
||||||
|
|
||||||
[UNRELEASED]: https://github.com/troglobit/finit/compare/1.16.0...HEAD
|
[UNRELEASED]: https://github.com/troglobit/finit/compare/1.16.0...HEAD
|
||||||
|
[1.16.1]: https://github.com/troglobit/finit/compare/1.16.0...1.16.1
|
||||||
[1.16.0]: https://github.com/troglobit/finit/compare/1.15.3...1.16.0
|
[1.16.0]: https://github.com/troglobit/finit/compare/1.15.3...1.16.0
|
||||||
[1.15.3]: https://github.com/troglobit/finit/compare/1.15.2...1.15.3
|
[1.15.3]: https://github.com/troglobit/finit/compare/1.15.2...1.15.3
|
||||||
[1.15.2]: https://github.com/troglobit/finit/compare/1.15.1...1.15.2
|
[1.15.2]: https://github.com/troglobit/finit/compare/1.15.1...1.15.2
|
||||||
|
12
README.md
12
README.md
@@ -37,9 +37,11 @@ Editline is maintained collaboratively at [GitHub][].
|
|||||||
Example
|
Example
|
||||||
-------
|
-------
|
||||||
|
|
||||||
Here is a very brief example to illustrate how one can use Editline to
|
Below is a very brief example to illustrate how one can use Editline to
|
||||||
create a simple CLI, use Ctrl-D to exit the program. More examples are
|
create a simple CLI, Ctrl-D exits the program. A slightly more advanced
|
||||||
availble in the source tree.
|
example is Jush, <https://github.com/troglobit/jush/>, a small and very
|
||||||
|
simplistic UNIX shell. The Editline sources also include an `examples/`
|
||||||
|
sub-directory.
|
||||||
|
|
||||||
1. Build and install the library, preferably using a [release tarball][]
|
1. Build and install the library, preferably using a [release tarball][]
|
||||||
The configure script defaults to a `/usr/local` prefix.
|
The configure script defaults to a `/usr/local` prefix.
|
||||||
@@ -221,7 +223,7 @@ similar to the [BSD license][]. Rich's current version is however under
|
|||||||
the Apache license. For details on the licensing terms of this version
|
the Apache license. For details on the licensing terms of this version
|
||||||
of the software, see [License][].
|
of the software, see [License][].
|
||||||
|
|
||||||
This version of the editline library was forked from the [Minix 3][]
|
This version of the editline library was forked from the [Minix 2][]
|
||||||
source tree and is *not* related to the similarily named NetBSD version
|
source tree and is *not* related to the similarily named NetBSD version
|
||||||
that [Jess Thrysøe][jess] disitributes to the world outside *BSD. The
|
that [Jess Thrysøe][jess] disitributes to the world outside *BSD. The
|
||||||
libraries have much in common, but the latter is heavily refactored and
|
libraries have much in common, but the latter is heavily refactored and
|
||||||
@@ -255,7 +257,7 @@ Outstanding issues are listed in the [TODO.md][] file.
|
|||||||
[FSF readline]: http://www.gnu.org/software/readline/
|
[FSF readline]: http://www.gnu.org/software/readline/
|
||||||
[Rich Salz]: https://github.com/richsalz/editline/
|
[Rich Salz]: https://github.com/richsalz/editline/
|
||||||
[comp.sources.unix]: http://ftp.cs.toronto.edu/pub/white/pub/rc/editline.shar
|
[comp.sources.unix]: http://ftp.cs.toronto.edu/pub/white/pub/rc/editline.shar
|
||||||
[Minix 3]: http://www.cise.ufl.edu/~cop4600/cgi-bin/lxr/http/source.cgi/lib/editline/
|
[Minix 2]: http://www.cise.ufl.edu/~cop4600/cgi-bin/lxr/http/source.cgi/lib/editline/
|
||||||
[jess]: http://thrysoee.dk/editline/
|
[jess]: http://thrysoee.dk/editline/
|
||||||
[BSD license]: http://en.wikipedia.org/wiki/BSD_licenses
|
[BSD license]: http://en.wikipedia.org/wiki/BSD_licenses
|
||||||
[libeditline]: http://packages.qa.debian.org/e/editline.html
|
[libeditline]: http://packages.qa.debian.org/e/editline.html
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
AC_INIT(editline, 1.16.0, https://github.com/troglobit/editline/issues)
|
AC_INIT(editline, 1.16.1, https://github.com/troglobit/editline/issues)
|
||||||
AM_INIT_AUTOMAKE([1.11 foreign no-dist-gzip dist-xz])
|
AM_INIT_AUTOMAKE([1.11 foreign no-dist-gzip dist-xz])
|
||||||
AM_SILENT_RULES([yes])
|
AM_SILENT_RULES([yes])
|
||||||
|
|
||||||
@@ -69,8 +69,8 @@ AS_IF([test "x$enable_unique_history" != "xno"],
|
|||||||
AS_IF([test "x$enable_terminal_bell" != "xno"],
|
AS_IF([test "x$enable_terminal_bell" != "xno"],
|
||||||
AC_DEFINE(CONFIG_ANSI_ARROWS, 1, [Define to include ANSI arrow keys support.]))
|
AC_DEFINE(CONFIG_ANSI_ARROWS, 1, [Define to include ANSI arrow keys support.]))
|
||||||
|
|
||||||
AS_IF([test "x$enable_terminal_bell" != "xno"],
|
AS_IF([test "x$enable_eof" != "xno"],
|
||||||
AC_DEFINE(CONFIG_EOF, 1, [Define to enable EOF (Ctrl-C) key.]))
|
AC_DEFINE(CONFIG_EOF, 1, [Define to enable EOF (Ctrl-D) key.]))
|
||||||
|
|
||||||
AS_IF([test "x$enable_sigint" != "xno"],
|
AS_IF([test "x$enable_sigint" != "xno"],
|
||||||
AC_DEFINE(CONFIG_SIGINT, 1, [Define to enable SIGINT (Ctrl-C) key.]))
|
AC_DEFINE(CONFIG_SIGINT, 1, [Define to enable SIGINT (Ctrl-C) key.]))
|
||||||
|
7
debian/changelog
vendored
7
debian/changelog
vendored
@@ -1,3 +1,10 @@
|
|||||||
|
editline (1.16.1) unstable; urgency=medium
|
||||||
|
|
||||||
|
* Minor bug fix and documentation update release.
|
||||||
|
* Add missing pkg-config .pc file to -dev package
|
||||||
|
|
||||||
|
-- Joachim Nilsson <troglobit@gmail.com> Fri, 07 Jun 2019 11:45:50 +0200
|
||||||
|
|
||||||
editline (1.16.0) unstable; urgency=medium
|
editline (1.16.0) unstable; urgency=medium
|
||||||
|
|
||||||
* New upstream release, v1.60.0
|
* New upstream release, v1.60.0
|
||||||
|
2
debian/control
vendored
2
debian/control
vendored
@@ -21,7 +21,7 @@ Description: development files for libeditline
|
|||||||
Package: libeditline1
|
Package: libeditline1
|
||||||
Architecture: any
|
Architecture: any
|
||||||
Section: libs
|
Section: libs
|
||||||
Depends: ${shlibs:Depends}, ${misc:Depends}
|
Depends: ${misc:Depends}
|
||||||
Description: line editing library similar to readline
|
Description: line editing library similar to readline
|
||||||
This is a line-editing library. It can be linked into almost any program
|
This is a line-editing library. It can be linked into almost any program
|
||||||
to provide command-line editing and recall. It is call-compatible with a
|
to provide command-line editing and recall. It is call-compatible with a
|
||||||
|
1
debian/libeditline-dev.install
vendored
1
debian/libeditline-dev.install
vendored
@@ -1,3 +1,4 @@
|
|||||||
usr/include/*.h
|
usr/include/*.h
|
||||||
usr/lib/*/libeditline*.*a
|
usr/lib/*/libeditline*.*a
|
||||||
|
usr/lib/*/pkgconfig/*
|
||||||
usr/share/man/man3/*
|
usr/share/man/man3/*
|
||||||
|
@@ -98,39 +98,59 @@ el_status_t list_possible(void)
|
|||||||
return el_ring_bell();
|
return el_ring_bell();
|
||||||
}
|
}
|
||||||
|
|
||||||
el_status_t do_break(void)
|
|
||||||
{
|
|
||||||
puts("Breakout!");
|
|
||||||
return CSeof;
|
|
||||||
}
|
|
||||||
|
|
||||||
el_status_t do_exit(void)
|
|
||||||
{
|
|
||||||
puts("Bye bye!");
|
|
||||||
return CSeof;
|
|
||||||
}
|
|
||||||
|
|
||||||
el_status_t do_suspend(void)
|
el_status_t do_suspend(void)
|
||||||
{
|
{
|
||||||
puts("Abort!");
|
puts("Abort!");
|
||||||
return CSstay;
|
return CSstay;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void breakit(int signo)
|
||||||
|
{
|
||||||
|
puts("Got SIGINT");
|
||||||
|
}
|
||||||
|
|
||||||
int main(void)
|
int main(void)
|
||||||
{
|
{
|
||||||
char *line;
|
char *line;
|
||||||
char *prompt = "cli> ";
|
char *prompt = "cli> ";
|
||||||
|
char *passwd = "Enter password: ";
|
||||||
|
|
||||||
|
signal(SIGINT, breakit);
|
||||||
|
|
||||||
/* Setup callbacks */
|
/* Setup callbacks */
|
||||||
rl_set_complete_func(&my_rl_complete);
|
rl_set_complete_func(&my_rl_complete);
|
||||||
rl_set_list_possib_func(&my_rl_list_possib);
|
rl_set_list_possib_func(&my_rl_list_possib);
|
||||||
|
|
||||||
el_bind_key('?', list_possible);
|
el_bind_key('?', list_possible);
|
||||||
el_bind_key(CTL('C'), do_break);
|
|
||||||
el_bind_key(CTL('D'), do_exit);
|
|
||||||
el_bind_key(CTL('Z'), do_suspend);
|
el_bind_key(CTL('Z'), do_suspend);
|
||||||
read_history(HISTORY);
|
read_history(HISTORY);
|
||||||
|
|
||||||
while ((line = readline(prompt)) != NULL) {
|
while ((line = readline(prompt))) {
|
||||||
|
int next = 0;
|
||||||
|
|
||||||
|
/* Use el_no_echo when reading passwords and similar */
|
||||||
|
if (!strncmp(line, "unlock", 6)) {
|
||||||
|
el_no_echo = 1;
|
||||||
|
while ((line = readline(passwd))) {
|
||||||
|
if (strncmp(line, "secret", 6)) {
|
||||||
|
printf("\nWrong password, please try again, it's secret.\n");
|
||||||
|
free(line);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
el_no_echo = 0;
|
||||||
|
|
||||||
|
printf("\nAchievement unlocked!\n");
|
||||||
|
free(line);
|
||||||
|
next = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (*line != '\0')
|
||||||
printf("\t\t\t|%s|\n", line);
|
printf("\t\t\t|%s|\n", line);
|
||||||
free(line);
|
free(line);
|
||||||
}
|
}
|
||||||
|
@@ -21,14 +21,11 @@
|
|||||||
|
|
||||||
#include "editline.h"
|
#include "editline.h"
|
||||||
|
|
||||||
void * xmalloc (size_t size);
|
|
||||||
void too_dangerous(char *caller);
|
void too_dangerous(char *caller);
|
||||||
void initialize_readline();
|
void initialize_readline();
|
||||||
int execute_line(char *line);
|
int execute_line(char *line);
|
||||||
int valid_argument(char *caller, char *arg);
|
int valid_argument(char *caller, char *arg);
|
||||||
|
|
||||||
typedef int rl_icpfunc_t (char *);
|
|
||||||
|
|
||||||
/* The names of functions that actually do the manipulation. */
|
/* The names of functions that actually do the manipulation. */
|
||||||
int com_list(char *);
|
int com_list(char *);
|
||||||
int com_view(char *);
|
int com_view(char *);
|
||||||
@@ -41,16 +38,13 @@ int com_help(char *);
|
|||||||
int com_cd(char *);
|
int com_cd(char *);
|
||||||
int com_quit(char *);
|
int com_quit(char *);
|
||||||
|
|
||||||
/* A structure which contains information on the commands this program
|
struct cmd {
|
||||||
can understand. */
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
char *name; /* User printable name of the function. */
|
char *name; /* User printable name of the function. */
|
||||||
rl_icpfunc_t *func; /* Function to call to do the job. */
|
int (*func)(char *); /* Function to call to do the job. */
|
||||||
char *doc; /* Documentation for this function. */
|
char *doc; /* Documentation for this function. */
|
||||||
} COMMAND;
|
};
|
||||||
|
|
||||||
COMMAND commands[] = {
|
struct cmd commands[] = {
|
||||||
{ "cd", com_cd, "Change to directory DIR"},
|
{ "cd", com_cd, "Change to directory DIR"},
|
||||||
{ "delete", com_delete, "Delete FILE"},
|
{ "delete", com_delete, "Delete FILE"},
|
||||||
{ "help", com_help, "Display this text"},
|
{ "help", com_help, "Display this text"},
|
||||||
@@ -63,43 +57,26 @@ COMMAND commands[] = {
|
|||||||
{ "stat", com_stat, "Print out statistics on FILE"},
|
{ "stat", com_stat, "Print out statistics on FILE"},
|
||||||
{ "view", com_view, "View the contents of FILE"},
|
{ "view", com_view, "View the contents of FILE"},
|
||||||
{ "history", com_history, "List editline history"},
|
{ "history", com_history, "List editline history"},
|
||||||
{ (char *)NULL, (rl_icpfunc_t *)NULL, (char *)NULL }
|
{ NULL, NULL, NULL },
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Forward declarations. */
|
/* Forward declarations. */
|
||||||
char *stripwhite();
|
char *stripwhite();
|
||||||
COMMAND *find_command ();
|
struct cmd *find_command();
|
||||||
|
|
||||||
/* The name of this program, as taken from argv[0]. */
|
|
||||||
char *progname;
|
|
||||||
|
|
||||||
/* When non-zero, this means the user is done using this program. */
|
/* When non-zero, this means the user is done using this program. */
|
||||||
int done;
|
int done;
|
||||||
|
|
||||||
char *
|
int main(int argc, char **argv)
|
||||||
dupstr (char* s)
|
|
||||||
{
|
|
||||||
char *r;
|
|
||||||
|
|
||||||
r = xmalloc (strlen (s) + 1);
|
|
||||||
strcpy (r, s);
|
|
||||||
return (r);
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
main (int argc __attribute__((__unused__)), char **argv)
|
|
||||||
{
|
{
|
||||||
char *line, *s;
|
char *line, *s;
|
||||||
|
|
||||||
progname = argv[0];
|
|
||||||
|
|
||||||
setlocale(LC_CTYPE, "");
|
setlocale(LC_CTYPE, "");
|
||||||
|
|
||||||
initialize_readline(); /* Bind our completer. */
|
initialize_readline(); /* Bind our completer. */
|
||||||
|
|
||||||
/* Loop reading and executing lines until the user quits. */
|
/* Loop reading and executing lines until the user quits. */
|
||||||
for ( ; done == 0; )
|
for (; done == 0;) {
|
||||||
{
|
|
||||||
line = readline("FileMan: ");
|
line = readline("FileMan: ");
|
||||||
|
|
||||||
if (!line)
|
if (!line)
|
||||||
@@ -129,17 +106,15 @@ main (int argc __attribute__((__unused__)), char **argv)
|
|||||||
#endif
|
#endif
|
||||||
free(line);
|
free(line);
|
||||||
}
|
}
|
||||||
exit (0);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Execute a command line. */
|
/* Execute a command line. */
|
||||||
int
|
int execute_line(char *line)
|
||||||
execute_line (char *line)
|
|
||||||
{
|
{
|
||||||
register int i;
|
int i;
|
||||||
COMMAND *command;
|
struct cmd *command;
|
||||||
char *word;
|
char *word;
|
||||||
|
|
||||||
/* Isolate the command word. */
|
/* Isolate the command word. */
|
||||||
@@ -156,10 +131,9 @@ execute_line (char *line)
|
|||||||
|
|
||||||
command = find_command(word);
|
command = find_command(word);
|
||||||
|
|
||||||
if (!command)
|
if (!command) {
|
||||||
{
|
|
||||||
fprintf(stderr, "%s: No such command for FileMan.\n", word);
|
fprintf(stderr, "%s: No such command for FileMan.\n", word);
|
||||||
return (-1);
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get argument to command, if any. */
|
/* Get argument to command, if any. */
|
||||||
@@ -169,35 +143,34 @@ execute_line (char *line)
|
|||||||
word = line + i;
|
word = line + i;
|
||||||
|
|
||||||
/* Call the function. */
|
/* Call the function. */
|
||||||
return ((*(command->func)) (word));
|
return command->func(word);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Look up NAME as the name of a command, and return a pointer to that
|
/* Look up NAME as the name of a command, and return a pointer to that
|
||||||
command. Return a NULL pointer if NAME isn't a command name. */
|
command. Return a NULL pointer if NAME isn't a command name. */
|
||||||
COMMAND *
|
struct cmd *find_command(char *name)
|
||||||
find_command (char *name)
|
|
||||||
{
|
{
|
||||||
register int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; commands[i].name; i++)
|
for (i = 0; commands[i].name; i++)
|
||||||
if (strcmp(name, commands[i].name) == 0)
|
if (strcmp(name, commands[i].name) == 0)
|
||||||
return (&commands[i]);
|
return &commands[i];
|
||||||
|
|
||||||
return ((COMMAND *)NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Strip whitespace from the start and end of STRING. Return a pointer
|
/*
|
||||||
into STRING. */
|
* Strip whitespace from the start and end of STRING. Return a pointer
|
||||||
char *
|
* into STRING.
|
||||||
stripwhite (char *string)
|
*/
|
||||||
|
char *stripwhite(char *string)
|
||||||
{
|
{
|
||||||
register char *s, *t;
|
char *s, *t;
|
||||||
|
|
||||||
for (s = string; isspace (*s); s++)
|
for (s = string; isspace(*s); s++) ;
|
||||||
;
|
|
||||||
|
|
||||||
if (*s == 0)
|
if (*s == 0)
|
||||||
return (s);
|
return s;
|
||||||
|
|
||||||
t = s + strlen(s) - 1;
|
t = s + strlen(s) - 1;
|
||||||
while (t > s && isspace(*t))
|
while (t > s && isspace(*t))
|
||||||
@@ -216,11 +189,12 @@ stripwhite (char *string)
|
|||||||
char *command_generator(const char *, int);
|
char *command_generator(const char *, int);
|
||||||
char **fileman_completion(const char *, int, int);
|
char **fileman_completion(const char *, int, int);
|
||||||
|
|
||||||
/* Tell the GNU Readline library how to complete. We want to try to
|
/*
|
||||||
complete on command names if this is the first word in the line, or
|
* Tell the GNU Readline library how to complete. We want to try to
|
||||||
on filenames if not. */
|
* complete on command names if this is the first word in the line, or
|
||||||
void
|
* on filenames if not.
|
||||||
initialize_readline ()
|
*/
|
||||||
|
void initialize_readline(void)
|
||||||
{
|
{
|
||||||
/* Allow conditional parsing of the ~/.inputrc file. */
|
/* Allow conditional parsing of the ~/.inputrc file. */
|
||||||
rl_readline_name = "FileMan";
|
rl_readline_name = "FileMan";
|
||||||
@@ -229,17 +203,16 @@ initialize_readline ()
|
|||||||
rl_attempted_completion_function = fileman_completion;
|
rl_attempted_completion_function = fileman_completion;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Attempt to complete on the contents of TEXT. START and END
|
/*
|
||||||
bound the region of rl_line_buffer that contains the word to
|
* Attempt to complete on the contents of TEXT. START and END
|
||||||
complete. TEXT is the word to complete. We can use the entire
|
* bound the region of rl_line_buffer that contains the word to
|
||||||
contents of rl_line_buffer in case we want to do some simple
|
* complete. TEXT is the word to complete. We can use the entire
|
||||||
parsing. Returnthe array of matches, or NULL if there aren't any. */
|
* contents of rl_line_buffer in case we want to do some simple
|
||||||
char **
|
* parsing. Returnthe array of matches, or NULL if there aren't any.
|
||||||
fileman_completion (const char* text, int start, int end __attribute__((__unused__)))
|
*/
|
||||||
|
char **fileman_completion(const char *text, int start, int end)
|
||||||
{
|
{
|
||||||
char **matches;
|
char **matches = NULL;
|
||||||
|
|
||||||
matches = (char **)NULL;
|
|
||||||
|
|
||||||
/* If this word is at the start of the line, then it is a command
|
/* If this word is at the start of the line, then it is a command
|
||||||
to complete. Otherwise it is the name of a file in the current
|
to complete. Otherwise it is the name of a file in the current
|
||||||
@@ -247,16 +220,13 @@ fileman_completion (const char* text, int start, int end __attribute__((__unused
|
|||||||
if (start == 0)
|
if (start == 0)
|
||||||
matches = rl_completion_matches(text, command_generator);
|
matches = rl_completion_matches(text, command_generator);
|
||||||
|
|
||||||
return (matches);
|
return matches;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Generator function for command completion. STATE lets us
|
/* Generator function for command completion. STATE lets us
|
||||||
know whether to start from scratch; without any state
|
know whether to start from scratch; without any state
|
||||||
(i.e. STATE == 0), then we start at the top of the list. */
|
(i.e. STATE == 0), then we start at the top of the list. */
|
||||||
char *
|
char *command_generator(const char *text, int state)
|
||||||
command_generator (text, state)
|
|
||||||
const char *text;
|
|
||||||
int state;
|
|
||||||
{
|
{
|
||||||
static int list_index, len;
|
static int list_index, len;
|
||||||
char *name;
|
char *name;
|
||||||
@@ -264,24 +234,22 @@ command_generator (text, state)
|
|||||||
/* If this is a new word to complete, initialize now. This
|
/* If this is a new word to complete, initialize now. This
|
||||||
includes saving the length of TEXT for efficiency, and
|
includes saving the length of TEXT for efficiency, and
|
||||||
initializing the index variable to 0. */
|
initializing the index variable to 0. */
|
||||||
if (!state)
|
if (!state) {
|
||||||
{
|
|
||||||
list_index = 0;
|
list_index = 0;
|
||||||
len = strlen(text);
|
len = strlen(text);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the next name which partially matches from the
|
/* Return the next name which partially matches from the
|
||||||
command list. */
|
command list. */
|
||||||
while ((name = commands[list_index].name))
|
while ((name = commands[list_index].name)) {
|
||||||
{
|
|
||||||
list_index++;
|
list_index++;
|
||||||
|
|
||||||
if (strncmp(name, text, len) == 0)
|
if (strncmp(name, text, len) == 0)
|
||||||
return (dupstr(name));
|
return strdup(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If no names matched, then return NULL. */
|
/* If no names matched, then return NULL. */
|
||||||
return ((char *)NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* **************************************************************** */
|
/* **************************************************************** */
|
||||||
@@ -295,34 +263,32 @@ command_generator (text, state)
|
|||||||
static char syscom[1024];
|
static char syscom[1024];
|
||||||
|
|
||||||
/* List the file(s) named in arg. */
|
/* List the file(s) named in arg. */
|
||||||
int
|
int com_list(char *arg)
|
||||||
com_list (char *arg)
|
|
||||||
{
|
{
|
||||||
if (!arg)
|
if (!arg)
|
||||||
arg = "";
|
arg = "";
|
||||||
|
|
||||||
sprintf(syscom, "ls -FClg %s", arg);
|
sprintf(syscom, "ls -FClg %s", arg);
|
||||||
return (system (syscom));
|
|
||||||
|
return system(syscom);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int com_view(char *arg)
|
||||||
com_view (char *arg)
|
|
||||||
{
|
{
|
||||||
if (!valid_argument("view", arg))
|
if (!valid_argument("view", arg))
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
sprintf(syscom, "more %s", arg);
|
sprintf(syscom, "more %s", arg);
|
||||||
return (system (syscom));
|
|
||||||
|
return system(syscom);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int com_history(char *arg)
|
||||||
com_history(char* arg __attribute__((__unused__)))
|
|
||||||
{
|
{
|
||||||
const char *he;
|
const char *he;
|
||||||
|
|
||||||
/* rewind history */
|
/* rewind history */
|
||||||
while (el_prev_hist())
|
while (el_prev_hist()) ;
|
||||||
;
|
|
||||||
|
|
||||||
for (he = el_next_hist(); he != NULL; he = el_next_hist())
|
for (he = el_next_hist(); he != NULL; he = el_next_hist())
|
||||||
printf("%s\n", he);
|
printf("%s\n", he);
|
||||||
@@ -330,25 +296,22 @@ com_history(char* arg __attribute__((__unused__)))
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int com_rename(char *arg)
|
||||||
com_rename (char *arg __attribute__((__unused__)))
|
|
||||||
{
|
{
|
||||||
too_dangerous("rename");
|
too_dangerous("rename");
|
||||||
return (1);
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int com_stat(char *arg)
|
||||||
com_stat (char *arg)
|
|
||||||
{
|
{
|
||||||
struct stat finfo;
|
struct stat finfo;
|
||||||
|
|
||||||
if (!valid_argument("stat", arg))
|
if (!valid_argument("stat", arg))
|
||||||
return (1);
|
return 1;
|
||||||
|
|
||||||
if (stat (arg, &finfo) == -1)
|
if (stat(arg, &finfo) == -1) {
|
||||||
{
|
|
||||||
perror(arg);
|
perror(arg);
|
||||||
return (1);
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("Statistics for `%s':\n", arg);
|
printf("Statistics for `%s':\n", arg);
|
||||||
@@ -356,47 +319,41 @@ com_stat (char *arg)
|
|||||||
printf("%s has %ld link%s, and is %lld byte%s in length.\n", arg,
|
printf("%s has %ld link%s, and is %lld byte%s in length.\n", arg,
|
||||||
(long)finfo.st_nlink,
|
(long)finfo.st_nlink,
|
||||||
(finfo.st_nlink == 1) ? "" : "s",
|
(finfo.st_nlink == 1) ? "" : "s",
|
||||||
(long long) finfo.st_size,
|
(long long)finfo.st_size, (finfo.st_size == 1) ? "" : "s");
|
||||||
(finfo.st_size == 1) ? "" : "s");
|
|
||||||
printf("Inode Last Change at: %s", ctime(&finfo.st_ctime));
|
printf("Inode Last Change at: %s", ctime(&finfo.st_ctime));
|
||||||
printf(" Last access at: %s", ctime(&finfo.st_atime));
|
printf(" Last access at: %s", ctime(&finfo.st_atime));
|
||||||
printf(" Last modified at: %s", ctime(&finfo.st_mtime));
|
printf(" Last modified at: %s", ctime(&finfo.st_mtime));
|
||||||
return (0);
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int com_delete(char *arg)
|
||||||
com_delete (char *arg __attribute__((__unused__)))
|
|
||||||
{
|
{
|
||||||
too_dangerous("delete");
|
too_dangerous("delete");
|
||||||
return (1);
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Print out help for ARG, or for all of the commands if ARG is
|
/* Print out help for ARG, or for all of the commands if ARG is
|
||||||
not present. */
|
not present. */
|
||||||
int
|
int com_help(char *arg)
|
||||||
com_help (char *arg)
|
|
||||||
{
|
{
|
||||||
register int i;
|
int i;
|
||||||
int printed = 0;
|
int printed = 0;
|
||||||
|
|
||||||
for (i = 0; commands[i].name; i++)
|
for (i = 0; commands[i].name; i++) {
|
||||||
{
|
if (!*arg || (strcmp(arg, commands[i].name) == 0)) {
|
||||||
if (!*arg || (strcmp (arg, commands[i].name) == 0))
|
printf("%s\t\t%s.\n", commands[i].name,
|
||||||
{
|
commands[i].doc);
|
||||||
printf ("%s\t\t%s.\n", commands[i].name, commands[i].doc);
|
|
||||||
printed++;
|
printed++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!printed)
|
if (!printed) {
|
||||||
{
|
|
||||||
printf("No commands match `%s'. Possibilties are:\n", arg);
|
printf("No commands match `%s'. Possibilties are:\n", arg);
|
||||||
|
|
||||||
for (i = 0; commands[i].name; i++)
|
for (i = 0; commands[i].name; i++) {
|
||||||
{
|
|
||||||
/* Print in six columns. */
|
/* Print in six columns. */
|
||||||
if (printed == 6)
|
if (printed == 6) {
|
||||||
{
|
|
||||||
printed = 0;
|
printed = 0;
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
@@ -408,32 +365,29 @@ com_help (char *arg)
|
|||||||
if (printed)
|
if (printed)
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
return (0);
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change to the directory ARG. */
|
/* Change to the directory ARG. */
|
||||||
int
|
int com_cd(char *arg)
|
||||||
com_cd (char *arg)
|
|
||||||
{
|
|
||||||
if (chdir (arg) == -1)
|
|
||||||
{
|
{
|
||||||
|
if (chdir(arg) == -1) {
|
||||||
perror(arg);
|
perror(arg);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
com_pwd("");
|
com_pwd("");
|
||||||
return (0);
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Print out the current working directory. */
|
/* Print out the current working directory. */
|
||||||
int
|
int com_pwd(char *ignore)
|
||||||
com_pwd (char* ignore __attribute__((__unused__)))
|
|
||||||
{
|
{
|
||||||
char dir[1024], *s;
|
char dir[1024], *s;
|
||||||
|
|
||||||
s = (char*)getcwd(dir, sizeof(dir) - 1);
|
s = getcwd(dir, sizeof(dir) - 1);
|
||||||
if (s == 0)
|
if (!s) {
|
||||||
{
|
|
||||||
printf("Error getting pwd: %s\n", dir);
|
printf("Error getting pwd: %s\n", dir);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@@ -444,44 +398,34 @@ com_pwd (char* ignore __attribute__((__unused__)))
|
|||||||
|
|
||||||
/* The user wishes to quit using this program. Just set DONE
|
/* The user wishes to quit using this program. Just set DONE
|
||||||
non-zero. */
|
non-zero. */
|
||||||
int
|
int com_quit(char *arg)
|
||||||
com_quit (char *arg __attribute__((__unused__)))
|
|
||||||
{
|
{
|
||||||
done = 1;
|
done = 1;
|
||||||
return (0);
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Function which tells you that you can't do this. */
|
/* Function which tells you that you can't do this. */
|
||||||
void
|
void too_dangerous(char *caller)
|
||||||
too_dangerous (char *caller)
|
|
||||||
{
|
{
|
||||||
fprintf (stderr,
|
fprintf(stderr, "%s: Too dangerous for me to distribute.\n", caller);
|
||||||
"%s: Too dangerous for me to distribute.\n",
|
|
||||||
caller);
|
|
||||||
fprintf(stderr, "Write it yourself.\n");
|
fprintf(stderr, "Write it yourself.\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return non-zero if ARG is a valid argument for CALLER,
|
/* Return non-zero if ARG is a valid argument for CALLER,
|
||||||
else print an error message and return zero. */
|
else print an error message and return zero. */
|
||||||
int
|
int valid_argument(char *caller, char *arg)
|
||||||
valid_argument (char *caller, char *arg)
|
|
||||||
{
|
|
||||||
if (!arg || !*arg)
|
|
||||||
{
|
{
|
||||||
|
if (!arg || !*arg) {
|
||||||
fprintf(stderr, "%s: Argument required.\n", caller);
|
fprintf(stderr, "%s: Argument required.\n", caller);
|
||||||
return (0);
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (1);
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void *
|
/**
|
||||||
xmalloc (size_t size)
|
* Local Variables:
|
||||||
{
|
* c-file-style: "k&r"
|
||||||
register void *value = (void*)malloc(size);
|
* c-basic-offset: 4
|
||||||
if (value == 0)
|
* End:
|
||||||
fprintf(stderr, "virtual memory exhausted");
|
*/
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
@@ -114,6 +114,8 @@ extern void add_history (const char *line);
|
|||||||
extern int read_history (const char *filename);
|
extern int read_history (const char *filename);
|
||||||
extern int write_history (const char *filename);
|
extern int write_history (const char *filename);
|
||||||
|
|
||||||
|
extern rl_getc_func_t *rl_set_getc_func(rl_getc_func_t *func);
|
||||||
|
|
||||||
extern rl_completion_func_t *rl_attempted_completion_function;
|
extern rl_completion_func_t *rl_attempted_completion_function;
|
||||||
extern rl_complete_func_t *rl_set_complete_func (rl_complete_func_t *func);
|
extern rl_complete_func_t *rl_set_complete_func (rl_complete_func_t *func);
|
||||||
extern rl_list_possib_func_t *rl_set_list_possib_func (rl_list_possib_func_t *func);
|
extern rl_list_possib_func_t *rl_set_list_possib_func (rl_list_possib_func_t *func);
|
||||||
|
442
man/editline.3
442
man/editline.3
@@ -1,200 +1,274 @@
|
|||||||
.TH EDITLINE 3
|
.Dd April 27, 2019
|
||||||
.SH NAME
|
.Dt EDITLINE 3
|
||||||
editline \- command-line editing library with history
|
.Os
|
||||||
.SH SYNOPSIS
|
.Sh NAME
|
||||||
.nf
|
.Nm editline
|
||||||
.B "char *readline(char *prompt);"
|
.Nd command-line editing library with history
|
||||||
.fi
|
.Sh LIBRARY
|
||||||
.SH DESCRIPTION
|
.Lb libeditline
|
||||||
.I Editline
|
.Sh SYNOPSIS
|
||||||
is a library that provides an line-editing interface with history.
|
.In editline.h
|
||||||
It is intended to be functionally equivalent with the
|
.Fn "char *readline" "const char *prompt"
|
||||||
.I readline
|
.Fn "void add_history" "const char *line"
|
||||||
library provided by the Free Software Foundation, but much smaller.
|
.Fn "int read_history" "const char *filename"
|
||||||
The bulk of this manual page describes the user interface.
|
.Fn "int write_history" "const char *filename"
|
||||||
.PP
|
.Sh DESCRIPTION
|
||||||
|
.Nm
|
||||||
|
is a library that provides n line-editing interface with history. It
|
||||||
|
is intended to be functionally equivalent with the
|
||||||
|
.Nm readline
|
||||||
|
library provided by the Free Software Foundation, but much smaller. The
|
||||||
|
bulk of this manual page describes the basic user interface. More APIs,
|
||||||
|
both native and for
|
||||||
|
.Nm readline
|
||||||
|
compatibility ,
|
||||||
|
are also available. See the
|
||||||
|
.Cm editline.h
|
||||||
|
header file for details.
|
||||||
|
.Pp
|
||||||
The
|
The
|
||||||
.I readline()
|
.Fn readline
|
||||||
function displays the given
|
function displays the given
|
||||||
.I prompt
|
.Fa prompt
|
||||||
on stdout, waits for user input on stdin and then
|
on stdout, waits for user input on stdin and then returns a line of text
|
||||||
returns a line of text with the trailing newline removed. The data is returned in a
|
with the trailing newline removed. The data is returned in a buffer
|
||||||
buffer allocated with
|
allocated with
|
||||||
.IR malloc (3),
|
.Xr malloc 3 ,
|
||||||
so the space should be released with
|
so the space should be released with
|
||||||
.IR free (3)
|
.Xr free 3
|
||||||
when the calling program is done with it.
|
when the calling program is done with it.
|
||||||
.PP
|
.Pp
|
||||||
Each line returned is copied to the internal history list, unless it happens
|
Each line returned is automatically saved in the internal history list,
|
||||||
to be equal to the previous line. This is configurable if you are building editline
|
unless it happens to be equal to the previous line. This is
|
||||||
from source.
|
configurable if you are building editline from source, i.e. if you would
|
||||||
.SS User Interface
|
rather like to call
|
||||||
A program that uses this library provides a simple emacs-like editing interface to
|
.Fn add_history
|
||||||
its users. A line may be edited before it is sent to the calling program by typing
|
manually.
|
||||||
either control characters or escape sequences. A control character, shown as a caret
|
.Pp
|
||||||
followed by a letter, is typed by holding down the ``control'' key while the letter
|
The
|
||||||
is typed. For example, ``^A'' is a control-A. An escape sequence is entered by
|
.Fn read_history
|
||||||
typing the ``escape'' key followed by one or more characters. The escape key is
|
and
|
||||||
abbreviated as ``ESC''. Note that unlike control keys, case matters in escape
|
.Fn write_history
|
||||||
sequences; ``ESC\ F'' is not the same as ``ESC\ f''.
|
functions can be used to load and store the history of your application.
|
||||||
.PP
|
.Em Note:
|
||||||
An editing command may be typed anywhere on the line, not just at the beginning. In
|
these APIs do not do any tilde or environment variable expansion of the
|
||||||
addition, a return may also be typed anywhere on the line, not just at the end.
|
given filename.
|
||||||
.PP
|
.Ss User Interface
|
||||||
|
A program that uses this library provides a simple emacs-like editing
|
||||||
|
interface to its users. A line may be edited before it is sent to the
|
||||||
|
calling program by typing either control characters or escape sequences.
|
||||||
|
A control character, shown as a caret followed by a letter, is typed by
|
||||||
|
holding down the control key while the letter is typed. For example,
|
||||||
|
.Cm ^A
|
||||||
|
is a control-A. An escape sequence is entered by typing the escape key
|
||||||
|
followed by one or more characters. The escape key is abbreviated as
|
||||||
|
.Cm ESC .
|
||||||
|
Note that unlike control keys, case matters in escape sequences;
|
||||||
|
.Cm ESC F
|
||||||
|
is not the same as
|
||||||
|
.Cm ESC f .
|
||||||
|
.Pp
|
||||||
|
An editing command may be typed anywhere on the line, not just at the
|
||||||
|
beginning. In addition, a return may also be typed anywhere on the
|
||||||
|
line, not just at the end.
|
||||||
|
.Pp
|
||||||
Most editing commands may be given a repeat count,
|
Most editing commands may be given a repeat count,
|
||||||
.IR n ,
|
.Ar n ,
|
||||||
where
|
where
|
||||||
.I n
|
.Ar n
|
||||||
is a number.
|
is a number. To enter a repeat count, type the escape key, the number,
|
||||||
To enter a repeat count, type the escape key, the number, and then
|
and then the command to execute. For example,
|
||||||
the command to execute.
|
.Cm ESC 4 ^f
|
||||||
For example, ``ESC\ 4\ ^f'' moves forward four characters.
|
moves forward four characters. If a command may be given a repeat count
|
||||||
If a command may be given a repeat count then the text ``[n]'' is given at the
|
then the text
|
||||||
end of its description.
|
.Cm [n]
|
||||||
.PP
|
is given at the end of its description.
|
||||||
|
.Pp
|
||||||
The following control characters are accepted:
|
The following control characters are accepted:
|
||||||
.RS
|
.Pp
|
||||||
.nf
|
.Bl -tag -width "ESC DEL " -compact
|
||||||
.ta \w'ESC DEL 'u
|
.It ^A
|
||||||
^A Move to the beginning of the line
|
Move to the beginning of the line
|
||||||
^B Move left (backwards) [n]
|
.It ^B
|
||||||
^D Delete character [n]
|
Move left (backwards) [n]
|
||||||
^E Move to end of line
|
.It ^D
|
||||||
^F Move right (forwards) [n]
|
Delete character [n]
|
||||||
^G Ring the bell
|
.It ^E
|
||||||
^H Delete character before cursor (backspace key) [n]
|
Move to end of line
|
||||||
^I Complete filename (tab key); see below
|
.It ^F
|
||||||
^J Done with line (return key)
|
Move right (forwards) [n]
|
||||||
^K Kill to end of line (or column [n])
|
.It ^G
|
||||||
^L Redisplay line
|
Ring the bell
|
||||||
^M Done with line (alternate return key)
|
.It ^H
|
||||||
^N Get next line from history [n]
|
Delete character before cursor (backspace key) [n]
|
||||||
^P Get previous line from history [n]
|
.It ^I
|
||||||
^R Search backward (forward if [n]) through history for
|
Complete filename (tab key); see below
|
||||||
\& text; must start line if text begins with an uparrow
|
.It ^J
|
||||||
^T Transpose characters
|
Done with line (return key)
|
||||||
^V Insert next character, even if it is an edit command
|
.It ^K
|
||||||
^W Wipe to the mark
|
Kill to end of line (or column [n])
|
||||||
^X^X Exchange current location and mark
|
.It ^L
|
||||||
^Y Yank back last killed text
|
Redisplay line
|
||||||
^[ Start an escape sequence (escape key)
|
.It ^M
|
||||||
^]c Move forward to next character ``c''
|
Done with line (alternate return key)
|
||||||
^? Delete character before cursor (delete key) [n]
|
.It ^N
|
||||||
.fi
|
Get next line from history [n]
|
||||||
.RE
|
.It ^P
|
||||||
.PP
|
Get previous line from history [n]
|
||||||
The following escape sequences are provided.
|
.It ^R
|
||||||
.RS
|
Search backward (forward if [n]) through history for text; prefixing the
|
||||||
.nf
|
string with a caret (^) forces it to match only at the beginning of a
|
||||||
.ta \w'ESC DEL 'u
|
history line
|
||||||
ESC\ ^H Delete previous word (backspace key) [n]
|
.It ^T
|
||||||
ESC\ DEL Delete previous word (delete key) [n]
|
Transpose characters
|
||||||
ESC\ SP Set the mark (space key); see ^X^X and ^Y above
|
.It ^V
|
||||||
ESC\ \. Get the last (or [n]'th) word from previous line
|
Insert next character, even if it is an edit command
|
||||||
ESC\ ? Show possible completions; see below
|
.It ^W
|
||||||
ESC\ < Move to start of history
|
Wipe to the mark
|
||||||
ESC\ > Move to end of history
|
.It ^X^X
|
||||||
ESC\ b Move backward a word [n]
|
Exchange current location and mark
|
||||||
ESC\ d Delete word under cursor [n]
|
.It ^Y
|
||||||
ESC\ f Move forward a word [n]
|
Yank back last killed text
|
||||||
ESC\ l Make word lowercase [n]
|
.It ^[
|
||||||
ESC\ m Toggle if 8bit chars display normally or with an
|
Start an escape sequence (escape key)
|
||||||
\& ``M\-'' prefix
|
.It ^]c
|
||||||
ESC\ u Make word uppercase [n]
|
Move forward to next character
|
||||||
ESC\ y Yank back last killed text
|
.Cm c
|
||||||
ESC\ v Show library version
|
.It ^?
|
||||||
ESC\ w Make area up to mark yankable
|
Delete character before cursor (delete key) [n]
|
||||||
ESC\ nn Set repeat count to the number nn
|
.El
|
||||||
ESC\ C Read from environment variable ``_C_'', where C is
|
.Pp
|
||||||
\& an uppercase letter
|
The following escape sequences are provided:
|
||||||
.fi
|
.Pp
|
||||||
.RE
|
.Bl -tag -width "ESC DEL " -compact
|
||||||
.PP
|
.It ESC ^H
|
||||||
|
Delete previous word (backspace key) [n]
|
||||||
|
.It ESC DEL
|
||||||
|
Delete previous word (delete key) [n]
|
||||||
|
.It ESC SP
|
||||||
|
Set the mark (space key); see ^X^X and ^Y above
|
||||||
|
.It ESC\ .
|
||||||
|
Get the last (or [n]'th) word from previous line
|
||||||
|
.It ESC\ ?
|
||||||
|
Show possible completions; see below
|
||||||
|
.It ESC <
|
||||||
|
Move to start of history
|
||||||
|
.It ESC >
|
||||||
|
Move to end of history
|
||||||
|
.It ESC b
|
||||||
|
Move backward a word [n]
|
||||||
|
.It ESC d
|
||||||
|
Delete word under cursor [n]
|
||||||
|
.It ESC f
|
||||||
|
Move forward a word [n]
|
||||||
|
.It ESC l
|
||||||
|
Make word lowercase [n]
|
||||||
|
.It ESC m
|
||||||
|
Toggle if 8bit chars display normally or with an
|
||||||
|
.Ar M-
|
||||||
|
prefix
|
||||||
|
.It ESC u
|
||||||
|
Make word uppercase [n]
|
||||||
|
.It ESC y
|
||||||
|
Yank back last killed text
|
||||||
|
.It ESC v
|
||||||
|
Show library version
|
||||||
|
.It ESC w
|
||||||
|
Make area up to mark yankable
|
||||||
|
.It ESC nn
|
||||||
|
Set repeat count to the number nn
|
||||||
|
.It ESC C
|
||||||
|
Read from environment variable
|
||||||
|
.Ar $C ,
|
||||||
|
where
|
||||||
|
.Ar C
|
||||||
|
is an uppercase letter
|
||||||
|
.El
|
||||||
|
.Pp
|
||||||
The
|
The
|
||||||
.I editline
|
.Nm
|
||||||
library has a small macro facility.
|
library has a small macro facility. If you type the escape key followed
|
||||||
If you type the escape key followed by an uppercase letter,
|
by an uppercase letter,
|
||||||
.IR C ,
|
.Ar C ,
|
||||||
then the contents of the environment variable
|
then the contents of the environment variable
|
||||||
.I _C_
|
.Ar $C
|
||||||
are read in as if you had typed them at the keyboard.
|
are read in as if you had typed them at the keyboard. For example, if
|
||||||
For example, if the variable
|
the variable
|
||||||
.I _L_
|
.Ar $L
|
||||||
contains the following:
|
contains the following:
|
||||||
.PP
|
.Pp
|
||||||
.RS
|
.Dl ^A^Kecho '^V^[[H^V^[[2J'^M
|
||||||
^A^Kecho '^V^[[H^V^[[2J'^M
|
.Pp
|
||||||
.RE
|
Then typing
|
||||||
.PP
|
.Cm ESC L
|
||||||
Then typing ``ESC L'' will move to the beginning of the line, kill the
|
will move to the beginning of the line, kill the entire line, enter the
|
||||||
entire line, enter the echo command needed to clear the terminal (if your
|
echo command needed to clear the terminal (if your terminal is like a
|
||||||
terminal is like a VT-100), and send the line back to the shell.
|
VT-100), and send the line back to the shell.
|
||||||
.PP
|
.Pp
|
||||||
The
|
The
|
||||||
.I editline
|
.Nm
|
||||||
library also does filename completion.
|
library also does filename completion. Suppose the root directory has
|
||||||
Suppose the root directory has the following files in it:
|
the following files in it:
|
||||||
.PP
|
.Pp
|
||||||
.RS
|
.Dl bin vmunix
|
||||||
.nf
|
.Dl core vmunix.old
|
||||||
.ta \w'core 'u
|
.Pp
|
||||||
bin vmunix
|
If you type
|
||||||
core vmunix.old
|
.Cm rm /v
|
||||||
.fi
|
and then the tab key,
|
||||||
.RE
|
.Nm
|
||||||
.PP
|
will then finish off as much of the name as possible by adding
|
||||||
If you type ``rm\ /v'' and then the tab key.
|
.Ar munix .
|
||||||
.I Editline
|
Because the name is not unique, it will then beep. If you type the
|
||||||
will then finish off as much of the name as possible by adding ``munix''.
|
escape key and a question mark, it will display the two choices. If you
|
||||||
Because the name is not unique, it will then beep.
|
then type a period and a tab, the library will finish off the filename
|
||||||
If you type the escape key and a question mark, it will display the
|
|
||||||
two choices.
|
|
||||||
If you then type a period and a tab, the library will finish off the filename
|
|
||||||
for you:
|
for you:
|
||||||
.PP
|
.Pp
|
||||||
.RS
|
.Bd -ragged -offset indent
|
||||||
.nf
|
rm /v[TAB]
|
||||||
.RI "rm /v[TAB]" munix ".[TAB]" old
|
.Em munix
|
||||||
.fi
|
\&.[TAB]
|
||||||
.RE
|
.Em old
|
||||||
.PP
|
.Ed
|
||||||
The tab key is shown by ``[TAB]'' and the automatically-entered text
|
.Pp
|
||||||
is shown in italics.
|
The tab key is shown by [TAB] and the automatically-entered text
|
||||||
.SH USAGE
|
is shown in italics, or underline.
|
||||||
|
.Sh USAGE
|
||||||
To include
|
To include
|
||||||
.I readline()
|
.Nm
|
||||||
in your program, simply call it as you do any other function. Just make sure to link
|
in your program, call it as you do any other function and link your
|
||||||
your program with libeditline.
|
program with
|
||||||
.SS Example
|
.Ar -leditline .
|
||||||
|
.Ss Example
|
||||||
The following brief example lets you enter a line and edit it, then displays it.
|
The following brief example lets you enter a line and edit it, then displays it.
|
||||||
.nf
|
.Pp
|
||||||
.B ""
|
.Bd -literal -offset indent
|
||||||
.B #include <stdlib.h>
|
#include <stdio.h>
|
||||||
.B ""
|
#include <stdlib.h>
|
||||||
.B extern char *readline(char *prompt);
|
#include <editline.h>
|
||||||
.B ""
|
|
||||||
.B int main(void)
|
int main(void)
|
||||||
.B {
|
{
|
||||||
.RS
|
char *p;
|
||||||
.B char *p;
|
|
||||||
.B ""
|
while ((p = readline("CLI> "))) {
|
||||||
.B while ((p = readline("CLI>"))) {
|
puts(p);
|
||||||
.RS
|
free(p);
|
||||||
.B puts(p);
|
}
|
||||||
.B free(p);
|
|
||||||
.RE
|
return 0;
|
||||||
.B }
|
}
|
||||||
.B ""
|
.El
|
||||||
.B return 0;
|
.Sh AUTHORS
|
||||||
.RE
|
The original editline library was posted to comp.sources.unix newsgroup
|
||||||
.B }
|
by created by Simmule R. Turner and Rich Salz in 1992. It now exists in
|
||||||
.fi
|
several forks: Debian, Minix, Heimdal, Festival speech tools, Mozilla,
|
||||||
.SH AUTHORS
|
Google Gadgets for Linux, and many other places. The original manual
|
||||||
The original editline library was created by Simmule R. Turner and Rich
|
page was made by David W. Sanderson.
|
||||||
Salz. It now exists in several forks: Heimdal, Festival speech tools,
|
.Pp
|
||||||
Mozilla, Google Gadgets for Linux, and many other places. The original
|
This version was originally based on the Minix 2 sources, but has since
|
||||||
manual page was made by David W. Sanderson. This version is maintained
|
evolved to include patches from all relevant forks. It is currently
|
||||||
by Joachim Nilsson at GitHub,
|
maintained by Joachim Nilsson at GitHub,
|
||||||
.Aq http://github.com/troglobit/editline
|
.Aq http://github.com/troglobit/editline
|
||||||
.SH BUGS
|
.Sh BUGS
|
||||||
Does not handle multiple lines or unicode characters well.
|
Does not handle multiple lines or unicode characters well.
|
||||||
|
@@ -3,4 +3,4 @@ libeditline_la_SOURCES = editline.c editline.h complete.c sysunix.c unix.h
|
|||||||
libeditline_la_CFLAGS = -std=gnu99
|
libeditline_la_CFLAGS = -std=gnu99
|
||||||
libeditline_la_CFLAGS += -W -Wall -Wextra -Wundef -Wunused -Wstrict-prototypes
|
libeditline_la_CFLAGS += -W -Wall -Wextra -Wundef -Wunused -Wstrict-prototypes
|
||||||
libeditline_la_CFLAGS += -Werror-implicit-function-declaration -Wshadow -Wcast-qual
|
libeditline_la_CFLAGS += -Werror-implicit-function-declaration -Wshadow -Wcast-qual
|
||||||
libeditline_la_LDFLAGS = $(AM_LDFLAGS) -version-info 1:0:0
|
libeditline_la_LDFLAGS = $(AM_LDFLAGS) -version-info 1:1:0
|
||||||
|
@@ -91,6 +91,7 @@ static const char *el_input = NILSTR;
|
|||||||
static char *Yanked;
|
static char *Yanked;
|
||||||
static char *Screen;
|
static char *Screen;
|
||||||
static char NEWLINE[]= CRLF;
|
static char NEWLINE[]= CRLF;
|
||||||
|
static char CLEAR[]= "\ec";
|
||||||
static const char *el_term = "dumb";
|
static const char *el_term = "dumb";
|
||||||
static int Repeat;
|
static int Repeat;
|
||||||
static int old_point;
|
static int old_point;
|
||||||
@@ -283,13 +284,16 @@ void el_print_columns(int ac, char **av)
|
|||||||
int skip;
|
int skip;
|
||||||
int longest;
|
int longest;
|
||||||
int cols;
|
int cols;
|
||||||
|
int colwidth;
|
||||||
|
|
||||||
/* Find longest name, determine column count from that. */
|
/* Find longest name, determine column count from that. */
|
||||||
for (longest = 0, i = 0; i < ac; i++) {
|
for (longest = 0, i = 0; i < ac; i++) {
|
||||||
if ((j = strlen((char *)av[i])) > longest)
|
if ((j = strlen((char *)av[i])) > longest)
|
||||||
longest = j;
|
longest = j;
|
||||||
}
|
}
|
||||||
cols = tty_cols / (longest + 3);
|
colwidth = longest + 3;
|
||||||
|
if (colwidth > tty_cols) colwidth = tty_cols;
|
||||||
|
cols = tty_cols / colwidth;
|
||||||
|
|
||||||
tty_puts(NEWLINE);
|
tty_puts(NEWLINE);
|
||||||
for (skip = ac / cols + 1, i = 0; i < skip; i++) {
|
for (skip = ac / cols + 1, i = 0; i < skip; i++) {
|
||||||
@@ -298,7 +302,7 @@ void el_print_columns(int ac, char **av)
|
|||||||
tty_put(*p);
|
tty_put(*p);
|
||||||
|
|
||||||
if (j + skip < ac) {
|
if (j + skip < ac) {
|
||||||
while (++len < longest + 3)
|
while (++len < colwidth)
|
||||||
tty_put(' ');
|
tty_put(' ');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -532,26 +536,34 @@ int rl_insert_text(const char *text)
|
|||||||
return rl_point - mark;
|
return rl_point - mark;
|
||||||
}
|
}
|
||||||
|
|
||||||
static el_status_t redisplay(void)
|
static el_status_t redisplay(int cls)
|
||||||
{
|
{
|
||||||
/* XXX: Use "\r\e[K" to get really neat effect on ANSI capable terminals. */
|
if (cls && rl_point == 0 && rl_end == 0)
|
||||||
tty_puts(NEWLINE);
|
tty_puts(CLEAR);
|
||||||
|
else
|
||||||
|
tty_puts("\r\e[K");
|
||||||
|
|
||||||
tty_puts(rl_prompt);
|
tty_puts(rl_prompt);
|
||||||
tty_string(rl_line_buffer);
|
tty_string(rl_line_buffer);
|
||||||
|
|
||||||
return CSmove;
|
return CSmove;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static el_status_t refresh(void)
|
||||||
|
{
|
||||||
|
return redisplay(1);
|
||||||
|
}
|
||||||
|
|
||||||
int rl_refresh_line(int ignore1 __attribute__((unused)), int ignore2 __attribute__((unused)))
|
int rl_refresh_line(int ignore1 __attribute__((unused)), int ignore2 __attribute__((unused)))
|
||||||
{
|
{
|
||||||
redisplay();
|
redisplay(0);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static el_status_t toggle_meta_mode(void)
|
static el_status_t toggle_meta_mode(void)
|
||||||
{
|
{
|
||||||
rl_meta_chars = ! rl_meta_chars;
|
rl_meta_chars = ! rl_meta_chars;
|
||||||
return redisplay();
|
return redisplay(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -679,16 +691,17 @@ static el_status_t h_search_end(const char *p)
|
|||||||
rl_prompt = old_prompt;
|
rl_prompt = old_prompt;
|
||||||
Searching = 0;
|
Searching = 0;
|
||||||
|
|
||||||
if (p == NULL && el_intr_pending > 0) {
|
if (el_intr_pending > 0) {
|
||||||
el_intr_pending = 0;
|
el_intr_pending = 0;
|
||||||
clear_line();
|
clear_line();
|
||||||
return redisplay();
|
return redisplay(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
p = search_hist(p, search_move);
|
p = search_hist(p, search_move);
|
||||||
if (p == NULL) {
|
if (p == NULL) {
|
||||||
el_ring_bell();
|
el_ring_bell();
|
||||||
return redisplay();
|
clear_line();
|
||||||
|
return redisplay(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
return do_insert_hist(p);
|
return do_insert_hist(p);
|
||||||
@@ -888,7 +901,7 @@ static el_status_t end_line(void)
|
|||||||
|
|
||||||
static el_status_t del_char(void)
|
static el_status_t del_char(void)
|
||||||
{
|
{
|
||||||
return delete_string(Repeat == NO_ARG ? 1 : Repeat);
|
return delete_string(Repeat == NO_ARG ? CSeof : Repeat);
|
||||||
}
|
}
|
||||||
|
|
||||||
el_status_t el_del_char(void)
|
el_status_t el_del_char(void)
|
||||||
@@ -1126,7 +1139,7 @@ static void hist_add(const char *p)
|
|||||||
char *s;
|
char *s;
|
||||||
|
|
||||||
#ifdef CONFIG_UNIQUE_HISTORY
|
#ifdef CONFIG_UNIQUE_HISTORY
|
||||||
if (H.Pos && strcmp(p, H.Lines[H.Pos - 1]) == 0)
|
if (H.Size && strcmp(p, H.Lines[H.Size - 1]) == 0)
|
||||||
return;
|
return;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -1297,7 +1310,7 @@ void rl_clear_message(void)
|
|||||||
|
|
||||||
void rl_forced_update_display()
|
void rl_forced_update_display()
|
||||||
{
|
{
|
||||||
redisplay();
|
redisplay(0);
|
||||||
tty_flush();
|
tty_flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1843,7 +1856,7 @@ static el_keymap_t Map[64] = {
|
|||||||
{ CTL('I'), c_complete },
|
{ CTL('I'), c_complete },
|
||||||
{ CTL('J'), accept_line },
|
{ CTL('J'), accept_line },
|
||||||
{ CTL('K'), kill_line },
|
{ CTL('K'), kill_line },
|
||||||
{ CTL('L'), redisplay },
|
{ CTL('L'), refresh },
|
||||||
{ CTL('M'), accept_line },
|
{ CTL('M'), accept_line },
|
||||||
{ CTL('N'), h_next },
|
{ CTL('N'), h_next },
|
||||||
{ CTL('O'), el_ring_bell },
|
{ CTL('O'), el_ring_bell },
|
||||||
@@ -1933,12 +1946,19 @@ static el_status_t el_bind_key_in_map(int key, el_keymap_func_t function, el_key
|
|||||||
|
|
||||||
el_status_t el_bind_key(int key, el_keymap_func_t function)
|
el_status_t el_bind_key(int key, el_keymap_func_t function)
|
||||||
{
|
{
|
||||||
return el_bind_key_in_map(key, function, Map, ARRAY_ELEMENTS(Map));
|
return el_bind_key_in_map(key, function, Map, NELEMS(Map));
|
||||||
}
|
}
|
||||||
|
|
||||||
el_status_t el_bind_key_in_metamap(int key, el_keymap_func_t function)
|
el_status_t el_bind_key_in_metamap(int key, el_keymap_func_t function)
|
||||||
{
|
{
|
||||||
return el_bind_key_in_map(key, function, MetaMap, ARRAY_ELEMENTS(MetaMap));
|
return el_bind_key_in_map(key, function, MetaMap, NELEMS(MetaMap));
|
||||||
|
}
|
||||||
|
|
||||||
|
rl_getc_func_t *rl_set_getc_func(rl_getc_func_t *func)
|
||||||
|
{
|
||||||
|
rl_getc_func_t *old = rl_getc_function;
|
||||||
|
rl_getc_function = func;
|
||||||
|
return old;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -57,8 +57,10 @@
|
|||||||
#define MEM_INC 64
|
#define MEM_INC 64
|
||||||
#define SCREEN_INC 256
|
#define SCREEN_INC 256
|
||||||
|
|
||||||
/* http://stackoverflow.com/questions/1598773/is-there-a-standard-function-in-c-that-would-return-the-length-of-an-array/1598827#1598827 */
|
/* From The Practice of Programming, by Kernighan and Pike */
|
||||||
#define ARRAY_ELEMENTS(arr) ((sizeof(arr)/sizeof(0[arr])) / ((size_t)(!(sizeof(arr) % sizeof(0[arr])))))
|
#ifndef NELEMS
|
||||||
|
#define NELEMS(array) (sizeof(array) / sizeof(array[0]))
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
** Variables and routines internal to this package.
|
** Variables and routines internal to this package.
|
||||||
|
Reference in New Issue
Block a user