//
// aegis - project change supervisor
// Copyright (C) 2002-2008 Peter Miller
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see
// .
//
#include
#include
#include
#include
#include
#include
#include
#include
#include
struct complete_filename_ty
{
complete_ty inherited;
int regular_ok;
int special_ok;
};
static void
destructor(complete_ty *)
{
}
static void
read_dir(string_ty *path, string_list_ty *wlp)
{
wlp->clear();
read_whole_dir__wl(path->str_text, wlp);
}
static void
perform(complete_ty *cp, shell_ty *sh)
{
complete_filename_ty *this_thing;
string_ty *prefix;
string_ty *prefix_dir;
string_ty *prefix_ent;
size_t j;
string_list_ty names;
this_thing = (complete_filename_ty *)cp;
prefix = shell_prefix_get(sh);
//
// Break the prefix into directory part and filename part.
//
prefix_dir = os_dirname_relative(prefix);
prefix_ent = os_entryname_relative(prefix);
os_become_orig();
read_dir(prefix_dir, &names);
os_become_undo();
for (j = 0; j < names.nstrings; ++j)
{
string_ty *name;
int err;
struct stat st;
string_ty *path;
static struct stat stz;
name = names.string[j];
if (!str_leading_prefix(name, prefix_ent))
continue;
path = os_path_cat(prefix_dir, name);
#ifdef S_IFLNK
err = glue_lstat(path->str_text, &st);
#else
err = glue_stat(path->str_text, &st);
#endif
if (err)
st = stz;
if (S_ISDIR(st.st_mode))
{
string_ty *s;
//
// We emit directories with and without a slash, so that Bash
// doesn't thing we are finished, and put a space after the
// argument if there is only one directory in the answer.
//
shell_emit(sh, path);
s = str_format("%s/", path->str_text);
shell_emit(sh, s);
str_free(s);
}
else if (S_ISREG(st.st_mode))
{
if (this_thing->regular_ok)
shell_emit(sh, path);
}
else
{
if (this_thing->special_ok)
shell_emit(sh, path);
}
str_free(path);
}
str_free(prefix_ent);
str_free(prefix_dir);
}
static complete_vtbl_ty vtbl =
{
destructor,
perform,
sizeof(complete_filename_ty),
"filename",
};
complete_ty *
complete_filename(int dir_only)
{
complete_ty *result;
complete_filename_ty *this_thing;
result = complete_new(&vtbl);
this_thing = (complete_filename_ty *)result;
this_thing->regular_ok = !dir_only;
this_thing->special_ok = 0;
return result;
}