Files
fnx_web/svelte/src/filesystem/filemanager/SearchBar.svelte

290 lines
6.7 KiB
Svelte
Raw Normal View History

<script lang="ts">
2024-11-14 16:14:58 +01:00
import { onMount } from "svelte";
import { fs_search, fs_encode_path, fs_thumbnail_url } from "lib/FilesystemAPI.svelte";
import type { FSNavigator } from "filesystem/FSNavigator";
2025-10-13 16:05:50 +02:00
import { loading_finish, loading_start } from "lib/Loading";
2023-05-25 17:06:17 +02:00
2025-10-13 16:05:50 +02:00
let { nav }: {
nav: FSNavigator;
} = $props();
2023-05-25 17:06:17 +02:00
2025-10-13 16:05:50 +02:00
let search_bar: HTMLInputElement = $state()
let error = $state("")
let search_term = $state("")
let search_results: string[] = $state([])
let selected_result = $state(0)
2023-05-25 17:06:17 +02:00
let searching = false
let last_searched_term = ""
2025-10-13 16:05:50 +02:00
let last_limit = $state(10)
2023-05-25 17:06:17 +02:00
2024-11-14 16:14:58 +01:00
onMount(() => {
// Clear results when the user moves to a new directory
return nav.subscribe(nav => {
if (nav.initialized) {
clear_search(false)
}
})
})
2023-05-25 17:06:17 +02:00
const search = async (limit = 10) => {
if (search_term.length < 2 || search_term.length > 100) {
// These are the length limits defined by the API
search_results = []
return
} else if (search_term === last_searched_term && limit === last_limit) {
// If the term is the same we don't have to search
return
} else if (searching) {
// If a search is already running we also don't search
return
}
console.debug("Searching for", search_term)
2023-05-25 17:32:59 +02:00
error = ""
2023-05-25 17:06:17 +02:00
last_searched_term = search_term
last_limit = limit
2023-05-25 17:06:17 +02:00
searching = true
try {
2025-10-09 15:48:23 +02:00
loading_start()
search_results = await fs_search(nav.base.path, search_term, limit)
2023-05-25 17:06:17 +02:00
} catch (err) {
2023-05-30 23:35:18 +02:00
if (err.value) {
error = err.value
} else {
2023-05-25 17:32:59 +02:00
alert(err)
console.error(err)
}
2025-10-09 15:48:23 +02:00
} finally {
loading_finish()
2023-05-25 17:06:17 +02:00
}
2024-11-14 16:14:58 +01:00
if (search_results.length > 0 && selected_result > search_results.length-1) {
selected_result = search_results.length-1
}
2023-05-25 17:06:17 +02:00
searching = false
// It's possible that the user entered another letter while we were
// performing the search reqeust. If this happens we run the search function
// again
if (last_searched_term !== search_term) {
console.debug("Search term changed during search. Searching again")
await search()
}
}
const clear_search = (blur: boolean) => {
2024-11-14 16:14:58 +01:00
error = ""
search_term = ""
search_results = []
selected_result = 0
searching = false
last_searched_term = ""
last_limit = 10
// If blur is true we unfocus the search field. This should only happen when
// the user presses escape
if (blur) {
search_bar.blur()
}
}
// Cursor navigation events can only be prevented with keydown. But we want to
// use keyup for searching, so we use two listeners here
const input_keydown = (e: KeyboardEvent) => {
2024-11-14 16:14:58 +01:00
if (e.key === "Escape" || e.key === "ArrowUp" || e.key === "ArrowDown") {
e.preventDefault()
}
}
const input_keyup = (e: KeyboardEvent) => {
2023-05-29 12:33:22 +02:00
if (e.key === "Escape") {
2024-11-14 16:14:58 +01:00
clear_search(true)
} else if (e.key === "ArrowUp") {
if (selected_result > 0) {
selected_result--
}
} else if (e.key === "ArrowDown") {
if (selected_result+1 < search_results.length) {
selected_result++
}
2023-05-29 12:33:22 +02:00
} else {
search()
}
}
2024-11-14 16:14:58 +01:00
// Submitting opens the selected result
const submit_search = (e: Event) => {
e.preventDefault()
2023-05-25 17:06:17 +02:00
if (search_results.length !== 0) {
2024-11-14 16:14:58 +01:00
open_result(selected_result)
2023-05-25 17:06:17 +02:00
}
}
const open_result = (index: number, e?: MouseEvent) => {
if (e !== undefined) {
e.preventDefault()
}
nav.navigate(search_results[index], true)
2024-11-14 16:14:58 +01:00
clear_search(false)
}
const window_keydown = (e: KeyboardEvent) => {
if (e.ctrlKey || e.altKey || e.metaKey) {
return // prevent custom shortcuts from interfering with system shortcuts
} else if ((document.activeElement as any).type !== undefined && (document.activeElement as any).type === "text") {
return // Prevent shortcuts from interfering with input fields
}
2024-11-14 16:14:58 +01:00
if (e.key === "Escape" && search_term !== "") {
clear_search(true)
e.preventDefault()
} else if (e.key === "/" || e.key === "f") {
e.preventDefault()
e.stopPropagation()
search_bar.focus()
return
}
2023-05-25 17:06:17 +02:00
}
</script>
2025-10-13 16:05:50 +02:00
<svelte:window onkeydown={window_keydown} />
2024-11-14 16:14:58 +01:00
2023-05-25 17:32:59 +02:00
{#if error === "path_not_found" || error === "node_is_a_directory"}
<div class="highlight_yellow center">
Search index not found. The search index is a file called
'.search_index.gz' in your home directory. If you delete this file then
search will not work. The file is regenerated 10 minutes after modifying
a file in your filesystem.
</div>
{:else if error !== ""}
<div class="highlight_red center">
An error ocurred while executing the search request: {error}
</div>
{/if}
2024-11-14 16:14:58 +01:00
<div class="center">
<form class="search_form" onsubmit={submit_search}>
2023-05-29 12:33:22 +02:00
<i class="icon">search</i>
<input
2024-11-14 16:14:58 +01:00
bind:this={search_bar}
class="term"
type="text"
2024-11-14 22:13:19 +01:00
placeholder="Press / to search in {$nav.base.name}"
style="width: 100%;"
bind:value={search_term}
2025-10-13 16:05:50 +02:00
onkeydown={input_keydown}
onkeyup={input_keyup}
/>
2024-11-14 21:46:42 +01:00
{#if search_term !== ""}
<!-- Button needs to be of button type in order to not submit the form -->
2025-10-13 16:05:50 +02:00
<button onclick={() => clear_search(false)} type="button">
2024-11-14 21:46:42 +01:00
<i class="icon">close</i>
</button>
{/if}
2023-05-29 12:33:22 +02:00
</form>
2023-05-25 17:06:17 +02:00
2024-11-14 16:14:58 +01:00
<div class="results">
{#if search_term !== "" && search_results.length === 0}
No results found
{/if}
2024-11-14 16:14:58 +01:00
{#each search_results as result, index}
<a
href={"/d"+fs_encode_path(result)}
onclick={(e) => open_result(index, e)}
2024-11-14 16:14:58 +01:00
class="node"
class:node_selected={selected_result === index}
>
<img src={fs_thumbnail_url(result, 32, 32)} class="node_icon" alt="icon"/>
<span class="node_name">
<!-- Remove the search directory from the result -->
{result.slice($nav.base.path.length+1)}
</span>
</a>
{/each}
{#if search_results.length === last_limit}
<div class="node">
<div class="node_name" style="text-align: center;">
2025-10-13 16:05:50 +02:00
<button onclick={() => {search(last_limit + 100)}}>
2024-11-14 16:14:58 +01:00
<i class="icon">expand_more</i>
More results
</button>
</div>
</div>
2024-11-14 16:14:58 +01:00
{/if}
</div>
2023-05-25 17:06:17 +02:00
</div>
<style>
2023-05-25 17:32:59 +02:00
.center {
margin: auto;
width: 1000px;
max-width: 100%;
2024-11-18 17:09:27 +01:00
padding-top: 2px;
padding-bottom: 2px;
2023-05-25 17:32:59 +02:00
}
2023-05-29 12:33:22 +02:00
.search_form {
2023-05-25 17:06:17 +02:00
display: flex;
flex-direction: row;
align-items: center;
}
2024-11-18 17:09:27 +01:00
.search_form > * {
flex: 0 0 auto;
}
.search_form > .term {
2023-05-25 17:06:17 +02:00
flex: 1 1 auto;
}
.results {
display: flex;
flex-direction: column;
2023-05-25 17:06:17 +02:00
position: relative;
2024-11-14 16:14:58 +01:00
overflow-x: hidden;
overflow-y: auto;
max-height: 80vh;
2023-05-25 17:06:17 +02:00
text-align: left;
background: var(--body_color);
border-radius: 8px;
}
.results > * {
display: flex;
flex-direction: row;
2023-05-25 17:06:17 +02:00
}
.node {
display: flex;
flex-direction: row;
align-items: center;
2024-11-14 16:14:58 +01:00
gap: 4px;
2023-05-25 17:06:17 +02:00
text-decoration: none;
color: var(--text-color);
2024-11-14 16:14:58 +01:00
padding: 2px;
2023-05-25 17:06:17 +02:00
}
2024-11-14 16:14:58 +01:00
.node_selected {
background: var(--highlight_background);
color: var(--highlight_text_color);
2023-05-25 17:06:17 +02:00
}
.node:hover:not(.node_selected) {
background: var(--input_background);
color: var(--input_text);
text-decoration: none;
}
.node_icon {
flex: 0 0 auto;
2023-05-25 17:06:17 +02:00
height: 32px;
width: 32px;
vertical-align: middle;
border-radius: 4px;
}
.node_name {
flex: 1 1 auto;
2023-05-25 17:06:17 +02:00
width: 100%;
line-height: 1.2em;
word-break: break-all;
}
</style>