From 3558bbf126a36e6e96f0f3d037aa80dcbd9c53fa Mon Sep 17 00:00:00 2001 From: Jerzy Kozera Date: Sun, 26 Feb 2017 06:52:43 +0100 Subject: [PATCH] registry: Prioritize exact matches over fuzzy Fixes #677. --- src/libs/registry/docset.cpp | 44 ++++++++++++++++++++++++------------ 1 file changed, 29 insertions(+), 15 deletions(-) diff --git a/src/libs/registry/docset.cpp b/src/libs/registry/docset.cpp index 9f2886cce..4c0d88a60 100644 --- a/src/libs/registry/docset.cpp +++ b/src/libs/registry/docset.cpp @@ -38,6 +38,8 @@ #include #include +#include + #include using namespace Zeal::Registry; @@ -750,13 +752,8 @@ static int scoreFuzzy(const char *str, int index, int length) } } -static void sqliteScoreFunction(sqlite3_context *context, int argc, sqlite3_value **argv) +static inline int scoreFunction(const char *needleOrig, const char *haystackOrig) { - Q_UNUSED(argc); - - const char *needleOrig = reinterpret_cast(sqlite3_value_text(argv[0])); - const char *haystackOrig = reinterpret_cast(sqlite3_value_text(argv[1])); - const int needleLength = static_cast(qstrlen(needleOrig)); const int haystackLength = static_cast(qstrlen(haystackOrig)); @@ -790,18 +787,25 @@ static void sqliteScoreFunction(sqlite3_context *context, int argc, sqlite3_valu int score = 0; int matchIndex = -1; int matchLength; + int exactIndex = -1; + const char* exactMatch = std::strstr(haystack.data(), needle.data()); + + if (exactMatch != nullptr) { + exactIndex = exactMatch - haystack.data(); + } - matchFuzzy(needle.data(), needleLength, - haystack.data(), haystackLength, - &matchIndex, &matchLength); + if (exactIndex == -1) { + matchFuzzy(needle.data(), needleLength, + haystack.data(), haystackLength, + &matchIndex, &matchLength); + } - if (matchIndex == -1) { // no match + if (matchIndex == -1 && exactIndex == -1) { // no match // simply return 0 - sqlite3_result_int(context, 0); - return; - } else if (needleLength == matchLength) { + return 0; + } else if (exactIndex != -1) { // +100 to make sure exact matches are always on top. - score = scoreExact(matchIndex, matchLength, haystack.data(), haystackLength) + 100; + score = scoreExact(exactIndex, needleLength, haystack.data(), haystackLength) + 100; } else { score = scoreFuzzy(haystack.data(), matchIndex, matchLength); @@ -824,5 +828,15 @@ static void sqliteScoreFunction(sqlite3_context *context, int argc, sqlite3_valu } } - sqlite3_result_int(context, score); + return score; +} + +static void sqliteScoreFunction(sqlite3_context *context, int argc, sqlite3_value **argv) +{ + Q_UNUSED(argc); + + const char *needle = reinterpret_cast(sqlite3_value_text(argv[0])); + const char *haystack = reinterpret_cast(sqlite3_value_text(argv[1])); + + sqlite3_result_int(context, scoreFunction(needle, haystack)); }