Added autocompletion for commands from /bin
This commit is contained in:
parent
d8c665a625
commit
a6ccd9344f
@ -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.
|
||||
@ -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
|
||||
* Environmental variables
|
||||
* `Ctrl+Z` running programm with `fd`
|
||||
* Commands autocompletion
|
||||
* Commands autocompletion
|
||||
* Getting commands path from system `PATH` environment variable
|
@ -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
|
@ -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, " ");
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user