Common front-end components for Web using the Tesseract design system, written for React. https://make.modal.sh/tesseract/web/react/common
 
 
 
 

187 řádky
4.2 KiB

  1. import { Icon } from '../../../../react-common/src'
  2. import pkg from '../../../../../package.json'
  3. import styled from 'styled-components'
  4. import Link from 'next/link'
  5. import * as React from 'react'
  6. import Nav from '../Nav/Nav'
  7. const StyledLink = styled('a')({
  8. display: 'block',
  9. textDecoration: 'none',
  10. marginTop: '3rem',
  11. marginBottom: '3rem',
  12. })
  13. const Container = styled('span')({
  14. display: 'flex',
  15. alignItems: 'center',
  16. padding: '0 1rem',
  17. height: '2rem',
  18. margin: '0 0 0 auto',
  19. boxSizing: 'border-box',
  20. '@media (min-width: 720px)': {
  21. maxWidth: 'var(--max-width)',
  22. },
  23. })
  24. const Base = styled('aside')({
  25. '--max-width': 240,
  26. position: 'fixed',
  27. top: 0,
  28. left: '-100%',
  29. width: '100%',
  30. height: '100%',
  31. backgroundColor: 'var(--color-bg)',
  32. zIndex: 4,
  33. transitionProperty: 'color, background-color',
  34. transitionTimingFunction: 'ease',
  35. transitionDuration: '350ms',
  36. '@media (min-width: 720px)': {
  37. left: 0,
  38. width: `${100 / 4}%`,
  39. maxWidth: 'none',
  40. height: '100%',
  41. '+ *': {
  42. paddingLeft: `${100 / 4}%`,
  43. boxSizing: 'border-box',
  44. },
  45. },
  46. })
  47. const Actions = styled('div')({
  48. marginBottom: '4rem',
  49. display: 'flex',
  50. gap: '1rem',
  51. alignItems: 'center',
  52. })
  53. const ToggleWrapper = styled('label')({
  54. cursor: 'pointer',
  55. color: 'var(--color-accent)',
  56. display: 'inline-block',
  57. })
  58. const ToggleIcon = styled('span')({
  59. })
  60. const ToggleInput = styled('input')({
  61. position: 'absolute',
  62. left: -999999,
  63. })
  64. const NavWrapper = styled('nav')({
  65. '--size-link': '3rem',
  66. })
  67. const RepoLink = styled('a')({
  68. display: 'inline-grid',
  69. width: '1.5rem',
  70. height: '1.5rem',
  71. placeContent: 'center',
  72. })
  73. const Sidebar = ({
  74. data,
  75. brand: Brand,
  76. initialTheme = 'Dark',
  77. }) => {
  78. const [theme, setTheme] = React.useState(initialTheme)
  79. const toggleDarkMode = (b: string) => () => {
  80. setTheme(b)
  81. }
  82. React.useEffect(() => {
  83. let storageTheme = window.localStorage.getItem('tesseract-theme')
  84. || (
  85. window.matchMedia('(prefers-color-scheme: dark)').matches
  86. ? 'Dark'
  87. : 'Light'
  88. )
  89. window.localStorage.setItem('tesseract-theme', storageTheme)
  90. setTimeout(() => {
  91. setTheme(storageTheme)
  92. })
  93. }, [])
  94. React.useEffect(() => {
  95. window.localStorage.setItem('tesseract-theme', theme)
  96. }, [theme])
  97. React.useEffect(() => {
  98. const stylesheets = Array.from(window.document.querySelectorAll('link[title]')) as HTMLLinkElement[]
  99. stylesheets.forEach(s => {
  100. const enabled = s.title === theme
  101. s.setAttribute('rel', enabled ? 'stylesheet' : 'alternate stylesheet')
  102. if (enabled) {
  103. s.removeAttribute('disabled')
  104. } else {
  105. s.setAttribute('disabled', 'disabled')
  106. }
  107. })
  108. }, [theme])
  109. return (
  110. <Base>
  111. <NavWrapper>
  112. <Link
  113. href="/"
  114. passHref
  115. >
  116. <StyledLink>
  117. <Container>
  118. <Brand />
  119. </Container>
  120. </StyledLink>
  121. </Link>
  122. <Container>
  123. <Actions>
  124. <ToggleWrapper>
  125. <ToggleInput
  126. type="checkbox"
  127. defaultChecked={theme === 'Dark'}
  128. onChange={toggleDarkMode(theme === 'Dark' ? 'Light' : 'Dark')}
  129. />
  130. <ToggleIcon>
  131. {
  132. theme === 'Dark'
  133. && (
  134. <Icon
  135. label="Set Light Mode"
  136. name="moon"
  137. />
  138. )
  139. }
  140. {
  141. theme === 'Light'
  142. && (
  143. <Icon
  144. label="Set Dark Mode"
  145. name="sun"
  146. />
  147. )
  148. }
  149. </ToggleIcon>
  150. </ToggleWrapper>
  151. <RepoLink
  152. href={pkg.repository}
  153. target="_blank"
  154. rel="noopener noreferer"
  155. >
  156. <Icon
  157. name="code"
  158. label="Visit Repository"
  159. />
  160. </RepoLink>
  161. </Actions>
  162. </Container>
  163. <Nav
  164. data={data.nav}
  165. />
  166. </NavWrapper>
  167. </Base>
  168. )
  169. }
  170. export default Sidebar