Use RankedMatch for filename based completion (file and command completion)
This commit is contained in:
parent
1fd7e80f04
commit
6d5900af16
107
src/file.cc
107
src/file.cc
|
@ -3,6 +3,7 @@
|
||||||
#include "assert.hh"
|
#include "assert.hh"
|
||||||
#include "buffer.hh"
|
#include "buffer.hh"
|
||||||
#include "unicode.hh"
|
#include "unicode.hh"
|
||||||
|
#include "ranked_match.hh"
|
||||||
#include "regex.hh"
|
#include "regex.hh"
|
||||||
#include "string.hh"
|
#include "string.hh"
|
||||||
|
|
||||||
|
@ -333,8 +334,7 @@ void make_directory(StringView dir)
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Filter>
|
template<typename Filter>
|
||||||
Vector<String> list_files(StringView prefix, StringView dirname,
|
Vector<String> list_files(StringView dirname, Filter filter)
|
||||||
Filter filter)
|
|
||||||
{
|
{
|
||||||
char buffer[PATH_MAX+1];
|
char buffer[PATH_MAX+1];
|
||||||
format_to(buffer, "{}", dirname);
|
format_to(buffer, "{}", dirname);
|
||||||
|
@ -345,43 +345,39 @@ Vector<String> list_files(StringView prefix, StringView dirname,
|
||||||
auto closeDir = on_scope_end([=]{ closedir(dir); });
|
auto closeDir = on_scope_end([=]{ closedir(dir); });
|
||||||
|
|
||||||
Vector<String> result;
|
Vector<String> result;
|
||||||
Vector<String> subseq_result;
|
|
||||||
while (dirent* entry = readdir(dir))
|
while (dirent* entry = readdir(dir))
|
||||||
{
|
{
|
||||||
if (not filter(*entry))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
StringView filename = entry->d_name;
|
StringView filename = entry->d_name;
|
||||||
if (filename.empty())
|
if (filename.empty() or not filter(*entry))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
const bool match_prefix = prefix_match(filename, prefix);
|
|
||||||
const bool match_subseq = subsequence_match(filename, prefix);
|
|
||||||
struct stat st;
|
struct stat st;
|
||||||
if (match_prefix or match_subseq)
|
auto fmt_str = (dirname.empty() or dirname.back() == '/') ? "{}{}" : "{}/{}";
|
||||||
{
|
format_to(buffer, fmt_str, dirname, filename);
|
||||||
auto fmt_str = (dirname.empty() or dirname.back() == '/') ? "{}{}" : "{}/{}";
|
if (stat(buffer, &st) != 0)
|
||||||
format_to(buffer, fmt_str, dirname, filename);
|
continue;
|
||||||
if (stat(buffer, &st) != 0)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (S_ISDIR(st.st_mode))
|
if (S_ISDIR(st.st_mode))
|
||||||
filename = format_to(buffer, "{}/", filename);
|
filename = format_to(buffer, "{}/", filename);
|
||||||
if (prefix.length() != 0 or filename[0_byte] != '.')
|
result.push_back(filename.str());
|
||||||
{
|
|
||||||
if (match_prefix)
|
|
||||||
result.push_back(filename.str());
|
|
||||||
if (match_subseq)
|
|
||||||
subseq_result.push_back(filename.str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return result.empty() ? subseq_result : result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<String> list_files(StringView directory)
|
Vector<String> list_files(StringView directory)
|
||||||
{
|
{
|
||||||
return list_files("", directory, [](const dirent&) { return true; });
|
return list_files(directory, [](const dirent& entry) {
|
||||||
|
return StringView{entry.d_name}.substr(0_byte, 1_byte) != ".";
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
static CandidateList candidates(ConstArrayView<RankedMatch> matches, StringView dirname)
|
||||||
|
{
|
||||||
|
CandidateList res;
|
||||||
|
res.reserve(matches.size());
|
||||||
|
for (auto& match : matches)
|
||||||
|
res.push_back(dirname + match.candidate());
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
CandidateList complete_filename(StringView prefix,
|
CandidateList complete_filename(StringView prefix,
|
||||||
|
@ -394,17 +390,22 @@ CandidateList complete_filename(StringView prefix,
|
||||||
|
|
||||||
const bool check_ignored_regex = not ignored_regex.empty() and
|
const bool check_ignored_regex = not ignored_regex.empty() and
|
||||||
not regex_match(fileprefix.begin(), fileprefix.end(), ignored_regex);
|
not regex_match(fileprefix.begin(), fileprefix.end(), ignored_regex);
|
||||||
|
const bool include_hidden = fileprefix.substr(0_byte, 1_byte) == ".";
|
||||||
|
|
||||||
auto filter = [&ignored_regex, check_ignored_regex](const dirent& entry)
|
auto filter = [&ignored_regex, check_ignored_regex, include_hidden](const dirent& entry)
|
||||||
{
|
{
|
||||||
return not check_ignored_regex or
|
return (include_hidden or StringView{entry.d_name}.substr(0_byte, 1_byte) != ".") and
|
||||||
not regex_match(entry.d_name, ignored_regex);
|
(not check_ignored_regex or not regex_match(entry.d_name, ignored_regex));
|
||||||
};
|
};
|
||||||
Vector<String> res = list_files(fileprefix, dirname, filter);
|
auto files = list_files(dirname, filter);
|
||||||
for (auto& file : res)
|
Vector<RankedMatch> matches;
|
||||||
file = dirname + file;
|
for (auto& file : files)
|
||||||
std::sort(res.begin(), res.end());
|
{
|
||||||
return res;
|
if (RankedMatch match{file, fileprefix})
|
||||||
|
matches.push_back(match);
|
||||||
|
}
|
||||||
|
std::sort(matches.begin(), matches.end());
|
||||||
|
return candidates(matches, dirname);
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<String> complete_command(StringView prefix, ByteCount cursor_pos)
|
Vector<String> complete_command(StringView prefix, ByteCount cursor_pos)
|
||||||
|
@ -415,11 +416,11 @@ Vector<String> complete_command(StringView prefix, ByteCount cursor_pos)
|
||||||
|
|
||||||
if (not dirname.empty())
|
if (not dirname.empty())
|
||||||
{
|
{
|
||||||
char buffer[PATH_MAX+1];
|
auto filter = [&dirname](const dirent& entry)
|
||||||
auto filter = [&dirname, &buffer](const dirent& entry)
|
|
||||||
{
|
{
|
||||||
struct stat st;
|
char buffer[PATH_MAX+1];
|
||||||
format_to(buffer, "{}{}", dirname, entry.d_name);
|
format_to(buffer, "{}{}", dirname, entry.d_name);
|
||||||
|
struct stat st;
|
||||||
if (stat(buffer, &st) != 0)
|
if (stat(buffer, &st) != 0)
|
||||||
return false;
|
return false;
|
||||||
bool executable = (st.st_mode & S_IXUSR)
|
bool executable = (st.st_mode & S_IXUSR)
|
||||||
|
@ -427,11 +428,15 @@ Vector<String> complete_command(StringView prefix, ByteCount cursor_pos)
|
||||||
| (st.st_mode & S_IXOTH);
|
| (st.st_mode & S_IXOTH);
|
||||||
return S_ISDIR(st.st_mode) or (S_ISREG(st.st_mode) and executable);
|
return S_ISDIR(st.st_mode) or (S_ISREG(st.st_mode) and executable);
|
||||||
};
|
};
|
||||||
Vector<String> res = list_files(fileprefix, dirname, filter);
|
auto files = list_files(dirname, filter);
|
||||||
for (auto& file : res)
|
Vector<RankedMatch> matches;
|
||||||
file = dirname + file;
|
for (auto& file : files)
|
||||||
std::sort(res.begin(), res.end());
|
{
|
||||||
return res;
|
if (RankedMatch match{file, real_prefix})
|
||||||
|
matches.push_back(match);
|
||||||
|
}
|
||||||
|
std::sort(matches.begin(), matches.end());
|
||||||
|
return candidates(matches, dirname);
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef decltype(stat::st_mtim) TimeSpec;
|
typedef decltype(stat::st_mtim) TimeSpec;
|
||||||
|
@ -443,7 +448,7 @@ Vector<String> complete_command(StringView prefix, ByteCount cursor_pos)
|
||||||
};
|
};
|
||||||
static UnorderedMap<String, CommandCache, MemoryDomain::Commands> command_cache;
|
static UnorderedMap<String, CommandCache, MemoryDomain::Commands> command_cache;
|
||||||
|
|
||||||
Vector<String> res;
|
Vector<RankedMatch> matches;
|
||||||
for (auto dir : split(getenv("PATH"), ':'))
|
for (auto dir : split(getenv("PATH"), ':'))
|
||||||
{
|
{
|
||||||
auto dirname = ((not dir.empty() and dir.back() == '/') ? dir.substr(0, dir.length()-1) : dir).str();
|
auto dirname = ((not dir.empty() and dir.back() == '/') ? dir.substr(0, dir.length()-1) : dir).str();
|
||||||
|
@ -467,19 +472,19 @@ Vector<String> complete_command(StringView prefix, ByteCount cursor_pos)
|
||||||
return S_ISREG(st.st_mode) and executable;
|
return S_ISREG(st.st_mode) and executable;
|
||||||
};
|
};
|
||||||
|
|
||||||
cache.commands = list_files("", dirname, filter);
|
cache.commands = list_files(dirname, filter);
|
||||||
memcpy(&cache.mtim, &st.st_mtim, sizeof(TimeSpec));
|
memcpy(&cache.mtim, &st.st_mtim, sizeof(TimeSpec));
|
||||||
}
|
}
|
||||||
for (auto& cmd : cache.commands)
|
for (auto& cmd : cache.commands)
|
||||||
{
|
{
|
||||||
if (prefix_match(cmd, fileprefix))
|
if (RankedMatch match{cmd, fileprefix})
|
||||||
res.push_back(cmd);
|
matches.push_back(match);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
std::sort(res.begin(), res.end());
|
std::sort(matches.begin(), matches.end());
|
||||||
auto it = std::unique(res.begin(), res.end());
|
auto it = std::unique(matches.begin(), matches.end());
|
||||||
res.erase(it, res.end());
|
matches.erase(it, matches.end());
|
||||||
return res;
|
return candidates(matches, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
timespec get_fs_timestamp(StringView filename)
|
timespec get_fs_timestamp(StringView filename)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user