mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 07:21:16 +08:00
153 lines
4.6 KiB
TypeScript
153 lines
4.6 KiB
TypeScript
import * as React from 'react';
|
|
import { AddonPanel, Link, Placeholder } from '@storybook/components';
|
|
import { Call, CallStates, ControlStates } from '@storybook/instrumenter';
|
|
import { styled } from '@storybook/theming';
|
|
|
|
import { Subnav } from './Subnav';
|
|
import { Interaction } from './Interaction';
|
|
|
|
export interface Controls {
|
|
start: (args: any) => void;
|
|
back: (args: any) => void;
|
|
goto: (args: any) => void;
|
|
next: (args: any) => void;
|
|
end: (args: any) => void;
|
|
rerun: (args: any) => void;
|
|
}
|
|
|
|
interface InteractionsPanelProps {
|
|
active: boolean;
|
|
controls: Controls;
|
|
controlStates: ControlStates;
|
|
interactions: (Call & {
|
|
status?: CallStates;
|
|
childCallIds: Call['id'][];
|
|
isCollapsed: boolean;
|
|
toggleCollapsed: () => void;
|
|
})[];
|
|
fileName?: string;
|
|
hasException?: boolean;
|
|
caughtException?: Error;
|
|
isPlaying?: boolean;
|
|
pausedAt?: Call['id'];
|
|
calls: Map<string, any>;
|
|
endRef?: React.Ref<HTMLDivElement>;
|
|
onScrollToEnd?: () => void;
|
|
isRerunAnimating: boolean;
|
|
setIsRerunAnimating: React.Dispatch<React.SetStateAction<boolean>>;
|
|
}
|
|
|
|
const Container = styled.div<{ withException: boolean }>(({ theme, withException }) => ({
|
|
minHeight: '100%',
|
|
background: withException ? theme.background.warning : theme.background.content,
|
|
}));
|
|
|
|
const CaughtException = styled.div(({ theme }) => ({
|
|
padding: 15,
|
|
fontSize: theme.typography.size.s2 - 1,
|
|
lineHeight: '19px',
|
|
}));
|
|
const CaughtExceptionCode = styled.code(({ theme }) => ({
|
|
margin: '0 1px',
|
|
padding: 3,
|
|
fontSize: theme.typography.size.s1 - 1,
|
|
lineHeight: 1,
|
|
verticalAlign: 'top',
|
|
background: 'rgba(0, 0, 0, 0.05)',
|
|
border: `1px solid ${theme.color.border}`,
|
|
borderRadius: 3,
|
|
}));
|
|
const CaughtExceptionTitle = styled.div({
|
|
paddingBottom: 4,
|
|
fontWeight: 'bold',
|
|
});
|
|
const CaughtExceptionDescription = styled.p({
|
|
margin: 0,
|
|
padding: '0 0 20px',
|
|
});
|
|
const CaughtExceptionStack = styled.pre(({ theme }) => ({
|
|
margin: 0,
|
|
padding: 0,
|
|
fontSize: theme.typography.size.s1 - 1,
|
|
}));
|
|
|
|
export const InteractionsPanel: React.FC<InteractionsPanelProps> = React.memo(
|
|
({
|
|
calls,
|
|
controls,
|
|
controlStates,
|
|
interactions,
|
|
fileName,
|
|
hasException,
|
|
caughtException,
|
|
isPlaying,
|
|
pausedAt,
|
|
onScrollToEnd,
|
|
endRef,
|
|
isRerunAnimating,
|
|
setIsRerunAnimating,
|
|
...panelProps
|
|
}) => (
|
|
<AddonPanel {...panelProps}>
|
|
<Container withException={!!caughtException}>
|
|
{controlStates.debugger && (interactions.length > 0 || hasException) && (
|
|
<Subnav
|
|
controls={controls}
|
|
controlStates={controlStates}
|
|
status={
|
|
// eslint-disable-next-line no-nested-ternary
|
|
isPlaying ? CallStates.ACTIVE : hasException ? CallStates.ERROR : CallStates.DONE
|
|
}
|
|
storyFileName={fileName}
|
|
onScrollToEnd={onScrollToEnd}
|
|
isRerunAnimating={isRerunAnimating}
|
|
setIsRerunAnimating={setIsRerunAnimating}
|
|
/>
|
|
)}
|
|
<div>
|
|
{interactions.map((call) => (
|
|
<Interaction
|
|
key={call.id}
|
|
call={call}
|
|
callsById={calls}
|
|
controls={controls}
|
|
controlStates={controlStates}
|
|
childCallIds={call.childCallIds}
|
|
isCollapsed={call.isCollapsed}
|
|
toggleCollapsed={call.toggleCollapsed}
|
|
pausedAt={pausedAt}
|
|
/>
|
|
))}
|
|
</div>
|
|
{caughtException && !caughtException.message?.startsWith('ignoredException') && (
|
|
<CaughtException>
|
|
<CaughtExceptionTitle>
|
|
Caught exception in <CaughtExceptionCode>play</CaughtExceptionCode> function
|
|
</CaughtExceptionTitle>
|
|
<CaughtExceptionDescription>
|
|
This story threw an error after it finished rendering which means your interactions
|
|
couldn't be run. Go to this story's play function in {fileName} to fix.
|
|
</CaughtExceptionDescription>
|
|
<CaughtExceptionStack>
|
|
{caughtException.stack || `${caughtException.name}: ${caughtException.message}`}
|
|
</CaughtExceptionStack>
|
|
</CaughtException>
|
|
)}
|
|
<div ref={endRef} />
|
|
{!isPlaying && !caughtException && interactions.length === 0 && (
|
|
<Placeholder>
|
|
No interactions found
|
|
<Link
|
|
href="https://storybook.js.org/docs/react/writing-stories/play-function"
|
|
target="_blank"
|
|
withArrow
|
|
>
|
|
Learn how to add interactions to your story
|
|
</Link>
|
|
</Placeholder>
|
|
)}
|
|
</Container>
|
|
</AddonPanel>
|
|
)
|
|
);
|