Skip to content

Commit

Permalink
Add markdown tabs (#113)
Browse files Browse the repository at this point in the history
  • Loading branch information
julieg18 authored Sep 30, 2021
1 parent c22ab08 commit a43ff69
Show file tree
Hide file tree
Showing 3 changed files with 187 additions and 3 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import React, { createContext, useState } from 'react'

interface ITogglesData {
[key: string]: { texts: string[]; checkedInd: number }
}

interface ITogglesContext {
addNewToggle?: (id: string, texts: string[]) => void
updateToggleInd?: (id: string, newInd: number) => void
togglesData?: ITogglesData
}

export const TogglesContext = createContext<ITogglesContext>({})

export const TogglesProvider: React.FC = ({ children }) => {
const [togglesData, setTogglesData] = useState({})

const addNewToggle = (id: string, texts: string[]): void => {
const togglesDataCopy: ITogglesData = { ...togglesData }
togglesDataCopy[id] = { texts, checkedInd: 0 }
setTogglesData(togglesDataCopy)
}

const updateToggleInd = (id: string, newInd: number): void => {
const togglesDataCopy: ITogglesData = { ...togglesData }
const selectedTabText = togglesDataCopy[id].texts[newInd]
togglesDataCopy[id] = { ...togglesDataCopy[id], checkedInd: newInd }

for (const [toggleId, { texts }] of Object.entries(togglesDataCopy)) {
if (texts.includes(selectedTabText)) {
togglesDataCopy[toggleId] = {
...togglesDataCopy[toggleId],
checkedInd: togglesDataCopy[id].texts.indexOf(selectedTabText)
}
}
}

setTogglesData(togglesDataCopy)
}

return (
<TogglesContext.Provider
value={{ addNewToggle, updateToggleInd, togglesData }}
>
{children}
</TogglesContext.Provider>
)
}
92 changes: 89 additions & 3 deletions src/components/pages/Documentation/Markdown/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,13 @@ import React, {
useCallback,
useEffect,
useRef,
useState,
ReactNode,
ReactElement
ReactElement,
useContext
} from 'react'
import cn from 'classnames'
import { nanoid } from 'nanoid'
import { navigate } from '@reach/router'
import rehypeReact from 'rehype-react'
import Collapsible from 'react-collapsible'
Expand All @@ -17,6 +20,7 @@ import { getPathWithSource } from '../../../../utils/shared/sidebar'
import sharedStyles from '../styles.module.css'
import 'github-markdown-css/github-markdown.css'
import styles from './styles.module.css'
import { TogglesContext, TogglesProvider } from './ToggleProvider'

const isInsideCodeBlock = (node: Element): boolean => {
while (node?.parentNode) {
Expand Down Expand Up @@ -117,6 +121,84 @@ const Card: React.FC<{
)
}

const ToggleTab: React.FC<{
id: string
title: string
ind: number
onChange: () => void
checked: boolean
}> = ({ children, id, checked, ind, onChange, title }) => {
const inputId = `tab-${id}-${ind}`

return (
<>
<input
id={inputId}
type="radio"
name={`toggle-${id}`}
onChange={onChange}
checked={checked}
/>
<label className={styles.tabHeading} htmlFor={inputId}>
{title}
</label>
{children}
</>
)
}

const Toggle: React.FC<{
children: Array<{ props: { title: string } } | string>
}> = ({ children }) => {
const [toggleId, setToggleId] = useState('')
const {
addNewToggle = (): null => null,
updateToggleInd = (): null => null,
togglesData = {}
} = useContext(TogglesContext)
const tabs: Array<{ props: { title: string } } | string> = children.filter(
child => child !== '\n'
)
const tabsTitles = tabs.map(tab =>
typeof tab === 'object' ? tab.props.title : ''
)

useEffect(() => {
if (toggleId === '') {
const newId = nanoid()
addNewToggle(newId, tabsTitles)
setToggleId(newId)
}

if (toggleId && !togglesData[toggleId]) {
addNewToggle(toggleId, tabsTitles)
}
}, [togglesData])

return (
<div className={styles.toggle}>
{tabs.map((tab, i) => (
<ToggleTab
ind={i}
key={i}
title={tabsTitles[i]}
id={toggleId}
checked={
i === (togglesData[toggleId] ? togglesData[toggleId].checkedInd : 0)
}
onChange={(): void => updateToggleInd(toggleId, i)}
>
{tab}
</ToggleTab>
))}
</div>
)
}

const Tab: React.FC = ({ children }) => (
<div className={styles.tab}>{children}</div>
)

const renderAst = new rehypeReact({
// eslint-disable-next-line @typescript-eslint/no-explicit-any
createElement: React.createElement as any,
Expand All @@ -125,7 +207,9 @@ const renderAst = new rehypeReact({
details: Details,
a: Link,
card: Card,
cards: Cards
cards: Cards,
toggle: Toggle,
tab: Tab
}
}).Compiler

Expand Down Expand Up @@ -192,7 +276,9 @@ const Markdown: React.FC<IMarkdownProps> = ({
<i className={cn(sharedStyles.buttonIcon, styles.githubIcon)} /> Edit on
GitHub
</Link>
<div className="markdown-body">{renderAst(htmlAst)}</div>
<TogglesProvider>
<div className="markdown-body">{renderAst(htmlAst)}</div>
</TogglesProvider>
<div className={styles.navButtons}>
<Link className={styles.navButton} href={prev || '#'}>
<i className={cn(styles.navButtonIcon, styles.prev)} />
Expand Down
50 changes: 50 additions & 0 deletions src/components/pages/Documentation/Markdown/styles.module.css
Original file line number Diff line number Diff line change
Expand Up @@ -430,3 +430,53 @@ a.card {
background-color: var(--color-light-blue);
}
}

.toggle {
display: flex;
flex-wrap: wrap;

input {
height: 0;
opacity: 0;
position: absolute;
width: 0;
overflow: hidden;
}

input:checked + label {
color: var(--color-azure);
border-color: var(--color-azure);
}

input:checked + label + .tab {
height: initial;
opacity: initial;
position: static;
width: 100%;
overflow: visible;
}

.tabHeading {
padding: 12px 16px 10px;
background-color: transparent;
border: none;
border-bottom: 2px solid transparent;
font-weight: bold;
font-size: 16px;
font-family: var(--font-brandon);
order: -1;

&:hover {
cursor: pointer;
}
}
}

.tab {
margin: 10px 0 0;
height: 0;
opacity: 0;
position: absolute;
overflow: hidden;
width: 0;
}

0 comments on commit a43ff69

Please sign in to comment.