diff options
| author | 2021-08-26 15:04:19 -0700 | |
|---|---|---|
| committer | 2021-08-26 15:04:44 -0700 | |
| commit | fdd701dd88529ee0d4911283e35be86b000ed1fc (patch) | |
| tree | 61d450d04633eece1c12b350b9ef3512dc2fa7b7 /examples/docs/src/components | |
| parent | 84c18d3030c98b2770144a8a27d1cca7bbfedc1d (diff) | |
| parent | 2e8db7ad2384b756894eac6be72bcf720f7f28fa (diff) | |
| download | astro-fdd701dd88529ee0d4911283e35be86b000ed1fc.tar.gz astro-fdd701dd88529ee0d4911283e35be86b000ed1fc.tar.zst astro-fdd701dd88529ee0d4911283e35be86b000ed1fc.zip | |
Merge branch 'okikio/main' (#1111)
Diffstat (limited to 'examples/docs/src/components')
24 files changed, 1025 insertions, 265 deletions
| diff --git a/examples/docs/src/components/AvatarList.astro b/examples/docs/src/components/AvatarList.astro deleted file mode 100644 index aafcb371b..000000000 --- a/examples/docs/src/components/AvatarList.astro +++ /dev/null @@ -1,74 +0,0 @@ -<!-- Thanks to @5t3ph for https://smolcss.dev/#smol-avatar-list! --> - -<ul class="avatar-list"> -  <li><a href="https://smolcss.dev/#smol-avatar-list"><img alt="Avatar 1" width="64" height="64" src='https://avataaars.io/?avatarStyle=Transparent&topType=LongHairBun&accessoriesType=Blank&hairColor=Auburn&facialHairType=BeardMedium&facialHairColor=Auburn&clotheType=ShirtCrewNeck&clotheColor=Blue01&eyeType=Side&eyebrowType=RaisedExcitedNatural&mouthType=Serious&skinColor=Tanned' /></a></li> -  <li><a href="https://smolcss.dev/#smol-avatar-list"><img alt="Avatar 2" width="64" height="64" src='https://avataaars.io/?avatarStyle=Transparent&topType=LongHairDreads&accessoriesType=Blank&hairColor=Brown&facialHairType=Blank&clotheType=ShirtScoopNeck&clotheColor=PastelGreen&eyeType=Default&eyebrowType=DefaultNatural&mouthType=Smile&skinColor=Tanned' /></a></li> -  <li><a href="https://smolcss.dev/#smol-avatar-list"><img alt="Avatar 3" width="64" height="64" src='https://avataaars.io/?avatarStyle=Transparent&topType=LongHairCurly&hairColor=BrownDark&facialHairType=Blank&clotheType=GraphicShirt&clotheColor=Pink&graphicType=Diamond&eyeType=Side&eyebrowType=Default&mouthType=Default&skinColor=Brown'/></a></li> -</ul> - -<style> -.avatar-list { -  --avatar-size: 2.5rem; -  --avatar-count: 3; - -  display: grid; -  list-style: none; -  /* Default to displaying most of the avatar to -  enable easier access on touch devices, ensuring -  the WCAG touch target size is met or exceeded */ -  grid-template-columns: repeat( -    var(--avatar-count), -    max(44px, calc(var(--avatar-size) / 1.15)) -  ); -  /* `padding` matches added visual dimensions of -  the `box-shadow` to help create a more accurate -  computed component size */ -  padding: 0.08em; -  font-size: var(--avatar-size); -} - -@media (any-hover: hover) and (any-pointer: fine) { -  .avatar-list { -    /* We create 1 extra cell to enable the computed  -    width to match the final visual width */ -    grid-template-columns: repeat( -      calc(var(--avatar-count) + 1), -      calc(var(--avatar-size) / 1.75) -    ); -  } -} - -.avatar-list li { -  width: var(--avatar-size); -  height: var(--avatar-size); -} - -.avatar-list li:hover ~ li a, -.avatar-list li:focus-within ~ li a { -  transform: translateX(33%); -} - -.avatar-list img, -.avatar-list a { -  display: block; -  border-radius: 50%; -} - -.avatar-list a { -  transition: transform 180ms ease-in-out; -} - -.avatar-list img { -  width: 100%; -  height: 100%; -  object-fit: cover; -  background-color: #fff; -  box-shadow: 0 0 0 0.05em #fff, 0 0 0 0.08em rgba(0, 0, 0, 0.15); -} - -.avatar-list a:focus { -  outline: 2px solid transparent; -  /* Double-layer trick to work for dark and light backgrounds */ -  box-shadow: 0 0 0 0.08em var(--theme-accent), 0 0 0 0.12em white; -} -</style> diff --git a/examples/docs/src/components/DocSidebar.tsx b/examples/docs/src/components/DocSidebar.tsx deleted file mode 100644 index 076d460cc..000000000 --- a/examples/docs/src/components/DocSidebar.tsx +++ /dev/null @@ -1,61 +0,0 @@ -import type { FunctionalComponent } from 'preact'; -import { h } from 'preact'; -import { useState, useEffect, useRef } from 'preact/hooks'; -import EditOnGithub from './EditOnGithub'; - -const DocSidebar: FunctionalComponent<{ headers: any[]; editHref: string }> = ({ headers = [], editHref }) => { -  const itemOffsets = useRef([]); -  const [activeId, setActiveId] = useState<string>(undefined); - -  useEffect(() => { -    const getItemOffsets = () => { -      const titles = document.querySelectorAll('article :is(h2, h3, h4)'); -      itemOffsets.current = Array.from(titles).map((title) => ({ -        id: title.id, -        topOffset: title.getBoundingClientRect().top + window.scrollY, -      })); -    }; - -    const onScroll = () => { -      const itemIndex = itemOffsets.current.findIndex((item) => item.topOffset > window.scrollY + window.innerHeight / 3); -      if (itemIndex === 0) { -        setActiveId(undefined); -      } else if (itemIndex === -1) { -        setActiveId(itemOffsets.current[itemOffsets.current.length - 1].id); -      } else { -        setActiveId(itemOffsets.current[itemIndex - 1].id); -      } -    }; - -    getItemOffsets(); -    window.addEventListener('resize', getItemOffsets); -    window.addEventListener('scroll', onScroll); - -    return () => { -      window.removeEventListener('resize', getItemOffsets); -      window.removeEventListener('scroll', onScroll); -    }; -  }, []); - -  return ( -    <nav> -      <div> -        <h4>Contents</h4> -        <ul> -          {headers -            .filter(({ depth }) => depth > 1 && depth < 5) -            .map((header) => ( -              <li class={`header-link depth-${header.depth} ${activeId === header.slug ? 'active' : ''}`.trim()}> -                <a href={`#${header.slug}`}>{header.text}</a> -              </li> -            ))} -        </ul> -      </div> -      <div> -        <EditOnGithub href={editHref} /> -      </div> -    </nav> -  ); -}; - -export default DocSidebar; diff --git a/examples/docs/src/components/EditOnGithub.tsx b/examples/docs/src/components/EditOnGithub.tsx deleted file mode 100644 index f7478934f..000000000 --- a/examples/docs/src/components/EditOnGithub.tsx +++ /dev/null @@ -1,26 +0,0 @@ -import type { FunctionalComponent } from 'preact'; -import { h } from 'preact'; - -const EditOnGithub: FunctionalComponent<{ href: string }> = ({ href }) => { -  return ( -    <a class="edit-on-github" href={href}> -      <svg -        preserveAspectRatio="xMidYMid meet" -        height="1em" -        width="1em" -        fill="currentColor" -        xmlns="http://www.w3.org/2000/svg" -        viewBox="0 0 438.549 438.549" -        stroke="none" -        class="icon-7f6730be--text-3f89f380" -      > -        <g> -          <path d="M409.132 114.573c-19.608-33.596-46.205-60.194-79.798-79.8-33.598-19.607-70.277-29.408-110.063-29.408-39.781 0-76.472 9.804-110.063 29.408-33.596 19.605-60.192 46.204-79.8 79.8C9.803 148.168 0 184.854 0 224.63c0 47.78 13.94 90.745 41.827 128.906 27.884 38.164 63.906 64.572 108.063 79.227 5.14.954 8.945.283 11.419-1.996 2.475-2.282 3.711-5.14 3.711-8.562 0-.571-.049-5.708-.144-15.417a2549.81 2549.81 0 0 1-.144-25.406l-6.567 1.136c-4.187.767-9.469 1.092-15.846 1-6.374-.089-12.991-.757-19.842-1.999-6.854-1.231-13.229-4.086-19.13-8.559-5.898-4.473-10.085-10.328-12.56-17.556l-2.855-6.57c-1.903-4.374-4.899-9.233-8.992-14.559-4.093-5.331-8.232-8.945-12.419-10.848l-1.999-1.431c-1.332-.951-2.568-2.098-3.711-3.429-1.142-1.331-1.997-2.663-2.568-3.997-.572-1.335-.098-2.43 1.427-3.289 1.525-.859 4.281-1.276 8.28-1.276l5.708.853c3.807.763 8.516 3.042 14.133 6.851 5.614 3.806 10.229 8.754 13.846 14.842 4.38 7.806 9.657 13.754 15.846 17.847 6.184 4.093 12.419 6.136 18.699 6.136 6.28 0 11.704-.476 16.274-1.423 4.565-.952 8.848-2.383 12.847-4.285 1.713-12.758 6.377-22.559 13.988-29.41-10.848-1.14-20.601-2.857-29.264-5.14-8.658-2.286-17.605-5.996-26.835-11.14-9.235-5.137-16.896-11.516-22.985-19.126-6.09-7.614-11.088-17.61-14.987-29.979-3.901-12.374-5.852-26.648-5.852-42.826 0-23.035 7.52-42.637 22.557-58.817-7.044-17.318-6.379-36.732 1.997-58.24 5.52-1.715 13.706-.428 24.554 3.853 10.85 4.283 18.794 7.952 23.84 10.994 5.046 3.041 9.089 5.618 12.135 7.708 17.705-4.947 35.976-7.421 54.818-7.421s37.117 2.474 54.823 7.421l10.849-6.849c7.419-4.57 16.18-8.758 26.262-12.565 10.088-3.805 17.802-4.853 23.134-3.138 8.562 21.509 9.325 40.922 2.279 58.24 15.036 16.18 22.559 35.787 22.559 58.817 0 16.178-1.958 30.497-5.853 42.966-3.9 12.471-8.941 22.457-15.125 29.979-6.191 7.521-13.901 13.85-23.131 18.986-9.232 5.14-18.182 8.85-26.84 11.136-8.662 2.286-18.415 4.004-29.263 5.146 9.894 8.562 14.842 22.077 14.842 40.539v60.237c0 3.422 1.19 6.279 3.572 8.562 2.379 2.279 6.136 2.95 11.276 1.995 44.163-14.653 80.185-41.062 108.068-79.226 27.88-38.161 41.825-81.126 41.825-128.906-.01-39.771-9.818-76.454-29.414-110.049z"></path> -        </g> -      </svg> -      <span>Edit on GitHub</span> -    </a> -  ); -}; - -export default EditOnGithub; diff --git a/examples/docs/src/components/Footer/AvatarList.astro b/examples/docs/src/components/Footer/AvatarList.astro new file mode 100644 index 000000000..589e296b9 --- /dev/null +++ b/examples/docs/src/components/Footer/AvatarList.astro @@ -0,0 +1,151 @@ +--- +// fetch all commits for just this page's path +const path = "docs/" + Astro.props.path; +const url = `https://api.github.com/repos/snowpackjs/astro/commits?path=${path}`; +const commitsURL = `https://github.com/snowpackjs/astro/commits/main/${path}`; + +async function getCommits(url) { +  try { +    const token = import.meta.env.SNOWPACK_PUBLIC_GITHUB_TOKEN; +    if (!token) { +      throw new Error( +        'Cannot find "SNOWPACK_PUBLIC_GITHUB_TOKEN" used for escaping rate-limiting.' +      ); +    } + +    const auth = `Basic ${Buffer.from(token, "binary").toString("base64")}`; + +    const res = await fetch(url, { +      method: "GET", +      headers: { +        Authorization: auth, +        "User-Agent": "astro-docs/1.0", +      }, +    }); + +    const data = await res.json(); + +    if (!res.ok) { +      throw new Error( +        `Request to fetch commits failed. Reason: ${res.statusText} +       Message: ${data.message}` +      ); +    } + +    return data; +  } catch (e) { +    console.warn(`[error]  /src/components/AvatarList.astro  +    ${e?.message ?? e}`); +    return new Array(); +  } +} + +function removeDups(arr) { +  if (!arr) { +    return new Array(); +  } +  let map = new Map(); + +  for (let item of arr) { +    let author = item.author; +    // Deduplicate based on author.id +    map.set(author.id, { login: author.login, id: author.id }); +  } + +  return Array.from(map.values()); +} + +const data = await getCommits(url); +const unique = removeDups(data); +const recentContributors = unique.slice(0, 3); // only show avatars for the 3 most recent contributors +const additionalContributors = unique.length - recentContributors.length; // list the rest of them as # of extra contributors + +--- +<!-- Thanks to @5t3ph for https://smolcss.dev/#smol-avatar-list! --> +<div class="contributors"> +<ul class="avatar-list" style={`--avatar-count: ${recentContributors.length}`}> + +{recentContributors.map((item) => ( +      <li><a href={`https://github.com/${item.login}`}><img alt={`Contributor ${item.login}`} title={`Contributor ${item.login}`} width="64" height="64" src={`https://avatars.githubusercontent.com/u/${item.id}`}/></a></li> +       +))} +  </ul> +  {additionalContributors > 0 && <span><a href={commitsURL}>{`and ${additionalContributors} additional contributor${additionalContributors > 1 ? 's' : ''}.`}</a></span>} +  {unique.length === 0 && <a href={commitsURL}>Contributors</a>} +</div> + +<style> +.avatar-list { +  --avatar-size: 2.5rem; +  --avatar-count: 3; + +  display: grid; +  list-style: none; +  /* Default to displaying most of the avatar to +  enable easier access on touch devices, ensuring +  the WCAG touch target size is met or exceeded */ +  grid-template-columns: repeat( +    var(--avatar-count), +    max(44px, calc(var(--avatar-size) / 1.15)) +  ); +  /* `padding` matches added visual dimensions of +  the `box-shadow` to help create a more accurate +  computed component size */ +  padding: 0.08em; +  font-size: var(--avatar-size); +} + +@media (any-hover: hover) and (any-pointer: fine) { +  .avatar-list { +    /* We create 1 extra cell to enable the computed  +    width to match the final visual width */ +    grid-template-columns: repeat( +      calc(var(--avatar-count) + 1), +      calc(var(--avatar-size) / 1.75) +    ); +  } +} + +.avatar-list li { +  width: var(--avatar-size); +  height: var(--avatar-size); +} + +.avatar-list li:hover ~ li a, +.avatar-list li:focus-within ~ li a { +  transform: translateX(33%); +} + +.avatar-list img, +.avatar-list a { +  display: block; +  border-radius: 50%; +} + +.avatar-list a { +  transition: transform 180ms ease-in-out; +} + +.avatar-list img { +  width: 100%; +  height: 100%; +  object-fit: cover; +  background-color: #fff; +  box-shadow: 0 0 0 0.05em #fff, 0 0 0 0.08em rgba(0, 0, 0, 0.15); +} + +.avatar-list a:focus { +  outline: 2px solid transparent; +  /* Double-layer trick to work for dark and light backgrounds */ +  box-shadow: 0 0 0 0.08em var(--theme-accent), 0 0 0 0.12em white; +} + +.contributors { +  display: flex; +  align-items: center; +} + +.contributors > * + * { +  margin-left: .75rem; +} +</style> diff --git a/examples/docs/src/components/ArticleFooter.astro b/examples/docs/src/components/Footer/Footer.astro index 8078e2cc3..48de51054 100644 --- a/examples/docs/src/components/ArticleFooter.astro +++ b/examples/docs/src/components/Footer/Footer.astro @@ -1,9 +1,10 @@  ---  import AvatarList from './AvatarList.astro'; +const { path } = Astro.props;  ---  <footer> -  <AvatarList /> +  <AvatarList path={path} />  </footer>  <style> diff --git a/examples/docs/src/components/HeadCommon.astro b/examples/docs/src/components/HeadCommon.astro new file mode 100644 index 000000000..83045c0d1 --- /dev/null +++ b/examples/docs/src/components/HeadCommon.astro @@ -0,0 +1,40 @@ +<!-- Global Metadata --> +<meta name="viewport" content="width=device-width"> + +<link rel="icon" type="image/svg+xml" href="/favicon.svg" /> +<link rel="alternate icon" type="image/x-icon" href="/favicon.ico" /> + +<link rel="sitemap" href="/sitemap.xml"/> + +<!-- Global CSS --> +<link rel="stylesheet" href="/theme.css" /> +<link rel="stylesheet" href="/code.css" /> +<link rel="stylesheet" href="/index.css" /> + +<!-- Preload Fonts --> +<link rel="preconnect" href="https://fonts.googleapis.com"> +<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> +<link href="https://fonts.googleapis.com/css2?family=IBM+Plex+Mono:ital@0;1&display=swap" rel="stylesheet"> + +<!-- Scrollable a11y code helper --> +<script type="module" src="/make-scrollable-code-focusable.js" /> + +<!-- This is intentionally inlined to avoid FOUC --> +<script> +  const root = document.documentElement; +  const theme = localStorage.getItem('theme'); +  if (theme === 'dark' || (!theme) && window.matchMedia('(prefers-color-scheme: dark)').matches) { +    root.classList.add('theme-dark'); +  } else { +    root.classList.remove('theme-dark'); +  } +</script> + +<!-- Global site tag (gtag.js) - Google Analytics --> +<!-- <script async src="https://www.googletagmanager.com/gtag/js?id=G-TEL60V1WM9"></script> +<script> +  window.dataLayer = window.dataLayer || []; +  function gtag(){dataLayer.push(arguments);} +  gtag('js', new Date()); +  gtag('config', 'G-TEL60V1WM9'); +</script> -->
\ No newline at end of file diff --git a/examples/docs/src/components/HeadSEO.astro b/examples/docs/src/components/HeadSEO.astro new file mode 100644 index 000000000..268809a87 --- /dev/null +++ b/examples/docs/src/components/HeadSEO.astro @@ -0,0 +1,40 @@ +--- +import {SITE, OPEN_GRAPH} from '../config.ts'; +export interface Props { +  content: any, +  site: any, +  canonicalURL: URL | string, +}; +const { content = {}, canonicalURL } = Astro.props; +const formattedContentTitle = content.title ? `${content.title} 🚀 ${SITE.title}` : SITE.title; +const imageSrc = content?.image?.src ?? OPEN_GRAPH.image.src; +const canonicalImageSrc = new URL(imageSrc, Astro.site); +const imageAlt = content?.image?.alt ?? OPEN_GRAPH.image.alt; +--- +<!-- Page Metadata --> +<link rel="canonical" href={canonicalURL}/> + +<!-- OpenGraph Tags --> +<meta property="og:title" content={formattedContentTitle}/> +<meta property="og:type" content="article"/> +<meta property="og:url" content={canonicalURL}/> +<meta property="og:locale" content={content.ogLocale ?? SITE.defaultLanguage}/> +<meta property="og:image" content={canonicalImageSrc}/> +<meta property="og:image:alt" content={imageAlt}/> +<meta property="og:description" content={content.description ? content.description : SITE.description}/> +<meta property="og:site_name" content={SITE.title}/> + +<!-- Twitter Tags --> +<meta name="twitter:card" content="summary_large_image"/> +<meta name="twitter:site" content={OPEN_GRAPH.twitter}/> +<meta name="twitter:title" content={formattedContentTitle}/> +<meta name="twitter:description" content={content.description ? content.description : SITE.description}/> +<meta name="twitter:image" content={canonicalImageSrc}/> +<meta name="twitter:image:alt" content={imageAlt}/> + +<!--  +  TODO: Add json+ld data, maybe https://schema.org/APIReference makes sense?  +  Docs: https://developers.google.com/search/docs/advanced/structured-data/intro-structured-data +  https://www.npmjs.com/package/schema-dts seems like a great resource for implementing this. +  Even better, there's a React component that integrates with `schema-dts`: https://github.com/google/react-schemaorg +--> diff --git a/examples/docs/src/components/Header/AstroLogo.astro b/examples/docs/src/components/Header/AstroLogo.astro new file mode 100644 index 000000000..124865f1b --- /dev/null +++ b/examples/docs/src/components/Header/AstroLogo.astro @@ -0,0 +1,18 @@ +--- +const {size} = Astro.props; +--- +<svg class="logo" width={size} height={size} viewBox="0 0 256 256" fill="none" xmlns="http://www.w3.org/2000/svg"> +    <style> +        #flame { +            fill: var(--theme-text-accent); +        } +        #a { +            fill: var(--theme-text-accent); +        } +    </style> +    <title>Logo</title> +    <path id="a" fill-rule="evenodd" clip-rule="evenodd" +        d="M163.008 18.929c1.944 2.413 2.935 5.67 4.917 12.181l43.309 142.27a180.277 180.277 0 00-51.778-17.53l-28.198-95.29a3.67 3.67 0 00-7.042.01l-27.857 95.232a180.225 180.225 0 00-52.01 17.557l43.52-142.281c1.99-6.502 2.983-9.752 4.927-12.16a15.999 15.999 0 016.484-4.798c2.872-1.154 6.271-1.154 13.07-1.154h31.085c6.807 0 10.211 0 13.086 1.157a16.004 16.004 0 016.487 4.806z" /> +    <path id="flame" fill-rule="evenodd" clip-rule="evenodd" +        d="M168.19 180.151c-7.139 6.105-21.39 10.268-37.804 10.268-20.147 0-37.033-6.272-41.513-14.707-1.602 4.835-1.961 10.367-1.961 13.902 0 0-1.056 17.355 11.015 29.426 0-6.268 5.081-11.349 11.349-11.349 10.743 0 10.731 9.373 10.721 16.977v.679c0 11.542 7.054 21.436 17.086 25.606a23.27 23.27 0 01-2.339-10.2c0-11.008 6.463-15.107 13.974-19.87 5.976-3.79 12.616-8.001 17.192-16.449a31.024 31.024 0 003.743-14.82c0-3.299-.513-6.479-1.463-9.463z" /> +</svg>
\ No newline at end of file diff --git a/examples/docs/src/components/Header/Header.astro b/examples/docs/src/components/Header/Header.astro new file mode 100644 index 000000000..a50957240 --- /dev/null +++ b/examples/docs/src/components/Header/Header.astro @@ -0,0 +1,132 @@ +--- +import { getLanguageFromURL, KNOWN_LANGUAGE_CODES } from '../../languages.ts'; +import * as CONFIG from '../../config.ts'; +import AstroLogo from './AstroLogo.astro'; +import SkipToContent from './SkipToContent.astro'; +import SidebarToggle from './SidebarToggle.tsx'; +import LanguageSelect from './LanguageSelect.jsx'; +import Search from "./Search.jsx"; + +const {currentPage} = Astro.props; +const lang = currentPage && getLanguageFromURL(currentPage); +--- +<style> +    header { +      z-index: 11; +      height: var(--theme-navbar-height); +      width: 100%; +      background-color: var(--theme-navbar-bg); +      display: flex; +      align-items: center; +      justify-content: center; +      overflow: hidden; +      position: sticky; +      top: 0; +    } + + +    .logo { +      display: flex; +      overflow: hidden; +      width: 30px; +      font-size: 2rem; +      flex-shrink: 0; +      font-weight: 600; +      line-height: 1; +      color: hsla(var(--color-base-white), 100%, 1); +      gap: 0.25em; +      z-index: -1; +    } + +    .logo a { +      padding: 0.5em 0.25em; +      margin: -0.5em -0.25em; +      text-decoration: none; +      font-weight: bold; +    } + +    .logo a { +      transition: color 100ms ease-out; +      color: var(--theme-text); +    } + +    .logo a:hover, +    .logo a:focus { +      color: var(--theme-text-accent); +    } + +    .logo h1 { +      font: inherit; +      color: inherit; +      margin: 0; +    } + +    .nav-wrapper { +      display: flex; +      align-items: center; +      justify-content: flex-end; +      gap: 1em; +      width: 100%; +      max-width: 82em; +      padding: 0 1rem; +    } + +    @media (min-width: 50em) { +        header { +            position: static; +            padding: 2rem 0rem; +        } +        .logo { +            width: auto; +            margin: 0; +            z-index: 0; +        } +        .menu-toggle { +            display: none; +        } +        .logo { +            width: auto; +        } +    } + +    /** Style Algolia */ +    :root { +      --docsearch-primary-color: var(--theme-accent); +      --docsearch-logo-color: var(--theme-text); +    } + +    .search-item { +      display: none; +      position: relative; +      z-index: 10; +      flex-grow: 1; +      padding-right: 0.7rem; +      display: flex; +      max-width: 200px; +    } +    :global(.search-item > *) { +      flex-grow: 1; +    } +    @media (min-width: 50em) { +      .search-item { +        max-width: 400px; +      } +    } +</style> +<header> +    <SkipToContent /> +    <nav class="nav-wrapper" title="Top Navigation"> +    <div class="menu-toggle"> +        <SidebarToggle client:idle/> +    </div> +    <div class="logo flex"> +        <AstroLogo size={40} /> +        <a href="/"> +          <h1>Documentation</h1> +        </a> +    </div> +    <div style="flex-grow: 1;"></div> +    {KNOWN_LANGUAGE_CODES.length > 1 && <LanguageSelect lang={lang} client:idle />} +    {CONFIG.ALGOLIA && <div class="search-item"><Search client:idle /></div>} +    </nav> +</header>
\ No newline at end of file diff --git a/examples/docs/src/components/Header/LanguageSelect.css b/examples/docs/src/components/Header/LanguageSelect.css new file mode 100644 index 000000000..4e878714b --- /dev/null +++ b/examples/docs/src/components/Header/LanguageSelect.css @@ -0,0 +1,47 @@ +.language-select { +  flex-grow: 1; +  width: 48px; +  box-sizing: border-box; +  margin: 0; +  padding: 0.33em 0.5em; +  overflow: visible; +  font-weight: 500; +  font-size: 1rem; +  font-family: inherit; +  line-height: inherit; +  background-color: var(--theme-bg); +  border-color: var(--theme-text-lighter); +  color: var(--theme-text-light); +  border-style: solid; +  border-width: 1px; +  border-radius: 0.25rem; +  outline: 0; +  cursor: pointer; +  transition-timing-function: ease-out; +  transition-duration: 0.2s; +  transition-property: border-color, color; +  -webkit-font-smoothing: antialiased; +  padding-left: 30px; +  padding-right: 1rem; +} +.language-select-wrapper .language-select:hover, +.language-select-wrapper .language-select:focus { +  color: var(--theme-text); +  border-color: var(--theme-text-light); +} +.language-select-wrapper { +  color: var(--theme-text-light); +  position: relative; +} +.language-select-wrapper > svg { +  position: absolute; +  top: 7px; +  left: 10px; +  pointer-events: none; +} + +@media (min-width: 50em) { +  .language-select { +    width: 100%; +  } +} diff --git a/examples/docs/src/components/Header/LanguageSelect.tsx b/examples/docs/src/components/Header/LanguageSelect.tsx new file mode 100644 index 000000000..8b9807fe8 --- /dev/null +++ b/examples/docs/src/components/Header/LanguageSelect.tsx @@ -0,0 +1,38 @@ +import type { FunctionalComponent } from 'preact'; +import { h } from 'preact'; +import './LanguageSelect.css'; +import { KNOWN_LANGUAGES, langPathRegex } from '../../languages'; + +const LanguageSelect: FunctionalComponent<{ lang: string }> = ({ lang }) => { +  return ( +    <div class="language-select-wrapper"> +      <svg aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 88.6 77.3" height="1.2em" width="1.2em"> +        <path fill="currentColor" d="M61,24.6h7.9l18.7,51.6h-7.7l-5.4-15.5H54.3l-5.6,15.5h-7.2L61,24.6z M72.6,55l-8-22.8L56.3,55H72.6z" /> +        <path +          fill="currentColor" +          d="M53.6,60.6c-10-4-16-9-22-14c0,0,1.3,1.3,0,0c-6,5-20,13-20,13l-4-6c8-5,10-6,19-13c-2.1-1.9-12-13-13-19h8          c4,9,10,14,10,14c10-8,10-19,10-19h8c0,0-1,13-12,24l0,0c5,5,10,9,19,13L53.6,60.6z M1.6,16.6h56v-8h-23v-7h-9v7h-24V16.6z" +        /> +      </svg> +      <select +        class="language-select" +        value={lang} +        onChange={(e) => { +          const newLang = e.target.value; +          let actualDest = window.location.pathname.replace(langPathRegex, '/'); +          if (actualDest == '/') actualDest = `/introduction`; +          window.location.pathname = '/' + newLang + actualDest; +        }} +      > +        {Object.keys(KNOWN_LANGUAGES).map((key) => { +          return ( +            <option value={KNOWN_LANGUAGES[key]}> +              <span>{key}</span> +            </option> +          ); +        })} +      </select> +    </div> +  ); +}; + +export default LanguageSelect; diff --git a/examples/docs/src/components/Header/Search.css b/examples/docs/src/components/Header/Search.css new file mode 100644 index 000000000..2056c2c8f --- /dev/null +++ b/examples/docs/src/components/Header/Search.css @@ -0,0 +1,76 @@ +/** Style Algolia */ +:root { +  --docsearch-primary-color: var(--theme-accent); +  --docsearch-logo-color: var(--theme-text); +} +.search-input { +  flex-grow: 1; +  box-sizing: border-box; +  width: 100%; +  margin: 0; +  padding: 0.33em 0.5em; +  overflow: visible; +  font-weight: 500; +  font-size: 1rem; +  font-family: inherit; +  line-height: inherit; +  background-color: var(--theme-divider); +  border-color: var(--theme-divider); +  color: var(--theme-text-light); +  border-style: solid; +  border-width: 1px; +  border-radius: 0.25rem; +  outline: 0; +  cursor: pointer; +  transition-timing-function: ease-out; +  transition-duration: 0.2s; +  transition-property: border-color, color; +  -webkit-font-smoothing: antialiased; +} +.search-input:hover, +.search-input:focus { +  color: var(--theme-text); +  border-color: var(--theme-text-light); +} +.search-input:hover::placeholder, +.search-input:focus::placeholder { +  color: var(--theme-text-light); +} +.search-input::placeholder { +  color: var(--theme-text-light); +} +.search-hint { +  position: absolute; +  top: 7px; +  right: 19px; +  padding: 3px 5px; +  display: none; +  display: none; +  align-items: center; +  justify-content: center; +  letter-spacing: 0.125em; +  font-size: 13px; +  font-family: var(--font-mono); +  pointer-events: none; +  border-color: var(--theme-text-lighter); +  color: var(--theme-text-light); +  border-style: solid; +  border-width: 1px; +  border-radius: 0.25rem; +  line-height: 14px; +} + +@media (min-width: 50em) { +  .search-hint { +    display: flex; +  } +} + +/* ------------------------------------------------------------ *\ +	DocSearch (Algolia) +\* ------------------------------------------------------------ */ + +.DocSearch-Modal .DocSearch-Hit a { +  box-shadow: none; +  border: 1px solid var(--theme-accent); +} diff --git a/examples/docs/src/components/Header/Search.tsx b/examples/docs/src/components/Header/Search.tsx new file mode 100644 index 000000000..19ee513f1 --- /dev/null +++ b/examples/docs/src/components/Header/Search.tsx @@ -0,0 +1,77 @@ +/* jsxImportSource: react */ +import { useState, useCallback, useRef } from 'react'; +import { createPortal } from 'react-dom'; +import { DocSearchModal, useDocSearchKeyboardEvents } from '@docsearch/react'; +import * as CONFIG from '../../config.js'; +import '@docsearch/css/dist/style.css'; +import './Search.css'; + +export default function Search() { +  const [isOpen, setIsOpen] = useState(false); +  const searchButtonRef = useRef(); +  const [initialQuery, setInitialQuery] = useState(null); + +  const onOpen = useCallback(() => { +    setIsOpen(true); +  }, [setIsOpen]); + +  const onClose = useCallback(() => { +    setIsOpen(false); +  }, [setIsOpen]); + +  const onInput = useCallback( +    (e) => { +      setIsOpen(true); +      setInitialQuery(e.key); +    }, +    [setIsOpen, setInitialQuery] +  ); + +  useDocSearchKeyboardEvents({ +    isOpen, +    onOpen, +    onClose, +    onInput, +    searchButtonRef, +  }); + +  return ( +    <> +      <button type="button" ref={searchButtonRef} onClick={onOpen} className="search-input"> +        <svg width="24" height="24" fill="none"> +          <path d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round" /> +        </svg> +        <span>Search</span> +        <span className="search-hint"> +          <span className="sr-only">Press </span> +          <kbd>/</kbd> +          <span className="sr-only"> to search</span> +        </span> +      </button> +      {isOpen && +        createPortal( +          <DocSearchModal +            initialQuery={initialQuery} +            initialScrollY={window.scrollY} +            onClose={onClose} +            indexName={(CONFIG as any).ALGOLIA.indexName} +            apiKey={(CONFIG as any).ALGOLIA.apiKey} +            transformItems={(items) => { +              return items.map((item) => { +                // We transform the absolute URL into a relative URL to +                // work better on localhost, preview URLS. +                const a = document.createElement('a'); +                a.href = item.url; +                const hash = a.hash === '#overview' ? '' : a.hash; +                return { +                  ...item, +                  url: `${a.pathname}${hash}`, +                }; +              }); +            }} +          />, +          document.body +        )} +    </> +  ); +} diff --git a/examples/docs/src/components/Header/SidebarToggle.tsx b/examples/docs/src/components/Header/SidebarToggle.tsx new file mode 100644 index 000000000..97fece6b2 --- /dev/null +++ b/examples/docs/src/components/Header/SidebarToggle.tsx @@ -0,0 +1,27 @@ +import type { FunctionalComponent } from 'preact'; +import { h, Fragment } from 'preact'; +import { useState, useEffect } from 'preact/hooks'; + +const MenuToggle: FunctionalComponent = () => { +  const [sidebarShown, setSidebarShown] = useState(false); + +  useEffect(() => { +    const body = document.getElementsByTagName('body')[0]; +    if (sidebarShown) { +      body.classList.add('mobile-sidebar-toggle'); +    } else { +      body.classList.remove('mobile-sidebar-toggle'); +    } +  }, [sidebarShown]); + +  return ( +    <button type="button" aria-pressed={sidebarShown ? 'true' : 'false'} id="menu-toggle" onClick={() => setSidebarShown(!sidebarShown)}> +      <svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" fill="none" viewBox="0 0 24 24" stroke="currentColor"> +        <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16" /> +      </svg> +      <span className="sr-only">Toggle sidebar</span> +    </button> +  ); +}; + +export default MenuToggle; diff --git a/examples/docs/src/components/Header/SkipToContent.astro b/examples/docs/src/components/Header/SkipToContent.astro new file mode 100644 index 000000000..6df3a72ed --- /dev/null +++ b/examples/docs/src/components/Header/SkipToContent.astro @@ -0,0 +1,21 @@ +<style> +.skiplink, +.skiplink:focus, +.skiplink:focus-visible { +  position: absolute; +  padding: 0.25em; +  font-size: larger; +  top: 0; +  left: 0; +  right: 0; +  z-index: 9; +  display: block; +  text-align: center; +  background-color: var(--theme-text-accent); +  color: var(--theme-bg); +  border-radius: 0.25em; +  outline: var(--theme-bg) solid 1px; +  outline-offset: 0; +} +</style> +<a href="#article" class="sr-only skiplink"><span>Skip to Content</span></a> diff --git a/examples/docs/src/components/LeftSidebar/LeftSidebar.astro b/examples/docs/src/components/LeftSidebar/LeftSidebar.astro new file mode 100644 index 000000000..e979dc80e --- /dev/null +++ b/examples/docs/src/components/LeftSidebar/LeftSidebar.astro @@ -0,0 +1,109 @@ +--- +import { getLanguageFromURL } from '../../languages.ts'; +import { SIDEBAR } from '../../config.ts'; +const {currentPage} = Astro.props; +const currentPageMatch = currentPage.slice(1); +const langCode = getLanguageFromURL(currentPage); +// SIDEBAR is a flat array. Group it by sections to properly render.  +const sidebarSections = SIDEBAR[langCode].reduce((col, item) => { +  if (item.header) { +    col.push({...item, children: []}); +  } else { +    col[col.length-1].children.push(item); +  } +  return col; +}, []); + +--- + +<nav aria-labelledby="grid-left"> +  <ul class="nav-groups">   +    {sidebarSections.map(section => ( +      <li> +        <div class="nav-group"> +          <h2 class="nav-group-title">{section.text}</h2> +          <ul> +            {section.children.map(child => ( +              <li class="nav-link"><a href={`${Astro.site.pathname}${child.link}`} aria-current={`${currentPageMatch === child.link ? 'page' : 'false'}`}>{child.text}</a></li> +            ))} +          </ul> +        </div> +      </li> +    ))} +  </ul> +</nav> + +<script> +  window.addEventListener('DOMContentLoaded', (event) => { +    var target = document.querySelector('[aria-current="page"]'); +    if (target && (target.offsetTop > (window.innerHeight - 100))) { +      document.querySelector('.nav-groups').scrollTop = target.offsetTop; +    } +  }); +</script> + +<style> +  nav { +    width: 100%; +    margin-right: 1rem; +  } +  .nav-groups { +    height: 100%; +    padding: 2rem 0; +    overflow-x: visible; +    overflow-y: auto; +    max-height: 100vh; +  } + +  .nav-groups > li + li { +    margin-top: 2rem; +  } + +  .nav-groups > :first-child { +    padding-top: var(--doc-padding); +  } + +  .nav-groups > :last-child { +    padding-bottom: 2rem; +    margin-bottom: var(--theme-navbar-height); +  } + +  .nav-group-title { +    font-size: 1.0rem; +    font-weight: 700; +    padding: 0.1rem 1rem; +    text-transform: uppercase; +    margin-bottom: 0.5rem; +  } + +  .nav-link a { +    font-size: 1.0rem; +    margin: 1px; +    padding: 0.3rem 1rem; +    font: inherit; +    color: inherit; +    text-decoration: none; +    display: block; +  } +  .nav-link a:hover, +  .nav-link a:focus { +    background-color: var(--theme-bg-hover); +  } + +  .nav-link a[aria-current="page"] { +    color: var(--theme-text-accent); +    background-color: var(--theme-bg-accent); +    font-weight: 600; +  } + +  :global(:root.theme-dark) .nav-link a[aria-current="page"] { +    color: hsla(var(--color-base-white), 100%, 1); +  } + +  @media (min-width: 50em) { +    .nav-groups { +      padding: 0; +    } +  } + +</style> diff --git a/examples/docs/src/components/Note.astro b/examples/docs/src/components/Note.astro deleted file mode 100644 index c57ede3a0..000000000 --- a/examples/docs/src/components/Note.astro +++ /dev/null @@ -1,52 +0,0 @@ ---- -export interface Props { -  title: string; -  type?: 'tip' | 'warning' | 'error' -} -const { type = 'tip', title } = Astro.props; ---- - -<aside class={`note type-${type}`}> -  {title && <label>{title}</label>} -  <slot /> -</aside> - -<style> -  .note { -    --padding-block: 1rem; -    --padding-inline: 1.25rem; - -    display: flex; -    flex-direction: column; - -    padding: var(--padding-block) var(--padding-inline); -    margin-left: calc(var(--padding-inline) * -1); -    margin-right: calc(var(--padding-inline) * -1); -     -    background: var(--theme-bg-offset); -    border-left: calc(var(--padding-inline) / 2) solid var(--color); -    border-radius: 0; -  } - -  .note label { -    font-weight: 500; -    color: var(--color); -  } - -  /* .note :global(a) { -    color: var(--color); -  } */ - -  .note.type-tip { -    --color: var(--color-green); -    --color-rgb: var(--color-green-rgb); -  } -  .note.type-warning { -    --color: var(--color-yellow); -    --color-rgb: var(--color-yellow-rgb); -  } -  .note.type-error { -    --color: var(--color-red); -    --color-rgb: var(--color-red-rgb); -  } -</style> diff --git a/examples/docs/src/components/PageContent/PageContent.astro b/examples/docs/src/components/PageContent/PageContent.astro new file mode 100644 index 000000000..fd1e9d242 --- /dev/null +++ b/examples/docs/src/components/PageContent/PageContent.astro @@ -0,0 +1,41 @@ +--- +const {content, githubEditUrl} = Astro.props; +const title = content.title; +const headers = content.astro.headers; +import MoreMenu from '../RightSidebar/MoreMenu.astro'; +import TableOfContents from '../RightSidebar/TableOfContents.tsx'; +--- +<style> +      .content { +        padding: 0; +        max-width: 75ch; +        width: 100%; +        height: 100%; +        display: flex; +        flex-direction: column; +      } +      .content > section { +        margin-bottom: 4rem; +      } +      .block { +        display: block; +      } + +      @media (min-width: 50em) { +        .sm\:hidden { +          display: none; +        } +      } +</style> +<article id="article" class="content"> +    <section class="main-section"> +        <h1 class="content-title" id="overview">{title}</h1> +        <nav class="block sm:hidden"> +            <TableOfContents client:media="(max-width: 50em)" headers={headers}/> +        </nav> +        <slot /> +    </section> +    <nav class="block sm:hidden"> +        <MoreMenu editHref={githubEditUrl}/> +    </nav> +</article>
\ No newline at end of file diff --git a/examples/docs/src/components/RightSidebar/MoreMenu.astro b/examples/docs/src/components/RightSidebar/MoreMenu.astro new file mode 100644 index 000000000..6be2d86ee --- /dev/null +++ b/examples/docs/src/components/RightSidebar/MoreMenu.astro @@ -0,0 +1,68 @@ +--- +import ThemeToggleButton from './ThemeToggleButton.jsx'; +const {editHref} = Astro.props; +--- +<style> +  .edit-on-github { +    text-decoration: none; +    font: inherit; +    color: inherit; +    font-size: 1rem; +  } +</style> +<h2 class="heading">More</h2> +<ul> +  <li class={`header-link depth-2`}> +    <a class="edit-on-github" href={editHref} target="_blank"> +      <svg +        aria-hidden="true" +        focusable="false" +        data-prefix="fas" +        data-icon="pen" +        class="svg-inline--fa fa-pen fa-w-16" +        role="img" +        xmlns="http://www.w3.org/2000/svg" +        viewBox="0 0 512 512" +        height="1em" +        width="1em" +      > +        <path +          fill="currentColor" +          d="M290.74 93.24l128.02 128.02-277.99 277.99-114.14 12.6C11.35 513.54-1.56 500.62.14 485.34l12.7-114.22 277.9-277.88zm207.2-19.06l-60.11-60.11c-18.75-18.75-49.16-18.75-67.91 0l-56.55 56.55 128.02 128.02 56.55-56.55c18.75-18.76 18.75-49.16 0-67.91z" +        ></path> +      </svg> +      <span>Edit this page</span> +    </a> +  </li> +  <li class={`header-link depth-2`}> +    <a href="https://github.com/snowpackjs/astro/blob/main/CONTRIBUTING.md#translations" target="_blank"> +      <svg aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 88.6 77.3" height="1.24em" width="1.24em" style="margin: -2px;"> <path fill="currentColor" d="M61,24.6h7.9l18.7,51.6h-7.7l-5.4-15.5H54.3l-5.6,15.5h-7.2L61,24.6z M72.6,55l-8-22.8L56.3,55H72.6z" /> <path fill="currentColor" d="M53.6,60.6c-10-4-16-9-22-14c0,0,1.3,1.3,0,0c-6,5-20,13-20,13l-4-6c8-5,10-6,19-13c-2.1-1.9-12-13-13-19h8          c4,9,10,14,10,14c10-8,10-19,10-19h8c0,0-1,13-12,24l0,0c5,5,10,9,19,13L53.6,60.6z M1.6,16.6h56v-8h-23v-7h-9v7h-24V16.6z" /> </svg> +      <span>Translate this page</span> +    </a> +  </li> +  <li class={`header-link depth-2`}> +    <a href="https://astro.build/chat" target="_blank"> +      <svg +        aria-hidden="true" +        focusable="false" +        data-prefix="fas" +        data-icon="comment-alt" +        class="svg-inline--fa fa-comment-alt fa-w-16" +        role="img" +        xmlns="http://www.w3.org/2000/svg" +        viewBox="0 0 512 512" +        height="1em" +        width="1em" +      > +        <path +          fill="currentColor" +          d="M448 0H64C28.7 0 0 28.7 0 64v288c0 35.3 28.7 64 64 64h96v84c0 9.8 11.2 15.5 19.1 9.7L304 416h144c35.3 0 64-28.7 64-64V64c0-35.3-28.7-64-64-64z" +        ></path> +      </svg> +      <span>Join our community</span> +    </a> +  </li> +</ul> +<div style="margin: 2rem 0; text-align: center;"> +  <ThemeToggleButton client:visible /> +</div> diff --git a/examples/docs/src/components/RightSidebar/RightSidebar.astro b/examples/docs/src/components/RightSidebar/RightSidebar.astro new file mode 100644 index 000000000..ed1dd37cc --- /dev/null +++ b/examples/docs/src/components/RightSidebar/RightSidebar.astro @@ -0,0 +1,25 @@ +--- +import TableOfContents from './TableOfContents.jsx'; +import MoreMenu from './MoreMenu.astro'; +const {content, githubEditUrl} = Astro.props; +const headers = content.astro.headers; +--- +<style> +  .sidebar-nav { +    width: 100%; +    position: sticky; +    top: 0; +  } +  .sidebar-nav-inner { +    height: 100%; +    padding: 0; +    padding-top: var(--doc-padding); +    overflow: auto; +  } +</style> +<nav class="sidebar-nav" aria-labelledby="grid-right"> +  <div class="sidebar-nav-inner"> +    <TableOfContents client:media="(min-width: 50em)" headers={headers} /> +    <MoreMenu editHref={githubEditUrl} /> +  </div> +</nav>
\ No newline at end of file diff --git a/examples/docs/src/components/RightSidebar/TableOfContents.tsx b/examples/docs/src/components/RightSidebar/TableOfContents.tsx new file mode 100644 index 000000000..d8ea998d4 --- /dev/null +++ b/examples/docs/src/components/RightSidebar/TableOfContents.tsx @@ -0,0 +1,45 @@ +import type { FunctionalComponent } from 'preact'; +import { h, Fragment } from 'preact'; +import { useState, useEffect, useRef } from 'preact/hooks'; + +const TableOfContents: FunctionalComponent<{ headers: any[] }> = ({ headers = [] }) => { +  const itemOffsets = useRef([]); +  const [activeId, setActiveId] = useState<string>(undefined); + +  useEffect(() => { +    const getItemOffsets = () => { +      const titles = document.querySelectorAll('article :is(h1, h2, h3, h4)'); +      itemOffsets.current = Array.from(titles).map((title) => ({ +        id: title.id, +        topOffset: title.getBoundingClientRect().top + window.scrollY, +      })); +    }; + +    getItemOffsets(); +    window.addEventListener('resize', getItemOffsets); + +    return () => { +      window.removeEventListener('resize', getItemOffsets); +    }; +  }, []); + +  return ( +    <> +      <h2 class="heading">On this page</h2> +      <ul> +        <li class={`header-link depth-2 ${activeId === 'overview' ? 'active' : ''}`.trim()}> +          <a href="#overview">Overview</a> +        </li> +        {headers +          .filter(({ depth }) => depth > 1 && depth < 4) +          .map((header) => ( +            <li class={`header-link depth-${header.depth} ${activeId === header.slug ? 'active' : ''}`.trim()}> +              <a href={`#${header.slug}`}>{header.text}</a> +            </li> +          ))} +      </ul> +    </> +  ); +}; + +export default TableOfContents; diff --git a/examples/docs/src/components/RightSidebar/ThemeToggleButton.css b/examples/docs/src/components/RightSidebar/ThemeToggleButton.css new file mode 100644 index 000000000..7de231d1b --- /dev/null +++ b/examples/docs/src/components/RightSidebar/ThemeToggleButton.css @@ -0,0 +1,37 @@ +.theme-toggle { +  display: inline-flex; +  align-items: center; +  gap: 0.25em; +  padding: 0.33em 0.67em; +  border-radius: 99em; +  background-color: var(--theme-code-inline-bg); +} + +.theme-toggle > label:focus-within { +  outline: 2px solid transparent; +  box-shadow: 0 0 0 0.08em var(--theme-accent), 0 0 0 0.12em white; +} + +.theme-toggle > label { +  color: var(--theme-code-inline-text); +  position: relative; +  display: flex; +  align-items: center; +  justify-content: center; +  opacity: 0.5; +} + +.theme-toggle .checked { +  color: var(--theme-accent); +  opacity: 1; +} + +input[name='theme-toggle'] { +  position: absolute; +  opacity: 0; +  top: 0; +  right: 0; +  bottom: 0; +  left: 0; +  z-index: -1; +} diff --git a/examples/docs/src/components/ThemeToggle.tsx b/examples/docs/src/components/RightSidebar/ThemeToggleButton.tsx index 5a5061c15..75ea775f4 100644 --- a/examples/docs/src/components/ThemeToggle.tsx +++ b/examples/docs/src/components/RightSidebar/ThemeToggleButton.tsx @@ -1,19 +1,13 @@  import type { FunctionalComponent } from 'preact';  import { h, Fragment } from 'preact';  import { useState, useEffect } from 'preact/hooks'; +import './ThemeToggleButton.css'; -const themes = ['system', 'light', 'dark']; +const themes = ['light', 'dark'];  const icons = [    <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 20 20" fill="currentColor">      <path -      fill-rule="evenodd" -      d="M3 5a2 2 0 012-2h10a2 2 0 012 2v8a2 2 0 01-2 2h-2.22l.123.489.804.804A1 1 0 0113 18H7a1 1 0 01-.707-1.707l.804-.804L7.22 15H5a2 2 0 01-2-2V5zm5.771 7H5V5h10v7H8.771z" -      clip-rule="evenodd" -    /> -  </svg>, -  <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 20 20" fill="currentColor"> -    <path        fillRule="evenodd"        d="M10 2a1 1 0 011 1v1a1 1 0 11-2 0V3a1 1 0 011-1zm4 8a4 4 0 11-8 0 4 4 0 018 0zm-.464 4.95l.707.707a1 1 0 001.414-1.414l-.707-.707a1 1 0 00-1.414 1.414zm2.12-10.607a1 1 0 010 1.414l-.706.707a1 1 0 11-1.414-1.414l.707-.707a1 1 0 011.414 0zM17 11a1 1 0 100-2h-1a1 1 0 100 2h1zm-7 4a1 1 0 011 1v1a1 1 0 11-2 0v-1a1 1 0 011-1zM5.05 6.464A1 1 0 106.465 5.05l-.708-.707a1 1 0 00-1.414 1.414l.707.707zm1.414 8.486l-.707.707a1 1 0 01-1.414-1.414l.707-.707a1 1 0 011.414 1.414zM4 11a1 1 0 100-2H3a1 1 0 000 2h1z"        clipRule="evenodd" @@ -25,42 +19,48 @@ const icons = [  ];  const ThemeToggle: FunctionalComponent = () => { -  const [theme, setTheme] = useState(themes[0]); - -  useEffect(() => { -    const user = localStorage.getItem('theme'); -    if (!user) return; -    setTheme(user); -  }, []); +  const [theme, setTheme] = useState(() => { +    if (import.meta.env.SSR) { +      return undefined; +    } +    if (typeof localStorage !== 'undefined' && localStorage.getItem('theme')) { +      return localStorage.getItem('theme'); +    } +    if (window.matchMedia('(prefers-color-scheme: dark)').matches) { +      return 'dark'; +    } +    return 'light'; +  });    useEffect(() => {      const root = document.documentElement; -    if (theme === 'system') { -      localStorage.removeItem('theme'); -      if (window.matchMedia('(prefers-color-scheme: dark)').matches) { -        root.classList.add('theme-dark'); -      } else { -        root.classList.remove('theme-dark'); -      } +    if (theme === 'light') { +      root.classList.remove('theme-dark');      } else { -      localStorage.setItem('theme', theme); -      if (theme === 'light') { -        root.classList.remove('theme-dark'); -      } else { -        root.classList.add('theme-dark'); -      } +      root.classList.add('theme-dark');      }    }, [theme]);    return ( -    <div id="theme-toggle"> +    <div class="theme-toggle">        {themes.map((t, i) => {          const icon = icons[i];          const checked = t === theme;          return ( -          <label className={checked ? 'checked' : ''}> +          <label className={checked ? ' checked' : ''}>              {icon} -            <input type="radio" name="theme-toggle" checked={checked} value={t} title={`Use ${t} theme`} aria-label={`Use ${t} theme`} onChange={() => setTheme(t)} /> +            <input +              type="radio" +              name="theme-toggle" +              checked={checked} +              value={t} +              title={`Use ${t} theme`} +              aria-label={`Use ${t} theme`} +              onChange={() => { +                localStorage.setItem('theme', t); +                setTheme(t); +              }} +            />            </label>          );        })} diff --git a/examples/docs/src/components/SiteSidebar.astro b/examples/docs/src/components/SiteSidebar.astro deleted file mode 100644 index 0fbad0c83..000000000 --- a/examples/docs/src/components/SiteSidebar.astro +++ /dev/null @@ -1,20 +0,0 @@ ---- -import { sidebar } from '../config.ts'; ---- - -<nav> -  <ul class="nav-groups"> -    {sidebar.map(category => ( -      <li> -        <div class="nav-group"> -          <h4 class="nav-group-title"><a href={`${Astro.site}${category.link}`}>{category.text}</a></h4> -          <ul> -            {category.children.map(child => ( -              <li class="nav-link"><a href={`${Astro.site}${child.link}`}>{child.text}</a></li> -            ))} -          </ul> -        </div> -      </li> -    ))} -  </ul> -</nav> | 
