1import { Handle } from "@atcute/lexicons";
2import { Meta, MetaProvider } from "@solidjs/meta";
3import { A, RouteSectionProps, useLocation, useNavigate } from "@solidjs/router";
4import { createEffect, createSignal, ErrorBoundary, onMount, Show, Suspense } from "solid-js";
5import { AccountManager } from "./components/account.jsx";
6import { RecordEditor } from "./components/create.jsx";
7import { DropdownMenu, MenuProvider, NavMenu } from "./components/dropdown.jsx";
8import { agent } from "./components/login.jsx";
9import { NavBar } from "./components/navbar.jsx";
10import { Search, SearchButton, showSearch } from "./components/search.jsx";
11import { themeEvent, ThemeSelection } from "./components/theme.jsx";
12import { resolveHandle } from "./utils/api.js";
13
14export const isTouchDevice = "ontouchstart" in window || navigator.maxTouchPoints > 1;
15
16export const [notif, setNotif] = createSignal<{
17 show: boolean;
18 icon?: string;
19 text?: string;
20}>({ show: false });
21
22const headers: Record<string, string> = {
23 "did:plc:ia76kvnndjutgedggx2ibrem": "bunny.jpg",
24 "did:plc:oisofpd7lj26yvgiivf3lxsi": "puppy.jpg",
25 "did:plc:vwzwgnygau7ed7b7wt5ux7y2": "water.webp",
26 "did:plc:uu5axsmbm2or2dngy4gwchec": "city.webp",
27 "did:plc:aokggmp5jzj4nc5jifhiplqc": "bridge.jpg",
28 "did:plc:bnqkww7bjxaacajzvu5gswdf": "forest.jpg",
29 "did:plc:p2cp5gopk7mgjegy6wadk3ep": "aurora.jpg",
30};
31
32const Layout = (props: RouteSectionProps<unknown>) => {
33 const location = useLocation();
34 const navigate = useNavigate();
35 let timeout: number;
36
37 if (location.search.includes("hrt=true")) localStorage.setItem("hrt", "true");
38 else if (location.search.includes("hrt=false")) localStorage.setItem("hrt", "false");
39 if (location.search.includes("sailor=true")) localStorage.setItem("sailor", "true");
40 else if (location.search.includes("sailor=false")) localStorage.setItem("sailor", "false");
41
42 createEffect(async () => {
43 if (props.params.repo && !props.params.repo.startsWith("did:")) {
44 const did = await resolveHandle(props.params.repo as Handle);
45 navigate(location.pathname.replace(props.params.repo, did));
46 }
47 });
48
49 createEffect(() => {
50 if (notif().show) {
51 clearTimeout(timeout);
52 timeout = setTimeout(() => setNotif({ show: false }), 3000);
53 }
54 });
55
56 onMount(() => {
57 window.matchMedia("(prefers-color-scheme: dark)").addEventListener("change", themeEvent);
58
59 if (localStorage.getItem("sailor") === "true") {
60 const style = document.createElement("style");
61 style.textContent = `
62 html, * {
63 cursor: url(/cursor.cur), pointer;
64 }
65
66 .star {
67 position: fixed;
68 pointer-events: none;
69 z-index: 9999;
70 font-size: 20px;
71 animation: sparkle 0.8s ease-out forwards;
72 }
73
74 @keyframes sparkle {
75 0% {
76 opacity: 1;
77 transform: translate(0, 0) rotate(0deg) scale(1);
78 }
79 100% {
80 opacity: 0;
81 transform: translate(var(--tx), var(--ty)) rotate(180deg) scale(0);
82 }
83 }
84 `;
85 document.head.appendChild(style);
86
87 let lastTime = 0;
88 const throttleDelay = 30;
89
90 document.addEventListener("mousemove", (e) => {
91 const now = Date.now();
92 if (now - lastTime < throttleDelay) return;
93 lastTime = now;
94
95 const star = document.createElement("div");
96 star.className = "star";
97 star.textContent = "✨";
98 star.style.left = e.pageX + "px";
99 star.style.top = e.pageY + "px";
100
101 const tx = (Math.random() - 0.5) * 50;
102 const ty = (Math.random() - 0.5) * 50;
103 star.style.setProperty("--tx", tx + "px");
104 star.style.setProperty("--ty", ty + "px");
105
106 document.body.appendChild(star);
107
108 setTimeout(() => star.remove(), 800);
109 });
110 }
111 });
112
113 return (
114 <div
115 id="main"
116 class="mx-auto mb-8 flex max-w-lg flex-col items-center p-4 text-neutral-900 dark:text-neutral-200"
117 >
118 <MetaProvider>
119 <Show when={location.pathname !== "/"}>
120 <Meta name="robots" content="noindex, nofollow" />
121 </Show>
122 </MetaProvider>
123 <header
124 class={`dark:shadow-dark-700 dark:bg-dark-300 mb-3 flex w-full items-center justify-between rounded-lg border-[0.5px] border-neutral-300 bg-neutral-50 bg-size-[95%] bg-right bg-no-repeat p-2 shadow-xs [--header-bg:#fafafa] dark:border-neutral-700 dark:[--header-bg:#2d2d2d] ${localStorage.getItem("hrt") === "true" ? "bg-[linear-gradient(to_left,transparent_10%,var(--header-bg)_85%),linear-gradient(to_bottom,#5BCEFA90_0%,#5BCEFA90_20%,#F5A9B890_20%,#F5A9B890_40%,#FFFFFF90_40%,#FFFFFF90_60%,#F5A9B890_60%,#F5A9B890_80%,#5BCEFA90_80%,#5BCEFA90_100%)]" : ""}`}
125 style={{
126 "background-image":
127 props.params.repo in headers ?
128 `linear-gradient(to left, transparent 10%, var(--header-bg) 85%), url(/headers/${headers[props.params.repo]})`
129 : undefined,
130 }}
131 >
132 <A
133 href="/"
134 style='font-feature-settings: "cv05"'
135 class="flex items-center gap-1 text-xl font-semibold"
136 >
137 <span>🎃</span>
138 <span class="bg-linear-to-r from-orange-400 via-orange-600 to-orange-800 bg-clip-text text-transparent">
139 PDSls
140 </span>
141 </A>
142 <div class="dark:bg-dark-300/60 relative flex items-center gap-1 rounded-lg bg-neutral-50/60">
143 <Show when={location.pathname !== "/"}>
144 <SearchButton />
145 </Show>
146 <Show when={agent()}>
147 <RecordEditor create={true} />
148 </Show>
149 <AccountManager />
150 <MenuProvider>
151 <DropdownMenu
152 icon="lucide--menu text-xl"
153 buttonClass="rounded-lg p-1"
154 menuClass="top-10 p-3"
155 >
156 <NavMenu href="/jetstream" label="Jetstream" />
157 <NavMenu href="/firehose" label="Firehose" />
158 <NavMenu href="/settings" label="Settings" />
159 <ThemeSelection />
160 </DropdownMenu>
161 </MenuProvider>
162 </div>
163 </header>
164 <div class="flex w-full flex-col items-center gap-3 text-pretty">
165 <Show when={showSearch() || location.pathname === "/"}>
166 <Search />
167 </Show>
168 <Show when={props.params.pds}>
169 <NavBar params={props.params} />
170 </Show>
171 <Show keyed when={location.pathname}>
172 <ErrorBoundary
173 fallback={(err) => <div class="mt-3 wrap-break-word">Error: {err.message}</div>}
174 >
175 <Suspense
176 fallback={
177 <span class="iconify lucide--loader-circle mt-3 animate-spin text-xl"></span>
178 }
179 >
180 {props.children}
181 </Suspense>
182 </ErrorBoundary>
183 </Show>
184 </div>
185 <Show when={notif().show}>
186 <button
187 class="dark:shadow-dark-700 dark:bg-dark-100 fixed bottom-10 z-50 flex items-center rounded-lg border-[0.5px] border-neutral-300 bg-neutral-50 p-2 shadow-md dark:border-neutral-700"
188 onClick={() => setNotif({ show: false })}
189 >
190 <span class={`iconify ${notif().icon} mr-1`}></span>
191 {notif().text}
192 </button>
193 </Show>
194 </div>
195 );
196};
197
198export { Layout };