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
* Running commands in separate process and termination them with `ctrl+c`
* `cd` and `exit` builtin commands
* Files autocompletion on `Tab` keypress
* Files and commands from `/bin` autocompletion on `Tab` keypress
# Builtin commands
* `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error.
@ -20,3 +20,4 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
* Environmental variables
* `Ctrl+Z` running programm with `fd`
* Commands autocompletion
* Getting commands path from system `PATH` environment variable

View File

@ -10,6 +10,7 @@
#include <stdio.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);
#endif

View File

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