Skip to content

Commit 1792937

Browse files
committed
add Sublime-like fuzzy search to file browser
1 parent 5aa8aa1 commit 1792937

File tree

4 files changed

+96
-10
lines changed

4 files changed

+96
-10
lines changed

src/base/base_strings.c

+79
Original file line numberDiff line numberDiff line change
@@ -1716,6 +1716,85 @@ fuzzy_match_range_list_copy(Arena *arena, FuzzyMatchRangeList *src)
17161716
return dst;
17171717
}
17181718

1719+
internal ScoredFuzzyMatchRangeList
1720+
scored_fuzzy_match_find(Arena *arena, String8 needle, String8 haystack)
1721+
{
1722+
Temp scratch = scratch_begin(0, 0);
1723+
// We're going to implement a very simple scoring mechanism similar to that described in
1724+
// https://www.forrestthewoods.com/blog/reverse_engineering_sublime_texts_fuzzy_match/.
1725+
#define scored_fuzzy_match_unmatched -1
1726+
#define scored_fuzzy_match_consecutive 5
1727+
#define scored_fuzzy_match_unmatched_leading -3
1728+
ScoredFuzzyMatchRangeList invalid = {0};
1729+
ScoredFuzzyMatchRangeList result = {0};
1730+
// Simplify to a single needle which has common delimiters removed.
1731+
String8List needles = str8_split(scratch.arena, needle, (U8*)" ", 1, 0);
1732+
needle = str8_list_join(scratch.arena, &needles, 0);
1733+
if (needle.size == 0)
1734+
{
1735+
scratch_end(scratch);
1736+
return invalid;
1737+
}
1738+
String8 tmp_str = str8(needle.str, 1);
1739+
U64 find_pos = 0;
1740+
find_pos = str8_find_needle(haystack, find_pos, tmp_str, StringMatchFlag_CaseInsensitive);
1741+
if (find_pos >= haystack.size)
1742+
{
1743+
scratch_end(scratch);
1744+
return invalid;
1745+
}
1746+
// Leading character penalty.
1747+
// Only go to a max of 3 based on the article.
1748+
result.score += Min(find_pos, 3) * scored_fuzzy_match_unmatched_leading;
1749+
// We also want to deduct for additional unmatched characters between start and find_pos.
1750+
if (find_pos > 3)
1751+
{
1752+
result.score += (find_pos - 3) * scored_fuzzy_match_unmatched;
1753+
}
1754+
Rng1U64 range = r1u64(find_pos, find_pos + 1);
1755+
FuzzyMatchRangeNode *n = push_array(arena, FuzzyMatchRangeNode, 1);
1756+
n->range = range;
1757+
SLLQueuePush(result.list.first, result.list.last, n);
1758+
result.list.count += 1;
1759+
// Match the rest.
1760+
U64 prev_found = find_pos;
1761+
U64 search_start = 0;
1762+
find_pos += 1;
1763+
for (U64 idx = 1; idx < needle.size; ++idx)
1764+
{
1765+
tmp_str = str8(needle.str + idx, 1);
1766+
search_start = find_pos;
1767+
find_pos = str8_find_needle(haystack, find_pos, tmp_str, StringMatchFlag_CaseInsensitive);
1768+
if (find_pos >= haystack.size)
1769+
{
1770+
scratch_end(scratch);
1771+
return invalid;
1772+
}
1773+
// Compute consecutive bonus.
1774+
if (prev_found + 1 == find_pos)
1775+
{
1776+
result.score += scored_fuzzy_match_consecutive;
1777+
// We can reuse the existing node and simply extend it.
1778+
result.list.last->range.max = find_pos + 1;
1779+
}
1780+
else
1781+
{
1782+
result.score += (find_pos - search_start) * scored_fuzzy_match_unmatched;
1783+
Rng1U64 range = r1u64(find_pos, find_pos + 1);
1784+
FuzzyMatchRangeNode *n = push_array(arena, FuzzyMatchRangeNode, 1);
1785+
n->range = range;
1786+
SLLQueuePush(result.list.first, result.list.last, n);
1787+
result.list.count += 1;
1788+
}
1789+
prev_found = find_pos;
1790+
find_pos += 1;
1791+
}
1792+
// Compute final unmatched characters.
1793+
result.score += (haystack.size - find_pos) * scored_fuzzy_match_unmatched;
1794+
scratch_end(scratch);
1795+
return result;
1796+
}
1797+
17191798
////////////////////////////////
17201799
//~ NOTE(allen): Serialization Helpers
17211800

src/base/base_strings.h

+8
Original file line numberDiff line numberDiff line change
@@ -148,6 +148,13 @@ struct FuzzyMatchRangeList
148148
U64 total_dim;
149149
};
150150

151+
typedef struct ScoredFuzzyMatchRangeList ScoredFuzzyMatchRangeList;
152+
struct ScoredFuzzyMatchRangeList
153+
{
154+
FuzzyMatchRangeList list;
155+
S32 score;
156+
};
157+
151158
////////////////////////////////
152159
//~ rjf: Character Classification & Conversion Functions
153160

@@ -342,6 +349,7 @@ internal Vec4F32 rgba_from_hex_string_4f32(String8 hex_string);
342349

343350
internal FuzzyMatchRangeList fuzzy_match_find(Arena *arena, String8 needle, String8 haystack);
344351
internal FuzzyMatchRangeList fuzzy_match_range_list_copy(Arena *arena, FuzzyMatchRangeList *src);
352+
internal ScoredFuzzyMatchRangeList scored_fuzzy_match_find(Arena *arena, String8 needles, String8 haystack);
345353

346354
////////////////////////////////
347355
//~ NOTE(allen): Serialization Helpers

src/df/gfx/df_views.c

+8-9
Original file line numberDiff line numberDiff line change
@@ -26,16 +26,15 @@ df_qsort_compare_file_info__default(DF_FileInfo *a, DF_FileInfo *b)
2626
internal int
2727
df_qsort_compare_file_info__default_filtered(DF_FileInfo *a, DF_FileInfo *b)
2828
{
29-
int result = 0;
30-
if(a->filename.size < b->filename.size)
29+
if (a->match_ranges.score > b->match_ranges.score)
3130
{
32-
result = -1;
31+
return -1;
3332
}
34-
else if(a->filename.size > b->filename.size)
33+
if (a->match_ranges.score < b->match_ranges.score)
3534
{
36-
result = +1;
35+
return 1;
3736
}
38-
return result;
37+
return 0;
3938
}
4039

4140
internal int
@@ -2039,8 +2038,8 @@ DF_VIEW_UI_FUNCTION_DEF(FileSystem)
20392038
OS_FileIter *it = os_file_iter_begin(scratch.arena, path_query.path, 0);
20402039
for(OS_FileInfo info = {0}; os_file_iter_next(scratch.arena, it, &info);)
20412040
{
2042-
FuzzyMatchRangeList match_ranges = fuzzy_match_find(fs->cached_files_arena, path_query.search, info.name);
2043-
B32 fits_search = (path_query.search.size == 0 || match_ranges.count == match_ranges.needle_part_count);
2041+
ScoredFuzzyMatchRangeList match_ranges = scored_fuzzy_match_find(fs->cached_files_arena, path_query.search, info.name);
2042+
B32 fits_search = (path_query.search.size == 0 || match_ranges.list.count != 0);
20442043
B32 fits_dir_only = !!(info.props.flags & FilePropertyFlag_IsFolder) || !dir_selection;
20452044
if(fits_search && fits_dir_only)
20462045
{
@@ -2339,7 +2338,7 @@ DF_VIEW_UI_FUNCTION_DEF(FileSystem)
23392338
UI_PrefWidth(ui_pct(1, 0))
23402339
{
23412340
UI_Box *box = ui_build_box_from_stringf(UI_BoxFlag_DrawText, "%S##%p", file->filename, view);
2342-
ui_box_equip_fuzzy_match_ranges(box, &file->match_ranges);
2341+
ui_box_equip_fuzzy_match_ranges(box, &file->match_ranges.list);
23432342
}
23442343
}
23452344

src/df/gfx/df_views.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ struct DF_FileInfo
2222
{
2323
String8 filename;
2424
FileProperties props;
25-
FuzzyMatchRangeList match_ranges;
25+
ScoredFuzzyMatchRangeList match_ranges;
2626
};
2727

2828
typedef struct DF_FileInfoNode DF_FileInfoNode;

0 commit comments

Comments
 (0)