2015-10-22 20:49:08 +02:00
|
|
|
#include "ranked_match.hh"
|
|
|
|
|
2015-10-30 14:57:46 +01:00
|
|
|
#include "utf8_iterator.hh"
|
2015-10-29 14:36:30 +01:00
|
|
|
#include "unit_tests.hh"
|
|
|
|
|
2015-10-22 20:49:08 +02:00
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2015-10-30 14:57:46 +01:00
|
|
|
using Utf8It = utf8::iterator<const char*>;
|
|
|
|
|
2015-10-29 14:36:30 +01:00
|
|
|
static int count_word_boundaries_match(StringView candidate, StringView query)
|
2015-10-22 20:49:08 +02:00
|
|
|
{
|
2015-10-29 14:36:30 +01:00
|
|
|
int count = 0;
|
2015-10-30 14:57:46 +01:00
|
|
|
Utf8It qit{query.begin(), query};
|
|
|
|
Codepoint prev = 0;
|
|
|
|
for (Utf8It it{candidate.begin(), candidate}; it != candidate.end(); ++it)
|
2015-10-22 20:49:08 +02:00
|
|
|
{
|
2015-10-30 14:57:46 +01:00
|
|
|
const Codepoint c = *it;
|
2015-10-29 14:36:30 +01:00
|
|
|
const bool is_word_boundary = prev == 0 or
|
2015-10-30 14:57:46 +01:00
|
|
|
(!iswalnum(prev) and iswalnum(c)) or
|
2015-10-29 14:36:30 +01:00
|
|
|
(islower(prev) and isupper(c));
|
|
|
|
prev = c;
|
2015-10-22 20:49:08 +02:00
|
|
|
|
2015-10-29 14:36:30 +01:00
|
|
|
if (not is_word_boundary)
|
|
|
|
continue;
|
2015-10-22 20:49:08 +02:00
|
|
|
|
2015-10-30 14:57:46 +01:00
|
|
|
const Codepoint lc = tolower(c);
|
|
|
|
for (; qit != query.end(); ++qit)
|
2015-10-22 20:49:08 +02:00
|
|
|
{
|
2015-10-30 14:57:46 +01:00
|
|
|
const Codepoint qc = *qit;
|
2015-10-29 14:36:30 +01:00
|
|
|
if (qc == (islower(qc) ? lc : c))
|
|
|
|
{
|
|
|
|
++count;
|
2015-10-30 14:57:46 +01:00
|
|
|
++qit;
|
2015-10-29 14:36:30 +01:00
|
|
|
break;
|
|
|
|
}
|
2015-10-22 20:49:08 +02:00
|
|
|
}
|
2015-10-30 14:57:46 +01:00
|
|
|
if (qit == query.end())
|
2015-10-29 14:36:30 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
2015-10-22 20:49:08 +02:00
|
|
|
|
2015-10-30 14:57:46 +01:00
|
|
|
static bool smartcase_eq(Codepoint query, Codepoint candidate)
|
2015-10-29 14:36:30 +01:00
|
|
|
{
|
|
|
|
return query == (islower(query) ? tolower(candidate) : candidate);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool subsequence_match_smart_case(StringView str, StringView subseq)
|
|
|
|
{
|
2015-10-30 14:57:46 +01:00
|
|
|
Utf8It it{str.begin(), str};
|
|
|
|
for (Utf8It subseq_it{subseq.begin(), subseq}; subseq_it != subseq.end(); ++subseq_it)
|
2015-10-29 14:36:30 +01:00
|
|
|
{
|
|
|
|
if (it == str.end())
|
|
|
|
return false;
|
2015-10-30 14:57:46 +01:00
|
|
|
while (not smartcase_eq(*subseq_it, *it))
|
2015-10-29 14:36:30 +01:00
|
|
|
{
|
|
|
|
if (++it == str.end())
|
|
|
|
return false;
|
|
|
|
}
|
2015-10-22 20:49:08 +02:00
|
|
|
++it;
|
|
|
|
}
|
2015-10-29 14:36:30 +01:00
|
|
|
return true;
|
2015-10-22 20:49:08 +02:00
|
|
|
}
|
|
|
|
|
2015-10-27 22:25:18 +01:00
|
|
|
RankedMatch::RankedMatch(StringView candidate, StringView query)
|
|
|
|
{
|
2015-10-29 14:36:30 +01:00
|
|
|
if (candidate.empty() or query.length() > candidate.length())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (query.empty())
|
2015-10-27 22:25:18 +01:00
|
|
|
{
|
|
|
|
m_candidate = candidate;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-10-29 14:36:30 +01:00
|
|
|
if (not subsequence_match_smart_case(candidate, query))
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_candidate = candidate;
|
|
|
|
|
|
|
|
m_first_char_match = smartcase_eq(query[0], candidate[0]);
|
|
|
|
m_word_boundary_match_count = count_word_boundaries_match(candidate, query);
|
|
|
|
m_only_word_boundary = m_word_boundary_match_count == query.length();
|
|
|
|
m_prefix = std::equal(query.begin(), query.end(), candidate.begin(), smartcase_eq);
|
2015-10-27 22:25:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool RankedMatch::operator<(const RankedMatch& other) const
|
|
|
|
{
|
2015-10-29 14:36:30 +01:00
|
|
|
if (m_only_word_boundary or other.m_only_word_boundary)
|
|
|
|
return m_only_word_boundary and other.m_only_word_boundary ?
|
|
|
|
m_word_boundary_match_count > other.m_word_boundary_match_count
|
|
|
|
: m_only_word_boundary;
|
|
|
|
|
|
|
|
if (m_prefix != other.m_prefix)
|
|
|
|
return m_prefix;
|
|
|
|
|
|
|
|
if (m_word_boundary_match_count != other.m_word_boundary_match_count)
|
|
|
|
return m_word_boundary_match_count > other.m_word_boundary_match_count;
|
|
|
|
|
|
|
|
if (m_first_char_match != other.m_first_char_match)
|
|
|
|
return m_first_char_match;
|
|
|
|
|
|
|
|
return std::lexicographical_compare(
|
2015-10-30 14:57:46 +01:00
|
|
|
Utf8It{m_candidate.begin(), m_candidate}, Utf8It{m_candidate.end(), m_candidate},
|
|
|
|
Utf8It{other.m_candidate.begin(), other.m_candidate}, Utf8It{other.m_candidate.end(), other.m_candidate},
|
|
|
|
[](Codepoint a, Codepoint b) {
|
2015-10-29 14:36:30 +01:00
|
|
|
const bool low_a = islower(a), low_b = islower(b);
|
|
|
|
return low_a == low_b ? a < b : low_a;
|
|
|
|
});
|
2015-10-27 22:25:18 +01:00
|
|
|
}
|
|
|
|
|
2015-10-29 14:36:30 +01:00
|
|
|
UnitTest test_ranked_match{[] {
|
|
|
|
kak_assert(count_word_boundaries_match("run_all_tests", "rat") == 3);
|
|
|
|
}};
|
|
|
|
|
2015-10-22 20:49:08 +02:00
|
|
|
}
|