import global from 'global'; import React, { FC, useMemo, ComponentProps, useCallback, forwardRef } from 'react'; import { Icons, WithTooltip, Spaced, TooltipLinkList } from '@storybook/components'; import { styled } from '@storybook/theming'; import { transparentize } from 'polished'; import { useStorybookApi } from '@storybook/api'; import { MenuItemIcon } from './Menu'; import { RefType } from './types'; import { getStateType } from './utils'; const { document, window: globalWindow } = global; export type ClickHandler = ComponentProps['links'][number]['onClick']; export interface IndicatorIconProps { type: ReturnType; } export interface CurrentVersionProps { url: string; versions: RefType['versions']; } const IndicatorPlacement = styled.aside(({ theme }) => ({ height: 16, display: 'flex', alignItems: 'center', '& > * + *': { marginLeft: theme.layoutMargin, }, })); const IndicatorClickTarget = styled.button(({ theme }) => ({ height: 20, width: 20, padding: 0, margin: 0, display: 'flex', alignItems: 'center', justifyContent: 'center', background: 'transparent', outline: 'none', border: '1px solid transparent', borderRadius: '100%', cursor: 'pointer', color: theme.base === 'light' ? transparentize(0.3, theme.color.defaultText) : transparentize(0.6, theme.color.defaultText), '&:hover': { color: theme.barSelectedColor, }, '&:focus': { color: theme.barSelectedColor, borderColor: theme.color.secondary, }, svg: { height: 10, width: 10, transition: 'all 150ms ease-out', color: 'inherit', }, })); const MessageTitle = styled.span(({ theme }) => ({ fontWeight: theme.typography.weight.bold, })); const Message = styled.a(({ theme }) => ({ textDecoration: 'none', lineHeight: '16px', padding: 15, display: 'flex', flexDirection: 'row', alignItems: 'flex-start', color: theme.color.defaultText, '&:not(:last-child)': { borderBottom: `1px solid ${theme.appBorderColor}`, }, '&:hover': { background: theme.background.hoverable, color: theme.color.darker, }, '&:link': { color: theme.color.darker, }, '&:active': { color: theme.color.darker, }, '&:focus': { color: theme.color.darker, }, '& > *': { flex: 1, }, '& > svg': { marginTop: 3, width: 16, height: 16, marginRight: 10, flex: 'unset', }, })); export const MessageWrapper = styled.div({ width: 280, boxSizing: 'border-box', borderRadius: 8, overflow: 'hidden', }); const BlueIcon = styled(Icons)(({ theme }) => ({ color: theme.color.secondary, })); const YellowIcon = styled(Icons)(({ theme }) => ({ color: theme.color.gold, })); const RedIcon = styled(Icons)(({ theme }) => ({ color: theme.color.negative, })); const GreenIcon = styled(Icons)(({ theme }) => ({ color: theme.color.green, })); const Version = styled.div(({ theme }) => ({ display: 'flex', alignItems: 'center', fontSize: theme.typography.size.s1, fontWeight: theme.typography.weight.regular, color: theme.base === 'light' ? transparentize(0.3, theme.color.defaultText) : transparentize(0.6, theme.color.defaultText), '& > * + *': { marginLeft: 4, }, svg: { height: 10, width: 10, }, })); const CurrentVersion: FC = ({ url, versions }) => { const currentVersionId = useMemo(() => { const c = Object.entries(versions).find(([k, v]) => v === url); return c && c[0] ? c[0] : 'current'; }, [url, versions]); return ( {currentVersionId} ); }; export const RefIndicator = React.memo( forwardRef }>( ({ state, ...ref }, forwardedRef) => { const api = useStorybookApi(); const list = useMemo(() => Object.values(ref.stories || {}), [ref.stories]); const componentCount = useMemo( () => list.filter((v) => v.type === 'component').length, [list] ); const leafCount = useMemo( () => list.filter((v) => v.type === 'docs' || v.type === 'story').length, [list] ); const changeVersion = useCallback( ((event, item) => { event.preventDefault(); api.changeRefVersion(ref.id, item.href); }) as ClickHandler, [] ); return ( {state === 'loading' && } {(state === 'error' || state === 'empty') && ( )} {state === 'ready' && ( )} {state === 'auth' && } {ref.type === 'auto-inject' && state !== 'error' && ( )} {state !== 'loading' && } } > {ref.versions && Object.keys(ref.versions).length ? ( ({ left: href === ref.url ? : , id, title: id, href, onClick: changeVersion, }))} /> } > ) : null} ); } ) ); const ReadyMessage: FC<{ url: string; componentCount: number; leafCount: number; }> = ({ url, componentCount, leafCount }) => (
View external Storybook
Explore {componentCount} components and {leafCount} stories in a new browser tab.
); const LoginRequiredMessage: FC = ({ loginUrl, id }) => { const open = useCallback((e) => { e.preventDefault(); const childWindow = globalWindow.open(loginUrl, `storybook_auth_${id}`, 'resizable,scrollbars'); // poll for window to close const timer = setInterval(() => { if (!childWindow) { clearInterval(timer); } else if (childWindow.closed) { clearInterval(timer); document.location.reload(); } }, 1000); }, []); return (
Log in required
You need to authenticate to view this Storybook's components.
); }; const ReadDocsMessage: FC = () => (
Read Composition docs
Learn how to combine multiple Storybooks into one.
); const ErrorOccurredMessage: FC<{ url: string }> = ({ url }) => (
Something went wrong
This external Storybook didn't load. Debug it in a new tab now.
); const LoadingMessage: FC<{ url: string }> = ({ url }) => (
Please wait
This Storybook is loading.
); const PerformanceDegradedMessage: FC = () => (
Reduce lag
Learn how to speed up Composition performance.
);