-
-
Notifications
You must be signed in to change notification settings - Fork 1.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[docs-infra] Type interface API pages #14138
Changes from all commits
39899ee
c0b2091
e3edf37
9f5c0d9
9ae0301
a888a54
86cab09
2616d37
f7775cd
942c6b6
c398094
3b03078
2e5669e
3cafec7
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,33 +6,37 @@ import Typography from '@mui/material/Typography'; | |
import Alert from '@mui/material/Alert'; | ||
import VerifiedRoundedIcon from '@mui/icons-material/VerifiedRounded'; | ||
import { alpha } from '@mui/material/styles'; | ||
import { useTranslate, useUserLanguage } from '@mui/docs/i18n'; | ||
import { Translate, useTranslate, useUserLanguage } from '@mui/docs/i18n'; | ||
import { HighlightedCode } from '@mui/docs/HighlightedCode'; | ||
import { MarkdownElement } from '@mui/docs/MarkdownElement'; | ||
import { SectionTitle } from '@mui/docs/SectionTitle'; | ||
import { SectionTitle, SectionTitleProps } from '@mui/docs/SectionTitle'; | ||
import AppLayoutDocs from 'docs/src/modules/components/AppLayoutDocs'; | ||
import PropertiesSection from 'docs/src/modules/components/ApiPage/sections/PropertiesSection'; | ||
import { DEFAULT_API_LAYOUT_STORAGE_KEYS } from 'docs/src/modules/components/ApiPage/sections/ToggleDisplayOption'; | ||
import { PropertyDefinition } from 'docs/src/modules/components/ApiPage/definitions/properties'; | ||
import { LayoutStorageKeys } from 'docs/src/modules/components/ApiPage'; | ||
import { | ||
DEFAULT_API_LAYOUT_STORAGE_KEYS, | ||
ApiDisplayOptions, | ||
} from 'docs/src/modules/components/ApiPage/sections/ToggleDisplayOption'; | ||
import { | ||
InterfaceApiTranslation, | ||
InterfaceApiContent, | ||
} from 'docsx/scripts/api/buildInterfacesDocumentation'; | ||
import { TableOfContentsEntry } from '@mui/internal-markdown'; | ||
import kebabCase from 'lodash/kebabCase'; | ||
|
||
export function getTranslatedHeader(t, header) { | ||
type HeaderHash = 'demos' | 'import'; | ||
|
||
export function getTranslatedHeader(t: Translate, header: HeaderHash) { | ||
const translations = { | ||
demos: t('api-docs.demos'), | ||
import: t('api-docs.import'), | ||
}; | ||
|
||
// TODO Drop runtime type-checking once we type-check this file | ||
if (!translations.hasOwnProperty(header)) { | ||
throw new TypeError( | ||
`Unable to translate header '${header}'. Did you mean one of '${Object.keys( | ||
translations, | ||
).join("', '")}'`, | ||
); | ||
} | ||
|
||
return translations[header] || header; | ||
} | ||
|
||
function Heading(props) { | ||
function Heading(props: Pick<SectionTitleProps<HeaderHash>, 'hash' | 'level'>) { | ||
const { hash, level = 'h2' } = props; | ||
const t = useTranslate(); | ||
|
||
|
@@ -44,7 +48,62 @@ Heading.propTypes = { | |
level: PropTypes.string, | ||
}; | ||
|
||
export default function ApiPage(props) { | ||
interface ApiPageProps { | ||
descriptions: { | ||
[lang: string]: InterfaceApiTranslation & { | ||
// Table of Content added by the mapApiPageTranslations function | ||
componentDescriptionToc: TableOfContentsEntry[]; | ||
}; | ||
}; | ||
pageContent: InterfaceApiContent; | ||
defaultLayout?: ApiDisplayOptions; | ||
/** | ||
* The localStorage key used to save the user layout for each section. | ||
* It's useful to dave different preferences on different pages. | ||
* For example, the data grid has a different key that the core. | ||
*/ | ||
layoutStorageKey?: LayoutStorageKeys; | ||
} | ||
|
||
interface GetInterfaceApiDefinitionsParams { | ||
interfaceName: string; | ||
properties: InterfaceApiContent['properties']; | ||
propertiesDescriptions: InterfaceApiTranslation['propertiesDescriptions']; | ||
/** | ||
* Add indicators that the properties is optional instead of showing it is required. | ||
*/ | ||
showOptionalAbbr?: boolean; | ||
} | ||
|
||
export function getInterfaceApiDefinitions( | ||
params: GetInterfaceApiDefinitionsParams, | ||
): PropertyDefinition[] { | ||
const { properties, propertiesDescriptions, interfaceName, showOptionalAbbr = false } = params; | ||
|
||
return Object.entries(properties).map(([propertyName, propertyData]) => { | ||
const isRequired = propertyData.required && !showOptionalAbbr; | ||
const isOptional = !propertyData.required && showOptionalAbbr; | ||
|
||
const typeName = propertyData.type.description; | ||
const propDefault = propertyData.default; | ||
const propDescription = propertiesDescriptions[propertyName]; | ||
|
||
return { | ||
propName: propertyName, | ||
hash: `${kebabCase(interfaceName)}-prop-${propertyName}`, | ||
propertyName, | ||
description: propDescription?.description, | ||
isOptional, | ||
isRequired, | ||
typeName, | ||
propDefault, | ||
isProPlan: propertyData.isProPlan, | ||
isPremiumPlan: propertyData.isPremiumPlan, | ||
}; | ||
}); | ||
} | ||
|
||
export default function ApiPage(props: ApiPageProps) { | ||
const { | ||
descriptions, | ||
pageContent, | ||
|
@@ -54,21 +113,17 @@ export default function ApiPage(props) { | |
const t = useTranslate(); | ||
const userLanguage = useUserLanguage(); | ||
|
||
const { demos, filename = '', properties } = pageContent; | ||
const { demos, properties } = pageContent; | ||
|
||
const { componentDescription, propertiesDescriptions, interfaceDescription } = | ||
descriptions[userLanguage]; | ||
const description = t('api-docs.pageDescription').replace(/{{name}}/, pageContent.name); | ||
|
||
// Prefer linking the .tsx or .d.ts for the "Edit this page" link. | ||
const apiSourceLocation = filename.replace('.js', '.d.ts'); | ||
const { propertiesDescriptions, interfaceDescription } = descriptions[userLanguage]; | ||
const description = t('api-docs.interfacePageDescription').replace(/{{name}}/, pageContent.name); | ||
|
||
return ( | ||
<AppLayoutDocs | ||
description={description} | ||
disableToc={false} | ||
toc={[]} | ||
location={apiSourceLocation} | ||
location="" | ||
title={`${pageContent.name} API`} | ||
disableAd | ||
> | ||
|
@@ -79,7 +134,7 @@ export default function ApiPage(props) { | |
component="p" | ||
className="description" | ||
gutterBottom | ||
dangerouslySetInnerHTML={{ __html: interfaceDescription }} | ||
dangerouslySetInnerHTML={{ __html: description }} | ||
/> | ||
<Heading hash="demos" /> | ||
{demos && ( | ||
|
@@ -131,26 +186,29 @@ export default function ApiPage(props) { | |
language="jsx" | ||
/> | ||
|
||
{componentDescription ? ( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Wondering what There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, component descript does not exist for interfaces. It has <Typography
variant="h5"
component="p"
className="description"
gutterBottom
dangerouslySetInnerHTML={{ __html: interfaceDescription }}
/> It's visible for example in GridActionsColDef API I replace the usual API page description
But that could probably be improved. I will check with other X memebers what they want for this page |
||
{interfaceDescription ? ( | ||
<React.Fragment> | ||
<br /> | ||
<br /> | ||
<span | ||
dangerouslySetInnerHTML={{ | ||
__html: componentDescription, | ||
__html: interfaceDescription, | ||
}} | ||
/> | ||
</React.Fragment> | ||
) : null} | ||
|
||
<PropertiesSection | ||
properties={properties} | ||
propertiesDescriptions={propertiesDescriptions} | ||
componentName={pageContent.name} | ||
properties={getInterfaceApiDefinitions({ | ||
propertiesDescriptions, | ||
properties, | ||
interfaceName: pageContent.name, | ||
showOptionalAbbr: true, | ||
})} | ||
title="api-docs.properties" | ||
titleHash="properties" | ||
defaultLayout={defaultLayout} | ||
layoutStorageKey={layoutStorageKey.props} | ||
showOptionalAbbr | ||
/> | ||
</MarkdownElement> | ||
<svg style={{ display: 'none' }} xmlns="http://www.w3.org/2000/svg"> | ||
|
@@ -162,15 +220,13 @@ export default function ApiPage(props) { | |
); | ||
} | ||
|
||
ApiPage.propTypes = { | ||
defaultLayout: PropTypes.oneOf(['collapsed', 'expanded', 'table']), | ||
descriptions: PropTypes.object.isRequired, | ||
layoutStorageKey: PropTypes.shape({ | ||
props: PropTypes.string, | ||
}), | ||
pageContent: PropTypes.object.isRequired, | ||
}; | ||
|
||
if (process.env.NODE_ENV !== 'production') { | ||
ApiPage.propTypes = exactProp(ApiPage.propTypes); | ||
ApiPage.propTypes = exactProp({ | ||
defaultLayout: PropTypes.oneOf(['collapsed', 'expanded', 'table']), | ||
descriptions: PropTypes.object.isRequired, | ||
layoutStorageKey: PropTypes.shape({ | ||
props: PropTypes.string, | ||
}), | ||
pageContent: PropTypes.object.isRequired, | ||
}); | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is there a reason not to have a
plan?: 'pro' | 'premium' | 'community'
here?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No other reason than "that's what the script returns"
I tried to do the update, but that would require a PR on the core repo to update properties section component
https://github.com/mui/material-ui/blob/master/docs/src/modules/components/ApiPage/sections/PropertiesSection.tsx/#L66-L69
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would probably make sense to update it at some point, but it's out of the scope of this PR then 👍