From 517a4627dcff8e1fd3008cbdb5c0369503851f0d Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 17:00:02 +0200 Subject: [PATCH 1/9] chore(official-storybook): add argTypesRegex to preview --- examples/official-storybook/preview.js | 1 + 1 file changed, 1 insertion(+) 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 }), From 46af46efdfcdeba4a306eeb95926899d76a55107 Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 17:01:00 +0200 Subject: [PATCH 2/9] feat(core): add interactions debugging feature flag --- examples/official-storybook/main.ts | 1 + lib/core-common/src/types.ts | 5 +++++ 2 files changed, 6 insertions(+) diff --git a/examples/official-storybook/main.ts b/examples/official-storybook/main.ts index c9553782464..758ffdf655b 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, + debugModeInteractions: true, }, }; diff --git a/lib/core-common/src/types.ts b/lib/core-common/src/types.ts index 4aeed0b0f06..1f62509f5aa 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 debugging functionality in Addon-interactions. + */ + debugModeInteractions?: boolean; + /** * Use Storybook 7.0 babel config scheme */ From 501c9e15d45a50419e4e3b0a7a646f5ebf118e5c Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 17:01:47 +0200 Subject: [PATCH 3/9] chore(addon-interactions): fix typo in stories parameter --- addons/interactions/src/components/MatcherResult.stories.tsx | 2 +- addons/interactions/src/components/MethodCall.stories.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) 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', }, }; From 6397e959980f8d8b59e2a61f806891b8cd215080 Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 17:04:20 +0200 Subject: [PATCH 4/9] feat(addon-interactions): put debugging behind a flag --- addons/interactions/src/Panel.stories.tsx | 81 +++++++++ addons/interactions/src/Panel.tsx | 165 +++++++++++++----- .../Interaction/Interaction.stories.tsx | 40 +---- .../components/Interaction/Interaction.tsx | 8 +- addons/interactions/src/mocks/index.ts | 30 ++++ 5 files changed, 244 insertions(+), 80 deletions(-) create mode 100644 addons/interactions/src/Panel.stories.tsx create mode 100644 addons/interactions/src/mocks/index.ts diff --git a/addons/interactions/src/Panel.stories.tsx b/addons/interactions/src/Panel.stories.tsx new file mode 100644 index 00000000000..87573086e08 --- /dev/null +++ b/addons/interactions/src/Panel.stories.tsx @@ -0,0 +1,81 @@ +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 Playing: Story = { + args: { + isPlaying: true, + interactions: [getCall(CallStates.WAITING)], + }, +}; + +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..fede04c66b7 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'; +const { FEATURES } = global; + interface PanelProps { active: boolean; } +interface PanelPropsy { + 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,6 +45,79 @@ const TabIcon = styled(StatusIcon)({ marginLeft: 5, }); +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); @@ -57,57 +154,39 @@ export const Panel: React.FC = (props) => { const [fileName] = storyFilePath.toString().split('/').slice(-1); const scrollToTarget = () => scrollTarget?.scrollIntoView({ behavior: 'smooth', block: 'end' }); + const isDebuggingEnabled = FEATURES.debugModeInteractions === 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); 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 - - - )} - + emit(EVENTS.START, { storyId })} + onPrevious={() => emit(EVENTS.BACK, { storyId })} + onNext={() => emit(EVENTS.NEXT, { storyId })} + onEnd={() => emit(EVENTS.END, { storyId })} + onInteractionClick={(callId) => emit(EVENTS.GOTO, { storyId, callId })} + onScrollToEnd={scrollTarget && scrollToTarget} + {...props} + /> ); }; diff --git a/addons/interactions/src/components/Interaction/Interaction.stories.tsx b/addons/interactions/src/components/Interaction/Interaction.stories.tsx index 8c1a2ed4042..e2f3c73542c 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'; @@ -16,56 +17,27 @@ export default { }, } 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/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 }; +}; From 3b7fbc89a55b69ec795d91cb946f6e8baffc584e Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 17:17:05 +0200 Subject: [PATCH 5/9] refactor(addon-interactions): rename types --- addons/interactions/src/Panel.tsx | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/addons/interactions/src/Panel.tsx b/addons/interactions/src/Panel.tsx index fede04c66b7..8184f0cd27f 100644 --- a/addons/interactions/src/Panel.tsx +++ b/addons/interactions/src/Panel.tsx @@ -13,11 +13,11 @@ import { Interaction } from './components/Interaction/Interaction'; const { FEATURES } = global; -interface PanelProps { +interface AddonPanelProps { active: boolean; } -interface PanelPropsy { +interface InteractionsPanelProps { active: boolean; showTabIcon?: boolean; interactions: (Call & { state?: CallStates })[]; @@ -45,7 +45,7 @@ const TabIcon = styled(StatusIcon)({ marginLeft: 5, }); -export const AddonPanelPure: React.FC = React.memo( +export const AddonPanelPure: React.FC = React.memo( ({ showTabIcon, interactions, @@ -118,7 +118,7 @@ export const AddonPanelPure: React.FC = React.memo( } ); -export const Panel: React.FC = (props) => { +export const Panel: React.FC = (props) => { const [isLocked, setLock] = React.useState(false); const [isPlaying, setPlaying] = React.useState(true); const [scrollTarget, setScrollTarget] = React.useState(); From 8697bf609a468c85f01b3c36e2827da9a47c5664 Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 21:08:16 +0200 Subject: [PATCH 6/9] refactor(addon-interactions): wrap callbacks in react hook --- addons/interactions/src/Panel.tsx | 19 ++++++++++++++----- 1 file changed, 14 insertions(+), 5 deletions(-) diff --git a/addons/interactions/src/Panel.tsx b/addons/interactions/src/Panel.tsx index 8184f0cd27f..b84a1a57c36 100644 --- a/addons/interactions/src/Panel.tsx +++ b/addons/interactions/src/Panel.tsx @@ -167,6 +167,15 @@ export const Panel: React.FC = (props) => { 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 ( = (props) => { calls={calls.current} endRef={endRef} isDebuggingEnabled={isDebuggingEnabled} - onStart={() => emit(EVENTS.START, { storyId })} - onPrevious={() => emit(EVENTS.BACK, { storyId })} - onNext={() => emit(EVENTS.NEXT, { storyId })} - onEnd={() => emit(EVENTS.END, { storyId })} - onInteractionClick={(callId) => emit(EVENTS.GOTO, { storyId, callId })} + onStart={onStart} + onPrevious={onPrevious} + onNext={onNext} + onEnd={onEnd} + onInteractionClick={onInteractionClick} onScrollToEnd={scrollTarget && scrollToTarget} {...props} /> From 217fa08e11d7d6c589a93bf1610836eeab7617e4 Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 21:11:50 +0200 Subject: [PATCH 7/9] refactor(addon-interactions): rename flag --- addons/interactions/src/Panel.tsx | 2 +- examples/official-storybook/main.ts | 2 +- lib/core-common/src/types.ts | 4 ++-- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/addons/interactions/src/Panel.tsx b/addons/interactions/src/Panel.tsx index b84a1a57c36..266a9212b80 100644 --- a/addons/interactions/src/Panel.tsx +++ b/addons/interactions/src/Panel.tsx @@ -154,7 +154,7 @@ export const Panel: React.FC = (props) => { const [fileName] = storyFilePath.toString().split('/').slice(-1); const scrollToTarget = () => scrollTarget?.scrollIntoView({ behavior: 'smooth', block: 'end' }); - const isDebuggingEnabled = FEATURES.debugModeInteractions === true; + const isDebuggingEnabled = FEATURES.interactionsDebugger === true; const isDebugging = log.some((item) => pendingStates.includes(item.state)); const hasPrevious = log.some((item) => completedStates.includes(item.state)); diff --git a/examples/official-storybook/main.ts b/examples/official-storybook/main.ts index 758ffdf655b..d4861400a64 100644 --- a/examples/official-storybook/main.ts +++ b/examples/official-storybook/main.ts @@ -38,7 +38,7 @@ const config: StorybookConfig = { logLevel: 'debug', features: { modernInlineRender: true, - debugModeInteractions: true, + interactionsDebugger: true, }, }; diff --git a/lib/core-common/src/types.ts b/lib/core-common/src/types.ts index 1f62509f5aa..12df892b75e 100644 --- a/lib/core-common/src/types.ts +++ b/lib/core-common/src/types.ts @@ -315,9 +315,9 @@ export interface StorybookConfig { breakingChangesV7?: boolean; /** - * Enable the debugging functionality in Addon-interactions. + * Enable the step debugger functionality in Addon-interactions. */ - debugModeInteractions?: boolean; + interactionsDebugger?: boolean; /** * Use Storybook 7.0 babel config scheme From 75021c67357f925005c1ae34527bcfb76dd733cb Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Tue, 26 Oct 2021 21:29:10 +0200 Subject: [PATCH 8/9] fix: add missing boolean in interaction stories --- .../src/components/Interaction/Interaction.stories.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/addons/interactions/src/components/Interaction/Interaction.stories.tsx b/addons/interactions/src/components/Interaction/Interaction.stories.tsx index e2f3c73542c..751841e7c76 100644 --- a/addons/interactions/src/components/Interaction/Interaction.stories.tsx +++ b/addons/interactions/src/components/Interaction/Interaction.stories.tsx @@ -14,6 +14,7 @@ export default { args: { callsById: new Map(), isDisabled: false, + isDebuggingEnabled: true, }, } as ComponentMeta; From 65d5f625167a62c3a963508d2311a5b85afc7e20 Mon Sep 17 00:00:00 2001 From: Yann Braga Date: Wed, 27 Oct 2021 15:52:07 +0200 Subject: [PATCH 9/9] chore(addon-interactions): add paused story --- addons/interactions/src/Panel.stories.tsx | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/addons/interactions/src/Panel.stories.tsx b/addons/interactions/src/Panel.stories.tsx index 87573086e08..53a3bc7504b 100644 --- a/addons/interactions/src/Panel.stories.tsx +++ b/addons/interactions/src/Panel.stories.tsx @@ -56,13 +56,20 @@ export const Passing: Story = { }, }; -export const Playing: Story = { +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,