2016-05-03 22:18:24 +00:00
|
|
|
// Helper functions for working with wcstring.
|
2016-05-18 22:30:21 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2016-05-27 21:41:16 +00:00
|
|
|
#include "wcstringutil.h"
|
2014-09-22 02:18:56 +00:00
|
|
|
|
2019-09-22 22:33:08 +00:00
|
|
|
#include <wctype.h>
|
|
|
|
|
2020-01-15 21:16:43 +00:00
|
|
|
#include <locale>
|
2019-10-13 22:50:48 +00:00
|
|
|
|
2020-01-15 21:16:43 +00:00
|
|
|
#include "common.h"
|
2020-07-30 00:16:51 +00:00
|
|
|
#include "flog.h"
|
2014-09-22 02:18:56 +00:00
|
|
|
|
2019-05-05 10:09:25 +00:00
|
|
|
wcstring_range wcstring_tok(wcstring &str, const wcstring &needle, wcstring_range last) {
|
2020-01-15 21:16:43 +00:00
|
|
|
using size_type = wcstring::size_type;
|
2014-09-22 02:18:56 +00:00
|
|
|
size_type pos = last.second == wcstring::npos ? wcstring::npos : last.first;
|
|
|
|
if (pos != wcstring::npos && last.second != wcstring::npos) pos += last.second;
|
|
|
|
if (pos != wcstring::npos && pos != 0) ++pos;
|
2016-05-03 22:18:24 +00:00
|
|
|
if (pos == wcstring::npos || pos >= str.size()) {
|
2014-09-22 02:18:56 +00:00
|
|
|
return std::make_pair(wcstring::npos, wcstring::npos);
|
|
|
|
}
|
|
|
|
|
2016-05-03 22:18:24 +00:00
|
|
|
if (needle.empty()) {
|
2014-09-22 02:18:56 +00:00
|
|
|
return std::make_pair(pos, wcstring::npos);
|
|
|
|
}
|
|
|
|
|
|
|
|
pos = str.find_first_not_of(needle, pos);
|
|
|
|
if (pos == wcstring::npos) return std::make_pair(wcstring::npos, wcstring::npos);
|
|
|
|
|
|
|
|
size_type next_pos = str.find_first_of(needle, pos);
|
2016-05-03 22:18:24 +00:00
|
|
|
if (next_pos == wcstring::npos) {
|
2014-09-22 02:18:56 +00:00
|
|
|
return std::make_pair(pos, wcstring::npos);
|
|
|
|
}
|
2016-05-04 22:19:47 +00:00
|
|
|
|
|
|
|
str[next_pos] = L'\0';
|
|
|
|
return std::make_pair(pos, next_pos - pos);
|
2014-09-22 02:18:56 +00:00
|
|
|
}
|
2018-03-09 20:52:12 +00:00
|
|
|
|
|
|
|
wcstring truncate(const wcstring &input, int max_len, ellipsis_type etype) {
|
2019-11-19 01:08:16 +00:00
|
|
|
if (input.size() <= static_cast<size_t>(max_len)) {
|
2018-03-09 20:52:12 +00:00
|
|
|
return input;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (etype == ellipsis_type::None) {
|
|
|
|
return input.substr(0, max_len);
|
|
|
|
}
|
|
|
|
if (etype == ellipsis_type::Prettiest) {
|
2019-04-28 22:00:36 +00:00
|
|
|
const wchar_t *ellipsis_str = get_ellipsis_str();
|
2019-03-12 21:06:01 +00:00
|
|
|
return input.substr(0, max_len - std::wcslen(ellipsis_str)).append(ellipsis_str);
|
2018-03-09 20:52:12 +00:00
|
|
|
}
|
|
|
|
wcstring output = input.substr(0, max_len - 1);
|
2019-04-28 22:00:36 +00:00
|
|
|
output.push_back(get_ellipsis_char());
|
2018-03-09 20:52:12 +00:00
|
|
|
return output;
|
|
|
|
}
|
2018-03-12 00:36:10 +00:00
|
|
|
|
2019-08-25 20:37:06 +00:00
|
|
|
wcstring trim(wcstring input) { return trim(std::move(input), L"\t\v \r\n"); }
|
2018-10-01 21:48:38 +00:00
|
|
|
|
2019-08-25 20:37:06 +00:00
|
|
|
wcstring trim(wcstring input, const wchar_t *any_of) {
|
|
|
|
wcstring result = std::move(input);
|
|
|
|
size_t suffix = result.find_last_not_of(any_of);
|
|
|
|
if (suffix == wcstring::npos) {
|
2018-03-12 00:36:10 +00:00
|
|
|
return wcstring{};
|
|
|
|
}
|
2019-08-25 20:37:06 +00:00
|
|
|
result.erase(suffix + 1);
|
2018-03-12 00:36:10 +00:00
|
|
|
|
2019-08-25 20:37:06 +00:00
|
|
|
auto prefix = result.find_first_not_of(any_of);
|
|
|
|
assert(prefix != wcstring::npos && "Should have one non-trimmed character");
|
|
|
|
result.erase(0, prefix);
|
2018-03-12 00:36:10 +00:00
|
|
|
return result;
|
|
|
|
}
|
2019-09-22 22:33:08 +00:00
|
|
|
|
|
|
|
wcstring wcstolower(wcstring input) {
|
|
|
|
wcstring result = std::move(input);
|
|
|
|
std::transform(result.begin(), result.end(), result.begin(), towlower);
|
|
|
|
return result;
|
|
|
|
}
|
2020-01-15 21:16:43 +00:00
|
|
|
|
|
|
|
bool string_prefixes_string(const wchar_t *proposed_prefix, const wcstring &value) {
|
|
|
|
return string_prefixes_string(proposed_prefix, value.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_prefixes_string(const wcstring &proposed_prefix, const wcstring &value) {
|
|
|
|
size_t prefix_size = proposed_prefix.size();
|
|
|
|
return prefix_size <= value.size() && value.compare(0, prefix_size, proposed_prefix) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_prefixes_string(const wchar_t *proposed_prefix, const wchar_t *value) {
|
|
|
|
for (size_t idx = 0; proposed_prefix[idx] != L'\0'; idx++) {
|
|
|
|
// Note if the prefix is longer than value, then we will compare a nonzero prefix character
|
|
|
|
// against a zero value character, and so we'll return false;
|
|
|
|
if (proposed_prefix[idx] != value[idx]) return false;
|
|
|
|
}
|
|
|
|
// We must have that proposed_prefix[idx] == L'\0', so we have a prefix match.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_prefixes_string(const char *proposed_prefix, const std::string &value) {
|
|
|
|
return string_prefixes_string(proposed_prefix, value.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_prefixes_string(const char *proposed_prefix, const char *value) {
|
|
|
|
for (size_t idx = 0; proposed_prefix[idx] != L'\0'; idx++) {
|
|
|
|
if (proposed_prefix[idx] != value[idx]) return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_prefixes_string_case_insensitive(const wcstring &proposed_prefix,
|
|
|
|
const wcstring &value) {
|
|
|
|
size_t prefix_size = proposed_prefix.size();
|
|
|
|
return prefix_size <= value.size() &&
|
|
|
|
wcsncasecmp(proposed_prefix.c_str(), value.c_str(), prefix_size) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_suffixes_string(const wcstring &proposed_suffix, const wcstring &value) {
|
|
|
|
size_t suffix_size = proposed_suffix.size();
|
|
|
|
return suffix_size <= value.size() &&
|
|
|
|
value.compare(value.size() - suffix_size, suffix_size, proposed_suffix) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_suffixes_string(const wchar_t *proposed_suffix, const wcstring &value) {
|
|
|
|
size_t suffix_size = std::wcslen(proposed_suffix);
|
|
|
|
return suffix_size <= value.size() &&
|
|
|
|
value.compare(value.size() - suffix_size, suffix_size, proposed_suffix) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_suffixes_string_case_insensitive(const wcstring &proposed_suffix,
|
|
|
|
const wcstring &value) {
|
|
|
|
size_t suffix_size = proposed_suffix.size();
|
|
|
|
return suffix_size <= value.size() && wcsncasecmp(value.c_str() + (value.size() - suffix_size),
|
|
|
|
proposed_suffix.c_str(), suffix_size) == 0;
|
|
|
|
}
|
|
|
|
|
2020-11-27 23:43:07 +00:00
|
|
|
/// Returns true if needle, represented as a subsequence, is contained within haystack.
|
|
|
|
/// Note subsequence is not substring: "foo" is a subsequence of "follow" for example.
|
|
|
|
static bool subsequence_in_string(const wcstring &needle, const wcstring &haystack) {
|
|
|
|
// Impossible if haystack is larger than string.
|
|
|
|
if (haystack.size() > haystack.size()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Empty strings are considered to be subsequences of everything.
|
|
|
|
if (needle.empty()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto ni = needle.begin();
|
|
|
|
for (auto hi = haystack.begin(); ni != needle.end() && hi != haystack.end(); ++hi) {
|
|
|
|
if (*ni == *hi) {
|
|
|
|
++ni;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// We succeeded if we exhausted our sequence.
|
|
|
|
assert(ni <= needle.end());
|
|
|
|
return ni == needle.end();
|
|
|
|
}
|
|
|
|
|
2020-11-28 00:35:19 +00:00
|
|
|
string_fuzzy_match_t::string_fuzzy_match_t(enum fuzzy_type_t t, size_t distance_first,
|
2020-11-27 23:43:07 +00:00
|
|
|
size_t distance_second)
|
|
|
|
: type(t), match_distance_first(distance_first), match_distance_second(distance_second) {}
|
|
|
|
|
|
|
|
string_fuzzy_match_t string_fuzzy_match_string(const wcstring &string,
|
|
|
|
const wcstring &match_against,
|
2020-11-28 00:35:19 +00:00
|
|
|
fuzzy_type_t limit_type) {
|
2020-11-27 23:43:07 +00:00
|
|
|
// Distances are generally the amount of text not matched.
|
2020-11-28 00:35:19 +00:00
|
|
|
string_fuzzy_match_t result(fuzzy_type_t::none, 0, 0);
|
2020-11-27 23:43:07 +00:00
|
|
|
size_t location;
|
2020-11-28 00:35:19 +00:00
|
|
|
if (limit_type >= fuzzy_type_t::exact && string == match_against) {
|
|
|
|
result.type = fuzzy_type_t::exact;
|
|
|
|
} else if (limit_type >= fuzzy_type_t::prefix &&
|
|
|
|
string_prefixes_string(string, match_against)) {
|
|
|
|
result.type = fuzzy_type_t::prefix;
|
2020-11-27 23:43:07 +00:00
|
|
|
assert(match_against.size() >= string.size());
|
|
|
|
result.match_distance_first = match_against.size() - string.size();
|
2020-11-28 00:35:19 +00:00
|
|
|
} else if (limit_type >= fuzzy_type_t::exact_icase &&
|
2020-11-27 23:43:07 +00:00
|
|
|
wcscasecmp(string.c_str(), match_against.c_str()) == 0) {
|
2020-11-28 00:35:19 +00:00
|
|
|
result.type = fuzzy_type_t::exact_icase;
|
|
|
|
} else if (limit_type >= fuzzy_type_t::prefix_icase &&
|
2020-11-27 23:43:07 +00:00
|
|
|
string_prefixes_string_case_insensitive(string, match_against)) {
|
2020-11-28 00:35:19 +00:00
|
|
|
result.type = fuzzy_type_t::prefix_icase;
|
2020-11-27 23:43:07 +00:00
|
|
|
assert(match_against.size() >= string.size());
|
|
|
|
result.match_distance_first = match_against.size() - string.size();
|
2020-11-28 00:35:19 +00:00
|
|
|
} else if (limit_type >= fuzzy_type_t::substr &&
|
2020-11-27 23:43:07 +00:00
|
|
|
(location = match_against.find(string)) != wcstring::npos) {
|
|
|
|
// String is contained within match against.
|
2020-11-28 00:35:19 +00:00
|
|
|
result.type = fuzzy_type_t::substr;
|
2020-11-27 23:43:07 +00:00
|
|
|
assert(match_against.size() >= string.size());
|
|
|
|
result.match_distance_first = match_against.size() - string.size();
|
|
|
|
result.match_distance_second = location; // prefer earlier matches
|
2020-11-28 00:35:19 +00:00
|
|
|
} else if (limit_type >= fuzzy_type_t::substr_icase &&
|
2020-11-27 23:43:07 +00:00
|
|
|
(location = ifind(match_against, string, true)) != wcstring::npos) {
|
|
|
|
// A case-insensitive version of the string is in the match against.
|
2020-11-28 00:35:19 +00:00
|
|
|
result.type = fuzzy_type_t::substr_icase;
|
2020-11-27 23:43:07 +00:00
|
|
|
assert(match_against.size() >= string.size());
|
|
|
|
result.match_distance_first = match_against.size() - string.size();
|
|
|
|
result.match_distance_second = location; // prefer earlier matches
|
2020-11-28 00:35:19 +00:00
|
|
|
} else if (limit_type >= fuzzy_type_t::subseq && subsequence_in_string(string, match_against)) {
|
|
|
|
result.type = fuzzy_type_t::subseq;
|
2020-11-27 23:43:07 +00:00
|
|
|
assert(match_against.size() >= string.size());
|
|
|
|
result.match_distance_first = match_against.size() - string.size();
|
|
|
|
// It would be nice to prefer matches with greater matching runs here.
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-01-15 21:16:43 +00:00
|
|
|
template <bool Fuzzy, typename T>
|
|
|
|
size_t ifind_impl(const T &haystack, const T &needle) {
|
|
|
|
using char_t = typename T::value_type;
|
|
|
|
std::locale locale;
|
|
|
|
|
|
|
|
auto ieq = [&locale](char_t c1, char_t c2) {
|
|
|
|
if (c1 == c2 || std::toupper(c1, locale) == std::toupper(c2, locale)) return true;
|
|
|
|
|
|
|
|
// In fuzzy matching treat treat `-` and `_` as equal (#3584).
|
|
|
|
if (Fuzzy) {
|
|
|
|
if ((c1 == '-' || c1 == '_') && (c2 == '-' || c2 == '_')) return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
auto result = std::search(haystack.begin(), haystack.end(), needle.begin(), needle.end(), ieq);
|
|
|
|
if (result != haystack.end()) {
|
|
|
|
return result - haystack.begin();
|
|
|
|
}
|
|
|
|
return T::npos;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t ifind(const wcstring &haystack, const wcstring &needle, bool fuzzy) {
|
|
|
|
return fuzzy ? ifind_impl<true>(haystack, needle) : ifind_impl<false>(haystack, needle);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t ifind(const std::string &haystack, const std::string &needle, bool fuzzy) {
|
|
|
|
return fuzzy ? ifind_impl<true>(haystack, needle) : ifind_impl<false>(haystack, needle);
|
|
|
|
}
|
|
|
|
|
|
|
|
wcstring_list_t split_string(const wcstring &val, wchar_t sep) {
|
|
|
|
wcstring_list_t out;
|
|
|
|
size_t pos = 0, end = val.size();
|
|
|
|
while (pos <= end) {
|
|
|
|
size_t next_pos = val.find(sep, pos);
|
|
|
|
if (next_pos == wcstring::npos) {
|
|
|
|
next_pos = end;
|
|
|
|
}
|
|
|
|
out.emplace_back(val, pos, next_pos - pos);
|
|
|
|
pos = next_pos + 1; // skip the separator, or skip past the end
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
wcstring join_strings(const wcstring_list_t &vals, wchar_t sep) {
|
|
|
|
if (vals.empty()) return wcstring{};
|
|
|
|
|
|
|
|
// Reserve the size we will need.
|
|
|
|
// count-1 separators, plus the length of all strings.
|
|
|
|
size_t size = vals.size() - 1;
|
|
|
|
for (const wcstring &s : vals) {
|
|
|
|
size += s.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the string.
|
|
|
|
wcstring result;
|
|
|
|
result.reserve(size);
|
|
|
|
bool first = true;
|
|
|
|
for (const wcstring &s : vals) {
|
|
|
|
if (!first) {
|
|
|
|
result.push_back(sep);
|
|
|
|
}
|
|
|
|
result.append(s);
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2020-07-30 00:16:51 +00:00
|
|
|
|
|
|
|
void wcs2string_bad_char(wchar_t wc) {
|
|
|
|
FLOGF(char_encoding, L"Wide character U+%4X has no narrow representation", wc);
|
|
|
|
}
|