chore(shared): Consolidate tab components
This commit is contained in:
parent
44a3b194cc
commit
c98a67885c
6 changed files with 50 additions and 43 deletions
|
@ -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,29 +428,29 @@ 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 className="flex flex-row items-center mt-4">
|
||||||
))}
|
<textarea
|
||||||
</div>
|
id={id}
|
||||||
<div className="flex flex-row items-center mt-4">
|
rows="5"
|
||||||
{activeTab === 'edit' ? (
|
className="textarea textarea-bordered textarea-lg w-full"
|
||||||
<textarea
|
value={current}
|
||||||
id={id}
|
placeholder={placeholder}
|
||||||
rows="5"
|
onChange={(evt) => update(evt.target.value)}
|
||||||
className="textarea textarea-bordered textarea-lg w-full"
|
/>
|
||||||
value={current}
|
</div>
|
||||||
placeholder={placeholder}
|
</Tab>
|
||||||
onChange={(evt) => update(evt.target.value)}
|
<Tab key="preview">
|
||||||
/>
|
<div className="flex flex-row items-center mt-4">
|
||||||
) : (
|
|
||||||
<div className="text-left px-4 border w-full">
|
|
||||||
<Markdown>{current}</Markdown>
|
<Markdown>{current}</Markdown>
|
||||||
</div>
|
</div>
|
||||||
)}
|
</Tab>
|
||||||
</div>
|
</Tabs>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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'
|
||||||
|
|
|
@ -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'
|
||||||
|
|
|
@ -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)}
|
|
@ -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
|
||||||
|
|
|
@ -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,22 +33,24 @@ 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">
|
||||||
{missingMeasurements ? (
|
<h2>{t('account:measurements')}</h2>
|
||||||
<Popout note dense noP>
|
{missingMeasurements ? (
|
||||||
<h5>{t('weLackSomeMeasies', { nr: missingMeasurements.length })}</h5>
|
<Popout note dense noP>
|
||||||
<ol className="list list-inside ml-4 list-decimal">
|
<h5>{t('weLackSomeMeasies', { nr: missingMeasurements.length })}</h5>
|
||||||
{missingMeasurements.map((m, i) => (
|
<ol className="list list-inside ml-4 list-decimal">
|
||||||
<li key={i}>{t(`measurements:${m}`)}</li>
|
{missingMeasurements.map((m, i) => (
|
||||||
))}
|
<li key={i}>{t(`measurements:${m}`)}</li>
|
||||||
</ol>
|
))}
|
||||||
<p className="text-lg">{t('youCanPickOrEnter')}</p>
|
</ol>
|
||||||
</Popout>
|
<p className="text-lg">{t('youCanPickOrEnter')}</p>
|
||||||
) : (
|
</Popout>
|
||||||
<Popout note compact ense oP>
|
) : (
|
||||||
<span className="text-lg">{t('measiesOk')}</span>
|
<Popout note compact ense oP>
|
||||||
</Popout>
|
<span className="text-lg">{t('measiesOk')}</span>
|
||||||
)}
|
</Popout>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
<Tabs tabs={tabs}>
|
<Tabs tabs={tabs}>
|
||||||
<Tab key="choose">
|
<Tab key="choose">
|
||||||
<SetPicker design={design} clickHandler={loadMeasurements} />
|
<SetPicker design={design} clickHandler={loadMeasurements} />
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue