|  |  | @ -4,8 +4,8 @@ import useLocation from "wouter/use-location"; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | import { ReactComponent as Logo } from "../assets/logo.svg"; | 
			
		
	
		
			
				
					|  |  |  | import { ReactComponent as Right } from "../assets/arrow-right.svg"; | 
			
		
	
		
			
				
					|  |  |  | import { getTimeout } from "../util"; | 
			
		
	
		
			
				
					|  |  |  | import Menu from "./Menu"; | 
			
		
	
		
			
				
					|  |  |  | import { get, getTimeout } from "../util"; | 
			
		
	
		
			
				
					|  |  |  | import Menu, { MenuEntries } from "./Menu"; | 
			
		
	
		
			
				
					|  |  |  | import useMediaQuery from "../util/useMediaQuery"; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | const [timer, clear] = getTimeout(); | 
			
		
	
	
		
			
				
					|  |  | @ -17,18 +17,9 @@ const Container: React.FC<{ | 
			
		
	
		
			
				
					|  |  |  | 		| React.ReactElement | 
			
		
	
		
			
				
					|  |  |  | 	)[]; | 
			
		
	
		
			
				
					|  |  |  | 	hideNav?: boolean; | 
			
		
	
		
			
				
					|  |  |  | 	end?: boolean; | 
			
		
	
		
			
				
					|  |  |  | 	next?: string; | 
			
		
	
		
			
				
					|  |  |  | 	className?: string; | 
			
		
	
		
			
				
					|  |  |  | }> = ({ | 
			
		
	
		
			
				
					|  |  |  | 	children: _children, | 
			
		
	
		
			
				
					|  |  |  | 	hideNav = false, | 
			
		
	
		
			
				
					|  |  |  | 	end = false, | 
			
		
	
		
			
				
					|  |  |  | 	next, | 
			
		
	
		
			
				
					|  |  |  | 	className, | 
			
		
	
		
			
				
					|  |  |  | 	...props | 
			
		
	
		
			
				
					|  |  |  | }) => { | 
			
		
	
		
			
				
					|  |  |  | 	const [, navigate] = useLocation(); | 
			
		
	
		
			
				
					|  |  |  | }> = ({ children: _children, hideNav = false, className, ...props }) => { | 
			
		
	
		
			
				
					|  |  |  | 	const [location, navigate] = useLocation(); | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	const mobile = useMediaQuery("(max-width: 50rem)"); | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  | @ -116,10 +107,9 @@ const Container: React.FC<{ | 
			
		
	
		
			
				
					|  |  |  | 		return () => window.removeEventListener("resize", handleResize); | 
			
		
	
		
			
				
					|  |  |  | 	}, []); | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	// on first render
 | 
			
		
	
		
			
				
					|  |  |  | 	useLayoutEffect(handleResize, []); | 
			
		
	
		
			
				
					|  |  |  | 	type MouseKb = React.MouseEvent | React.KeyboardEvent | KeyboardEvent; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	const handleNext: React.MouseEventHandler<HTMLButtonElement> = e => { | 
			
		
	
		
			
				
					|  |  |  | 	const animateArrow = (e: MouseKb) => { | 
			
		
	
		
			
				
					|  |  |  | 		if (containerChild.current) { | 
			
		
	
		
			
				
					|  |  |  | 			( | 
			
		
	
		
			
				
					|  |  |  | 				[...containerChild.current.children] as (HTMLElement | SVGElement)[] | 
			
		
	
	
		
			
				
					|  |  | @ -130,10 +120,46 @@ const Container: React.FC<{ | 
			
		
	
		
			
				
					|  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  | 		document.body.style.maxHeight = "100vh"; | 
			
		
	
		
			
				
					|  |  |  | 		document.body.style.overflow = "hidden"; | 
			
		
	
		
			
				
					|  |  |  | 		e.currentTarget.style.width = "0"; | 
			
		
	
		
			
				
					|  |  |  | 		try { | 
			
		
	
		
			
				
					|  |  |  | 			const target = e.currentTarget! as HTMLButtonElement; | 
			
		
	
		
			
				
					|  |  |  | 			target.style.width = "0"; | 
			
		
	
		
			
				
					|  |  |  | 		} catch {} | 
			
		
	
		
			
				
					|  |  |  | 	}; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	const current = MenuEntries.findIndex(([, path]) => location === path); | 
			
		
	
		
			
				
					|  |  |  | 	const next = get.next(MenuEntries, current)[1]; | 
			
		
	
		
			
				
					|  |  |  | 	const prev = get.prev(MenuEntries, current)[1]; | 
			
		
	
		
			
				
					|  |  |  | 	const end = current === MenuEntries.length - 1; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	const handlePrev = (e: MouseKb) => { | 
			
		
	
		
			
				
					|  |  |  | 		animateArrow(e); | 
			
		
	
		
			
				
					|  |  |  | 		timer(() => prev && navigate(prev), 300); | 
			
		
	
		
			
				
					|  |  |  | 	}; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	const handleNext = (e: MouseKb) => { | 
			
		
	
		
			
				
					|  |  |  | 		animateArrow(e); | 
			
		
	
		
			
				
					|  |  |  | 		timer(() => next && navigate(next), 300); | 
			
		
	
		
			
				
					|  |  |  | 	}; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	function kbnav(e: KeyboardEvent) { | 
			
		
	
		
			
				
					|  |  |  | 		switch (e.key) { | 
			
		
	
		
			
				
					|  |  |  | 			case "ArrowLeft": | 
			
		
	
		
			
				
					|  |  |  | 				return handlePrev(e); | 
			
		
	
		
			
				
					|  |  |  | 			case "ArrowRight": | 
			
		
	
		
			
				
					|  |  |  | 				return handleNext(e); | 
			
		
	
		
			
				
					|  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	useEffect(() => { | 
			
		
	
		
			
				
					|  |  |  | 		window.addEventListener("keydown", kbnav); | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 		// cleanup
 | 
			
		
	
		
			
				
					|  |  |  | 		return () => window.removeEventListener("keydown", kbnav); | 
			
		
	
		
			
				
					|  |  |  | 	}, []); | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	// on first render
 | 
			
		
	
		
			
				
					|  |  |  | 	useLayoutEffect(handleResize, []); | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	return ( | 
			
		
	
		
			
				
					|  |  |  | 		<div | 
			
		
	
		
			
				
					|  |  |  | 			className={css` | 
			
		
	
	
		
			
				
					|  |  | @ -247,41 +273,39 @@ const Container: React.FC<{ | 
			
		
	
		
			
				
					|  |  |  | 					<Menu show={showMenu} setShowMenu={setShowMenu} /> | 
			
		
	
		
			
				
					|  |  |  | 				</span> | 
			
		
	
		
			
				
					|  |  |  | 			)} | 
			
		
	
		
			
				
					|  |  |  | 			{next && ( | 
			
		
	
		
			
				
					|  |  |  | 				<button | 
			
		
	
		
			
				
					|  |  |  | 					onClick={handleNext} | 
			
		
	
		
			
				
					|  |  |  | 					ref={nextBtn} | 
			
		
	
		
			
				
					|  |  |  | 					title={end ? "Back to start" : "Next page"} | 
			
		
	
		
			
				
					|  |  |  | 			<button | 
			
		
	
		
			
				
					|  |  |  | 				onClick={handleNext} | 
			
		
	
		
			
				
					|  |  |  | 				ref={nextBtn} | 
			
		
	
		
			
				
					|  |  |  | 				title={end ? "Back to start" : "Next page"} | 
			
		
	
		
			
				
					|  |  |  | 				className={css` | 
			
		
	
		
			
				
					|  |  |  | 					position: fixed; | 
			
		
	
		
			
				
					|  |  |  | 					right: 14vw; | 
			
		
	
		
			
				
					|  |  |  | 					bottom: 10vh; | 
			
		
	
		
			
				
					|  |  |  | 					z-index: 500; | 
			
		
	
		
			
				
					|  |  |  | 					background: none; | 
			
		
	
		
			
				
					|  |  |  | 					padding: 0; | 
			
		
	
		
			
				
					|  |  |  | 					font-weight: 500; | 
			
		
	
		
			
				
					|  |  |  | 					cursor: pointer; | 
			
		
	
		
			
				
					|  |  |  | 					letter-spacing: 0.2rem; | 
			
		
	
		
			
				
					|  |  |  | 					border: none; | 
			
		
	
		
			
				
					|  |  |  | 					overflow: hidden; | 
			
		
	
		
			
				
					|  |  |  | 					width: 0; | 
			
		
	
		
			
				
					|  |  |  | 					transition: all 300ms; | 
			
		
	
		
			
				
					|  |  |  | 					overflow: hidden; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 					${end ? "rotate: 180deg;" : ""} | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 					&:hover * { | 
			
		
	
		
			
				
					|  |  |  | 						fill: var(--primary-colour); | 
			
		
	
		
			
				
					|  |  |  | 					} | 
			
		
	
		
			
				
					|  |  |  | 				`}>
 | 
			
		
	
		
			
				
					|  |  |  | 				<Right | 
			
		
	
		
			
				
					|  |  |  | 					className={css` | 
			
		
	
		
			
				
					|  |  |  | 						position: fixed; | 
			
		
	
		
			
				
					|  |  |  | 						right: 14vw; | 
			
		
	
		
			
				
					|  |  |  | 						bottom: 10vh; | 
			
		
	
		
			
				
					|  |  |  | 						z-index: 500; | 
			
		
	
		
			
				
					|  |  |  | 						background: none; | 
			
		
	
		
			
				
					|  |  |  | 						padding: 0; | 
			
		
	
		
			
				
					|  |  |  | 						font-weight: 500; | 
			
		
	
		
			
				
					|  |  |  | 						cursor: pointer; | 
			
		
	
		
			
				
					|  |  |  | 						letter-spacing: 0.2rem; | 
			
		
	
		
			
				
					|  |  |  | 						border: none; | 
			
		
	
		
			
				
					|  |  |  | 						overflow: hidden; | 
			
		
	
		
			
				
					|  |  |  | 						width: 0; | 
			
		
	
		
			
				
					|  |  |  | 						transition: all 300ms; | 
			
		
	
		
			
				
					|  |  |  | 						overflow: hidden; | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 						${end ? "rotate: 180deg;" : ""} | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 						&:hover * { | 
			
		
	
		
			
				
					|  |  |  | 							fill: var(--primary-colour); | 
			
		
	
		
			
				
					|  |  |  | 						} | 
			
		
	
		
			
				
					|  |  |  | 					`}>
 | 
			
		
	
		
			
				
					|  |  |  | 					<Right | 
			
		
	
		
			
				
					|  |  |  | 						className={css` | 
			
		
	
		
			
				
					|  |  |  | 							height: "2rem"; | 
			
		
	
		
			
				
					|  |  |  | 							width: "2rem"; | 
			
		
	
		
			
				
					|  |  |  | 						`}
 | 
			
		
	
		
			
				
					|  |  |  | 					/> | 
			
		
	
		
			
				
					|  |  |  | 				</button> | 
			
		
	
		
			
				
					|  |  |  | 			)} | 
			
		
	
		
			
				
					|  |  |  | 						height: "2rem"; | 
			
		
	
		
			
				
					|  |  |  | 						width: "2rem"; | 
			
		
	
		
			
				
					|  |  |  | 					`}
 | 
			
		
	
		
			
				
					|  |  |  | 				/> | 
			
		
	
		
			
				
					|  |  |  | 			</button> | 
			
		
	
		
			
				
					|  |  |  | 			<div | 
			
		
	
		
			
				
					|  |  |  | 				className={cx( | 
			
		
	
		
			
				
					|  |  |  | 					css` | 
			
		
	
	
		
			
				
					|  |  | 
 |