diff --git a/addons/interactions/src/Panel.stories.tsx b/addons/interactions/src/Panel.stories.tsx new file mode 100644 index 00000000000..53a3bc7504b --- /dev/null +++ b/addons/interactions/src/Panel.stories.tsx @@ -0,0 +1,88 @@ +import React from 'react'; +import { ComponentStoryObj, ComponentMeta } from '@storybook/react'; +import { CallStates } from '@storybook/instrumenter'; +import { styled } from '@storybook/theming'; + +import { getCall } from './mocks'; +import { AddonPanelPure } from './Panel'; + +const StyledWrapper = styled.div(({ theme }) => ({ + backgroundColor: theme.background.content, + color: theme.color.defaultText, + display: 'block', + height: '100%', + position: 'absolute', + left: 0, + right: 0, + bottom: 0, + overflow: 'auto', +})); + +export default { + title: 'Addons/Interactions/Panel', + component: AddonPanelPure, + decorators: [ + (Story: any) => ( + + + + ), + ], + parameters: { + layout: 'fullscreen', + }, + args: { + calls: new Map(), + endRef: null, + fileName: 'addon-interactions.stories.tsx', + hasException: false, + hasNext: false, + hasPrevious: true, + interactions: [getCall(CallStates.DONE)], + isDisabled: false, + isPlaying: false, + showTabIcon: false, + isDebuggingEnabled: true, + // prop for the AddonPanel used as wrapper of Panel + active: true, + }, +} as ComponentMeta; + +type Story = ComponentStoryObj; + +export const Passing: Story = { + args: { + interactions: [getCall(CallStates.DONE)], + }, +}; + +export const Paused: Story = { + args: { + isPlaying: true, + interactions: [getCall(CallStates.WAITING)], + }, +}; + +export const Playing: Story = { + args: { + isPlaying: true, + interactions: [getCall(CallStates.ACTIVE)], + }, +}; + +export const Failed: Story = { + args: { + hasException: true, + interactions: [getCall(CallStates.ERROR)], + }, +}; + +export const WithDebuggingDisabled: Story = { + args: { isDebuggingEnabled: false }, +}; + +export const NoInteractions: Story = { + args: { + interactions: [], + }, +}; diff --git a/addons/interactions/src/Panel.tsx b/addons/interactions/src/Panel.tsx index bb8b9293b4c..266a9212b80 100644 --- a/addons/interactions/src/Panel.tsx +++ b/addons/interactions/src/Panel.tsx @@ -6,14 +6,38 @@ import { STORY_RENDER_PHASE_CHANGED } from '@storybook/core-events'; import { AddonPanel, Link, Placeholder } from '@storybook/components'; import { EVENTS, Call, CallStates, LogItem } from '@storybook/instrumenter'; import { styled } from '@storybook/theming'; + import { StatusIcon } from './components/StatusIcon/StatusIcon'; import { Subnav } from './components/Subnav/Subnav'; import { Interaction } from './components/Interaction/Interaction'; -interface PanelProps { +const { FEATURES } = global; + +interface AddonPanelProps { active: boolean; } +interface InteractionsPanelProps { + active: boolean; + showTabIcon?: boolean; + interactions: (Call & { state?: CallStates })[]; + isDisabled?: boolean; + hasPrevious?: boolean; + hasNext?: boolean; + fileName?: string; + hasException?: boolean; + isPlaying?: boolean; + calls: Map; + endRef?: React.Ref; + isDebuggingEnabled?: boolean; + onStart?: () => void; + onPrevious?: () => void; + onNext?: () => void; + onEnd?: () => void; + onScrollToEnd?: () => void; + onInteractionClick?: (callId: string) => void; +} + const pendingStates = [CallStates.ACTIVE, CallStates.WAITING]; const completedStates = [CallStates.DONE, CallStates.ERROR]; @@ -21,7 +45,80 @@ const TabIcon = styled(StatusIcon)({ marginLeft: 5, }); -export const Panel: React.FC = (props) => { +export const AddonPanelPure: React.FC = React.memo( + ({ + showTabIcon, + interactions, + isDisabled, + hasPrevious, + hasNext, + fileName, + hasException, + isPlaying, + onStart, + onPrevious, + onNext, + onEnd, + onScrollToEnd, + calls, + onInteractionClick, + endRef, + isDebuggingEnabled, + ...panelProps + }) => { + return ( + + {showTabIcon && + ReactDOM.createPortal( + , + global.document.getElementById('tabbutton-interactions') + )} + {isDebuggingEnabled && interactions.length > 0 && ( + + )} + {interactions.map((call) => ( + onInteractionClick(call.id)} + /> + ))} +
+ {!isPlaying && interactions.length === 0 && ( + + No interactions found + + Learn how to add interactions to your story + + + )} + + ); + } +); + +export const Panel: React.FC = (props) => { const [isLocked, setLock] = React.useState(false); const [isPlaying, setPlaying] = React.useState(true); const [scrollTarget, setScrollTarget] = React.useState(); @@ -57,57 +154,48 @@ export const Panel: React.FC = (props) => { const [fileName] = storyFilePath.toString().split('/').slice(-1); const scrollToTarget = () => scrollTarget?.scrollIntoView({ behavior: 'smooth', block: 'end' }); + const isDebuggingEnabled = FEATURES.interactionsDebugger === true; + const isDebugging = log.some((item) => pendingStates.includes(item.state)); const hasPrevious = log.some((item) => completedStates.includes(item.state)); const hasNext = log.some((item) => item.state === CallStates.WAITING); const hasActive = log.some((item) => item.state === CallStates.ACTIVE); const hasException = log.some((item) => item.state === CallStates.ERROR); - const isDisabled = hasActive || isLocked || (isPlaying && !isDebugging); + const isDisabled = isDebuggingEnabled + ? hasActive || isLocked || (isPlaying && !isDebugging) + : true; - const tabButton = global.document.getElementById('tabbutton-interactions'); - const tabStatus = hasException ? CallStates.ERROR : CallStates.ACTIVE; const showTabIcon = isDebugging || (!isPlaying && hasException); + const onStart = React.useCallback(() => emit(EVENTS.START, { storyId }), [storyId]); + const onPrevious = React.useCallback(() => emit(EVENTS.BACK, { storyId }), [storyId]); + const onNext = React.useCallback(() => emit(EVENTS.NEXT, { storyId }), [storyId]); + const onEnd = React.useCallback(() => emit(EVENTS.END, { storyId }), [storyId]); + const onInteractionClick = React.useCallback( + (callId: string) => emit(EVENTS.GOTO, { storyId, callId }), + [storyId] + ); + return ( - - {tabButton && showTabIcon && ReactDOM.createPortal(, tabButton)} - {interactions.length > 0 && ( - emit(EVENTS.START, { storyId })} - onPrevious={() => emit(EVENTS.BACK, { storyId })} - onNext={() => emit(EVENTS.NEXT, { storyId })} - onEnd={() => emit(EVENTS.END, { storyId })} - onScrollToEnd={scrollTarget && scrollToTarget} - /> - )} - {interactions.map((call) => ( - emit(EVENTS.GOTO, { storyId, callId: call.id })} - isDisabled={isDisabled} - /> - ))} -
- {!isPlaying && interactions.length === 0 && ( - - No interactions found - - Learn how to add interactions to your story - - - )} - + ); }; diff --git a/addons/interactions/src/components/Interaction/Interaction.stories.tsx b/addons/interactions/src/components/Interaction/Interaction.stories.tsx index 8c1a2ed4042..751841e7c76 100644 --- a/addons/interactions/src/components/Interaction/Interaction.stories.tsx +++ b/addons/interactions/src/components/Interaction/Interaction.stories.tsx @@ -1,7 +1,8 @@ import { ComponentStoryObj, ComponentMeta } from '@storybook/react'; import { expect } from '@storybook/jest'; -import { Call, CallStates } from '@storybook/instrumenter'; +import { CallStates } from '@storybook/instrumenter'; import { userEvent, within } from '@storybook/testing-library'; +import { getCall } from '../../mocks'; import { Interaction } from './Interaction'; @@ -13,59 +14,31 @@ export default { args: { callsById: new Map(), isDisabled: false, + isDebuggingEnabled: true, }, } as ComponentMeta; -const getCallMock = (state: CallStates): Call => { - const defaultData = { - id: 'addons-interactions-accountform--standard-email-filled [3] change', - path: ['fireEvent'], - method: 'change', - storyId: 'addons-interactions-accountform--standard-email-filled', - args: [ - { - __callId__: 'addons-interactions-accountform--standard-email-filled [2] getByTestId', - retain: false, - }, - { - target: { - value: 'michael@chromatic.com', - }, - }, - ], - interceptable: true, - retain: false, - state, - }; - - const overrides = CallStates.ERROR - ? { exception: { callId: '', stack: '', message: "Things didn't work!" } } - : {}; - - return { ...defaultData, ...overrides }; -}; - export const Active: Story = { args: { - call: getCallMock(CallStates.ACTIVE), + call: getCall(CallStates.ACTIVE), }, }; export const Waiting: Story = { args: { - call: getCallMock(CallStates.WAITING), + call: getCall(CallStates.WAITING), }, }; export const Failed: Story = { args: { - call: getCallMock(CallStates.ERROR), + call: getCall(CallStates.ERROR), }, }; export const Done: Story = { args: { - call: getCallMock(CallStates.DONE), + call: getCall(CallStates.DONE), }, }; diff --git a/addons/interactions/src/components/Interaction/Interaction.tsx b/addons/interactions/src/components/Interaction/Interaction.tsx index c2bfe896488..f9a9dc366d3 100644 --- a/addons/interactions/src/components/Interaction/Interaction.tsx +++ b/addons/interactions/src/components/Interaction/Interaction.tsx @@ -65,11 +65,13 @@ export const Interaction = ({ callsById, onClick, isDisabled, + isDebuggingEnabled, }: { call: Call; callsById: Map; onClick: React.MouseEventHandler; isDisabled: boolean; + isDebuggingEnabled?: boolean; }) => { const [isHovered, setIsHovered] = React.useState(false); return ( @@ -77,9 +79,9 @@ export const Interaction = ({ setIsHovered(true)} - onMouseLeave={() => setIsHovered(false)} - disabled={isDisabled} + disabled={isDebuggingEnabled ? isDisabled : true} + onMouseEnter={() => isDebuggingEnabled && setIsHovered(true)} + onMouseLeave={() => isDebuggingEnabled && setIsHovered(false)} > diff --git a/addons/interactions/src/components/MatcherResult.stories.tsx b/addons/interactions/src/components/MatcherResult.stories.tsx index 80935e461d0..c81ac36eb20 100644 --- a/addons/interactions/src/components/MatcherResult.stories.tsx +++ b/addons/interactions/src/components/MatcherResult.stories.tsx @@ -22,7 +22,7 @@ export default { ), ], parameters: { - layout: 'fullscren', + layout: 'fullscreen', }, }; diff --git a/addons/interactions/src/components/MethodCall.stories.tsx b/addons/interactions/src/components/MethodCall.stories.tsx index 66cb61713a9..a2d09b71d6c 100644 --- a/addons/interactions/src/components/MethodCall.stories.tsx +++ b/addons/interactions/src/components/MethodCall.stories.tsx @@ -23,7 +23,7 @@ export default { ), ], parameters: { - layout: 'fullscren', + layout: 'fullscreen', }, }; diff --git a/addons/interactions/src/mocks/index.ts b/addons/interactions/src/mocks/index.ts new file mode 100644 index 00000000000..591fdd9cfbc --- /dev/null +++ b/addons/interactions/src/mocks/index.ts @@ -0,0 +1,30 @@ +import { CallStates, Call } from '@storybook/instrumenter'; + +export const getCall = (state: CallStates): Call => { + const defaultData = { + id: 'addons-interactions-accountform--standard-email-filled [3] change', + path: ['fireEvent'], + method: 'change', + storyId: 'addons-interactions-accountform--standard-email-filled', + args: [ + { + __callId__: 'addons-interactions-accountform--standard-email-filled [2] getByTestId', + retain: false, + }, + { + target: { + value: 'michael@chromatic.com', + }, + }, + ], + interceptable: true, + retain: false, + state, + }; + + const overrides = CallStates.ERROR + ? { exception: { callId: '', stack: '', message: "Things didn't work!" } } + : {}; + + return { ...defaultData, ...overrides }; +}; diff --git a/examples/official-storybook/main.ts b/examples/official-storybook/main.ts index c9553782464..d4861400a64 100644 --- a/examples/official-storybook/main.ts +++ b/examples/official-storybook/main.ts @@ -38,6 +38,7 @@ const config: StorybookConfig = { logLevel: 'debug', features: { modernInlineRender: true, + interactionsDebugger: true, }, }; diff --git a/examples/official-storybook/preview.js b/examples/official-storybook/preview.js index c5c2ff77a69..c2131c0ebc9 100644 --- a/examples/official-storybook/preview.js +++ b/examples/official-storybook/preview.js @@ -154,6 +154,7 @@ export const parameters = { restoreScroll: true, }, }, + actions: { argTypesRegex: '^on.*' }, options: { storySort: (a, b) => a[1].kind === b[1].kind ? 0 : a[1].id.localeCompare(b[1].id, undefined, { numeric: true }), diff --git a/lib/core-common/src/types.ts b/lib/core-common/src/types.ts index 4aeed0b0f06..12df892b75e 100644 --- a/lib/core-common/src/types.ts +++ b/lib/core-common/src/types.ts @@ -314,6 +314,11 @@ export interface StorybookConfig { */ breakingChangesV7?: boolean; + /** + * Enable the step debugger functionality in Addon-interactions. + */ + interactionsDebugger?: boolean; + /** * Use Storybook 7.0 babel config scheme */