lineedit+unicode: make TAB completion work again
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
parent
1302892a14
commit
044b18083b
@ -115,6 +115,9 @@ struct lineedit_statics {
|
|||||||
|
|
||||||
unsigned cursor;
|
unsigned cursor;
|
||||||
unsigned command_len;
|
unsigned command_len;
|
||||||
|
/* *int* maxsize: we want x in "if (x > S.maxsize)"
|
||||||
|
* to _not_ be promoted to unsigned */
|
||||||
|
int maxsize;
|
||||||
CHAR_T *command_ps;
|
CHAR_T *command_ps;
|
||||||
|
|
||||||
const char *cmdedit_prompt;
|
const char *cmdedit_prompt;
|
||||||
@ -390,7 +393,7 @@ static void put_prompt(void)
|
|||||||
static void redraw(int y, int back_cursor)
|
static void redraw(int y, int back_cursor)
|
||||||
{
|
{
|
||||||
if (y > 0) /* up to start y */
|
if (y > 0) /* up to start y */
|
||||||
printf("\033[%dA", y);
|
printf("\033[%uA", y);
|
||||||
bb_putchar('\r');
|
bb_putchar('\r');
|
||||||
put_prompt();
|
put_prompt();
|
||||||
input_end(); /* rewrite */
|
input_end(); /* rewrite */
|
||||||
@ -687,7 +690,7 @@ static void exe_n_cwd_tab_completion(char *command, int type)
|
|||||||
memmove(pos_buf+(is), pos_buf+(in), (MAX_LINELEN+1-(is)-(in)) * sizeof(pos_buf[0])); \
|
memmove(pos_buf+(is), pos_buf+(in), (MAX_LINELEN+1-(is)-(in)) * sizeof(pos_buf[0])); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
static int find_match(char *matchBuf, int *len_with_quotes)
|
static NOINLINE int find_match(char *matchBuf, int *len_with_quotes)
|
||||||
{
|
{
|
||||||
int i, j;
|
int i, j;
|
||||||
int command_mode;
|
int command_mode;
|
||||||
@ -922,12 +925,19 @@ static void input_tab(smallint *lastWasTab)
|
|||||||
#define matchBuf (S.input_tab__matchBuf)
|
#define matchBuf (S.input_tab__matchBuf)
|
||||||
int find_type;
|
int find_type;
|
||||||
int recalc_pos;
|
int recalc_pos;
|
||||||
|
#if ENABLE_FEATURE_ASSUME_UNICODE
|
||||||
|
/* cursor pos in command converted to multibyte form */
|
||||||
|
int cursor_mb;
|
||||||
|
#endif
|
||||||
|
|
||||||
*lastWasTab = TRUE; /* flop trigger */
|
*lastWasTab = TRUE; /* flop trigger */
|
||||||
|
|
||||||
/* Make a local copy of the string --
|
/* Make a local copy of the string --
|
||||||
* up to the position of the cursor */
|
* up to the position of the cursor */
|
||||||
save_string(matchBuf, cursor + 1);
|
save_string(matchBuf, cursor + 1);
|
||||||
|
#if ENABLE_FEATURE_ASSUME_UNICODE
|
||||||
|
cursor_mb = strlen(matchBuf);
|
||||||
|
#endif
|
||||||
tmp = matchBuf;
|
tmp = matchBuf;
|
||||||
|
|
||||||
find_type = find_match(matchBuf, &recalc_pos);
|
find_type = find_match(matchBuf, &recalc_pos);
|
||||||
@ -939,7 +949,7 @@ static void input_tab(smallint *lastWasTab)
|
|||||||
/* If the word starts with `~' and there is no slash in the word,
|
/* If the word starts with `~' and there is no slash in the word,
|
||||||
* then try completing this word as a username. */
|
* then try completing this word as a username. */
|
||||||
if (state->flags & USERNAME_COMPLETION)
|
if (state->flags & USERNAME_COMPLETION)
|
||||||
if (matchBuf[0] == '~' && strchr(matchBuf, '/') == 0)
|
if (matchBuf[0] == '~' && strchr(matchBuf, '/') == NULL)
|
||||||
username_tab_completion(matchBuf, NULL);
|
username_tab_completion(matchBuf, NULL);
|
||||||
#endif
|
#endif
|
||||||
/* Try to match any executable in our path and everything
|
/* Try to match any executable in our path and everything
|
||||||
@ -965,18 +975,20 @@ static void input_tab(smallint *lastWasTab)
|
|||||||
num_matches = n + 1;
|
num_matches = n + 1;
|
||||||
}
|
}
|
||||||
/* Did we find exactly one match? */
|
/* Did we find exactly one match? */
|
||||||
if (!matches || num_matches > 1) {
|
if (!matches || num_matches > 1) { /* no */
|
||||||
beep();
|
beep();
|
||||||
if (!matches)
|
if (!matches)
|
||||||
return; /* not found */
|
return; /* not found */
|
||||||
/* find minimal match */
|
/* find minimal match */
|
||||||
tmp1 = xstrdup(matches[0]);
|
tmp1 = xstrdup(matches[0]);
|
||||||
for (tmp = tmp1; *tmp; tmp++)
|
for (tmp = tmp1; *tmp; tmp++) {
|
||||||
for (len_found = 1; len_found < num_matches; len_found++)
|
for (len_found = 1; len_found < num_matches; len_found++) {
|
||||||
if (matches[len_found][(tmp - tmp1)] != *tmp) {
|
if (matches[len_found][tmp - tmp1] != *tmp) {
|
||||||
*tmp = '\0';
|
*tmp = '\0';
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
if (*tmp1 == '\0') { /* have unique */
|
if (*tmp1 == '\0') { /* have unique */
|
||||||
free(tmp1);
|
free(tmp1);
|
||||||
return;
|
return;
|
||||||
@ -994,29 +1006,44 @@ static void input_tab(smallint *lastWasTab)
|
|||||||
tmp[len_found+1] = '\0';
|
tmp[len_found+1] = '\0';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
len_found = strlen(tmp);
|
len_found = strlen(tmp);
|
||||||
/* have space to placed match? */
|
|
||||||
if ((len_found - strlen(matchBuf) + command_len) < MAX_LINELEN) {
|
|
||||||
/* before word for match */
|
|
||||||
//TODO:
|
|
||||||
#if !ENABLE_FEATURE_ASSUME_UNICODE
|
#if !ENABLE_FEATURE_ASSUME_UNICODE
|
||||||
command_ps[cursor - recalc_pos] = '\0';
|
/* have space to place the match? */
|
||||||
/* save tail line */
|
/* The result consists of three parts with these lengths: */
|
||||||
|
/* (cursor - recalc_pos) + len_found + (command_len - cursor) */
|
||||||
|
/* it simplifies into: */
|
||||||
|
if ((int)(len_found + command_len - recalc_pos) < S.maxsize) {
|
||||||
|
/* save tail */
|
||||||
strcpy(matchBuf, command_ps + cursor);
|
strcpy(matchBuf, command_ps + cursor);
|
||||||
/* add match */
|
/* add match and tail */
|
||||||
strcat(command_ps, tmp);
|
sprintf(&command_ps[cursor - recalc_pos], "%s%s", tmp, matchBuf);
|
||||||
/* add tail */
|
|
||||||
strcat(command_ps, matchBuf);
|
|
||||||
/* back to begin word for match */
|
|
||||||
input_backward(recalc_pos);
|
|
||||||
/* new pos */
|
|
||||||
recalc_pos = cursor + len_found;
|
|
||||||
/* new len */
|
|
||||||
command_len = strlen(command_ps);
|
command_len = strlen(command_ps);
|
||||||
|
/* new pos */
|
||||||
|
recalc_pos = cursor - recalc_pos + len_found;
|
||||||
/* write out the matched command */
|
/* write out the matched command */
|
||||||
#endif
|
|
||||||
redraw(cmdedit_y, command_len - recalc_pos);
|
redraw(cmdedit_y, command_len - recalc_pos);
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
{
|
||||||
|
char command[MAX_LINELEN];
|
||||||
|
int len = save_string(command, sizeof(command));
|
||||||
|
/* have space to place the match? */
|
||||||
|
/* (cursor_mb - recalc_pos) + len_found + (len - cursor_mb) */
|
||||||
|
if ((int)(len_found + len - recalc_pos) < MAX_LINELEN) {
|
||||||
|
/* save tail */
|
||||||
|
strcpy(matchBuf, command + cursor_mb);
|
||||||
|
/* where do we want to have cursor after all? */
|
||||||
|
strcpy(&command[cursor_mb - recalc_pos], tmp);
|
||||||
|
len = load_string(command, S.maxsize);
|
||||||
|
/* add match and tail */
|
||||||
|
sprintf(&command[cursor_mb - recalc_pos], "%s%s", tmp, matchBuf);
|
||||||
|
command_len = load_string(command, S.maxsize);
|
||||||
|
/* write out the matched command */
|
||||||
|
redraw(cmdedit_y, command_len - len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
free(tmp);
|
free(tmp);
|
||||||
#undef matchBuf
|
#undef matchBuf
|
||||||
} else {
|
} else {
|
||||||
@ -1024,7 +1051,8 @@ static void input_tab(smallint *lastWasTab)
|
|||||||
* just hit TAB again, print a list of all the
|
* just hit TAB again, print a list of all the
|
||||||
* available choices... */
|
* available choices... */
|
||||||
if (matches && num_matches > 0) {
|
if (matches && num_matches > 0) {
|
||||||
int sav_cursor = cursor; /* change goto_new_line() */
|
/* changed by goto_new_line() */
|
||||||
|
int sav_cursor = cursor;
|
||||||
|
|
||||||
/* Go to the next line */
|
/* Go to the next line */
|
||||||
goto_new_line();
|
goto_new_line();
|
||||||
@ -1641,6 +1669,7 @@ int FAST_FUNC read_line_input(const char *prompt, char *command, int maxsize, li
|
|||||||
// FIXME: audit & improve this
|
// FIXME: audit & improve this
|
||||||
if (maxsize > MAX_LINELEN)
|
if (maxsize > MAX_LINELEN)
|
||||||
maxsize = MAX_LINELEN;
|
maxsize = MAX_LINELEN;
|
||||||
|
S.maxsize = maxsize;
|
||||||
|
|
||||||
/* With null flags, no other fields are ever used */
|
/* With null flags, no other fields are ever used */
|
||||||
state = st ? st : (line_input_t*) &const_int_0;
|
state = st ? st : (line_input_t*) &const_int_0;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user