* refactor: landing screen * fix: code debt * feat: add settings screen * chore: clean up * feat: settings * feat: small updates
146 lines
4.6 KiB
TypeScript
146 lines
4.6 KiB
TypeScript
import { SearchIcon } from "@lume/icons";
|
|
import { type NostrEvent, Kind } from "@lume/types";
|
|
import { Spinner } from "@lume/ui";
|
|
import { Note } from "@/components/note";
|
|
import { User } from "@/components/user";
|
|
import { createFileRoute } from "@tanstack/react-router";
|
|
import { useEffect, useState } from "react";
|
|
import { toast } from "sonner";
|
|
import { useDebounce } from "use-debounce";
|
|
import { LumeEvent, LumeWindow } from "@lume/system";
|
|
|
|
export const Route = createFileRoute("/search")({
|
|
component: Screen,
|
|
});
|
|
|
|
function Screen() {
|
|
const [loading, setLoading] = useState(false);
|
|
const [events, setEvents] = useState<LumeEvent[]>([]);
|
|
const [search, setSearch] = useState("");
|
|
const [searchValue] = useDebounce(search, 500);
|
|
|
|
const searchEvents = async () => {
|
|
try {
|
|
setLoading(true);
|
|
|
|
const query = `https://api.nostr.wine/search?query=${searchValue}&kind=0,1`;
|
|
const res = await fetch(query);
|
|
const content = await res.json();
|
|
const events = content.data as NostrEvent[];
|
|
const lumeEvents = events.map((ev) => new LumeEvent(ev));
|
|
const sorted = lumeEvents.sort((a, b) => b.created_at - a.created_at);
|
|
|
|
setLoading(false);
|
|
setEvents(sorted);
|
|
} catch (e) {
|
|
setLoading(false);
|
|
toast.error(String(e));
|
|
}
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (searchValue.length >= 3 && searchValue.length < 500) {
|
|
searchEvents();
|
|
}
|
|
}, [searchValue]);
|
|
|
|
return (
|
|
<div data-tauri-drag-region className="flex flex-col w-full h-full">
|
|
<div className="relative flex flex-col h-24 border-b shrink-0 border-black/5 dark:border-white/5">
|
|
<div data-tauri-drag-region className="w-full h-4 shrink-0" />
|
|
<input
|
|
value={search}
|
|
onChange={(e) => setSearch(e.target.value)}
|
|
onKeyDown={(e) => {
|
|
if (e.key === "Enter") searchEvents();
|
|
}}
|
|
placeholder="Search anything..."
|
|
className="w-full h-20 px-3 pt-10 text-lg bg-transparent border-none focus:outline-none focus:ring-0 placeholder:text-neutral-500 dark:placeholder:text-neutral-600"
|
|
/>
|
|
</div>
|
|
<div className="flex-1 p-3 overflow-y-auto scrollbar-none">
|
|
{loading ? (
|
|
<div className="flex items-center justify-center w-full h-full">
|
|
<Spinner />
|
|
</div>
|
|
) : events.length ? (
|
|
<div className="flex flex-col gap-5">
|
|
<div className="flex flex-col gap-1.5">
|
|
<div className="text-sm font-medium text-neutral-700 dark:text-neutral-300 shrink-0">
|
|
Users
|
|
</div>
|
|
<div className="flex flex-col flex-1 gap-1">
|
|
{events
|
|
.filter((ev) => ev.kind === Kind.Metadata)
|
|
.map((event) => (
|
|
<SearchUser key={event.pubkey} event={event} />
|
|
))}
|
|
</div>
|
|
</div>
|
|
<div className="flex flex-col gap-1.5">
|
|
<div className="text-sm font-medium text-neutral-700 dark:text-neutral-300 shrink-0">
|
|
Notes
|
|
</div>
|
|
<div className="flex flex-col flex-1 gap-3">
|
|
{events
|
|
.filter((ev) => ev.kind === Kind.Text)
|
|
.map((event) => (
|
|
<SearchNote key={event.id} event={event} />
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
) : null}
|
|
{!loading && !events.length ? (
|
|
<div className="flex flex-col items-center justify-center h-full gap-3">
|
|
<div className="inline-flex items-center justify-center rounded-full size-16 bg-black/10 dark:bg-white/10">
|
|
<SearchIcon className="size-6" />
|
|
</div>
|
|
Try searching for people, notes, or keywords
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function SearchUser({ event }: { event: LumeEvent }) {
|
|
return (
|
|
<button
|
|
key={event.id}
|
|
type="button"
|
|
onClick={() => LumeWindow.openProfile(event.pubkey)}
|
|
className="col-span-1 p-2 rounded-lg hover:bg-black/10 dark:hover:bg-white/10"
|
|
>
|
|
<User.Provider pubkey={event.pubkey} embedProfile={event.content}>
|
|
<User.Root className="flex items-center gap-2">
|
|
<User.Avatar className="rounded-full size-9 shrink-0" />
|
|
<div className="inline-flex items-center gap-1.5">
|
|
<User.Name className="font-semibold" />
|
|
<User.NIP05 />
|
|
</div>
|
|
</User.Root>
|
|
</User.Provider>
|
|
</button>
|
|
);
|
|
}
|
|
|
|
function SearchNote({ event }: { event: LumeEvent }) {
|
|
return (
|
|
<div className="bg-white dark:bg-black/20 backdrop-blur-lg rounded-xl shadow-primary dark:ring-1 ring-neutral-800/50">
|
|
<Note.Provider event={event}>
|
|
<Note.Root>
|
|
<div className="flex items-center justify-between px-3 h-14">
|
|
<Note.User />
|
|
<Note.Menu />
|
|
</div>
|
|
<Note.Content className="px-3" quote={false} mention={false} />
|
|
<div className="flex items-center gap-4 px-3 mt-3 h-14">
|
|
<Note.Open />
|
|
</div>
|
|
</Note.Root>
|
|
</Note.Provider>
|
|
</div>
|
|
);
|
|
}
|