// @deno-types="https://deno.land/x/fuse@v6.4.1/dist/fuse.d.ts" import Fuse from "fuse"; import type { FilterOption } from "@silverbulletmd/silverbullet/type/client"; type FuseOption = FilterOption & { baseName: string; }; export const fuzzySearchAndSort = ( arr: FilterOption[], searchPhrase: string, ): FilterOption[] => { if (!searchPhrase) { return arr.sort((a, b) => (a.orderId || 0) - (b.orderId || 0)); } const enrichedArr: FuseOption[] = arr.map((item) => { return { ...item, baseName: item.name.split("/").pop()!, tags: item.tags?.join(" "), aliases: item.aliases?.join(" "), }; }); const fuse = new Fuse(enrichedArr, { keys: [{ name: "name", weight: 0.3, }, { name: "baseName", weight: 1, }, { name: "displayName", weight: 0.7, }, { name: "aliases", weight: 0.5, }, { name: "description", weight: 0.3, }], includeScore: true, shouldSort: true, isCaseSensitive: false, ignoreLocation: true, threshold: 0.6, sortFn: (a, b): number => { if (a.score === b.score) { const aOrder = enrichedArr[a.idx].orderId || 0; const bOrder = enrichedArr[b.idx].orderId || 0; if (aOrder !== bOrder) { return aOrder - bOrder; } } return a.score - b.score; }, }); const results = fuse.search(searchPhrase); // console.log("results", results); return results.map((r) => r.item); };