From 1142e1ddb10699a59790b1a144ca3570a307cc8a Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Mon, 9 Nov 2020 13:48:51 +1100 Subject: [PATCH 1/7] Add an example to highlight if a story renders too many times --- .../stories/core/rendering.stories.js | 43 +++++++++++++++++-- 1 file changed, 39 insertions(+), 4 deletions(-) diff --git a/examples/official-storybook/stories/core/rendering.stories.js b/examples/official-storybook/stories/core/rendering.stories.js index f446292b95f..ab6e3cabd64 100644 --- a/examples/official-storybook/stories/core/rendering.stories.js +++ b/examples/official-storybook/stories/core/rendering.stories.js @@ -1,20 +1,55 @@ -import React, { useRef } from 'react'; +import React, { useEffect, useRef } from 'react'; +import { useArgs } from '@storybook/client-api'; export default { title: 'Core/Rendering', }; // NOTE: in our example apps each component is mounted twice as we render in strict mode -let timesMounted = 0; +let timesCounterMounted = 0; export const Counter = () => { const countRef = useRef(); - if (!countRef.current) timesMounted += 1; + if (!countRef.current) timesCounterMounted += 1; countRef.current = (countRef.current || 0) + 1; return (
- Mounted: {timesMounted}, rendered (this mount): {countRef.current} + Mounted: {timesCounterMounted}, rendered (this mount): {countRef.current}
); }; + +// An example to test what happens when the story is remounted due to argChanges +let timesArgsChangeMounted = 0; +export const ArgsChange = () => { + const countRef = useRef(); + + if (!countRef.current) timesArgsChangeMounted += 1; + countRef.current = true; + + return ( +
+ Mounted: {timesArgsChangeMounted} (NOTE: we use strict mode so this number is 2x what you'd + expect -- it should be 2, not 4 though!) +
+ ); +}; + +ArgsChange.args = { + first: 0, +}; + +ArgsChange.decorators = [ + (StoryFn) => { + const [args, updateArgs] = useArgs(); + + useEffect(() => { + if (args.first === 0) { + updateArgs({ first: 1 }); + } + }, []); + + return ; + }, +]; From 243efbb7e3f551d6f1b62c7314d23d27d0db839f Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Fri, 23 Apr 2021 13:57:36 +1000 Subject: [PATCH 2/7] Bind the context *inside* the function --- lib/addons/src/types.ts | 14 +++++-- lib/client-api/src/decorators.ts | 64 +++++++++++++++++++------------ lib/client-api/src/story_store.ts | 4 +- 3 files changed, 52 insertions(+), 30 deletions(-) diff --git a/lib/addons/src/types.ts b/lib/addons/src/types.ts index 445e8b5731c..7d9e2e6e737 100644 --- a/lib/addons/src/types.ts +++ b/lib/addons/src/types.ts @@ -53,6 +53,7 @@ export interface StoryIdentifier { name: StoryName; } +export type StoryContextUpdate = Partial; export type StoryContext = StoryIdentifier & { [key: string]: any; parameters: Parameters; @@ -93,8 +94,13 @@ export interface OptionsParameter extends Object { export type StoryGetter = (context: StoryContext) => any; +// This is the type of story function passed to a decorator -- does not rely on being passed any context +export type PartialStoryFn = (p?: StoryContextUpdate) => ReturnType; +// This is a passArgsFirst: false user story function export type LegacyStoryFn = (p?: StoryContext) => ReturnType; +// This is a passArgsFirst: true user story function export type ArgsStoryFn = (a?: Args, p?: StoryContext) => ReturnType; +// This is either type of user story function export type StoryFn = LegacyStoryFn | ArgsStoryFn; export type StoryWrapper = ( @@ -136,16 +142,16 @@ export interface StoryApi { } export type DecoratorFunction = ( - fn: StoryFn, + fn: PartialStoryFn, c: StoryContext -) => ReturnType>; +) => ReturnType>; export type LoaderFunction = (c: StoryContext) => Promise>; export type DecorateStoryFunction = ( - storyFn: StoryFn, + storyFn: LegacyStoryFn, decorators: DecoratorFunction[] -) => StoryFn; +) => LegacyStoryFn; export interface ClientStoryApi { storiesOf(kind: StoryKind, module: NodeModule): StoryApi; diff --git a/lib/client-api/src/decorators.ts b/lib/client-api/src/decorators.ts index 8e51b7d580d..eef4b9bb89a 100644 --- a/lib/client-api/src/decorators.ts +++ b/lib/client-api/src/decorators.ts @@ -1,20 +1,6 @@ -import { StoryContext, StoryFn } from '@storybook/addons'; +import { StoryContext, StoryContextUpdate, PartialStoryFn, LegacyStoryFn } from '@storybook/addons'; import { DecoratorFunction } from './types'; -interface StoryContextUpdate { - [key: string]: any; -} - -const defaultContext: StoryContext = { - id: 'unspecified', - name: 'unspecified', - kind: 'unspecified', - parameters: {}, - args: {}, - argTypes: {}, - globals: {}, -}; - /** * When you call the story function inside a decorator, e.g.: * @@ -25,15 +11,43 @@ const defaultContext: StoryContext = { * This will override the `foo` property on the `innerContext`, which gets * merged in with the default context */ -export const decorateStory = (storyFn: StoryFn, decorator: DecoratorFunction) => { - return (context: StoryContext = defaultContext) => - decorator( - // You cannot override the parameters key, it is fixed - ({ parameters, ...innerContext }: StoryContextUpdate = {}) => - storyFn({ ...context, ...innerContext }), - context - ); +const bindWithContext = ( + storyFn: LegacyStoryFn, + getStoryContext: () => StoryContext +): PartialStoryFn => + // (NOTE: You cannot override the parameters key, it is fixed) + ({ parameters, ...innerContext }: StoryContextUpdate = {}) => + storyFn({ ...getStoryContext(), ...innerContext }); + +export const decorateStory = ( + storyFn: LegacyStoryFn, + decorator: DecoratorFunction, + getStoryContext: () => StoryContext +): LegacyStoryFn => { + // Bind the partially decorated storyFn so that when it is called it always knows about the story context, + // no matter what it is passed directly. This is because we cannot guarantee a decorator will + // pass the context down to the next decorator in the chain. + const boundStoryFunction = bindWithContext(storyFn, getStoryContext); + + return (context: StoryContext) => decorator(boundStoryFunction, context); }; -export const defaultDecorateStory = (storyFn: StoryFn, decorators: DecoratorFunction[]) => - decorators.reduce(decorateStory, storyFn); +export const defaultDecorateStory = ( + storyFn: LegacyStoryFn, + decorators: DecoratorFunction[] +): LegacyStoryFn => { + // We use a trick to avoid recreating the bound story function inside `decorateStory`. + // Instead we pass it a context "getter", which is defined once (at "decoration time") + // The getter reads a variable which is scoped to this call of `decorateStory` + // (ie to this story), so there is no possibility of overlap. + // This will break if you call the same story twice interleaved. + let contextStore: StoryContext; + const decoratedWithContextStore = decorators.reduce( + (story, decorator) => decorateStory(story, decorator, () => contextStore), + storyFn + ); + return (context) => { + contextStore = context; + return decoratedWithContextStore(context); + }; +}; diff --git a/lib/client-api/src/story_store.ts b/lib/client-api/src/story_store.ts index 1c85fcd2ae5..12a6f175dbe 100644 --- a/lib/client-api/src/story_store.ts +++ b/lib/client-api/src/story_store.ts @@ -394,7 +394,9 @@ export default class StoryStore { return acc; }, {} as Args), }; - return passArgsFirst ? (original as ArgsStoryFn)(mapped.args, mapped) : original(mapped); + return passArgsFirst + ? (original as ArgsStoryFn)(mapped.args, mapped) + : (original as LegacyStoryFn)(mapped); }; // lazily decorate the story when it's loaded From 9c2a1c67a1487ca4711569c0ee5ed2b4854a79fd Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Thu, 22 Apr 2021 21:31:37 +1000 Subject: [PATCH 3/7] Some minor comment improvements --- lib/client-api/src/decorators.ts | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/lib/client-api/src/decorators.ts b/lib/client-api/src/decorators.ts index eef4b9bb89a..97ff5351f7a 100644 --- a/lib/client-api/src/decorators.ts +++ b/lib/client-api/src/decorators.ts @@ -26,7 +26,7 @@ export const decorateStory = ( ): LegacyStoryFn => { // Bind the partially decorated storyFn so that when it is called it always knows about the story context, // no matter what it is passed directly. This is because we cannot guarantee a decorator will - // pass the context down to the next decorator in the chain. + // pass the context down to the next decorated story in the chain. const boundStoryFunction = bindWithContext(storyFn, getStoryContext); return (context: StoryContext) => decorator(boundStoryFunction, context); @@ -40,7 +40,8 @@ export const defaultDecorateStory = ( // Instead we pass it a context "getter", which is defined once (at "decoration time") // The getter reads a variable which is scoped to this call of `decorateStory` // (ie to this story), so there is no possibility of overlap. - // This will break if you call the same story twice interleaved. + // This will break if you call the same story twice interleaved + // (React might do it if you rendered the same story twice in the one ReactDom.render call, for instance) let contextStore: StoryContext; const decoratedWithContextStore = decorators.reduce( (story, decorator) => decorateStory(story, decorator, () => contextStore), @@ -48,6 +49,6 @@ export const defaultDecorateStory = ( ); return (context) => { contextStore = context; - return decoratedWithContextStore(context); + return decoratedWithContextStore(context); // Pass the context directly into the first decorator }; }; From 3934b7c60d81329754728da26d12651a2b831d6e Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Thu, 22 Apr 2021 22:46:55 +1000 Subject: [PATCH 4/7] Update name as per suggestion --- lib/client-api/src/decorators.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/client-api/src/decorators.ts b/lib/client-api/src/decorators.ts index 97ff5351f7a..0a03d61a469 100644 --- a/lib/client-api/src/decorators.ts +++ b/lib/client-api/src/decorators.ts @@ -16,8 +16,8 @@ const bindWithContext = ( getStoryContext: () => StoryContext ): PartialStoryFn => // (NOTE: You cannot override the parameters key, it is fixed) - ({ parameters, ...innerContext }: StoryContextUpdate = {}) => - storyFn({ ...getStoryContext(), ...innerContext }); + ({ parameters, ...contextUpdate }: StoryContextUpdate = {}) => + storyFn({ ...getStoryContext(), ...contextUpdate }); export const decorateStory = ( storyFn: LegacyStoryFn, From 256135d855f4ea633fcfb705c045e601db793413 Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Fri, 23 Apr 2021 14:11:29 +1000 Subject: [PATCH 5/7] Add tests for decorator context binding --- lib/client-api/src/decorators.test.ts | 48 +++++++++++++++++++++++++-- 1 file changed, 46 insertions(+), 2 deletions(-) diff --git a/lib/client-api/src/decorators.test.ts b/lib/client-api/src/decorators.test.ts index d7513feea95..aeacac331f9 100644 --- a/lib/client-api/src/decorators.test.ts +++ b/lib/client-api/src/decorators.test.ts @@ -2,7 +2,7 @@ import { StoryContext } from '@storybook/addons'; import { defaultDecorateStory } from './decorators'; -function makeContext(input: Record): StoryContext { +function makeContext(input: Record = {}): StoryContext { return { id: 'id', kind: 'kind', @@ -24,7 +24,7 @@ describe('client-api.decorators', () => { const decorated = defaultDecorateStory(() => order.push(4), decorators); expect(order).toEqual([]); - decorated(); + decorated(makeContext()); expect(order).toEqual([3, 2, 1, 4]); }); @@ -42,6 +42,50 @@ describe('client-api.decorators', () => { expect(contexts.map((c) => c.k)).toEqual([0, 3, 2, 1]); }); + it('does not recreate decorated story functions each time', () => { + const decoratedStories = []; + const decorators = [ + (s, c) => { + decoratedStories.push = s; + return s(); + }, + ]; + const decorated = defaultDecorateStory(() => 0, decorators); + + decorated(makeContext()); + decorated(makeContext()); + expect(decoratedStories[0]).toBe(decoratedStories[1]); + }); + + // NOTE: important point--this test would not work if we called `decoratedOne` twice simultaneously + // both story functions would receive {story: 2}. The assumption here is that we'll never render + // the same story twice at the same time. + it('does not interleave contexts if two decorated stories are call simultaneously', async () => { + const contexts = []; + let resolve; + const fence = new Promise((r) => { + resolve = r; + }); + const decorators = [ + async (s, c) => { + // The fence here simulates async-ness in react rendering an element (`` doesn't run `S()` straight away) + await fence; + s(); + }, + ]; + const decoratedOne = defaultDecorateStory((c) => contexts.push(c), decorators); + const decoratedTwo = defaultDecorateStory((c) => contexts.push(c), decorators); + + decoratedOne(makeContext({ story: 1 })); + decoratedTwo(makeContext({ story: 2 })); + + resolve(); + await fence; + + expect(contexts[0].story).toBe(1); + expect(contexts[1].story).toBe(2); + }); + it('merges contexts', () => { const contexts = []; const decorators = [(s, c) => contexts.push(c) && s({ c: 'd' })]; From eec9799343bc50359bb7c46d002c48a03dafb60b Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Fri, 23 Apr 2021 14:13:48 +1000 Subject: [PATCH 6/7] Add back (janky) default context --- lib/client-api/src/decorators.ts | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/lib/client-api/src/decorators.ts b/lib/client-api/src/decorators.ts index 0a03d61a469..6e77c93c1b0 100644 --- a/lib/client-api/src/decorators.ts +++ b/lib/client-api/src/decorators.ts @@ -1,6 +1,16 @@ import { StoryContext, StoryContextUpdate, PartialStoryFn, LegacyStoryFn } from '@storybook/addons'; import { DecoratorFunction } from './types'; +const defaultContext: StoryContext = { + id: 'unspecified', + name: 'unspecified', + kind: 'unspecified', + parameters: {}, + args: {}, + argTypes: {}, + globals: {}, +}; + /** * When you call the story function inside a decorator, e.g.: * @@ -47,7 +57,7 @@ export const defaultDecorateStory = ( (story, decorator) => decorateStory(story, decorator, () => contextStore), storyFn ); - return (context) => { + return (context = defaultContext) => { contextStore = context; return decoratedWithContextStore(context); // Pass the context directly into the first decorator }; From 02ba8bb873783e7afd4d8744425439dd05f04e9f Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Tue, 27 Apr 2021 15:48:26 +1000 Subject: [PATCH 7/7] Replace `.story` => `.value` --- lib/client-api/src/decorators.test.ts | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/lib/client-api/src/decorators.test.ts b/lib/client-api/src/decorators.test.ts index aeacac331f9..0907c0045f3 100644 --- a/lib/client-api/src/decorators.test.ts +++ b/lib/client-api/src/decorators.test.ts @@ -76,14 +76,14 @@ describe('client-api.decorators', () => { const decoratedOne = defaultDecorateStory((c) => contexts.push(c), decorators); const decoratedTwo = defaultDecorateStory((c) => contexts.push(c), decorators); - decoratedOne(makeContext({ story: 1 })); - decoratedTwo(makeContext({ story: 2 })); + decoratedOne(makeContext({ value: 1 })); + decoratedTwo(makeContext({ value: 2 })); resolve(); await fence; - expect(contexts[0].story).toBe(1); - expect(contexts[1].story).toBe(2); + expect(contexts[0].value).toBe(1); + expect(contexts[1].value).toBe(2); }); it('merges contexts', () => {