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

218 lines
4.7 KiB

  1. import * as React from 'react'
  2. import * as PropTypes from 'prop-types'
  3. import { Icon } from '../../../../react-common/src'
  4. import pkg from '../../../../../package.json'
  5. import styled from 'styled-components'
  6. import Link from 'next/link'
  7. import Nav from '../Nav/Nav'
  8. import { MouseEventHandler } from 'react'
  9. import ThemeToggle from '../ThemeToggle/ThemeToggle'
  10. const StyledLink = styled('a')({
  11. display: 'block',
  12. textDecoration: 'none',
  13. marginTop: '4rem',
  14. marginBottom: '3rem',
  15. })
  16. const Container = styled('span')({
  17. display: 'flex',
  18. alignItems: 'center',
  19. padding: '0 1rem',
  20. height: '2rem',
  21. margin: '0 0 0 auto',
  22. boxSizing: 'border-box',
  23. '@media (min-width: 720px)': {
  24. maxWidth: 'var(--max-width)',
  25. },
  26. })
  27. const Base = styled('aside')({
  28. '--max-width': 240,
  29. display: 'none',
  30. '@media only screen': {
  31. display: 'block',
  32. position: 'fixed',
  33. top: 0,
  34. width: '100%',
  35. height: '100%',
  36. backgroundColor: 'var(--color-bg)',
  37. zIndex: 4,
  38. transitionProperty: 'color, background-color, left',
  39. transitionTimingFunction: 'ease',
  40. transitionDuration: '350ms',
  41. overflow: 'auto',
  42. '@media (min-width: 720px)': {
  43. left: '0 !important',
  44. width: `${100 / 4}%`,
  45. maxWidth: 'none',
  46. height: '100%',
  47. '+ *': {
  48. paddingLeft: `${100 / 4}%`,
  49. boxSizing: 'border-box',
  50. },
  51. },
  52. },
  53. })
  54. const SidebarToggle = styled('a')({
  55. display: 'none',
  56. '@media only screen': {
  57. width: '4rem',
  58. height: '4rem',
  59. position: 'fixed',
  60. top: 0,
  61. left: 0,
  62. display: 'grid',
  63. placeContent: 'center',
  64. zIndex: 5,
  65. '@media (min-width: 720px)': {
  66. display: 'none',
  67. },
  68. '::before': {
  69. backgroundColor: 'var(--color-bg)',
  70. content: "''",
  71. position: 'absolute',
  72. top: 0,
  73. left: 0,
  74. width: '100%',
  75. height: '100%',
  76. opacity: 0.75,
  77. zIndex: -1,
  78. },
  79. },
  80. })
  81. const Actions = styled('div')({
  82. marginBottom: '4rem',
  83. display: 'grid',
  84. gap: '1rem',
  85. alignItems: 'center',
  86. gridTemplateColumns: 'auto auto',
  87. })
  88. const NavWrapper = styled('nav')({
  89. '--size-link': '3rem',
  90. marginBottom: '4rem',
  91. })
  92. const RepoLink = styled('a')({
  93. display: 'inline-grid',
  94. width: '1.5rem',
  95. height: '1.5rem',
  96. placeContent: 'center',
  97. })
  98. const propTypes = {
  99. data: PropTypes.shape({
  100. nav: PropTypes.array,
  101. }),
  102. brand: PropTypes.elementType,
  103. initialTheme: PropTypes.string,
  104. }
  105. type Props = PropTypes.InferProps<typeof propTypes>
  106. const Sidebar: React.FC<Props> = ({
  107. data,
  108. brand: BrandRaw,
  109. initialTheme,
  110. }) => {
  111. const [sidebar, setSidebar] = React.useState(false)
  112. const navRef = React.useRef<HTMLDivElement>(null)
  113. const Brand = BrandRaw!
  114. const toggleSidebar = (b: boolean): MouseEventHandler => (e) => {
  115. e.preventDefault()
  116. setSidebar(b)
  117. }
  118. const handleSidebarClose = (e: MouseEvent) => {
  119. let currentElement = e.target as HTMLElement
  120. while (currentElement !== e.currentTarget) {
  121. if (currentElement.tagName === 'A') {
  122. setSidebar(false)
  123. break
  124. }
  125. const { parentElement } = currentElement
  126. if (parentElement as HTMLElement === null) {
  127. break
  128. }
  129. currentElement = parentElement as HTMLElement
  130. }
  131. }
  132. React.useEffect(() => {
  133. navRef.current.addEventListener('click', handleSidebarClose, { capture: true })
  134. return () => {
  135. navRef.current.removeEventListener('click', handleSidebarClose, { capture: true })
  136. }
  137. }, [])
  138. return (
  139. <React.Fragment>
  140. <SidebarToggle
  141. onClick={toggleSidebar(!sidebar)}
  142. >
  143. {
  144. !sidebar && (
  145. <Icon
  146. name="menu"
  147. />
  148. )
  149. }
  150. {
  151. sidebar && (
  152. <Icon
  153. name="x"
  154. />
  155. )
  156. }
  157. </SidebarToggle>
  158. <Base
  159. ref={navRef}
  160. style={{
  161. left: sidebar ? 0 : '-100%',
  162. }}
  163. >
  164. <NavWrapper>
  165. <Link
  166. href="/"
  167. passHref
  168. >
  169. <StyledLink>
  170. <Container>
  171. <Brand />
  172. </Container>
  173. </StyledLink>
  174. </Link>
  175. <Container>
  176. <Actions>
  177. <ThemeToggle
  178. initialTheme={initialTheme}
  179. />
  180. <RepoLink
  181. href={pkg.repository}
  182. target="_blank"
  183. rel="noopener noreferrer"
  184. >
  185. <Icon
  186. name="code"
  187. label="Visit Repository"
  188. />
  189. </RepoLink>
  190. </Actions>
  191. </Container>
  192. <Nav
  193. data={data!.nav}
  194. />
  195. </NavWrapper>
  196. </Base>
  197. </React.Fragment>
  198. )
  199. }
  200. Sidebar.propTypes = propTypes
  201. export default Sidebar