1
0
Fork 0

chore(shared): Consolidate tab components

This commit is contained in:
Joost De Cock 2023-08-28 20:30:00 +02:00
parent 44a3b194cc
commit c98a67885c
6 changed files with 50 additions and 43 deletions

View file

@ -15,13 +15,14 @@ import { ResetIcon, DocsIcon, UploadIcon } from 'shared/components/icons.mjs'
import { ModalWrapper } from 'shared/components/wrappers/modal.mjs' import { ModalWrapper } from 'shared/components/wrappers/modal.mjs'
import { isDegreeMeasurement } from 'config/measurements.mjs' import { isDegreeMeasurement } from 'config/measurements.mjs'
import { measurementAsMm, measurementAsUnits, parseDistanceInput } from 'shared/utils.mjs' import { measurementAsMm, measurementAsUnits, parseDistanceInput } from 'shared/utils.mjs'
import { Tabs, Tab } from 'shared/components/tabs.mjs'
export const ns = ['account', 'measurements', 'designs'] export const ns = ['account', 'measurements', 'designs']
/* /*
* Helper component to display a tab heading * Helper component to display a tab heading
*/ */
export const Tab = ({ export const _Tab = ({
id, // The tab ID id, // The tab ID
label, // A label for the tab, if not set we'll use the ID label, // A label for the tab, if not set we'll use the ID
activeTab, // Which tab (id) is active activeTab, // Which tab (id) is active
@ -427,15 +428,13 @@ export const MarkdownInput = ({
}) => { }) => {
const [activeTab, setActiveTab] = useState('edit') const [activeTab, setActiveTab] = useState('edit')
const tabs = ['edit', 'preview']
return ( return (
<FormControl {...{ label, labelBL, labelBR, docs }} forId={id}> <FormControl {...{ label, labelBL, labelBR, docs }} forId={id}>
<div className="tabs w-full"> <Tabs tabs={tabs}>
{['edit', 'preview'].map((tab) => ( <Tab key="edit">
<Tab id={tab} key={tab} label={tab} {...{ activeTab, setActiveTab }} />
))}
</div>
<div className="flex flex-row items-center mt-4"> <div className="flex flex-row items-center mt-4">
{activeTab === 'edit' ? (
<textarea <textarea
id={id} id={id}
rows="5" rows="5"
@ -444,12 +443,14 @@ export const MarkdownInput = ({
placeholder={placeholder} placeholder={placeholder}
onChange={(evt) => update(evt.target.value)} onChange={(evt) => update(evt.target.value)}
/> />
) : ( </div>
<div className="text-left px-4 border w-full"> </Tab>
<Tab key="preview">
<div className="flex flex-row items-center mt-4">
<Markdown>{current}</Markdown> <Markdown>{current}</Markdown>
</div> </div>
)} </Tab>
</div> </Tabs>
</FormControl> </FormControl>
) )
} }

View file

@ -3,7 +3,7 @@ import { Highlight } from './highlight.mjs'
import { YouTube } from './youtube.mjs' import { YouTube } from './youtube.mjs'
//import { Figure } from './figure.mjs' //import { Figure } from './figure.mjs'
import { ReadMore } from './read-more.mjs' import { ReadMore } from './read-more.mjs'
import { Tab, Tabs } from './tabs.mjs' import { Tab, Tabs } from '../tabs.mjs'
import { TabbedExample as Example } from './tabbed-example.mjs' import { TabbedExample as Example } from './tabbed-example.mjs'
import { HttpMethod, HttpStatusCode } from './http.mjs' import { HttpMethod, HttpStatusCode } from './http.mjs'
import { ControlTip } from '../control/tip.mjs' import { ControlTip } from '../control/tip.mjs'

View file

@ -1,4 +1,4 @@
import { Tab, Tabs } from './tabs.mjs' import { Tab, Tabs } from '../tabs.mjs'
import Md from 'react-markdown' import Md from 'react-markdown'
import { pluginBundle } from '@freesewing/plugin-bundle' import { pluginBundle } from '@freesewing/plugin-bundle'
import { pluginFlip } from '@freesewing/plugin-flip' import { pluginFlip } from '@freesewing/plugin-flip'

View file

@ -4,8 +4,12 @@ export const Tabs = ({ tabs = '', active = 0, children }) => {
// Keep active tab in state // Keep active tab in state
const [activeTab, setActiveTab] = useState(active) const [activeTab, setActiveTab] = useState(active)
// Parse tab list /*
const tablist = tabs.split(',').map((tab) => tab.trim()) * Parse tab list
* Comma-seperated tabs passed as a string are how it works in MDX
*/
const tablist = Array.isArray(tabs) ? tabs : tabs.split(',').map((tab) => tab.trim())
if (!tablist) return null if (!tablist) return null
// Pass down activeTab and tabId for conditional rendering // Pass down activeTab and tabId for conditional rendering
@ -19,7 +23,7 @@ export const Tabs = ({ tabs = '', active = 0, children }) => {
{tablist.map((title, tabId) => ( {tablist.map((title, tabId) => (
<button <button
key={tabId} key={tabId}
className={`text-xl font-bold capitalize tab h-auto tab-bordered grow ${ className={`text-lg font-bold capitalize tab h-auto tab-bordered grow py-2 ${
activeTab === tabId ? 'tab-active' : '' activeTab === tabId ? 'tab-active' : ''
}`} }`}
onClick={() => setActiveTab(tabId)} onClick={() => setActiveTab(tabId)}

View file

@ -1,6 +1,6 @@
import Markdown from 'react-markdown' import Markdown from 'react-markdown'
import { formatMm } from 'shared/utils.mjs' import { formatMm } from 'shared/utils.mjs'
import { Tab, Tabs } from '../mdx/tabs.mjs' import { Tab, Tabs } from '../tabs.mjs'
export const Error = ({ err }) => { export const Error = ({ err }) => {
// Include the error name and message info if it isn't already at the top // Include the error name and message info if it isn't already at the top

View file

@ -7,7 +7,7 @@ import { useTranslation } from 'next-i18next'
import { useLoadingStatus } from 'shared/hooks/use-loading-status.mjs' import { useLoadingStatus } from 'shared/hooks/use-loading-status.mjs'
// Components // Components
import { SetPicker, ns as setsNs } from 'shared/components/account/sets.mjs' import { SetPicker, ns as setsNs } from 'shared/components/account/sets.mjs'
import { Tabs, Tab } from 'shared/components/mdx/tabs.mjs' import { Tabs, Tab } from 'shared/components/tabs.mjs'
import { MeasiesEditor } from './editor.mjs' import { MeasiesEditor } from './editor.mjs'
import { Popout } from 'shared/components/popout/index.mjs' import { Popout } from 'shared/components/popout/index.mjs'
@ -33,7 +33,8 @@ export const MeasiesView = ({ design, Design, settings, update, missingMeasureme
return ( return (
<div className="max-w-7xl mx-auto my-6"> <div className="max-w-7xl mx-auto my-6">
<LoadingStatus /> <LoadingStatus />
<h1 className="m-auto text-center">{t('account:measurements')}</h1> <div className="max-w-xl m-auto">
<h2>{t('account:measurements')}</h2>
{missingMeasurements ? ( {missingMeasurements ? (
<Popout note dense noP> <Popout note dense noP>
<h5>{t('weLackSomeMeasies', { nr: missingMeasurements.length })}</h5> <h5>{t('weLackSomeMeasies', { nr: missingMeasurements.length })}</h5>
@ -49,6 +50,7 @@ export const MeasiesView = ({ design, Design, settings, update, missingMeasureme
<span className="text-lg">{t('measiesOk')}</span> <span className="text-lg">{t('measiesOk')}</span>
</Popout> </Popout>
)} )}
</div>
<Tabs tabs={tabs}> <Tabs tabs={tabs}>
<Tab key="choose"> <Tab key="choose">
<SetPicker design={design} clickHandler={loadMeasurements} /> <SetPicker design={design} clickHandler={loadMeasurements} />