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(var(--ttheta1)) scale(1);
78 }
79 100% {
80 opacity: 0;
81 transform: translate(var(--tx), var(--ty)) rotate(var(--ttheta2)) 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.clientX + "px";
99 star.style.top = e.clientY + "px";
100
101 const tx = (Math.random() - 0.5) * 50;
102 const ty = (Math.random() - 0.5) * 50;
103 const ttheta1 = Math.random() * 360;
104 const ttheta2 = ttheta1 + (Math.random() - 0.5) * 540;
105 star.style.setProperty("--tx", tx + "px");
106 star.style.setProperty("--ty", ty + "px");
107 star.style.setProperty("--ttheta1", ttheta1 + "deg");
108 star.style.setProperty("--ttheta2", ttheta2 + "deg");
109
110 document.body.appendChild(star);
111
112 setTimeout(() => star.remove(), 800);
113 });
114 }
115 });
116
117 return (
118 <div
119 id="main"
120 class="mx-auto mb-8 flex max-w-lg flex-col items-center p-4 text-neutral-900 dark:text-neutral-200"
121 >
122 <MetaProvider>
123 <Show when={location.pathname !== "/"}>
124 <Meta name="robots" content="noindex, nofollow" />
125 </Show>
126 </MetaProvider>
127 <header
128 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%)]" : ""}`}
129 style={{
130 "background-image":
131 props.params.repo in headers ?
132 `linear-gradient(to left, transparent 10%, var(--header-bg) 85%), url(/headers/${headers[props.params.repo]})`
133 : undefined,
134 }}
135 >
136 <A
137 href="/"
138 style='font-feature-settings: "cv05"'
139 class="flex items-center gap-1 text-xl font-semibold"
140 >
141 <span>🎃</span>
142 <span class="bg-linear-to-r from-orange-400 via-orange-600 to-orange-800 bg-clip-text text-transparent">
143 PDSls
144 </span>
145 </A>
146 <div class="dark:bg-dark-300/60 relative flex items-center gap-1 rounded-lg bg-neutral-50/60">
147 <Show when={location.pathname !== "/"}>
148 <SearchButton />
149 </Show>
150 <Show when={agent()}>
151 <RecordEditor create={true} />
152 </Show>
153 <AccountManager />
154 <MenuProvider>
155 <DropdownMenu
156 icon="lucide--menu text-xl"
157 buttonClass="rounded-lg p-1"
158 menuClass="top-10 p-3"
159 >
160 <NavMenu href="/jetstream" label="Jetstream" />
161 <NavMenu href="/firehose" label="Firehose" />
162 <NavMenu href="/settings" label="Settings" />
163 <ThemeSelection />
164 </DropdownMenu>
165 </MenuProvider>
166 </div>
167 </header>
168 <div class="flex w-full flex-col items-center gap-3 text-pretty">
169 <Show when={showSearch() || location.pathname === "/"}>
170 <Search />
171 </Show>
172 <Show when={props.params.pds}>
173 <NavBar params={props.params} />
174 </Show>
175 <Show keyed when={location.pathname}>
176 <ErrorBoundary
177 fallback={(err) => <div class="mt-3 wrap-break-word">Error: {err.message}</div>}
178 >
179 <Suspense
180 fallback={
181 <span class="iconify lucide--loader-circle mt-3 animate-spin text-xl"></span>
182 }
183 >
184 {props.children}
185 </Suspense>
186 </ErrorBoundary>
187 </Show>
188 </div>
189 <Show when={notif().show}>
190 <button
191 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"
192 onClick={() => setNotif({ show: false })}
193 >
194 <span class={`iconify ${notif().icon} mr-1`}></span>
195 {notif().text}
196 </button>
197 </Show>
198 </div>
199 );
200};
201
202export { Layout };