libcubescript/tools/edit_readline.hh

88 lines
2.1 KiB
C++
Raw Normal View History

#ifdef CS_REPL_USE_READLINE
#ifndef CS_REPL_HAS_EDIT
#define CS_REPL_HAS_EDIT
/* use the GNU readline library */
2016-09-02 23:15:05 +02:00
#include <string.h>
#include <ostd/string.hh>
#include <ostd/maybe.hh>
#include <readline/readline.h>
#include <readline/history.h>
2016-09-11 23:33:02 +02:00
static CsState *rd_cs = nullptr;
2016-09-02 23:15:05 +02:00
static char *ln_complete_list(char const *buf, int state) {
static ostd::ConstCharRange cmd;
static ostd::PointerRange<CsIdent *> itr;
if (!state) {
cmd = get_complete_cmd(buf);
itr = rd_cs->get_idents();
2016-09-02 23:15:05 +02:00
}
for (; !itr.empty(); itr.pop_front()) {
CsIdent *id = itr.front();
if (!id->is_command()) {
continue;
}
ostd::ConstCharRange idname = id->get_name();
if (idname.size() <= cmd.size()) {
continue;
}
if (idname.slice(0, cmd.size()) == cmd) {
itr.pop_front();
return strdup(idname.data());
}
}
return nullptr;
}
static char **ln_complete(char const *buf, int, int) {
rl_attempted_completion_over = 1;
return rl_completion_matches(buf, ln_complete_list);
}
void ln_hint() {
2016-09-11 23:33:02 +02:00
CsCommand *cmd = get_hint_cmd(*rd_cs, rl_line_buffer);
if (!cmd) {
rl_redisplay();
return;
}
ostd::String old = rl_line_buffer;
ostd::String args = old;
args += " [";
fill_cmd_args(args, cmd->get_args());
args += "] ";
rl_extend_line_buffer(args.size());
rl_replace_line(args.data(), 0);
rl_redisplay();
rl_replace_line(old.data(), 0);
}
2016-09-11 23:33:02 +02:00
static void init_lineedit(CsState &cs, ostd::ConstCharRange) {
rd_cs = &cs;
2016-09-02 23:15:05 +02:00
rl_attempted_completion_function = ln_complete;
rl_redisplay_function = ln_hint;
}
2016-09-11 23:33:02 +02:00
static ostd::Maybe<ostd::String> read_line(CsState &, CsSvar *pr) {
auto line = readline(pr->get_value().data());
if (!line) {
return ostd::String();
}
ostd::String ret = line;
free(line);
2017-01-25 01:57:33 +01:00
return std::move(ret);
}
2016-09-11 23:33:02 +02:00
static void add_history(CsState &, ostd::ConstCharRange line) {
/* backed by ostd::String so it's terminated */
add_history(line.data());
}
#endif
#endif