Add more bash-like dot-dirs cmdedit feature

Close bug 648.
Restore my whitespace
This commit is contained in:
"Vladimir N. Oleynik" 2006-01-25 11:53:47 +00:00
parent 7d501a8673
commit fdb871c498
2 changed files with 108 additions and 97 deletions

View File

@ -563,6 +563,22 @@ static void cmdedit_init(void)
#ifdef CONFIG_FEATURE_COMMAND_TAB_COMPLETION #ifdef CONFIG_FEATURE_COMMAND_TAB_COMPLETION
static char **matches;
static int num_matches;
static char *add_char_to_match;
static void add_match(char *matched, int add_char)
{
int nm = num_matches;
int nm1 = nm + 1;
matches = xrealloc(matches, nm1 * sizeof(char *));
add_char_to_match = xrealloc(add_char_to_match, nm1);
matches[nm] = matched;
add_char_to_match[nm] = (char)add_char;
num_matches++;
}
static int is_execute(const struct stat *st) static int is_execute(const struct stat *st)
{ {
if ((!my_euid && (st->st_mode & (S_IXUSR | S_IXGRP | S_IXOTH))) || if ((!my_euid && (st->st_mode & (S_IXUSR | S_IXGRP | S_IXOTH))) ||
@ -574,19 +590,18 @@ static int is_execute(const struct stat *st)
#ifdef CONFIG_FEATURE_COMMAND_USERNAME_COMPLETION #ifdef CONFIG_FEATURE_COMMAND_USERNAME_COMPLETION
static char **username_tab_completion(char *ud, int *num_matches) static void username_tab_completion(char *ud, char *with_shash_flg)
{ {
struct passwd *entry; struct passwd *entry;
int userlen; int userlen;
char *temp;
ud++; /* ~user/... to user/... */ ud++; /* ~user/... to user/... */
userlen = strlen(ud); userlen = strlen(ud);
if (num_matches == 0) { /* "~/..." or "~user/..." */ if (with_shash_flg) { /* "~/..." or "~user/..." */
char *sav_ud = ud - 1; char *sav_ud = ud - 1;
char *home = 0; char *home = 0;
char *temp;
if (*ud == '/') { /* "~/..." */ if (*ud == '/') { /* "~/..." */
home = home_pwd_buf; home = home_pwd_buf;
@ -609,28 +624,18 @@ static char **username_tab_completion(char *ud, int *num_matches)
strcpy(sav_ud, temp2); strcpy(sav_ud, temp2);
} }
} }
return 0; /* void, result save to argument :-) */
} else { } else {
/* "~[^/]*" */ /* "~[^/]*" */
char **matches = (char **) NULL;
int nm = 0;
setpwent(); setpwent();
while ((entry = getpwent()) != NULL) { while ((entry = getpwent()) != NULL) {
/* Null usernames should result in all users as possible completions. */ /* Null usernames should result in all users as possible completions. */
if ( /*!userlen || */ !strncmp(ud, entry->pw_name, userlen)) { if ( /*!userlen || */ !strncmp(ud, entry->pw_name, userlen)) {
add_match(bb_xasprintf("~%s", entry->pw_name), '/');
temp = bb_xasprintf("~%s/", entry->pw_name);
matches = xrealloc(matches, (nm + 1) * sizeof(char *));
matches[nm++] = temp;
} }
} }
endpwent(); endpwent();
(*num_matches) = nm;
return (matches);
} }
} }
#endif /* CONFIG_FEATURE_COMMAND_USERNAME_COMPLETION */ #endif /* CONFIG_FEATURE_COMMAND_USERNAME_COMPLETION */
@ -693,7 +698,7 @@ static int path_parse(char ***p, int flags)
return npth; return npth;
} }
static char *add_quote_for_spec_chars(char *found) static char *add_quote_for_spec_chars(char *found, int add)
{ {
int l = 0; int l = 0;
char *s = xmalloc((strlen(found) + 1) * 2); char *s = xmalloc((strlen(found) + 1) * 2);
@ -703,19 +708,17 @@ static char *add_quote_for_spec_chars(char *found)
s[l++] = '\\'; s[l++] = '\\';
s[l++] = *found++; s[l++] = *found++;
} }
if(add)
s[l++] = (char)add;
s[l] = 0; s[l] = 0;
return s; return s;
} }
static char **exe_n_cwd_tab_completion(char *command, int *num_matches, static void exe_n_cwd_tab_completion(char *command, int type)
int type)
{ {
char **matches = 0;
DIR *dir; DIR *dir;
struct dirent *next; struct dirent *next;
char dirbuf[BUFSIZ]; char dirbuf[BUFSIZ];
int nm = *num_matches;
struct stat st; struct stat st;
char *path1[1]; char *path1[1];
char **paths = path1; char **paths = path1;
@ -738,7 +741,7 @@ static char **exe_n_cwd_tab_completion(char *command, int *num_matches,
dirbuf[(pfind - command) + 1] = 0; dirbuf[(pfind - command) + 1] = 0;
#ifdef CONFIG_FEATURE_COMMAND_USERNAME_COMPLETION #ifdef CONFIG_FEATURE_COMMAND_USERNAME_COMPLETION
if (dirbuf[0] == '~') /* ~/... or ~user/... */ if (dirbuf[0] == '~') /* ~/... or ~user/... */
username_tab_completion(dirbuf, 0); username_tab_completion(dirbuf, dirbuf);
#endif #endif
/* "strip" dirname in command */ /* "strip" dirname in command */
pfind++; pfind++;
@ -755,6 +758,7 @@ static char **exe_n_cwd_tab_completion(char *command, int *num_matches,
while ((next = readdir(dir)) != NULL) { while ((next = readdir(dir)) != NULL) {
char *str_found = next->d_name; char *str_found = next->d_name;
int add_chr = 0;
/* matched ? */ /* matched ? */
if (strncmp(str_found, pfind, strlen(pfind))) if (strncmp(str_found, pfind, strlen(pfind)))
@ -775,23 +779,22 @@ static char **exe_n_cwd_tab_completion(char *command, int *num_matches,
strcpy(found, next->d_name); /* only name */ strcpy(found, next->d_name); /* only name */
if (S_ISDIR(st.st_mode)) { if (S_ISDIR(st.st_mode)) {
/* name is directory */ /* name is directory */
str_found = found; char *e = found + strlen(found) - 1;
found = concat_path_file(found, "");
free(str_found); add_chr = '/';
str_found = add_quote_for_spec_chars(found); if(*e == '/')
*e = '\0';
} else { } else {
/* not put found file if search only dirs for cd */ /* not put found file if search only dirs for cd */
if (type == FIND_DIR_ONLY) if (type == FIND_DIR_ONLY)
goto cont; goto cont;
str_found = add_quote_for_spec_chars(found);
if (type == FIND_FILE_ONLY || if (type == FIND_FILE_ONLY ||
(type == FIND_EXE_ONLY && is_execute(&st))) (type == FIND_EXE_ONLY && is_execute(&st)))
strcat(str_found, " "); add_chr = ' ';
} }
/* Add it to the list */ /* Add it to the list */
matches = xrealloc(matches, (nm + 1) * sizeof(char *)); add_match(found, add_chr);
continue;
matches[nm++] = str_found;
cont: cont:
free(found); free(found);
} }
@ -801,16 +804,8 @@ cont:
free(paths[0]); /* allocated memory only in first member */ free(paths[0]); /* allocated memory only in first member */
free(paths); free(paths);
} }
*num_matches = nm;
return (matches);
} }
static int match_compare(const void *a, const void *b)
{
return strcmp(*(char **) a, *(char **) b);
}
#define QUOT (UCHAR_MAX+1) #define QUOT (UCHAR_MAX+1)
@ -984,16 +979,20 @@ static int find_match(char *matchBuf, int *len_with_quotes)
display by column original ideas from ls applet, display by column original ideas from ls applet,
very optimize by my :) very optimize by my :)
*/ */
static void showfiles(char **matches, int nfiles) static void showfiles(void)
{ {
int ncols, row; int ncols, row;
int column_width = 0; int column_width = 0;
int nfiles = num_matches;
int nrows = nfiles; int nrows = nfiles;
char str_add_chr[2];
int l;
/* find the longest file name- use that as the column width */ /* find the longest file name- use that as the column width */
for (row = 0; row < nrows; row++) { for (row = 0; row < nrows; row++) {
int l = strlen(matches[row]); l = strlen(matches[row]);
if(add_char_to_match[row])
l++;
if (column_width < l) if (column_width < l)
column_width = l; column_width = l;
} }
@ -1004,17 +1003,27 @@ static void showfiles(char **matches, int nfiles)
nrows /= ncols; nrows /= ncols;
if(nfiles % ncols) if(nfiles % ncols)
nrows++; /* round up fractionals */ nrows++; /* round up fractionals */
column_width = -column_width; /* for printf("%-Ns", ...); */
} else { } else {
ncols = 1; ncols = 1;
} }
str_add_chr[1] = 0;
for (row = 0; row < nrows; row++) { for (row = 0; row < nrows; row++) {
int n = row; int n = row;
int nc; int nc;
int acol;
for(nc = 1; nc < ncols && n+nrows < nfiles; n += nrows, nc++) for(nc = 1; nc < ncols && n+nrows < nfiles; n += nrows, nc++) {
printf("%*s", column_width, matches[n]); str_add_chr[0] = add_char_to_match[n];
printf("%s\n", matches[n]); acol = str_add_chr[0] ? column_width - 1 : column_width;
printf("%s%s", matches[n], str_add_chr);
l = strlen(matches[n]);
while(l < acol) {
putchar(' ');
l++;
}
}
str_add_chr[0] = add_char_to_match[n];
printf("%s%s\n", matches[n], str_add_chr);
} }
} }
@ -1022,21 +1031,20 @@ static void showfiles(char **matches, int nfiles)
static void input_tab(int *lastWasTab) static void input_tab(int *lastWasTab)
{ {
/* Do TAB completion */ /* Do TAB completion */
static int num_matches;
static char **matches;
if (lastWasTab == 0) { /* free all memory */ if (lastWasTab == 0) { /* free all memory */
if (matches) { if (matches) {
while (num_matches > 0) while (num_matches > 0)
free(matches[--num_matches]); free(matches[--num_matches]);
free(matches); free(matches);
matches = (char **) NULL; matches = (char **) NULL;
free(add_char_to_match);
add_char_to_match = NULL;
} }
return; return;
} }
if (! *lastWasTab) { if (! *lastWasTab) {
char *tmp; char *tmp, *tmp1;
int len_found; int len_found;
char matchBuf[BUFSIZ]; char matchBuf[BUFSIZ];
int find_type; int find_type;
@ -1059,64 +1067,68 @@ static void input_tab(int *lastWasTab)
* then try completing this word as a username. */ * then try completing this word as a username. */
if (matchBuf[0] == '~' && strchr(matchBuf, '/') == 0) if (matchBuf[0] == '~' && strchr(matchBuf, '/') == 0)
matches = username_tab_completion(matchBuf, &num_matches); username_tab_completion(matchBuf, NULL);
if (!matches)
#endif #endif
/* Try to match any executable in our path and everything /* Try to match any executable in our path and everything
* in the current working directory that matches. */ * in the current working directory that matches. */
if (!matches) exe_n_cwd_tab_completion(matchBuf, find_type);
matches = /* Remove duplicate found and sort */
exe_n_cwd_tab_completion(matchBuf,
&num_matches, find_type);
/* Remove duplicate found */
if(matches) { if(matches) {
int i, j; int i, j, n, srt;
/* bubble */ /* bubble */
for(i=0; i<(num_matches-1); i++) n = num_matches;
for(j=i+1; j<num_matches; j++) for(i=0; i<(n-1); i++)
if(matches[i]!=0 && matches[j]!=0 && for(j=i+1; j<n; j++)
strcmp(matches[i], matches[j])==0) { if(matches[i]!=NULL && matches[j]!=NULL) {
srt = strcmp(matches[i], matches[j]);
if(srt == 0) {
free(matches[j]); free(matches[j]);
matches[j]=0; matches[j]=0;
} else if(srt > 0) {
tmp1 = matches[i];
matches[i] = matches[j];
matches[j] = tmp1;
srt = add_char_to_match[i];
add_char_to_match[i] = add_char_to_match[j];
add_char_to_match[j] = srt;
} }
j=num_matches; }
num_matches = 0; j = n;
n = 0;
for(i=0; i<j; i++) for(i=0; i<j; i++)
if(matches[i]) { if(matches[i]) {
if(!strcmp(matches[i], "./")) matches[n]=matches[i];
matches[i][1]=0; add_char_to_match[n]=add_char_to_match[i];
else if(!strcmp(matches[i], "../")) n++;
matches[i][2]=0;
matches[num_matches++]=matches[i];
} }
num_matches = n;
} }
/* Did we find exactly one match? */ /* Did we find exactly one match? */
if (!matches || num_matches > 1) { if (!matches || num_matches > 1) {
char *tmp1;
beep(); beep();
if (!matches) if (!matches)
return; /* not found */ return; /* not found */
/* sort */
qsort(matches, num_matches, sizeof(char *), match_compare);
/* find minimal match */ /* find minimal match */
tmp = bb_xstrdup(matches[0]); tmp1 = bb_xstrdup(matches[0]);
for (tmp1 = tmp; *tmp1; tmp1++) 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][(tmp1 - tmp)] != *tmp1) { if (matches[len_found][(tmp - tmp1)] != *tmp) {
*tmp1 = 0; *tmp = 0;
break; break;
} }
if (*tmp == 0) { /* have unique */ if (*tmp1 == 0) { /* have unique */
free(tmp); free(tmp1);
return; return;
} }
tmp = add_quote_for_spec_chars(tmp1, 0);
free(tmp1);
} else { /* one match */ } else { /* one match */
tmp = matches[0]; tmp = add_quote_for_spec_chars(matches[0], add_char_to_match[0]);
/* for next completion current found */ /* for next completion current found */
*lastWasTab = FALSE; *lastWasTab = FALSE;
} }
len_found = strlen(tmp); len_found = strlen(tmp);
/* have space to placed match? */ /* have space to placed match? */
if ((len_found - strlen(matchBuf) + len) < BUFSIZ) { if ((len_found - strlen(matchBuf) + len) < BUFSIZ) {
@ -1138,7 +1150,6 @@ static void input_tab(int *lastWasTab)
/* write out the matched command */ /* write out the matched command */
redraw(cmdedit_y, len - recalc_pos); redraw(cmdedit_y, len - recalc_pos);
} }
if (tmp != matches[0])
free(tmp); free(tmp);
} else { } else {
/* Ok -- the last char was a TAB. Since they /* Ok -- the last char was a TAB. Since they
@ -1149,7 +1160,7 @@ static void input_tab(int *lastWasTab)
/* Go to the next line */ /* Go to the next line */
goto_new_line(); goto_new_line();
showfiles(matches, num_matches); showfiles();
redraw(0, len - sav_cursor); redraw(0, len - sav_cursor);
} }
} }