Added autocompletion for commands from /bin

This commit is contained in:
Dmitriy Shishkov 2020-07-09 12:54:34 +05:00
parent d8c665a625
commit a6ccd9344f
3 changed files with 24 additions and 15 deletions

View File

@ -8,7 +8,7 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
* Command input with `left` and `right` arrow, `home` and `end` keys navigation and `backspace`, `delete` support * Command input with `left` and `right` arrow, `home` and `end` keys navigation and `backspace`, `delete` support
* Running commands in separate process and termination them with `ctrl+c` * Running commands in separate process and termination them with `ctrl+c`
* `cd` and `exit` builtin commands * `cd` and `exit` builtin commands
* Files autocompletion on `Tab` keypress * Files and commands from `/bin` autocompletion on `Tab` keypress
# Builtin commands # Builtin commands
* `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error. * `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error.
@ -19,4 +19,5 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
* Replace linux `echo` command with builtin one with support of environmental variables * Replace linux `echo` command with builtin one with support of environmental variables
* Environmental variables * Environmental variables
* `Ctrl+Z` running programm with `fd` * `Ctrl+Z` running programm with `fd`
* Commands autocompletion * Commands autocompletion
* Getting commands path from system `PATH` environment variable

View File

@ -10,6 +10,7 @@
#include <stdio.h> #include <stdio.h>
#include <dirent.h> #include <dirent.h>
size_t get_dir_list(char ***dir_list, char *path);
void complete_line(int *pos, int *n, char **line, char **out); void complete_line(int *pos, int *n, char **line, char **out);
#endif #endif

View File

@ -2,16 +2,15 @@
#include "../include/shell.h" #include "../include/shell.h"
#include "../include/tree.h" #include "../include/tree.h"
size_t get_dir_list(char ***dir_list) size_t get_dir_list(char ***dir_list, char *path)
{ {
size_t n = 0; size_t n = 0;
*dir_list = malloc(sizeof(char *) * n); *dir_list = malloc(sizeof(char *) * n);
char *pwd = get_current_dir_name();
DIR *dir; DIR *dir;
struct dirent *ent; struct dirent *ent;
if ((dir = opendir(pwd)) == NULL) if ((dir = opendir(path)) == NULL)
{ {
perror("opendir"); perror("opendir");
return -1; return -1;
@ -33,11 +32,21 @@ void complete_line(int *pos, int *n, char **line, char **out)
(*line)[*pos] = '\0'; (*line)[*pos] = '\0';
*out = strdup("\x1b[2K"); *out = strdup("\x1b[2K");
char **dir_list; char **comp_list;
size_t sz = get_dir_list(&dir_list);
char *curr_path, *tmp_line = strdup(*line); char *curr_path, *tmp_line = strdup(*line);
if ((curr_path = strtok(tmp_line, " ")) != NULL && (*line)[*pos - 1] != ' ') size_t sz;
if ((curr_path = strtok(tmp_line, " ")) != NULL && (curr_path = strtok(NULL, " ")) != NULL)
{
sz = get_dir_list(&comp_list, get_current_dir_name());
}
else
{
sz = get_dir_list(&comp_list, "/usr/bin");
curr_path = *line;
}
if (*pos > 0 && (*line)[*pos - 1] != ' ')
{ {
curr_path = strdup(curr_path); curr_path = strdup(curr_path);
@ -52,17 +61,15 @@ void complete_line(int *pos, int *n, char **line, char **out)
struct tree_node *child_dirs_root = get_new_node(); struct tree_node *child_dirs_root = get_new_node();
for (size_t i = 0; i < sz; i++) for (size_t i = 0; i < sz; i++)
{ {
insert_tree(child_dirs_root, dir_list[i]); insert_tree(child_dirs_root, comp_list[i]);
} }
free(dir_list); sz = list_strings_containing(child_dirs_root, curr_path, &comp_list);
sz = list_strings_containing(child_dirs_root, curr_path, &dir_list);
} }
if (sz == 1) if (sz == 1)
{ {
*out = strdup(dir_list[0] + strlen(curr_path)); *out = strdup(comp_list[0] + strlen(curr_path));
*pos += strlen(*out); *pos += strlen(*out);
*n = *pos; *n = *pos;
@ -74,8 +81,8 @@ void complete_line(int *pos, int *n, char **line, char **out)
for (int i = 0; i < sz; i++) for (int i = 0; i < sz; i++)
{ {
*out = realloc(*out, strlen(*out) + strlen(dir_list[i]) + 2); *out = realloc(*out, strlen(*out) + strlen(comp_list[i]) + 2);
*out = strcat(*out, dir_list[i]); *out = strcat(*out, comp_list[i]);
*out = strcat(*out, " "); *out = strcat(*out, " ");
} }