// // 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; }