silverbullet/web/components/filter.tsx

241 lines
7.2 KiB
TypeScript
Raw Normal View History

2024-07-30 23:33:33 +08:00
import type { FeatherProps } from "preact-feather/types";
import type {
CompletionContext,
CompletionResult,
} from "@codemirror/autocomplete";
import type { FunctionalComponent } from "preact";
import { useEffect, useRef, useState } from "preact/hooks";
import type { FilterOption } from "@silverbulletmd/silverbullet/type/client";
import { MiniEditor } from "./mini_editor.tsx";
import { fuzzySearchAndSort } from "../fuse_search.ts";
2024-02-29 22:23:05 +08:00
import { deepEqual } from "../../plug-api/lib/json.ts";
import { AlwaysShownModal } from "./basic_modals.tsx";
2022-05-16 21:09:36 +08:00
export function FilterList({
placeholder,
options,
label,
onSelect,
onKeyPress,
completer,
vimMode,
darkMode,
2023-12-22 02:49:25 +08:00
preFilter,
phrasePreprocessor,
allowNew = false,
helpText = "",
2022-04-01 21:02:35 +08:00
completePrefix,
icon: Icon,
newHint,
}: {
placeholder: string;
options: FilterOption[];
label: string;
onKeyPress?: (key: string, currentText: string) => void;
onSelect: (option: FilterOption | undefined) => void;
2023-12-22 02:49:25 +08:00
preFilter?: (options: FilterOption[], phrase: string) => FilterOption[];
phrasePreprocessor?: (phrase: string) => string;
vimMode: boolean;
darkMode: boolean;
completer: (context: CompletionContext) => Promise<CompletionResult | null>;
allowNew?: boolean;
2022-04-01 21:02:35 +08:00
completePrefix?: string;
helpText: string;
newHint?: string;
icon?: FunctionalComponent<FeatherProps>;
}) {
const [text, setText] = useState("");
const [matchingOptions, setMatchingOptions] = useState(
fuzzySearchAndSort(
preFilter ? preFilter(options, "") : options,
"",
),
);
const [selectedOption, setSelectionOption] = useState(0);
const selectedElementRef = useRef<HTMLDivElement>(null);
function updateFilter(originalPhrase: string) {
2023-12-22 02:49:25 +08:00
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
);
2022-08-01 23:06:17 +08:00
if (allowNew && !foundExactMatch && originalPhrase) {
results.splice(1, 0, {
2022-05-16 21:09:36 +08:00
name: originalPhrase,
hint: newHint,
});
}
2023-08-21 00:02:13 +08:00
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 = (
<AlwaysShownModal
onCancel={() => {
onSelect(undefined);
}}
>
<div
className="sb-header"
onClick={(e) => {
// Allow tapping/clicking the header without closing it
e.stopPropagation();
}}
>
<label>{label}</label>
<MiniEditor
text={text}
vimMode={vimMode}
vimStartInInsertMode={true}
focus={true}
darkMode={darkMode}
completer={completer}
placeholderText={placeholder}
onEnter={(_newText, shiftDown) => {
onSelect(
shiftDown ? { name: text } : matchingOptions[selectedOption],
);
return true;
2023-07-24 15:36:10 +08:00
}}
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 /
2024-07-30 21:17:34 +08:00
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;
2022-08-01 19:05:30 +08:00
}
}
return false;
}}
/>
</div>
<div
className="sb-help-text"
dangerouslySetInnerHTML={{ __html: helpText }}
>
</div>
<div className="sb-result-list" tabIndex={-1}>
{matchingOptions && matchingOptions.length > 0
? matchingOptions.map((option, idx) => (
<div
key={"" + idx}
ref={selectedOption === idx ? selectedElementRef : undefined}
className={(selectedOption === idx
? "sb-option sb-selected-option"
: "sb-option") +
2024-08-02 23:14:40 +08:00
(option.cssClass
? " sb-decorated-object " + option.cssClass
: "")}
2024-07-30 21:17:34 +08:00
onMouseMove={() => {
if (selectedOption !== idx) {
setSelectionOption(idx);
}
}}
onClick={(e) => {
e.stopPropagation();
onSelect(option);
}}
>
{Icon && (
<span className="sb-icon">
<Icon width={16} height={16} />
</span>
)}
<span className="sb-name">
{option.name}
</span>
{option.hint && <span className="sb-hint">{option.hint}</span>}
<div className="sb-description">{option.description}</div>
</div>
))
: null}
</div>
</AlwaysShownModal>
);
useEffect(() => {
selectedElementRef.current?.scrollIntoView({
block: "nearest",
});
});
return returnEl;
}