import { FeatherProps } from "preact-feather/types"; import { CompletionContext, CompletionResult } from "@codemirror/autocomplete"; import { FunctionalComponent } from "preact"; import { useEffect, useRef, useState } from "preact/hooks"; import { FilterOption } from "$lib/web.ts"; import { MiniEditor } from "./mini_editor.tsx"; import { fuzzySearchAndSort } from "../fuse_search.ts"; import { deepEqual } from "../../plug-api/lib/json.ts"; import { AlwaysShownModal } from "./basic_modals.tsx"; export function FilterList({ placeholder, options, label, onSelect, onKeyPress, completer, vimMode, darkMode, preFilter, phrasePreprocessor, allowNew = false, helpText = "", completePrefix, icon: Icon, newHint, }: { placeholder: string; options: FilterOption[]; label: string; onKeyPress?: (key: string, currentText: string) => void; onSelect: (option: FilterOption | undefined) => void; preFilter?: (options: FilterOption[], phrase: string) => FilterOption[]; phrasePreprocessor?: (phrase: string) => string; vimMode: boolean; darkMode: boolean; completer: (context: CompletionContext) => Promise; allowNew?: boolean; completePrefix?: string; helpText: string; newHint?: string; icon?: FunctionalComponent; }) { const [text, setText] = useState(""); const [matchingOptions, setMatchingOptions] = useState( fuzzySearchAndSort( preFilter ? preFilter(options, "") : options, "", ), ); const [selectedOption, setSelectionOption] = useState(0); const selectedElementRef = useRef(null); function updateFilter(originalPhrase: string) { const prefilteredOptions = preFilter ? preFilter(options, originalPhrase) : options; if (phrasePreprocessor) { originalPhrase = phrasePreprocessor(originalPhrase); } const results = fuzzySearchAndSort(prefilteredOptions, originalPhrase); const foundExactMatch = !!results.find((result) => result.name === originalPhrase ); if (allowNew && !foundExactMatch && originalPhrase) { results.splice(1, 0, { name: originalPhrase, hint: newHint, }); } if (!deepEqual(matchingOptions, results)) { // Only do this (=> rerender of UI) if the results have changed setMatchingOptions(results); setSelectionOption(0); } } useEffect(() => { updateFilter(text); }, [options, text]); useEffect(() => { function closer() { onSelect(undefined); } document.addEventListener("click", closer); return () => { document.removeEventListener("click", closer); }; }, []); const returnEl = ( { onSelect(undefined); }} >
{ // Allow tapping/clicking the header without closing it e.stopPropagation(); }} > { onSelect( shiftDown ? { name: text } : matchingOptions[selectedOption], ); return true; }} onEscape={() => { onSelect(undefined); }} onChange={(text) => { setText(text); }} onKeyUp={(view, e) => { // This event is triggered after the key has been processed by CM already if (onKeyPress) { onKeyPress(e.key, view.state.sliceDoc()); } return false; }} onKeyDown={(view, e) => { switch (e.key) { case "ArrowUp": setSelectionOption(Math.max(0, selectedOption - 1)); return true; case "ArrowDown": setSelectionOption( Math.min(matchingOptions.length - 1, selectedOption + 1), ); return true; case "PageUp": setSelectionOption(Math.max(0, selectedOption - 5)); return true; case "PageDown": setSelectionOption( Math.min(matchingOptions.length - 1, selectedOption + 5), ); return true; case "Home": setSelectionOption(0); return true; case "End": setSelectionOption(matchingOptions.length - 1); return true; case " ": { const text = view.state.sliceDoc(); if (e.shiftKey) { // Operate on the highlighted option, ignoring prompt const option = matchingOptions[selectedOption].name; // Get the folder it's nested in, keeping the trailing / const folderPath = option.slice(0, option.lastIndexOf("/") + 1); // If the option wasn't in a folder, make it a folder setText(folderPath !== "" ? folderPath : option + "/"); return true; } else if (completePrefix && text === "") { setText(completePrefix); // updateFilter(completePrefix); return true; } break; } } return false; }} />
{matchingOptions && matchingOptions.length > 0 ? matchingOptions.map((option, idx) => (
{ if (selectedOption !== idx) { setSelectionOption(idx); } }} onClick={(e) => { e.stopPropagation(); onSelect(option); }} > {Icon && ( )} {option.name} {option.hint && {option.hint}}
{option.description}
)) : null}
); useEffect(() => { selectedElementRef.current?.scrollIntoView({ block: "nearest", }); }); return returnEl; }