atmosphere explorer
at main 205 lines 8.8 kB view raw
1import { Handle } from "@atcute/lexicons"; 2import { Meta, MetaProvider, Title } from "@solidjs/meta"; 3import { A, RouteSectionProps, useLocation, useNavigate } from "@solidjs/router"; 4import { createEffect, ErrorBoundary, onCleanup, onMount, Show, Suspense } from "solid-js"; 5import { AccountManager } from "./auth/account.jsx"; 6import { agent } from "./auth/state.js"; 7import { RecordEditor } from "./components/create"; 8import { DropdownMenu, MenuProvider, MenuSeparator, NavMenu } from "./components/dropdown.jsx"; 9import { NavBar } from "./components/navbar.jsx"; 10import { NotificationContainer } from "./components/notification.jsx"; 11import { PermissionPromptContainer } from "./components/permission-prompt.jsx"; 12import { Search, SearchButton } from "./components/search.jsx"; 13import { themeEvent } from "./components/theme.jsx"; 14import { resolveHandle } from "./utils/api.js"; 15import { plcDirectory } from "./views/settings.jsx"; 16 17export const canHover = window.matchMedia("(hover: hover) and (pointer: fine)").matches; 18 19const headers: Record<string, string> = { 20 "did:plc:ia76kvnndjutgedggx2ibrem": "bunny.jpg", 21 "did:plc:oisofpd7lj26yvgiivf3lxsi": "puppy.jpg", 22 "did:plc:vwzwgnygau7ed7b7wt5ux7y2": "water.webp", 23 "did:plc:uu5axsmbm2or2dngy4gwchec": "city.webp", 24 "did:plc:ucaezectmpny7l42baeyooxi": "almaty.webp", 25 "did:plc:355lbopbpckczt672hss2ra4": "kit.jpg", 26 "did:plc:q6ywj35eew5f3cdajho7bmq7": "dreary.jpg", 27}; 28 29const Layout = (props: RouteSectionProps<unknown>) => { 30 const location = useLocation(); 31 const navigate = useNavigate(); 32 33 if (location.search.includes("hrt=true")) localStorage.setItem("hrt", "true"); 34 else if (location.search.includes("hrt=false")) localStorage.setItem("hrt", "false"); 35 if (location.search.includes("sailor=true")) localStorage.setItem("sailor", "true"); 36 else if (location.search.includes("sailor=false")) localStorage.setItem("sailor", "false"); 37 38 createEffect(async () => { 39 if (props.params.repo && !props.params.repo.startsWith("did:")) { 40 const did = await resolveHandle(props.params.repo as Handle); 41 navigate(location.pathname.replace(props.params.repo, did), { replace: true }); 42 } 43 }); 44 45 onMount(() => { 46 window.matchMedia("(prefers-color-scheme: dark)").addEventListener("change", themeEvent); 47 48 const handleGoToRepo = (ev: KeyboardEvent) => { 49 if (document.querySelector("[data-modal]")) return; 50 if (ev.target instanceof HTMLInputElement || ev.target instanceof HTMLTextAreaElement) return; 51 52 if (ev.key === "g" && agent()?.sub) { 53 ev.preventDefault(); 54 navigate(`/at://${agent()!.sub}`); 55 } 56 }; 57 58 window.addEventListener("keydown", handleGoToRepo); 59 onCleanup(() => window.removeEventListener("keydown", handleGoToRepo)); 60 61 if (localStorage.getItem("sailor") === "true") { 62 const style = document.createElement("style"); 63 style.textContent = ` 64 html, * { 65 cursor: url(/cursor.cur), pointer; 66 } 67 68 .star { 69 position: fixed; 70 pointer-events: none; 71 z-index: 9999; 72 font-size: 20px; 73 animation: sparkle 0.8s ease-out forwards; 74 } 75 76 @keyframes sparkle { 77 0% { 78 opacity: 1; 79 transform: translate(0, 0) rotate(var(--ttheta1)) scale(1); 80 } 81 100% { 82 opacity: 0; 83 transform: translate(var(--tx), var(--ty)) rotate(var(--ttheta2)) scale(0); 84 } 85 } 86 `; 87 document.head.appendChild(style); 88 89 let lastTime = 0; 90 const throttleDelay = 30; 91 92 document.addEventListener("mousemove", (e) => { 93 const now = Date.now(); 94 if (now - lastTime < throttleDelay) return; 95 lastTime = now; 96 97 const star = document.createElement("div"); 98 star.className = "star"; 99 star.textContent = "✨"; 100 star.style.left = e.clientX + "px"; 101 star.style.top = e.clientY + "px"; 102 103 const tx = (Math.random() - 0.5) * 50; 104 const ty = (Math.random() - 0.5) * 50; 105 const ttheta1 = Math.random() * 360; 106 const ttheta2 = ttheta1 + (Math.random() - 0.5) * 540; 107 star.style.setProperty("--tx", tx + "px"); 108 star.style.setProperty("--ty", ty + "px"); 109 star.style.setProperty("--ttheta1", ttheta1 + "deg"); 110 star.style.setProperty("--ttheta2", ttheta2 + "deg"); 111 112 document.body.appendChild(star); 113 114 setTimeout(() => star.remove(), 800); 115 }); 116 } 117 }); 118 119 return ( 120 <MetaProvider> 121 <Title>PDSls</Title> 122 <Show when={location.pathname !== "/"}> 123 <Meta name="robots" content="noindex, nofollow" /> 124 </Show> 125 <div id="main" class="mx-auto mb-8 flex max-w-lg flex-col items-center p-3"> 126 <header 127 class={`dark:shadow-dark-700 mb-3 flex h-13 w-full items-center justify-between rounded-xl border-[0.5px] border-neutral-300 bg-neutral-50 bg-size-[95%] bg-right bg-no-repeat p-2 pl-3 shadow-xs [--header-bg:#fafafa] [--trans-blue:#5BCEFA90] [--trans-pink:#F5A9B890] [--trans-white:#FFFFFF90] dark:border-neutral-700 dark:bg-neutral-800 dark:[--header-bg:#262626] dark:[--trans-blue:#5BCEFAa0] dark:[--trans-pink:#F5A9B8a0] dark:[--trans-white:#FFFFFFa0] ${localStorage.getItem("hrt") === "true" ? "bg-[linear-gradient(to_left,transparent_10%,var(--header-bg)_85%),linear-gradient(to_bottom,var(--trans-blue)_0%,var(--trans-blue)_20%,var(--trans-pink)_20%,var(--trans-pink)_40%,var(--trans-white)_40%,var(--trans-white)_60%,var(--trans-pink)_60%,var(--trans-pink)_80%,var(--trans-blue)_80%,var(--trans-blue)_100%)]" : ""}`} 128 style={{ 129 "background-image": 130 props.params.repo && props.params.repo in headers ? 131 `linear-gradient(to left, transparent 20%, var(--header-bg) 85%), url(/headers/${headers[props.params.repo]})` 132 : undefined, 133 }} 134 > 135 <A 136 href="/" 137 style='font-feature-settings: "cv05"' 138 class="relative flex items-center gap-1 text-xl font-semibold" 139 > 140 <span class="iconify tabler--binary-tree-filled text-[#76c4e5]"></span> 141 <span>PDSls</span> 142 <Show when={localStorage.getItem("hrt") === "true"}> 143 <img 144 src="/ribbon.webp" 145 alt="" 146 class="pointer-events-none absolute -top-3 -right-4 w-8 rotate-15" 147 /> 148 </Show> 149 </A> 150 <div class="relative flex items-center gap-0.5 rounded-lg bg-neutral-50/60 p-1 dark:bg-neutral-800/60"> 151 <div class="mr-1"> 152 <SearchButton /> 153 </div> 154 <Show when={agent()}> 155 <RecordEditor create={true} scope="create" /> 156 </Show> 157 <AccountManager /> 158 <MenuProvider> 159 <DropdownMenu icon="lucide--menu text-lg" buttonClass="rounded-md p-1.5"> 160 <NavMenu href="/jetstream" label="Jetstream" icon="lucide--radio-tower" /> 161 <NavMenu href="/firehose" label="Firehose" icon="lucide--rss" /> 162 <NavMenu href="/spacedust" label="Spacedust" icon="lucide--sparkles" /> 163 <MenuSeparator /> 164 <NavMenu href="/labels" label="Labels" icon="lucide--tag" /> 165 <NavMenu href="/car" label="Archive tools" icon="lucide--folder-archive" /> 166 <MenuSeparator /> 167 <NavMenu href="/settings" label="Settings" icon="lucide--settings" /> 168 </DropdownMenu> 169 </MenuProvider> 170 </div> 171 </header> 172 <div class="flex w-full flex-col items-center gap-3 text-pretty"> 173 <Search /> 174 <Show when={props.params.pds}> 175 <NavBar params={props.params} /> 176 </Show> 177 <Show keyed when={location.pathname}> 178 <ErrorBoundary 179 fallback={(err) => <div class="mt-3 wrap-anywhere">Error: {err.message}</div>} 180 > 181 <Suspense 182 fallback={ 183 <span class="iconify lucide--loader-circle mt-3 animate-spin text-xl"></span> 184 } 185 > 186 {props.children} 187 </Suspense> 188 </ErrorBoundary> 189 </Show> 190 </div> 191 <NotificationContainer /> 192 <PermissionPromptContainer /> 193 <Show when={plcDirectory() !== "https://plc.directory"}> 194 <div class="dark:bg-dark-500 fixed right-0 bottom-0 left-0 z-10 flex items-center justify-center bg-neutral-100 px-3 py-1 text-xs"> 195 <span> 196 PLC directory: <span class="font-medium">{plcDirectory()}</span> 197 </span> 198 </div> 199 </Show> 200 </div> 201 </MetaProvider> 202 ); 203}; 204 205export { Layout };