From 3d7f249ab406d2cd152d0c6f9a225b050ed423d8 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 10:44:31 +0100
Subject: [PATCH 001/171] feat: add testing utilities in @storybook/store
---
lib/store/src/StoryStore.ts | 4 +-
lib/store/src/csf/index.ts | 5 +
.../src/{ => csf}/normalizeInputTypes.test.ts | 0
.../src/{ => csf}/normalizeInputTypes.ts | 0
.../src/csf/normalizeProjectAnnotations.ts | 22 +++++
.../src/{ => csf}/normalizeStory.test.ts | 0
lib/store/src/{ => csf}/normalizeStory.ts | 2 +-
lib/store/src/{ => csf}/prepareStory.test.ts | 2 +-
lib/store/src/{ => csf}/prepareStory.ts | 10 +-
.../src/{ => csf}/processCSFFile.test.ts | 0
lib/store/src/{ => csf}/processCSFFile.ts | 22 ++---
lib/store/src/csf/testing-utils/index.ts | 99 +++++++++++++++++++
lib/store/src/index.ts | 2 +-
13 files changed, 143 insertions(+), 25 deletions(-)
create mode 100644 lib/store/src/csf/index.ts
rename lib/store/src/{ => csf}/normalizeInputTypes.test.ts (100%)
rename lib/store/src/{ => csf}/normalizeInputTypes.ts (100%)
create mode 100644 lib/store/src/csf/normalizeProjectAnnotations.ts
rename lib/store/src/{ => csf}/normalizeStory.test.ts (100%)
rename lib/store/src/{ => csf}/normalizeStory.ts (97%)
rename lib/store/src/{ => csf}/prepareStory.test.ts (99%)
rename lib/store/src/{ => csf}/prepareStory.ts (96%)
rename lib/store/src/{ => csf}/processCSFFile.test.ts (100%)
rename lib/store/src/{ => csf}/processCSFFile.ts (74%)
create mode 100644 lib/store/src/csf/testing-utils/index.ts
diff --git a/lib/store/src/StoryStore.ts b/lib/store/src/StoryStore.ts
index ddfeb57c14c..c1a7e79cf52 100644
--- a/lib/store/src/StoryStore.ts
+++ b/lib/store/src/StoryStore.ts
@@ -17,8 +17,7 @@ import { SynchronousPromise } from 'synchronous-promise';
import { StoryIndexStore } from './StoryIndexStore';
import { ArgsStore } from './ArgsStore';
import { GlobalsStore } from './GlobalsStore';
-import { processCSFFile } from './processCSFFile';
-import { prepareStory } from './prepareStory';
+import { normalizeInputTypes, processCSFFile, prepareStory } from './csf';
import {
CSFFile,
ModuleImportFn,
@@ -32,7 +31,6 @@ import {
V2CompatIndexEntry,
} from './types';
import { HooksContext } from './hooks';
-import { normalizeInputTypes } from './normalizeInputTypes';
import { inferArgTypes } from './inferArgTypes';
import { inferControls } from './inferControls';
diff --git a/lib/store/src/csf/index.ts b/lib/store/src/csf/index.ts
new file mode 100644
index 00000000000..4e7870ccd0b
--- /dev/null
+++ b/lib/store/src/csf/index.ts
@@ -0,0 +1,5 @@
+export * from './normalizeInputTypes';
+export * from './normalizeStory';
+export * from './processCSFFile';
+export * from './prepareStory';
+export * from './testing-utils';
diff --git a/lib/store/src/normalizeInputTypes.test.ts b/lib/store/src/csf/normalizeInputTypes.test.ts
similarity index 100%
rename from lib/store/src/normalizeInputTypes.test.ts
rename to lib/store/src/csf/normalizeInputTypes.test.ts
diff --git a/lib/store/src/normalizeInputTypes.ts b/lib/store/src/csf/normalizeInputTypes.ts
similarity index 100%
rename from lib/store/src/normalizeInputTypes.ts
rename to lib/store/src/csf/normalizeInputTypes.ts
diff --git a/lib/store/src/csf/normalizeProjectAnnotations.ts b/lib/store/src/csf/normalizeProjectAnnotations.ts
new file mode 100644
index 00000000000..dd3eba96347
--- /dev/null
+++ b/lib/store/src/csf/normalizeProjectAnnotations.ts
@@ -0,0 +1,22 @@
+import { sanitize, AnyFramework } from '@storybook/csf';
+
+import { ModuleExports, NormalizedComponentAnnotations } from '../types';
+import { normalizeInputTypes } from './normalizeInputTypes';
+
+export function normalizeProjectAnnotations(
+ defaultExport: ModuleExports['default'],
+ title: string = defaultExport.title,
+ importPath?: string
+): NormalizedComponentAnnotations {
+ const { id, argTypes } = defaultExport;
+ return {
+ id: sanitize(id || title),
+ ...defaultExport,
+ title,
+ ...(argTypes && { argTypes: normalizeInputTypes(argTypes) }),
+ parameters: {
+ fileName: importPath,
+ ...defaultExport.parameters,
+ },
+ };
+}
diff --git a/lib/store/src/normalizeStory.test.ts b/lib/store/src/csf/normalizeStory.test.ts
similarity index 100%
rename from lib/store/src/normalizeStory.test.ts
rename to lib/store/src/csf/normalizeStory.test.ts
diff --git a/lib/store/src/normalizeStory.ts b/lib/store/src/csf/normalizeStory.ts
similarity index 97%
rename from lib/store/src/normalizeStory.ts
rename to lib/store/src/csf/normalizeStory.ts
index b0784d5a3c6..1d0f3b93c07 100644
--- a/lib/store/src/normalizeStory.ts
+++ b/lib/store/src/csf/normalizeStory.ts
@@ -11,7 +11,7 @@ import {
import dedent from 'ts-dedent';
import { logger } from '@storybook/client-logger';
import deprecate from 'util-deprecate';
-import { NormalizedStoryAnnotations } from './types';
+import { NormalizedStoryAnnotations } from '../types';
import { normalizeInputTypes } from './normalizeInputTypes';
const deprecatedStoryAnnotation = dedent`
diff --git a/lib/store/src/prepareStory.test.ts b/lib/store/src/csf/prepareStory.test.ts
similarity index 99%
rename from lib/store/src/prepareStory.test.ts
rename to lib/store/src/csf/prepareStory.test.ts
index 4dd1c7b379c..1400ed1ea49 100644
--- a/lib/store/src/prepareStory.test.ts
+++ b/lib/store/src/csf/prepareStory.test.ts
@@ -7,7 +7,7 @@ import {
SBScalarType,
StoryContext,
} from '@storybook/csf';
-import { NO_TARGET_NAME } from './args';
+import { NO_TARGET_NAME } from '../args';
import { prepareStory } from './prepareStory';
jest.mock('global', () => ({
diff --git a/lib/store/src/prepareStory.ts b/lib/store/src/csf/prepareStory.ts
similarity index 96%
rename from lib/store/src/prepareStory.ts
rename to lib/store/src/csf/prepareStory.ts
index 6547cf95981..16d5919aa57 100644
--- a/lib/store/src/prepareStory.ts
+++ b/lib/store/src/csf/prepareStory.ts
@@ -18,11 +18,11 @@ import {
Story,
NormalizedStoryAnnotations,
NormalizedProjectAnnotations,
-} from './types';
-import { combineParameters } from './parameters';
-import { applyHooks } from './hooks';
-import { defaultDecorateStory } from './decorators';
-import { groupArgsByTarget, NO_TARGET_NAME } from './args';
+} from '../types';
+import { combineParameters } from '../parameters';
+import { applyHooks } from '../hooks';
+import { defaultDecorateStory } from '../decorators';
+import { groupArgsByTarget, NO_TARGET_NAME } from '../args';
const argTypeDefaultValueWarning = deprecate(
() => {},
diff --git a/lib/store/src/processCSFFile.test.ts b/lib/store/src/csf/processCSFFile.test.ts
similarity index 100%
rename from lib/store/src/processCSFFile.test.ts
rename to lib/store/src/csf/processCSFFile.test.ts
diff --git a/lib/store/src/processCSFFile.ts b/lib/store/src/csf/processCSFFile.ts
similarity index 74%
rename from lib/store/src/processCSFFile.ts
rename to lib/store/src/csf/processCSFFile.ts
index dbb5e44a419..0e6d3207553 100644
--- a/lib/store/src/processCSFFile.ts
+++ b/lib/store/src/csf/processCSFFile.ts
@@ -1,10 +1,10 @@
-import { isExportStory, sanitize, Parameters, AnyFramework, ComponentTitle } from '@storybook/csf';
+import { isExportStory, Parameters, AnyFramework, ComponentTitle } from '@storybook/csf';
import { logger } from '@storybook/client-logger';
-import { ModuleExports, CSFFile, NormalizedComponentAnnotations } from './types';
+import { ModuleExports, CSFFile, NormalizedComponentAnnotations } from '../types';
import { normalizeStory } from './normalizeStory';
-import { normalizeInputTypes } from './normalizeInputTypes';
-import { Path } from '.';
+import { Path } from '..';
+import { normalizeProjectAnnotations } from './normalizeProjectAnnotations';
const checkGlobals = (parameters: Parameters) => {
const { globals, globalTypes } = parameters;
@@ -40,17 +40,11 @@ export function processCSFFile(
): CSFFile {
const { default: defaultExport, __namedExportsOrder, ...namedExports } = moduleExports;
- const { id, argTypes } = defaultExport;
- const meta: NormalizedComponentAnnotations = {
- id: sanitize(id || title),
- ...defaultExport,
+ const meta: NormalizedComponentAnnotations = normalizeProjectAnnotations(
+ defaultExport,
title,
- ...(argTypes && { argTypes: normalizeInputTypes(argTypes) }),
- parameters: {
- fileName: importPath,
- ...defaultExport.parameters,
- },
- };
+ importPath
+ );
checkDisallowedParameters(meta.parameters);
const csfFile: CSFFile = { meta, stories: {} };
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
new file mode 100644
index 00000000000..7ebeece296b
--- /dev/null
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -0,0 +1,99 @@
+import {
+ isExportStory,
+ AnyFramework,
+ AnnotatedStoryFn,
+ ComponentAnnotations,
+ Args,
+} from '@storybook/csf';
+
+import { prepareStory } from '../prepareStory';
+import { normalizeStory } from '../normalizeStory';
+import { normalizeProjectAnnotations } from '../normalizeProjectAnnotations';
+import { HooksContext } from '../../hooks';
+import { NormalizedProjectAnnotations } from '../..';
+
+if (process.env.NODE_ENV === 'test') {
+ // eslint-disable-next-line global-require
+ const { default: addons, mockChannel } = require('@storybook/addons');
+ addons.setChannel(mockChannel());
+}
+
+export type StoryFile = { default: any; __esModule?: boolean; __namedExportsOrder?: any };
+
+type Entries = {
+ [K in keyof T]: [K, T[K]];
+}[keyof T];
+export function objectEntries(t: T): Entries[] {
+ return Object.entries(t) as any;
+}
+
+export function composeStory<
+ TFramework extends AnyFramework = AnyFramework,
+ TArgs extends Args = Args
+>(
+ story: AnnotatedStoryFn,
+ meta: ComponentAnnotations,
+ globalConfig: NormalizedProjectAnnotations = {}
+) {
+ if (story === undefined) {
+ throw new Error('Expected a story but received undefined.');
+ }
+
+ const normalizedMeta = normalizeProjectAnnotations(meta);
+
+ const normalizedStory = normalizeStory(story.name, story, normalizedMeta);
+
+ const preparedStory = prepareStory(normalizedStory, normalizedMeta, globalConfig);
+
+ const defaultGlobals = Object.entries(globalConfig.globalTypes || {}).reduce(
+ (acc, [arg, { defaultValue }]) => {
+ if (defaultValue) {
+ acc[arg] = defaultValue;
+ }
+ return acc;
+ },
+ {} as Record
+ );
+
+ const composedStory = (extraArgs: Partial) => {
+ const context = {
+ ...preparedStory,
+ hooks: new HooksContext(),
+ globals: defaultGlobals,
+ args: { ...preparedStory.initialArgs, ...extraArgs },
+ } as any;
+
+ return preparedStory.unboundStoryFn(context);
+ };
+
+ composedStory.storyName = story.storyName || story.name;
+ composedStory.args = preparedStory.initialArgs;
+ composedStory.play = preparedStory.playFunction;
+ composedStory.parameters = preparedStory.parameters;
+
+ return composedStory;
+}
+
+export function composeStories(
+ storiesImport: TModule,
+ globalConfig: NormalizedProjectAnnotations,
+ composeStoryFn: typeof composeStory
+) {
+ const { default: meta, __esModule, __namedExportsOrder, ...stories } = storiesImport;
+ const composedStories = objectEntries(stories).reduce((storiesMap, [key, _story]) => {
+ if (!isExportStory(key as string, meta)) {
+ return storiesMap;
+ }
+
+ const storyName = String(key);
+ const story = _story as any;
+ story.storyName = storyName;
+
+ const result = Object.assign(storiesMap, {
+ [key]: composeStoryFn(story, meta, globalConfig),
+ });
+ return result;
+ }, {});
+
+ return composedStories;
+}
diff --git a/lib/store/src/index.ts b/lib/store/src/index.ts
index ab63007918c..f3bca023e57 100644
--- a/lib/store/src/index.ts
+++ b/lib/store/src/index.ts
@@ -1,12 +1,12 @@
export { StoryStore } from './StoryStore';
export { combineParameters } from './parameters';
export { filterArgTypes } from './filterArgTypes';
-export { normalizeInputTypes } from './normalizeInputTypes';
export type { PropDescriptor } from './filterArgTypes';
export { inferControls } from './inferControls';
export * from './types';
+export * from './csf';
export * from './hooks';
export * from './decorators';
export * from './args';
From a3b599dbe263690f79357c6dbfb4bcb896d22223 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 10:44:47 +0100
Subject: [PATCH 002/171] feat: add testing utilities in @storybook/react
---
app/react/src/client/index.ts | 1 +
app/react/src/client/testing/index.ts | 57 +++++++++++++++++++++++++++
app/react/src/client/testing/types.ts | 43 ++++++++++++++++++++
3 files changed, 101 insertions(+)
create mode 100644 app/react/src/client/testing/index.ts
create mode 100644 app/react/src/client/testing/types.ts
diff --git a/app/react/src/client/index.ts b/app/react/src/client/index.ts
index 54de76c8585..141f7e7f43f 100644
--- a/app/react/src/client/index.ts
+++ b/app/react/src/client/index.ts
@@ -9,6 +9,7 @@ export {
raw,
forceReRender,
} from './preview';
+export * from './testing';
export * from './preview/types-6-3';
diff --git a/app/react/src/client/testing/index.ts b/app/react/src/client/testing/index.ts
new file mode 100644
index 00000000000..7ce12904ea0
--- /dev/null
+++ b/app/react/src/client/testing/index.ts
@@ -0,0 +1,57 @@
+import {
+ composeStory as originalComposeStory,
+ composeStories as originalComposeStories,
+} from '@storybook/store';
+// eslint-disable-next-line import/no-extraneous-dependencies
+import { composeConfigs } from '@storybook/preview-web';
+import type { AnnotatedStoryFn } from '@storybook/csf';
+import { render } from '../preview/render';
+
+import type { Meta, ReactFramework } from '../preview/types-6-0';
+import type { StoriesWithPartialProps, GlobalConfig, StoryFile } from './types';
+
+const defaultGlobalConfig: GlobalConfig = {
+ render,
+};
+
+let globalStorybookConfig = {
+ ...defaultGlobalConfig,
+};
+
+/** Function that sets the globalConfig of your storybook. The global config is the preview module of your .storybook folder.
+ *
+ * It should be run a single time, so that your global config (e.g. decorators) is applied to your stories when using `composeStories` or `composeStory`.
+ *
+ * Example:
+ *```jsx
+ * // setup.js (for jest)
+ * import { setGlobalConfig } from '@storybook/testing-react';
+ * import * as globalStorybookConfig from './.storybook/preview';
+ *
+ * setGlobalConfig(globalStorybookConfig);
+ *```
+ *
+ * @param config - e.g. (import * as globalConfig from '../.storybook/preview')
+ */
+export function setGlobalConfig(config: GlobalConfig) {
+ globalStorybookConfig = composeConfigs([defaultGlobalConfig, config]) as GlobalConfig;
+}
+
+export function composeStory(
+ story: AnnotatedStoryFn,
+ meta: Meta,
+ globalConfig: GlobalConfig = globalStorybookConfig
+) {
+ const projectAnnotations = { ...defaultGlobalConfig, ...globalConfig };
+
+ return originalComposeStory(story, meta, projectAnnotations);
+}
+
+export function composeStories(
+ storiesImport: TModule,
+ globalConfig?: GlobalConfig
+) {
+ const composedStories = originalComposeStories(storiesImport, globalConfig, composeStory);
+
+ return (composedStories as unknown) as Omit, keyof StoryFile>;
+}
diff --git a/app/react/src/client/testing/types.ts b/app/react/src/client/testing/types.ts
new file mode 100644
index 00000000000..574495b0177
--- /dev/null
+++ b/app/react/src/client/testing/types.ts
@@ -0,0 +1,43 @@
+import { NormalizedProjectAnnotations } from '@storybook/store';
+import type {
+ StoryFn as OriginalStoryFn,
+ StoryObj,
+ Meta,
+ Args,
+ StoryContext,
+ ReactFramework,
+} from '../preview/types-6-0';
+
+/**
+ * Object representing the preview.ts module
+ *
+ * Used in storybook testing utilities.
+ * @see [Unit testing with Storybook](https://storybook.js.org/docs/react/workflows/unit-testing)
+ */
+export type GlobalConfig = NormalizedProjectAnnotations;
+
+export type StoryFile = { default: Meta; __esModule?: boolean; __namedExportsOrder?: any };
+
+export type TestingStory = StoryFn | StoryObj;
+
+export type TestingStoryPlayContext = Partial> &
+ Pick;
+
+export type TestingStoryPlayFn = (
+ context: TestingStoryPlayContext
+) => Promise | void;
+
+export type StoryFn = OriginalStoryFn & { play: TestingStoryPlayFn };
+
+/**
+ * T represents the whole es module of a stories file. K of T means named exports (basically the Story type)
+ * 1. pick the keys K of T that have properties that are Story
+ * 2. infer the actual prop type for each Story
+ * 3. reconstruct Story with Partial. Story -> Story>
+ */
+export type StoriesWithPartialProps = any;
+// {
+// [K in keyof T as T[K] extends TestingStory ? K : never]: T[K] extends TestingStory
+// ? StoryFn>
+// : unknown;
+// };
From f9f5c02697029c38726a914d1aab910cd06271b3 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 10:45:13 +0100
Subject: [PATCH 003/171] chore: add tests for testing utilities in
cra-ts-essentials
---
.../.storybook/{main.js => main.ts} | 0
.../.storybook/{preview.js => preview.tsx} | 17 +-
examples/cra-ts-essentials/package.json | 7 +-
examples/cra-ts-essentials/src/setupTests.ts | 4 +
.../components/AccountForm.stories.tsx | 109 ++++
.../components/AccountForm.test.tsx | 25 +
.../testing-react/components/AccountForm.tsx | 552 ++++++++++++++++++
.../components/Button.stories.tsx | 87 +++
.../testing-react/components/Button.test.tsx | 94 +++
.../testing-react/components/Button.tsx | 43 ++
.../__snapshots__/internals.test.tsx.snap | 127 ++++
.../testing-react/components/button.css | 30 +
.../components/internals.test.tsx | 112 ++++
examples/cra-ts-essentials/tsconfig.json | 14 +-
.../.storybook/{preview.js => preview.tsx} | 0
examples/react-ts/package.json | 1 +
examples/react-ts/src/AccountForm.stories.tsx | 55 +-
examples/react-ts/src/AccountForm.test.tsx | 24 +
yarn.lock | 23 +-
19 files changed, 1294 insertions(+), 30 deletions(-)
rename examples/cra-ts-essentials/.storybook/{main.js => main.ts} (100%)
rename examples/cra-ts-essentials/.storybook/{preview.js => preview.tsx} (55%)
create mode 100644 examples/cra-ts-essentials/src/setupTests.ts
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.stories.tsx
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.tsx
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/Button.tsx
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/button.css
create mode 100644 examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
rename examples/react-ts/.storybook/{preview.js => preview.tsx} (100%)
create mode 100644 examples/react-ts/src/AccountForm.test.tsx
diff --git a/examples/cra-ts-essentials/.storybook/main.js b/examples/cra-ts-essentials/.storybook/main.ts
similarity index 100%
rename from examples/cra-ts-essentials/.storybook/main.js
rename to examples/cra-ts-essentials/.storybook/main.ts
diff --git a/examples/cra-ts-essentials/.storybook/preview.js b/examples/cra-ts-essentials/.storybook/preview.tsx
similarity index 55%
rename from examples/cra-ts-essentials/.storybook/preview.js
rename to examples/cra-ts-essentials/.storybook/preview.tsx
index 305c8eb0be1..54093cc18b6 100644
--- a/examples/cra-ts-essentials/.storybook/preview.js
+++ b/examples/cra-ts-essentials/.storybook/preview.tsx
@@ -1,14 +1,25 @@
import React from 'react';
+import type { DecoratorFn } from '@storybook/react';
+import { ThemeProvider, convert, themes } from '@storybook/theming';
-export const decorators = [
- (StoryFn, { globals: { locale = 'en' } }) => (
+export const decorators: DecoratorFn[] = [
+ (StoryFn, { globals: { locale } }) => (
<>
- {locale}
+ Locale: {locale}
>
),
+ (StoryFn) => (
+
+
+
+ ),
];
+export const parameters = {
+ actions: { argTypesRegex: '^on[A-Z].*' },
+};
+
export const globalTypes = {
locale: {
name: 'Locale',
diff --git a/examples/cra-ts-essentials/package.json b/examples/cra-ts-essentials/package.json
index e00f7dc18c1..4c0059e96b1 100644
--- a/examples/cra-ts-essentials/package.json
+++ b/examples/cra-ts-essentials/package.json
@@ -8,7 +8,7 @@
"eject": "react-scripts eject",
"start": "react-scripts start",
"storybook": "start-storybook -p 9009 --no-manager-cache",
- "test": "react-scripts test"
+ "test": "SKIP_PREFLIGHT_CHECK=true react-scripts test"
},
"browserslist": {
"production": [
@@ -27,6 +27,7 @@
"@types/node": "^14.14.20 || ^16.0.0",
"@types/react": "^16.14.2",
"@types/react-dom": "16.9.10",
+ "formik": "2.2.9",
"global": "^4.4.0",
"react": "16.14.0",
"react-dom": "16.14.0",
@@ -38,8 +39,12 @@
"@storybook/addon-ie11": "0.0.7--canary.5e87b64.0",
"@storybook/addons": "6.5.0-alpha.23",
"@storybook/builder-webpack4": "6.5.0-alpha.23",
+ "@storybook/components": "6.5.0-alpha.23",
+ "@storybook/jest": "^0.0.5",
"@storybook/preset-create-react-app": "^3.1.6",
"@storybook/react": "6.5.0-alpha.23",
+ "@storybook/testing-library": "^0.0.7",
+ "@storybook/theming": "6.5.0-alpha.23",
"webpack": "4"
},
"storybook": {
diff --git a/examples/cra-ts-essentials/src/setupTests.ts b/examples/cra-ts-essentials/src/setupTests.ts
new file mode 100644
index 00000000000..c3f66482a56
--- /dev/null
+++ b/examples/cra-ts-essentials/src/setupTests.ts
@@ -0,0 +1,4 @@
+import { setGlobalConfig } from '@storybook/react';
+import * as globalStorybookConfig from '../.storybook/preview';
+
+setGlobalConfig(globalStorybookConfig);
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.stories.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.stories.tsx
new file mode 100644
index 00000000000..f3cab5e6d2f
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.stories.tsx
@@ -0,0 +1,109 @@
+/* eslint-disable storybook/await-interactions */
+import React from 'react';
+import type { ComponentMeta, ComponentStoryObj } from '@storybook/react';
+import { userEvent, within } from '@storybook/testing-library';
+
+import { AccountForm, AccountFormProps } from './AccountForm';
+
+export default {
+ title: 'CSF3/AccountForm',
+ component: AccountForm,
+ parameters: {
+ layout: 'centered',
+ },
+} as ComponentMeta;
+
+type Story = ComponentStoryObj;
+
+export const Standard: Story = {
+ args: { passwordVerification: false },
+};
+
+export const StandardEmailFilled: Story = {
+ ...Standard,
+ play: async ({ canvasElement }) => {
+ const canvas = within(canvasElement);
+ await userEvent.type(canvas.getByTestId('email'), 'michael@chromatic.com');
+ },
+};
+
+export const StandardEmailFailed: Story = {
+ ...Standard,
+ play: async ({ canvasElement }) => {
+ const canvas = within(canvasElement);
+ await userEvent.type(canvas.getByTestId('email'), 'michael@chromatic.com.com@com');
+ await userEvent.type(canvas.getByTestId('password1'), 'testpasswordthatwontfail');
+ await userEvent.click(canvas.getByTestId('submit'));
+ },
+};
+
+export const StandardPasswordFailed: Story = {
+ ...Standard,
+ play: async (context) => {
+ const canvas = within(context.canvasElement);
+ await StandardEmailFilled.play!(context);
+ await userEvent.type(canvas.getByTestId('password1'), 'asdf');
+ await userEvent.click(canvas.getByTestId('submit'));
+ },
+};
+
+export const StandardFailHover: Story = {
+ ...StandardPasswordFailed,
+ play: async (context) => {
+ const canvas = within(context.canvasElement);
+ await StandardPasswordFailed.play!(context);
+ await sleep(100);
+ await userEvent.hover(canvas.getByTestId('password-error-info'));
+ },
+};
+
+export const Verification: Story = {
+ args: { passwordVerification: true },
+};
+
+export const VerificationPasssword1: Story = {
+ ...Verification,
+ play: async (context) => {
+ const canvas = within(context.canvasElement);
+ await StandardEmailFilled.play!(context);
+ await userEvent.type(canvas.getByTestId('password1'), 'asdfasdf');
+ await userEvent.click(canvas.getByTestId('submit'));
+ },
+};
+
+export const VerificationPasswordMismatch: Story = {
+ ...Verification,
+ play: async (context) => {
+ const canvas = within(context.canvasElement);
+ await StandardEmailFilled.play!(context);
+ await userEvent.type(canvas.getByTestId('password1'), 'asdfasdf');
+ await userEvent.type(canvas.getByTestId('password2'), 'asdf1234');
+ await userEvent.click(canvas.getByTestId('submit'));
+ },
+};
+
+const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));
+
+export const VerificationSuccess: Story = {
+ ...Verification,
+ play: async (context) => {
+ const canvas = within(context.canvasElement);
+ await StandardEmailFilled.play!(context);
+ await sleep(1000);
+ await userEvent.type(canvas.getByTestId('password1'), 'asdfasdf', { delay: 50 });
+ await sleep(1000);
+ await userEvent.type(canvas.getByTestId('password2'), 'asdfasdf', { delay: 50 });
+ await sleep(1000);
+ await userEvent.click(canvas.getByTestId('submit'));
+ },
+};
+
+export const StandardWithRenderFunction: Story = {
+ ...Standard,
+ render: (args: AccountFormProps) => (
+
+
This uses a custom render
+
+
+ ),
+};
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx
new file mode 100644
index 00000000000..a15fe422477
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx
@@ -0,0 +1,25 @@
+import React from 'react';
+import { render, screen } from '@testing-library/react';
+
+import { composeStories, composeStory } from '@storybook/react';
+
+import * as stories from './AccountForm.stories';
+
+const { Standard } = composeStories(stories);
+
+test('renders form', async () => {
+ await render();
+ expect(screen.getByTestId('email')).not.toBe(null);
+});
+
+test('fills input from play function', async () => {
+ // @ts-ignore
+ const StandardEmailFilled = composeStory(stories.StandardEmailFilled, stories.default);
+ const { container } = await render();
+
+ // @ts-ignore
+ await StandardEmailFilled.play({ canvasElement: container });
+
+ const emailInput = screen.getByTestId('email') as HTMLInputElement;
+ expect(emailInput.value).toBe('michael@chromatic.com');
+});
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.tsx
new file mode 100644
index 00000000000..b646a174143
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.tsx
@@ -0,0 +1,552 @@
+import React, { FC, HTMLAttributes, useCallback, useState } from 'react';
+import { keyframes, styled } from '@storybook/theming';
+import {
+ ErrorMessage,
+ Field as FormikInput,
+ Form as FormikForm,
+ Formik,
+ FormikProps,
+} from 'formik';
+import { Icons, WithTooltip } from '@storybook/components';
+
+const errorMap = {
+ email: {
+ required: {
+ normal: 'Please enter your email address',
+ tooltip:
+ 'We do require an email address and a password as a minimum in order to be able to create an account for you to log in with',
+ },
+ format: {
+ normal: 'Please enter a correctly formatted email address',
+ tooltip:
+ 'Your email address is formatted incorrectly and is not correct - please double check for misspelling',
+ },
+ },
+ password: {
+ required: {
+ normal: 'Please enter a password',
+ tooltip: 'A password is requried to create an account',
+ },
+ length: {
+ normal: 'Please enter a password of minimum 6 characters',
+ tooltip:
+ 'For security reasons we enforce a password length of minimum 6 characters - but have no other requirements',
+ },
+ },
+ verifiedPassword: {
+ required: {
+ normal: 'Please verify your password',
+ tooltip:
+ 'Verification of your password is required to ensure no errors in the spelling of the password',
+ },
+ match: {
+ normal: 'Your passwords do not match',
+ tooltip:
+ 'Your verification password has to match your password to make sure you have not misspelled',
+ },
+ },
+};
+
+// https://emailregex.com/
+const email99RegExp = new RegExp(
+ // eslint-disable-next-line no-useless-escape
+ /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/
+);
+
+export interface AccountFormResponse {
+ success: boolean;
+}
+
+export interface AccountFormValues {
+ email: string;
+ password: string;
+}
+
+interface FormValues extends AccountFormValues {
+ verifiedPassword: string;
+}
+
+interface FormErrors {
+ email?: string;
+ emailTooltip?: string;
+ password?: string;
+ passwordTooltip?: string;
+ verifiedPassword?: string;
+ verifiedPasswordTooltip?: string;
+}
+
+export type AccountFormProps = {
+ passwordVerification?: boolean;
+ onSubmit?: (values: AccountFormValues) => void;
+ onTransactionStart?: (values: AccountFormValues) => void;
+ onTransactionEnd?: (values: AccountFormResponse) => void;
+};
+
+export const AccountForm: FC = ({
+ passwordVerification,
+ onSubmit,
+ onTransactionStart,
+ onTransactionEnd,
+}) => {
+ const [state, setState] = useState({
+ transacting: false,
+ transactionSuccess: false,
+ transactionFailure: false,
+ });
+
+ const handleFormSubmit = useCallback(
+ async ({ email, password }: FormValues, { setSubmitting, resetForm }) => {
+ if (onSubmit) {
+ onSubmit({ email, password });
+ }
+
+ if (onTransactionStart) {
+ onTransactionStart({ email, password });
+ }
+
+ setSubmitting(true);
+
+ setState({
+ ...state,
+ transacting: true,
+ });
+
+ await new Promise((r) => setTimeout(r, 2100));
+
+ const success = Math.random() < 1;
+
+ if (onTransactionEnd) {
+ onTransactionEnd({ success });
+ }
+
+ setSubmitting(false);
+ resetForm({ values: { email: '', password: '', verifiedPassword: '' } });
+
+ setState({
+ ...state,
+ transacting: false,
+ transactionSuccess: success === true,
+ transactionFailure: success === false,
+ });
+ },
+ [setState, onTransactionEnd, onTransactionStart]
+ );
+
+ return (
+
+
+
+ Storybook icon
+
+
+
+
+
+
+
+ Storybook
+
+
+
+
+
+ {!state.transactionSuccess && !state.transactionFailure && (
+ Create an account to join the Storybook community
+ )}
+
+ {state.transactionSuccess && !state.transactionFailure && (
+
+
+ Everything is perfect. Your account is ready and we should probably get you started!
+
+ So why don't you get started then?
+ {
+ setState({
+ transacting: false,
+ transactionSuccess: false,
+ transactionFailure: false,
+ });
+ }}
+ >
+ Go back
+
+
+ )}
+ {state.transactionFailure && !state.transactionSuccess && (
+
+ What a mess, this API is not working
+
+ Someone should probably have a stern talking to about this, but it won't be me - coz
+ I'm gonna head out into the nice weather
+
+ {
+ setState({
+ transacting: false,
+ transactionSuccess: false,
+ transactionFailure: false,
+ });
+ }}
+ >
+ Go back
+
+
+ )}
+ {!state.transactionSuccess && !state.transactionFailure && (
+ {
+ const errors: FormErrors = {};
+
+ if (!email) {
+ errors.email = errorMap.email.required.normal;
+ errors.emailTooltip = errorMap.email.required.tooltip;
+ } else {
+ const validEmail = email.match(email99RegExp);
+
+ if (validEmail === null) {
+ errors.email = errorMap.email.format.normal;
+ errors.emailTooltip = errorMap.email.format.tooltip;
+ }
+ }
+
+ if (!password) {
+ errors.password = errorMap.password.required.normal;
+ errors.passwordTooltip = errorMap.password.required.tooltip;
+ } else if (password.length < 6) {
+ errors.password = errorMap.password.length.normal;
+ errors.passwordTooltip = errorMap.password.length.tooltip;
+ }
+
+ if (passwordVerification && !verifiedPassword) {
+ errors.verifiedPassword = errorMap.verifiedPassword.required.normal;
+ errors.verifiedPasswordTooltip = errorMap.verifiedPassword.required.tooltip;
+ } else if (passwordVerification && password !== verifiedPassword) {
+ errors.verifiedPassword = errorMap.verifiedPassword.match.normal;
+ errors.verifiedPasswordTooltip = errorMap.verifiedPassword.match.tooltip;
+ }
+
+ return errors;
+ }}
+ >
+ {({ errors: _errors, isSubmitting, dirty }: FormikProps) => {
+ const errors = _errors as FormErrors;
+
+ return (
+
+ );
+ }}
+
+ )}
+
+
+ );
+};
+
+const Wrapper = styled.section(({ theme }) => ({
+ fontFamily: theme.typography.fonts.base,
+ display: 'flex',
+ flexDirection: 'column',
+ alignItems: 'center',
+ width: 450,
+ padding: 32,
+ backgroundColor: theme.background.content,
+ borderRadius: 7,
+}));
+
+const Brand = styled.div({
+ display: 'flex',
+ alignItems: 'center',
+ justifyContent: 'center',
+});
+
+const Title = styled.svg({
+ height: 40,
+ zIndex: 1,
+ left: -32,
+ position: 'relative',
+});
+
+const logoAnimation = keyframes({
+ '0': {
+ transform: 'rotateY(0deg)',
+ transformOrigin: '50% 5% 0',
+ },
+ '100%': {
+ transform: 'rotateY(360deg)',
+ transformOrigin: '50% 5% 0',
+ },
+});
+
+interface LogoProps {
+ transacting: boolean;
+}
+
+const Logo = styled.svg(
+ ({ transacting }) =>
+ transacting && {
+ animation: `${logoAnimation} 1250ms both infinite`,
+ },
+ { height: 40, zIndex: 10, marginLeft: 32 }
+);
+
+const Introduction = styled.p({
+ marginTop: 20,
+ textAlign: 'center',
+});
+
+const Content = styled.div({
+ display: 'flex',
+ alignItems: 'flex-start',
+ justifyContent: 'center',
+ width: 350,
+ minHeight: 189,
+ marginTop: 8,
+});
+
+const Presentation = styled.div({
+ textAlign: 'center',
+});
+
+const Form = styled(FormikForm)({
+ width: '100%',
+ alignSelf: 'flex-start',
+ '&[aria-disabled="true"]': {
+ opacity: 0.6,
+ },
+});
+
+const FieldWrapper = styled.div({
+ display: 'flex',
+ flexDirection: 'column',
+ justifyContent: 'stretch',
+ marginBottom: 10,
+});
+
+const Label = styled.label({
+ fontSize: 13,
+ fontWeight: 500,
+ marginBottom: 6,
+});
+
+const Input = styled.input(({ theme }) => ({
+ fontSize: 14,
+ color: theme.color.defaultText,
+ padding: '10px 15px',
+ borderRadius: 4,
+ appearance: 'none',
+ outline: 'none',
+ border: '0 none',
+ boxShadow: 'rgb(0 0 0 / 10%) 0px 0px 0px 1px inset',
+ '&:focus': {
+ boxShadow: 'rgb(30 167 253) 0px 0px 0px 1px inset',
+ },
+ '&:active': {
+ boxShadow: 'rgb(30 167 253) 0px 0px 0px 1px inset',
+ },
+ '&[aria-invalid="true"]': {
+ boxShadow: 'rgb(255 68 0) 0px 0px 0px 1px inset',
+ },
+}));
+
+const ErrorWrapper = styled.div({
+ display: 'flex',
+ alignItems: 'flex-start',
+ fontSize: 11,
+ marginTop: 6,
+ cursor: 'help',
+});
+
+const ErrorIcon = styled(Icons)(({ theme }) => ({
+ fill: theme.color.defaultText,
+ opacity: 0.8,
+ marginRight: 6,
+ marginLeft: 2,
+ marginTop: 1,
+}));
+
+const ErrorTooltip = styled.div(({ theme }) => ({
+ fontFamily: theme.typography.fonts.base,
+ fontSize: 13,
+ padding: 8,
+ maxWidth: 350,
+}));
+
+const Actions = styled.div({
+ alignSelf: 'stretch',
+ display: 'flex',
+ justifyContent: 'space-between',
+ marginTop: 24,
+});
+
+const Error = styled(ErrorMessage)({});
+
+interface ButtonProps {
+ dirty?: boolean;
+}
+
+const Button = styled.button({
+ backgroundColor: 'transparent',
+ border: '0 none',
+ outline: 'none',
+ appearance: 'none',
+ fontWeight: 500,
+ fontSize: 12,
+ flexBasis: '50%',
+ cursor: 'pointer',
+ padding: '11px 16px',
+ borderRadius: 4,
+ textTransform: 'uppercase',
+ '&:focus': {
+ textDecoration: 'underline',
+ fontWeight: 700,
+ },
+ '&:active': {
+ textDecoration: 'underline',
+ fontWeight: 700,
+ },
+ '&[aria-disabled="true"]': {
+ cursor: 'default',
+ },
+});
+
+const Submit = styled(Button)(({ theme, dirty }) => ({
+ marginRight: 8,
+ backgroundColor: theme.color.secondary,
+ color: theme.color.inverseText,
+ opacity: dirty ? 1 : 0.6,
+ boxShadow: 'rgb(30 167 253 / 10%) 0 0 0 1px inset',
+}));
+
+const Reset = styled(Button)(({ theme }) => ({
+ marginLeft: 8,
+ boxShadow: 'rgb(30 167 253) 0 0 0 1px inset',
+ color: theme.color.secondary,
+}));
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx
new file mode 100644
index 00000000000..cea434dba36
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx
@@ -0,0 +1,87 @@
+/* eslint-disable storybook/use-storybook-testing-library */
+import React from 'react';
+import { StoryFn as CSF2Story, StoryObj as CSF3Story, Meta } from '@storybook/react';
+import { screen } from '@testing-library/react';
+import userEvent from '@testing-library/user-event';
+
+import { Button, ButtonProps } from './Button';
+
+export default {
+ title: 'Example/Button',
+ component: Button,
+ argTypes: {
+ backgroundColor: { control: 'color' },
+ label: { defaultValue: 'Button' },
+ },
+} as Meta;
+
+const Template: CSF2Story = (args) => ;
+
+export const Primary: CSF3Story = {
+ args: {
+ children: 'foo',
+ size: 'large',
+ primary: true,
+ },
+};
+
+export const Secondary = Template.bind({});
+Secondary.args = {
+ children: 'Children coming from story args!',
+ primary: false,
+};
+
+const getCaptionForLocale = (locale: string) => {
+ switch (locale) {
+ case 'es':
+ return 'Hola!';
+ case 'fr':
+ return 'Bonjour!';
+ case 'kr':
+ return '안녕하세요!';
+ case 'pt':
+ return 'Olá!';
+ default:
+ return 'Hello!';
+ }
+};
+
+export const StoryWithLocale: CSF2Story = (args, { globals: { locale } }) => {
+ const caption = getCaptionForLocale(locale);
+ return ;
+};
+StoryWithLocale.storyName = 'WithLocale';
+
+export const StoryWithParamsAndDecorator: CSF2Story = (args) => {
+ return ;
+};
+StoryWithParamsAndDecorator.args = {
+ children: 'foo',
+};
+StoryWithParamsAndDecorator.parameters = {
+ layout: 'centered',
+};
+StoryWithParamsAndDecorator.decorators = [(StoryFn) => ];
+
+export const CSF3Button: CSF3Story = {
+ args: { children: 'foo' },
+};
+
+export const CSF3ButtonWithRender: CSF3Story = {
+ ...CSF3Button,
+ render: (args: ButtonProps) => (
+
+
I am a custom render function
+
+
+ ),
+};
+
+export const InputFieldFilled: CSF3Story = {
+ render: () => {
+ return ;
+ },
+ play: async (context) => {
+ await userEvent.type(screen.getByRole('textbox'), 'Hello world!');
+ },
+};
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
new file mode 100644
index 00000000000..36321d888f4
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
@@ -0,0 +1,94 @@
+/* eslint-disable @typescript-eslint/ban-ts-comment */
+/* eslint-disable no-shadow */
+import React from 'react';
+import { render, screen } from '@testing-library/react';
+
+import { composeStories, composeStory } from '@storybook/react';
+
+import * as stories from './Button.stories';
+
+// example with composeStories, returns an object with all stories composed with args/decorators
+const { Primary } = composeStories(stories);
+
+// example with composeStory, returns a single story composed with args/decorators
+const Secondary = composeStory(stories.Secondary, stories.default);
+
+test('renders primary button', () => {
+ render(Hello world);
+ const buttonElement = screen.getByText(/Hello world/i);
+ expect(buttonElement).not.toBeNull();
+});
+
+test('reuses args from composed story', () => {
+ render();
+
+ const buttonElement = screen.getByRole('button');
+ expect(buttonElement.textContent).toEqual(Secondary.args!.children);
+});
+
+test('onclick handler is called', async () => {
+ const onClickSpy = jest.fn();
+ render();
+ const buttonElement = screen.getByRole('button');
+ buttonElement.click();
+ expect(onClickSpy).toHaveBeenCalled();
+});
+
+test('reuses args from composeStories', () => {
+ const { getByText } = render();
+ const buttonElement = getByText(/foo/i);
+ expect(buttonElement).not.toBeNull();
+});
+
+describe('GlobalConfig', () => {
+ test('renders with default globalConfig', () => {
+ const WithEnglishText = composeStory(stories.StoryWithLocale, stories.default);
+ const { getByText } = render();
+ const buttonElement = getByText('Hello!');
+ expect(buttonElement).not.toBeNull();
+ });
+
+ test('renders with custom globalConfig', () => {
+ const WithPortugueseText = composeStory(
+ stories.StoryWithLocale,
+ stories.default,
+ // @ts-ignore
+ { globalTypes: { locale: { defaultValue: 'pt' } } }
+ );
+ const { getByText } = render();
+ const buttonElement = getByText('Olá!');
+ expect(buttonElement).not.toBeNull();
+ });
+});
+
+describe('CSF3', () => {
+ test('renders with inferred globalRender', () => {
+ // @ts-ignore
+ const Primary = composeStory(stories.CSF3Button, stories.default);
+
+ render(Hello world);
+ const buttonElement = screen.getByText(/Hello world/i);
+ expect(buttonElement).not.toBeNull();
+ });
+
+ test('renders with custom render function', () => {
+ // @ts-ignore
+ const Primary = composeStory(stories.CSF3ButtonWithRender, stories.default);
+
+ render();
+ expect(screen.getByTestId('custom-render')).not.toBeNull();
+ });
+
+ test('renders with play function', async () => {
+ // @ts-ignore
+ const InputFieldFilled = composeStory(stories.InputFieldFilled, stories.default);
+
+ const { container } = render();
+
+ // @ts-ignore
+ await InputFieldFilled.play({ canvasElement: container });
+
+ const input = screen.getByRole('textbox') as HTMLInputElement;
+ expect(input.value).toEqual('Hello world!');
+ });
+});
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.tsx
new file mode 100644
index 00000000000..d9986afb79a
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.tsx
@@ -0,0 +1,43 @@
+import React from 'react';
+import './button.css';
+
+export interface ButtonProps {
+ /**
+ * Is this the principal call to action on the page?
+ */
+ primary?: boolean;
+ /**
+ * What background color to use
+ */
+ backgroundColor?: string;
+ /**
+ * How large should the button be?
+ */
+ size?: 'small' | 'medium' | 'large';
+ /**
+ * Button contents
+ */
+ children: React.ReactNode;
+ /**
+ * Optional click handler
+ */
+ onClick?: () => void;
+}
+
+/**
+ * Primary UI component for user interaction
+ */
+export const Button: React.FC = (props) => {
+ const { primary = false, size = 'medium', backgroundColor, children, ...otherProps } = props;
+ const mode = primary ? 'storybook-button--primary' : 'storybook-button--secondary';
+ return (
+
+ );
+};
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap b/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
new file mode 100644
index 00000000000..68cf24c2043
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
@@ -0,0 +1,127 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`Renders CSF3Button story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
+exports[`Renders CSF3ButtonWithRender story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+ I am a custom render function
+
+
+
+
+
+`;
+
+exports[`Renders InputFieldFilled story 1`] = `
+
+
+
+`;
+
+exports[`Renders Primary story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
+exports[`Renders Secondary story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
+exports[`Renders StoryWithLocale story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
+exports[`Renders StoryWithParamsAndDecorator story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/button.css b/examples/cra-ts-essentials/src/stories/testing-react/components/button.css
new file mode 100644
index 00000000000..dc91dc76370
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/button.css
@@ -0,0 +1,30 @@
+.storybook-button {
+ font-family: 'Nunito Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+ font-weight: 700;
+ border: 0;
+ border-radius: 3em;
+ cursor: pointer;
+ display: inline-block;
+ line-height: 1;
+}
+.storybook-button--primary {
+ color: white;
+ background-color: #1ea7fd;
+}
+.storybook-button--secondary {
+ color: #333;
+ background-color: transparent;
+ box-shadow: rgba(0, 0, 0, 0.15) 0px 0px 0px 1px inset;
+}
+.storybook-button--small {
+ font-size: 12px;
+ padding: 10px 16px;
+}
+.storybook-button--medium {
+ font-size: 14px;
+ padding: 11px 20px;
+}
+.storybook-button--large {
+ font-size: 16px;
+ padding: 12px 24px;
+}
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
new file mode 100644
index 00000000000..e209c67651a
--- /dev/null
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
@@ -0,0 +1,112 @@
+/* eslint-disable @typescript-eslint/no-non-null-assertion */
+import React from 'react';
+import addons from '@storybook/addons';
+import { render, screen } from '@testing-library/react';
+
+import { composeStories, composeStory } from '@storybook/react';
+
+import * as stories from './Button.stories';
+
+import * as globalConfig from '../../../../.storybook/preview';
+
+const { StoryWithParamsAndDecorator } = composeStories(stories);
+
+test('returns composed args including default values from argtypes', () => {
+ expect(StoryWithParamsAndDecorator.args).toEqual({
+ ...stories.StoryWithParamsAndDecorator.args,
+ label: stories.default!.argTypes!.label!.defaultValue,
+ });
+});
+
+test('returns composed parameters from story', () => {
+ expect(StoryWithParamsAndDecorator.parameters).toEqual(
+ expect.objectContaining({
+ ...stories.StoryWithParamsAndDecorator.parameters,
+ ...globalConfig.parameters,
+ })
+ );
+});
+
+// common in addons that need to communicate between manager and preview
+test('should pass with decorators that need addons channel', () => {
+ // @ts-ignore
+ const PrimaryWithChannels = composeStory(stories.Primary, stories.default, {
+ decorators: [
+ (StoryFn: any) => {
+ addons.getChannel();
+ return ;
+ },
+ ],
+ });
+ render(Hello world);
+ const buttonElement = screen.getByText(/Hello world/i);
+ expect(buttonElement).not.toBeNull();
+});
+
+describe('Unsupported formats', () => {
+ test('should throw error StoryFn.story notation', () => {
+ const UnsupportedStory = () => hello world
;
+ UnsupportedStory.story = { parameters: {} };
+
+ const UnsupportedStoryModule: any = {
+ default: {},
+ UnsupportedStory,
+ };
+
+ expect(() => {
+ composeStories(UnsupportedStoryModule);
+ }).toThrow();
+ });
+
+ test('should throw error with non component stories', () => {
+ const UnsupportedStoryModule: any = {
+ default: {},
+ UnsupportedStory: 123,
+ };
+
+ expect(() => {
+ composeStories(UnsupportedStoryModule);
+ }).toThrow();
+ });
+});
+
+describe('non-story exports', () => {
+ test('should filter non-story exports with excludeStories', () => {
+ const StoryModuleWithNonStoryExports = {
+ default: {
+ title: 'Some/Component',
+ excludeStories: /.*Data/,
+ },
+ LegitimateStory: () => hello world
,
+ mockData: {},
+ };
+
+ const result = composeStories(StoryModuleWithNonStoryExports);
+ expect(Object.keys(result)).not.toContain('mockData');
+ });
+
+ test('should filter non-story exports with includeStories', () => {
+ const StoryModuleWithNonStoryExports = {
+ default: {
+ title: 'Some/Component',
+ includeStories: /.*Story/,
+ },
+ LegitimateStory: () => hello world
,
+ mockData: {},
+ };
+
+ const result = composeStories(StoryModuleWithNonStoryExports);
+ expect(Object.keys(result)).not.toContain('mockData');
+ });
+});
+
+// Batch snapshot testing
+const testCases = Object.values(composeStories(stories)).map((Story) => [
+ // The ! is necessary in Typescript only, as the property is part of a partial type
+ Story.storyName!,
+ Story,
+]);
+test.each(testCases)('Renders %s story', async (_storyName, Story) => {
+ const tree = await render();
+ expect(tree.baseElement).toMatchSnapshot();
+});
diff --git a/examples/cra-ts-essentials/tsconfig.json b/examples/cra-ts-essentials/tsconfig.json
index 450e0014a4e..4e81ac32da5 100644
--- a/examples/cra-ts-essentials/tsconfig.json
+++ b/examples/cra-ts-essentials/tsconfig.json
@@ -6,7 +6,7 @@
"experimentalDecorators": true,
"emitDecoratorMetadata": true,
"jsx": "react",
- "module": "commonjs",
+ "module": "esnext",
"skipLibCheck": true,
"allowSyntheticDefaultImports": true,
"esModuleInterop": true,
@@ -18,9 +18,17 @@
"lib": [
"es2017",
"dom"
- ]
+ ],
+ "allowJs": true,
+ "strict": true,
+ "forceConsistentCasingInFileNames": true,
+ "noFallthroughCasesInSwitch": true,
+ "moduleResolution": "node",
+ "resolveJsonModule": true,
+ "isolatedModules": true,
+ "noEmit": true
},
"include": [
"src"
]
-}
\ No newline at end of file
+}
diff --git a/examples/react-ts/.storybook/preview.js b/examples/react-ts/.storybook/preview.tsx
similarity index 100%
rename from examples/react-ts/.storybook/preview.js
rename to examples/react-ts/.storybook/preview.tsx
diff --git a/examples/react-ts/package.json b/examples/react-ts/package.json
index 731f86269da..f839df256c2 100644
--- a/examples/react-ts/package.json
+++ b/examples/react-ts/package.json
@@ -26,6 +26,7 @@
"@storybook/react": "6.5.0-alpha.23",
"@storybook/theming": "6.5.0-alpha.23",
"@testing-library/dom": "^7.31.2",
+ "@testing-library/react": "12.1.2",
"@testing-library/user-event": "^13.1.9",
"@types/babel__preset-env": "^7",
"@types/react": "^16.14.2",
diff --git a/examples/react-ts/src/AccountForm.stories.tsx b/examples/react-ts/src/AccountForm.stories.tsx
index d670d64ee45..1a4806881f0 100644
--- a/examples/react-ts/src/AccountForm.stories.tsx
+++ b/examples/react-ts/src/AccountForm.stories.tsx
@@ -1,11 +1,12 @@
+/* eslint-disable @typescript-eslint/no-non-null-assertion */
/* eslint-disable storybook/await-interactions */
/* eslint-disable storybook/use-storybook-testing-library */
// @TODO: use addon-interactions and remove the rule disable above
import React from 'react';
import { ComponentStoryObj, ComponentMeta } from '@storybook/react';
-import { screen } from '@testing-library/dom';
+import { screen } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
-import { AccountForm } from './AccountForm';
+import { AccountForm, AccountFormProps } from './AccountForm';
export default {
// Title not needed due to CSF3 auto-title
@@ -20,17 +21,19 @@ export default {
// Standard.args = { passwordVerification: false };
// Standard.play = () => userEvent.type(screen.getByTestId('email'), 'michael@chromatic.com');
-export const Standard: ComponentStoryObj = {
+type Story = ComponentStoryObj;
+
+export const Standard: Story = {
// render: (args: AccountFormProps) => ,
args: { passwordVerification: false },
};
-export const StandardEmailFilled = {
+export const StandardEmailFilled: Story = {
...Standard,
play: () => userEvent.type(screen.getByTestId('email'), 'michael@chromatic.com'),
};
-export const StandardEmailFailed = {
+export const StandardEmailFailed: Story = {
...Standard,
play: async () => {
await userEvent.type(screen.getByTestId('email'), 'michael@chromatic.com.com@com');
@@ -39,41 +42,41 @@ export const StandardEmailFailed = {
},
};
-export const StandardPasswordFailed = {
+export const StandardPasswordFailed: Story = {
...Standard,
- play: async () => {
- await StandardEmailFilled.play();
+ play: async (context) => {
+ await StandardEmailFilled.play!(context);
await userEvent.type(screen.getByTestId('password1'), 'asdf');
await userEvent.click(screen.getByTestId('submit'));
},
};
-export const StandardFailHover = {
+export const StandardFailHover: Story = {
...StandardPasswordFailed,
- play: async () => {
- await StandardPasswordFailed.play();
+ play: async (context) => {
+ await StandardPasswordFailed.play!(context);
await sleep(100);
await userEvent.hover(screen.getByTestId('password-error-info'));
},
};
-export const Verification: ComponentStoryObj = {
+export const Verification: Story = {
args: { passwordVerification: true },
};
-export const VerificationPasssword1 = {
+export const VerificationPasssword1: Story = {
...Verification,
- play: async () => {
- await StandardEmailFilled.play();
+ play: async (context) => {
+ await StandardEmailFilled.play!(context);
await userEvent.type(screen.getByTestId('password1'), 'asdfasdf');
await userEvent.click(screen.getByTestId('submit'));
},
};
-export const VerificationPasswordMismatch = {
+export const VerificationPasswordMismatch: Story = {
...Verification,
- play: async () => {
- await StandardEmailFilled.play();
+ play: async (context) => {
+ await StandardEmailFilled.play!(context);
await userEvent.type(screen.getByTestId('password1'), 'asdfasdf');
await userEvent.type(screen.getByTestId('password2'), 'asdf1234');
await userEvent.click(screen.getByTestId('submit'));
@@ -82,10 +85,10 @@ export const VerificationPasswordMismatch = {
const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));
-export const VerificationSuccess = {
+export const VerificationSuccess: Story = {
...Verification,
- play: async () => {
- await StandardEmailFilled.play();
+ play: async (context) => {
+ await StandardEmailFilled.play!(context);
await sleep(1000);
await userEvent.type(screen.getByTestId('password1'), 'asdfasdf', { delay: 50 });
await sleep(1000);
@@ -94,3 +97,13 @@ export const VerificationSuccess = {
await userEvent.click(screen.getByTestId('submit'));
},
};
+
+export const StandardWithRenderFunction: Story = {
+ ...Standard,
+ render: (args: AccountFormProps) => (
+
+
This uses a custom render
+
+
+ ),
+};
diff --git a/examples/react-ts/src/AccountForm.test.tsx b/examples/react-ts/src/AccountForm.test.tsx
new file mode 100644
index 00000000000..863fabe75e0
--- /dev/null
+++ b/examples/react-ts/src/AccountForm.test.tsx
@@ -0,0 +1,24 @@
+import React from 'react';
+import { render, screen } from '@testing-library/react';
+
+import { composeStories, composeStory } from '@storybook/react';
+
+import * as stories from './AccountForm.stories';
+
+const { Standard } = composeStories(stories);
+
+test('renders form', async () => {
+ await render();
+});
+
+test('fills input from play function', async () => {
+ // @ts-ignore
+ const StandardEmailFilled = composeStory(stories.StandardEmailFilled, stories.default);
+ const { container } = await render();
+
+ // @ts-ignore
+ await StandardEmailFilled.play({ canvasElement: container });
+
+ const emailInput = screen.getByTestId('email') as HTMLInputElement;
+ expect(emailInput.value).toBe('michael@chromatic.com');
+});
diff --git a/yarn.lock b/yarn.lock
index 1c885aabd8f..3161316ab6d 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -7766,6 +7766,7 @@ __metadata:
"@storybook/react": 6.5.0-alpha.23
"@storybook/theming": 6.5.0-alpha.23
"@testing-library/dom": ^7.31.2
+ "@testing-library/react": 12.1.2
"@testing-library/user-event": ^13.1.9
"@types/babel__preset-env": ^7
"@types/react": ^16.14.2
@@ -9093,7 +9094,7 @@ __metadata:
languageName: node
linkType: hard
-"@testing-library/dom@npm:^8.3.0":
+"@testing-library/dom@npm:^8.0.0, @testing-library/dom@npm:^8.3.0":
version: 8.11.2
resolution: "@testing-library/dom@npm:8.11.2"
dependencies:
@@ -9126,6 +9127,19 @@ __metadata:
languageName: node
linkType: hard
+"@testing-library/react@npm:12.1.2":
+ version: 12.1.2
+ resolution: "@testing-library/react@npm:12.1.2"
+ dependencies:
+ "@babel/runtime": ^7.12.5
+ "@testing-library/dom": ^8.0.0
+ peerDependencies:
+ react: "*"
+ react-dom: "*"
+ checksum: c8579252f5f0a23df368253108bbe5b4f26abb9ed5f514746ba6b2ce1a6d09592900526ef6284466af959b50fbb7afa1f37eb2ff629fc91abe70dade3da6cc9a
+ languageName: node
+ linkType: hard
+
"@testing-library/react@npm:^11.2.2":
version: 11.2.7
resolution: "@testing-library/react@npm:11.2.7"
@@ -17648,12 +17662,17 @@ __metadata:
"@storybook/addon-ie11": 0.0.7--canary.5e87b64.0
"@storybook/addons": 6.5.0-alpha.23
"@storybook/builder-webpack4": 6.5.0-alpha.23
+ "@storybook/components": 6.5.0-alpha.23
+ "@storybook/jest": ^0.0.5
"@storybook/preset-create-react-app": ^3.1.6
"@storybook/react": 6.5.0-alpha.23
+ "@storybook/testing-library": ^0.0.7
+ "@storybook/theming": 6.5.0-alpha.23
"@types/jest": ^26.0.16
"@types/node": ^14.14.20 || ^16.0.0
"@types/react": ^16.14.2
"@types/react-dom": 16.9.10
+ formik: 2.2.9
global: ^4.4.0
react: 16.14.0
react-dom: 16.14.0
@@ -23207,7 +23226,7 @@ __metadata:
languageName: node
linkType: hard
-"formik@npm:^2.2.9":
+"formik@npm:2.2.9, formik@npm:^2.2.9":
version: 2.2.9
resolution: "formik@npm:2.2.9"
dependencies:
From d9ac7cb0302c2a4e8f81d28754fca5579ecc0c8c Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 17:17:04 +0100
Subject: [PATCH 004/171] fix storyStore tests
---
lib/store/src/StoryStore.test.ts | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/lib/store/src/StoryStore.test.ts b/lib/store/src/StoryStore.test.ts
index 8b882b373f9..ce28c254f1a 100644
--- a/lib/store/src/StoryStore.test.ts
+++ b/lib/store/src/StoryStore.test.ts
@@ -1,18 +1,18 @@
import { AnyFramework, ProjectAnnotations } from '@storybook/csf';
import global from 'global';
-import { prepareStory } from './prepareStory';
-import { processCSFFile } from './processCSFFile';
+import { prepareStory } from './csf/prepareStory';
+import { processCSFFile } from './csf/processCSFFile';
import { StoryStore } from './StoryStore';
import { StoryIndex } from './types';
import { HooksContext } from './hooks';
// Spy on prepareStory/processCSFFile
-jest.mock('./prepareStory', () => ({
- prepareStory: jest.fn(jest.requireActual('./prepareStory').prepareStory),
+jest.mock('./csf/prepareStory', () => ({
+ prepareStory: jest.fn(jest.requireActual('./csf/prepareStory').prepareStory),
}));
-jest.mock('./processCSFFile', () => ({
- processCSFFile: jest.fn(jest.requireActual('./processCSFFile').processCSFFile),
+jest.mock('./csf/processCSFFile', () => ({
+ processCSFFile: jest.fn(jest.requireActual('./csf/processCSFFile').processCSFFile),
}));
jest.mock('global', () => ({
From 9ccbaacafad5b38985638df7a5fb86e582a3fa55 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 17:17:57 +0100
Subject: [PATCH 005/171] improve testing utility types
---
app/react/src/client/testing/types.ts | 21 +++++++++++----------
lib/store/src/csf/testing-utils/index.ts | 22 ++++++++++------------
2 files changed, 21 insertions(+), 22 deletions(-)
diff --git a/app/react/src/client/testing/types.ts b/app/react/src/client/testing/types.ts
index 574495b0177..ac8d58ce777 100644
--- a/app/react/src/client/testing/types.ts
+++ b/app/react/src/client/testing/types.ts
@@ -16,9 +16,7 @@ import type {
*/
export type GlobalConfig = NormalizedProjectAnnotations;
-export type StoryFile = { default: Meta; __esModule?: boolean; __namedExportsOrder?: any };
-
-export type TestingStory = StoryFn | StoryObj;
+export type StoryFile = { default: Meta; __esModule?: boolean; __namedExportsOrder?: any };
export type TestingStoryPlayContext = Partial> &
Pick;
@@ -29,15 +27,18 @@ export type TestingStoryPlayFn = (
export type StoryFn = OriginalStoryFn & { play: TestingStoryPlayFn };
+export type TestingStory = StoryFn | StoryObj;
+
/**
- * T represents the whole es module of a stories file. K of T means named exports (basically the Story type)
+ * T represents the whole ES module of a stories file. K of T means named exports (basically the Story type)
* 1. pick the keys K of T that have properties that are Story
* 2. infer the actual prop type for each Story
* 3. reconstruct Story with Partial. Story -> Story>
*/
-export type StoriesWithPartialProps = any;
-// {
-// [K in keyof T as T[K] extends TestingStory ? K : never]: T[K] extends TestingStory
-// ? StoryFn>
-// : unknown;
-// };
+export type StoriesWithPartialProps = {
+ // @TODO once we can use Typescript 4.0 do this to exclude nonStory exports:
+ // replace [K in keyof TModule] with [K in keyof TModule as TModule[K] extends TestingStory ? K : never]
+ [K in keyof TModule]: TModule[K] extends TestingStory
+ ? StoryFn>
+ : unknown;
+};
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index 7ebeece296b..bcbcb1f943b 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -4,6 +4,7 @@ import {
AnnotatedStoryFn,
ComponentAnnotations,
Args,
+ StoryContext,
} from '@storybook/csf';
import { prepareStory } from '../prepareStory';
@@ -18,14 +19,11 @@ if (process.env.NODE_ENV === 'test') {
addons.setChannel(mockChannel());
}
-export type StoryFile = { default: any; __esModule?: boolean; __namedExportsOrder?: any };
-
-type Entries = {
- [K in keyof T]: [K, T[K]];
-}[keyof T];
-export function objectEntries(t: T): Entries[] {
- return Object.entries(t) as any;
-}
+export type StoryFile = {
+ default: Record;
+ __esModule?: boolean;
+ __namedExportsOrder?: string[];
+};
export function composeStory<
TFramework extends AnyFramework = AnyFramework,
@@ -56,14 +54,14 @@ export function composeStory<
);
const composedStory = (extraArgs: Partial) => {
- const context = {
+ const context: Partial = {
...preparedStory,
hooks: new HooksContext(),
globals: defaultGlobals,
args: { ...preparedStory.initialArgs, ...extraArgs },
- } as any;
+ };
- return preparedStory.unboundStoryFn(context);
+ return preparedStory.unboundStoryFn(context as StoryContext);
};
composedStory.storyName = story.storyName || story.name;
@@ -80,7 +78,7 @@ export function composeStories(
composeStoryFn: typeof composeStory
) {
const { default: meta, __esModule, __namedExportsOrder, ...stories } = storiesImport;
- const composedStories = objectEntries(stories).reduce((storiesMap, [key, _story]) => {
+ const composedStories = Object.entries(stories).reduce((storiesMap, [key, _story]) => {
if (!isExportStory(key as string, meta)) {
return storiesMap;
}
From bef9e354825e7f60d98524ba840d485dfdcd165f Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 17:18:26 +0100
Subject: [PATCH 006/171] test(store): add tests for testing utils
---
lib/store/src/csf/testing-utils/index.test.ts | 67 +++++++++++++++++++
1 file changed, 67 insertions(+)
create mode 100644 lib/store/src/csf/testing-utils/index.test.ts
diff --git a/lib/store/src/csf/testing-utils/index.test.ts b/lib/store/src/csf/testing-utils/index.test.ts
new file mode 100644
index 00000000000..2eccf6c0b58
--- /dev/null
+++ b/lib/store/src/csf/testing-utils/index.test.ts
@@ -0,0 +1,67 @@
+import { composeStory, composeStories } from '.';
+
+describe('composeStory', () => {
+ const meta = {
+ title: 'Button',
+ parameters: {
+ firstAddon: true,
+ },
+ args: {
+ label: 'Hello World',
+ primary: true,
+ },
+ };
+
+ test('should return story with composed args and parameters', () => {
+ const Story = () => {};
+ Story.args = { primary: true };
+ Story.parameters = {
+ parameters: {
+ secondAddon: true,
+ },
+ };
+
+ const composedStory = composeStory(Story, meta);
+ expect(composedStory.args).toEqual({ ...Story.args, ...meta.args });
+ expect(composedStory.parameters).toEqual(
+ // why is this erroring in TS?
+ expect.objectContaining({ ...Story.parameters, ...meta.parameters })
+ );
+ });
+
+ test('should throw an error if Story is undefined', () => {
+ expect(() => {
+ composeStory(undefined, meta);
+ }).toThrow();
+ });
+});
+
+describe('composeStories', () => {
+ test('should call composeStoryFn with stories', () => {
+ const composeConfigFn = jest.fn((v) => v);
+ const module = {
+ default: {
+ title: 'Button',
+ },
+ StoryOne: () => {},
+ StoryTwo: () => {},
+ };
+ const globalConfig = {};
+ composeStories(module, globalConfig, composeConfigFn);
+ expect(composeConfigFn).toHaveBeenCalledWith(module.StoryOne, module.default, globalConfig);
+ expect(composeConfigFn).toHaveBeenCalledWith(module.StoryTwo, module.default, globalConfig);
+ });
+
+ test('should not call composeStoryFn for non-story exports', () => {
+ const composeConfigFn = jest.fn((v) => v);
+ const module = {
+ default: {
+ title: 'Button',
+ excludeStories: /Data/,
+ },
+ mockData: {},
+ };
+ composeStories(module, {}, composeConfigFn);
+ expect(composeConfigFn).not.toHaveBeenCalled();
+ });
+});
From 7571c0e55e58552b0baf2e6c95e31c911a807892 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Wed, 19 Jan 2022 17:33:50 +0100
Subject: [PATCH 007/171] chore: make example stories CSF version more explicit
in cra-ts-essentials
---
.../components/Button.stories.tsx | 34 +++---
.../testing-react/components/Button.test.tsx | 28 +++--
.../__snapshots__/internals.test.tsx.snap | 110 +++++++++---------
.../components/internals.test.tsx | 14 +--
lib/store/src/csf/testing-utils/index.ts | 1 +
5 files changed, 93 insertions(+), 94 deletions(-)
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx
index cea434dba36..bd5f81fb0d5 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.stories.tsx
@@ -17,16 +17,8 @@ export default {
const Template: CSF2Story = (args) => ;
-export const Primary: CSF3Story = {
- args: {
- children: 'foo',
- size: 'large',
- primary: true,
- },
-};
-
-export const Secondary = Template.bind({});
-Secondary.args = {
+export const CSF2Secondary = Template.bind({});
+CSF2Secondary.args = {
children: 'Children coming from story args!',
primary: false,
};
@@ -46,22 +38,30 @@ const getCaptionForLocale = (locale: string) => {
}
};
-export const StoryWithLocale: CSF2Story = (args, { globals: { locale } }) => {
+export const CSF2StoryWithLocale: CSF2Story = (args, { globals: { locale } }) => {
const caption = getCaptionForLocale(locale);
return ;
};
-StoryWithLocale.storyName = 'WithLocale';
+CSF2StoryWithLocale.storyName = 'WithLocale';
-export const StoryWithParamsAndDecorator: CSF2Story = (args) => {
+export const CSF2StoryWithParamsAndDecorator: CSF2Story = (args) => {
return ;
};
-StoryWithParamsAndDecorator.args = {
+CSF2StoryWithParamsAndDecorator.args = {
children: 'foo',
};
-StoryWithParamsAndDecorator.parameters = {
+CSF2StoryWithParamsAndDecorator.parameters = {
layout: 'centered',
};
-StoryWithParamsAndDecorator.decorators = [(StoryFn) => ];
+CSF2StoryWithParamsAndDecorator.decorators = [(StoryFn) => ];
+
+export const CSF3Primary: CSF3Story = {
+ args: {
+ children: 'foo',
+ size: 'large',
+ primary: true,
+ },
+};
export const CSF3Button: CSF3Story = {
args: { children: 'foo' },
@@ -77,7 +77,7 @@ export const CSF3ButtonWithRender: CSF3Story = {
),
};
-export const InputFieldFilled: CSF3Story = {
+export const CSF3InputFieldFilled: CSF3Story = {
render: () => {
return ;
},
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
index 36321d888f4..961ff4576dd 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
@@ -8,13 +8,13 @@ import { composeStories, composeStory } from '@storybook/react';
import * as stories from './Button.stories';
// example with composeStories, returns an object with all stories composed with args/decorators
-const { Primary } = composeStories(stories);
+const { CSF3Primary } = composeStories(stories);
// example with composeStory, returns a single story composed with args/decorators
-const Secondary = composeStory(stories.Secondary, stories.default);
+const Secondary = composeStory(stories.CSF2Secondary, stories.default);
test('renders primary button', () => {
- render(Hello world);
+ render(Hello world);
const buttonElement = screen.getByText(/Hello world/i);
expect(buttonElement).not.toBeNull();
});
@@ -23,7 +23,7 @@ test('reuses args from composed story', () => {
render();
const buttonElement = screen.getByRole('button');
- expect(buttonElement.textContent).toEqual(Secondary.args!.children);
+ expect(buttonElement.textContent).toEqual(Secondary.args.children);
});
test('onclick handler is called', async () => {
@@ -35,26 +35,24 @@ test('onclick handler is called', async () => {
});
test('reuses args from composeStories', () => {
- const { getByText } = render();
+ const { getByText } = render();
const buttonElement = getByText(/foo/i);
expect(buttonElement).not.toBeNull();
});
describe('GlobalConfig', () => {
test('renders with default globalConfig', () => {
- const WithEnglishText = composeStory(stories.StoryWithLocale, stories.default);
+ const WithEnglishText = composeStory(stories.CSF2StoryWithLocale, stories.default);
const { getByText } = render();
const buttonElement = getByText('Hello!');
expect(buttonElement).not.toBeNull();
});
test('renders with custom globalConfig', () => {
- const WithPortugueseText = composeStory(
- stories.StoryWithLocale,
- stories.default,
- // @ts-ignore
- { globalTypes: { locale: { defaultValue: 'pt' } } }
- );
+ const WithPortugueseText = composeStory(stories.CSF2StoryWithLocale, stories.default, {
+ // @TODO globals should be able to get custom stuff like custom globalTypes. Currently there is a type error
+ globalTypes: { locale: { defaultValue: 'pt' } } as any,
+ });
const { getByText } = render();
const buttonElement = getByText('Olá!');
expect(buttonElement).not.toBeNull();
@@ -81,12 +79,12 @@ describe('CSF3', () => {
test('renders with play function', async () => {
// @ts-ignore
- const InputFieldFilled = composeStory(stories.InputFieldFilled, stories.default);
+ const CSF3InputFieldFilled = composeStory(stories.CSF3InputFieldFilled, stories.default);
- const { container } = render();
+ const { container } = render();
// @ts-ignore
- await InputFieldFilled.play({ canvasElement: container });
+ await CSF3InputFieldFilled.play({ canvasElement: container });
const input = screen.getByRole('textbox') as HTMLInputElement;
expect(input.value).toEqual('Hello world!');
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap b/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
index 68cf24c2043..dd0a3d4183f 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
@@ -1,5 +1,58 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
+exports[`Renders CSF2Secondary story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
+exports[`Renders CSF2StoryWithLocale story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
+exports[`Renders CSF2StoryWithParamsAndDecorator story 1`] = `
+
+
+
+ Locale:
+ en
+
+
+
+
+`;
+
exports[`Renders CSF3Button story 1`] = `
@@ -43,7 +96,7 @@ exports[`Renders CSF3ButtonWithRender story 1`] = `
`;
-exports[`Renders InputFieldFilled story 1`] = `
+exports[`Renders CSF3InputFieldFilled story 1`] = `
@@ -55,7 +108,7 @@ exports[`Renders InputFieldFilled story 1`] = `
`;
-exports[`Renders Primary story 1`] = `
+exports[`Renders CSF3Primary story 1`] = `
@@ -72,56 +125,3 @@ exports[`Renders Primary story 1`] = `
`;
-
-exports[`Renders Secondary story 1`] = `
-
-
-
- Locale:
- en
-
-
-
-
-`;
-
-exports[`Renders StoryWithLocale story 1`] = `
-
-
-
- Locale:
- en
-
-
-
-
-`;
-
-exports[`Renders StoryWithParamsAndDecorator story 1`] = `
-
-
-
- Locale:
- en
-
-
-
-
-`;
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
index e209c67651a..9dfda44ecbf 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
@@ -9,19 +9,19 @@ import * as stories from './Button.stories';
import * as globalConfig from '../../../../.storybook/preview';
-const { StoryWithParamsAndDecorator } = composeStories(stories);
+const { CSF2StoryWithParamsAndDecorator } = composeStories(stories);
test('returns composed args including default values from argtypes', () => {
- expect(StoryWithParamsAndDecorator.args).toEqual({
- ...stories.StoryWithParamsAndDecorator.args,
- label: stories.default!.argTypes!.label!.defaultValue,
+ expect(CSF2StoryWithParamsAndDecorator.args).toEqual({
+ ...stories.CSF2StoryWithParamsAndDecorator.args,
+ label: stories.default.argTypes!.label!.defaultValue,
});
});
test('returns composed parameters from story', () => {
- expect(StoryWithParamsAndDecorator.parameters).toEqual(
+ expect(CSF2StoryWithParamsAndDecorator.parameters).toEqual(
expect.objectContaining({
- ...stories.StoryWithParamsAndDecorator.parameters,
+ ...stories.CSF2StoryWithParamsAndDecorator.parameters,
...globalConfig.parameters,
})
);
@@ -30,7 +30,7 @@ test('returns composed parameters from story', () => {
// common in addons that need to communicate between manager and preview
test('should pass with decorators that need addons channel', () => {
// @ts-ignore
- const PrimaryWithChannels = composeStory(stories.Primary, stories.default, {
+ const PrimaryWithChannels = composeStory(stories.CSF3Primary, stories.default, {
decorators: [
(StoryFn: any) => {
addons.getChannel();
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index bcbcb1f943b..d11f5ed68bd 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -66,6 +66,7 @@ export function composeStory<
composedStory.storyName = story.storyName || story.name;
composedStory.args = preparedStory.initialArgs;
+ // @TODO this should be partial play fn that works by just passing canvasElement
composedStory.play = preparedStory.playFunction;
composedStory.parameters = preparedStory.parameters;
From 4a606b5c2ba457c91a4c10ec5ab24df0516da247 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Fri, 21 Jan 2022 14:16:00 +0100
Subject: [PATCH 008/171] support csf 2 and csf 3 in testing utilities
---
app/react/src/client/testing/index.ts | 5 ++---
lib/store/src/csf/testing-utils/index.ts | 10 +++++++---
2 files changed, 9 insertions(+), 6 deletions(-)
diff --git a/app/react/src/client/testing/index.ts b/app/react/src/client/testing/index.ts
index 7ce12904ea0..902a4a40f90 100644
--- a/app/react/src/client/testing/index.ts
+++ b/app/react/src/client/testing/index.ts
@@ -4,11 +4,10 @@ import {
} from '@storybook/store';
// eslint-disable-next-line import/no-extraneous-dependencies
import { composeConfigs } from '@storybook/preview-web';
-import type { AnnotatedStoryFn } from '@storybook/csf';
import { render } from '../preview/render';
import type { Meta, ReactFramework } from '../preview/types-6-0';
-import type { StoriesWithPartialProps, GlobalConfig, StoryFile } from './types';
+import type { StoriesWithPartialProps, GlobalConfig, StoryFile, TestingStory } from './types';
const defaultGlobalConfig: GlobalConfig = {
render,
@@ -38,7 +37,7 @@ export function setGlobalConfig(config: GlobalConfig) {
}
export function composeStory(
- story: AnnotatedStoryFn,
+ story: TestingStory,
meta: Meta,
globalConfig: GlobalConfig = globalStorybookConfig
) {
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index d11f5ed68bd..acdbb1cf984 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -2,6 +2,7 @@ import {
isExportStory,
AnyFramework,
AnnotatedStoryFn,
+ StoryAnnotations,
ComponentAnnotations,
Args,
StoryContext,
@@ -25,11 +26,15 @@ export type StoryFile = {
__namedExportsOrder?: string[];
};
+type PartialPlayFn = (
+ context: Partial & Pick
+) => Promise | void;
+
export function composeStory<
TFramework extends AnyFramework = AnyFramework,
TArgs extends Args = Args
>(
- story: AnnotatedStoryFn,
+ story: AnnotatedStoryFn | StoryAnnotations,
meta: ComponentAnnotations,
globalConfig: NormalizedProjectAnnotations = {}
) {
@@ -66,8 +71,7 @@ export function composeStory<
composedStory.storyName = story.storyName || story.name;
composedStory.args = preparedStory.initialArgs;
- // @TODO this should be partial play fn that works by just passing canvasElement
- composedStory.play = preparedStory.playFunction;
+ composedStory.play = preparedStory.playFunction as PartialPlayFn;
composedStory.parameters = preparedStory.parameters;
return composedStory;
From 682dfe34d647071385010e373754ce3d4c6fd0b2 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Fri, 21 Jan 2022 14:25:16 +0100
Subject: [PATCH 009/171] chore(storybook): remove now unnecessary ts-ignores
---
.../stories/testing-react/components/AccountForm.test.tsx | 2 --
.../src/stories/testing-react/components/Button.test.tsx | 7 -------
.../stories/testing-react/components/internals.test.tsx | 1 -
3 files changed, 10 deletions(-)
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx
index a15fe422477..4dd5f7bef31 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/AccountForm.test.tsx
@@ -13,11 +13,9 @@ test('renders form', async () => {
});
test('fills input from play function', async () => {
- // @ts-ignore
const StandardEmailFilled = composeStory(stories.StandardEmailFilled, stories.default);
const { container } = await render();
- // @ts-ignore
await StandardEmailFilled.play({ canvasElement: container });
const emailInput = screen.getByTestId('email') as HTMLInputElement;
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
index 961ff4576dd..b35aac700fb 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/Button.test.tsx
@@ -1,5 +1,3 @@
-/* eslint-disable @typescript-eslint/ban-ts-comment */
-/* eslint-disable no-shadow */
import React from 'react';
import { render, screen } from '@testing-library/react';
@@ -50,7 +48,6 @@ describe('GlobalConfig', () => {
test('renders with custom globalConfig', () => {
const WithPortugueseText = composeStory(stories.CSF2StoryWithLocale, stories.default, {
- // @TODO globals should be able to get custom stuff like custom globalTypes. Currently there is a type error
globalTypes: { locale: { defaultValue: 'pt' } } as any,
});
const { getByText } = render();
@@ -61,7 +58,6 @@ describe('GlobalConfig', () => {
describe('CSF3', () => {
test('renders with inferred globalRender', () => {
- // @ts-ignore
const Primary = composeStory(stories.CSF3Button, stories.default);
render(Hello world);
@@ -70,7 +66,6 @@ describe('CSF3', () => {
});
test('renders with custom render function', () => {
- // @ts-ignore
const Primary = composeStory(stories.CSF3ButtonWithRender, stories.default);
render();
@@ -78,12 +73,10 @@ describe('CSF3', () => {
});
test('renders with play function', async () => {
- // @ts-ignore
const CSF3InputFieldFilled = composeStory(stories.CSF3InputFieldFilled, stories.default);
const { container } = render();
- // @ts-ignore
await CSF3InputFieldFilled.play({ canvasElement: container });
const input = screen.getByRole('textbox') as HTMLInputElement;
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx b/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
index 9dfda44ecbf..d03cc4e1bdf 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/internals.test.tsx
@@ -29,7 +29,6 @@ test('returns composed parameters from story', () => {
// common in addons that need to communicate between manager and preview
test('should pass with decorators that need addons channel', () => {
- // @ts-ignore
const PrimaryWithChannels = composeStory(stories.CSF3Primary, stories.default, {
decorators: [
(StoryFn: any) => {
From e12ccb14ca1153139b7c49d3d62bf80cedec7353 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Fri, 21 Jan 2022 14:57:09 +0100
Subject: [PATCH 010/171] add jsdoc to testing utilities
---
app/react/src/client/testing/index.ts | 53 ++++++++++++++++++++++++++-
1 file changed, 52 insertions(+), 1 deletion(-)
diff --git a/app/react/src/client/testing/index.ts b/app/react/src/client/testing/index.ts
index 902a4a40f90..6ebec57206c 100644
--- a/app/react/src/client/testing/index.ts
+++ b/app/react/src/client/testing/index.ts
@@ -24,7 +24,7 @@ let globalStorybookConfig = {
* Example:
*```jsx
* // setup.js (for jest)
- * import { setGlobalConfig } from '@storybook/testing-react';
+ * import { setGlobalConfig } from '@storybook/react';
* import * as globalStorybookConfig from './.storybook/preview';
*
* setGlobalConfig(globalStorybookConfig);
@@ -36,6 +36,32 @@ export function setGlobalConfig(config: GlobalConfig) {
globalStorybookConfig = composeConfigs([defaultGlobalConfig, config]) as GlobalConfig;
}
+/**
+ * Function that will receive a story along with meta (e.g. a default export from a .stories file)
+ * and optionally a globalConfig e.g. (import * from '../.storybook/preview)
+ * and will return a composed component that has all args/parameters/decorators/etc combined and applied to it.
+ *
+ *
+ * It's very useful for reusing a story in scenarios outside of Storybook like unit testing.
+ *
+ * Example:
+ *```jsx
+ * import { render } from '@testing-library/react';
+ * import { composeStory } from '@storybook/react';
+ * import Meta, { Primary as PrimaryStory } from './Button.stories';
+ *
+ * const Primary = composeStory(PrimaryStory, Meta);
+ *
+ * test('renders primary button with Hello World', () => {
+ * const { getByText } = render(Hello world);
+ * expect(getByText(/Hello world/i)).not.toBeNull();
+ * });
+ *```
+ *
+ * @param story
+ * @param meta - e.g. (import Meta from './Button.stories')
+ * @param [globalConfig] - e.g. (import * as globalConfig from '../.storybook/preview') this can be applied automatically if you use `setGlobalConfig` in your setup files.
+ */
export function composeStory(
story: TestingStory,
meta: Meta,
@@ -46,6 +72,31 @@ export function composeStory(
return originalComposeStory(story, meta, projectAnnotations);
}
+/**
+ * Function that will receive a stories import (e.g. `import * as stories from './Button.stories'`)
+ * and optionally a globalConfig (e.g. `import * from '../.storybook/preview`)
+ * and will return an object containing all the stories passed, but now as a composed component that has all args/parameters/decorators/etc combined and applied to it.
+ *
+ *
+ * It's very useful for reusing stories in scenarios outside of Storybook like unit testing.
+ *
+ * Example:
+ *```jsx
+ * import { render } from '@testing-library/react';
+ * import { composeStories } from '@storybook/react';
+ * import * as stories from './Button.stories';
+ *
+ * const { Primary, Secondary } = composeStories(stories);
+ *
+ * test('renders primary button with Hello World', () => {
+ * const { getByText } = render(Hello world);
+ * expect(getByText(/Hello world/i)).not.toBeNull();
+ * });
+ *```
+ *
+ * @param storiesImport - e.g. (import * as stories from './Button.stories')
+ * @param [globalConfig] - e.g. (import * as globalConfig from '../.storybook/preview') this can be applied automatically if you use `setGlobalConfig` in your setup files.
+ */
export function composeStories(
storiesImport: TModule,
globalConfig?: GlobalConfig
From 545b473e6b24a12b1c5f9d18d7740f6a9dfa2b78 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Mon, 24 Jan 2022 12:24:49 +0100
Subject: [PATCH 011/171] refactor(story-store): move
normalizeProjectAnnotations to csf
---
lib/store/src/StoryStore.ts | 26 +------------
lib/store/src/csf/index.ts | 2 +
.../src/csf/normalizeComponentAnnotations.ts | 22 +++++++++++
.../src/csf/normalizeProjectAnnotations.ts | 37 ++++++++++---------
lib/store/src/csf/processCSFFile.ts | 9 ++---
lib/store/src/csf/testing-utils/index.ts | 6 +--
6 files changed, 51 insertions(+), 51 deletions(-)
create mode 100644 lib/store/src/csf/normalizeComponentAnnotations.ts
diff --git a/lib/store/src/StoryStore.ts b/lib/store/src/StoryStore.ts
index c1a7e79cf52..f46602d2953 100644
--- a/lib/store/src/StoryStore.ts
+++ b/lib/store/src/StoryStore.ts
@@ -17,7 +17,7 @@ import { SynchronousPromise } from 'synchronous-promise';
import { StoryIndexStore } from './StoryIndexStore';
import { ArgsStore } from './ArgsStore';
import { GlobalsStore } from './GlobalsStore';
-import { normalizeInputTypes, processCSFFile, prepareStory } from './csf';
+import { processCSFFile, prepareStory, normalizeProjectAnnotations } from './csf';
import {
CSFFile,
ModuleImportFn,
@@ -31,35 +31,11 @@ import {
V2CompatIndexEntry,
} from './types';
import { HooksContext } from './hooks';
-import { inferArgTypes } from './inferArgTypes';
-import { inferControls } from './inferControls';
// TODO -- what are reasonable values for these?
const CSF_CACHE_SIZE = 1000;
const STORY_CACHE_SIZE = 10000;
-function normalizeProjectAnnotations({
- argTypes,
- globalTypes,
- argTypesEnhancers,
- ...annotations
-}: ProjectAnnotations): NormalizedProjectAnnotations {
- return {
- ...(argTypes && { argTypes: normalizeInputTypes(argTypes) }),
- ...(globalTypes && { globalTypes: normalizeInputTypes(globalTypes) }),
- argTypesEnhancers: [
- ...(argTypesEnhancers || []),
- inferArgTypes,
- // inferControls technically should only run if the user is using the controls addon,
- // and so should be added by a preset there. However, as it seems some code relies on controls
- // annotations (in particular the angular implementation's `cleanArgsDecorator`), for backwards
- // compatibility reasons, we will leave this in the store until 7.0
- inferControls,
- ],
- ...annotations,
- };
-}
-
export class StoryStore {
storyIndex: StoryIndexStore;
diff --git a/lib/store/src/csf/index.ts b/lib/store/src/csf/index.ts
index 4e7870ccd0b..eb26845e410 100644
--- a/lib/store/src/csf/index.ts
+++ b/lib/store/src/csf/index.ts
@@ -2,4 +2,6 @@ export * from './normalizeInputTypes';
export * from './normalizeStory';
export * from './processCSFFile';
export * from './prepareStory';
+export * from './normalizeComponentAnnotations';
+export * from './normalizeProjectAnnotations';
export * from './testing-utils';
diff --git a/lib/store/src/csf/normalizeComponentAnnotations.ts b/lib/store/src/csf/normalizeComponentAnnotations.ts
new file mode 100644
index 00000000000..ce3f821ac8e
--- /dev/null
+++ b/lib/store/src/csf/normalizeComponentAnnotations.ts
@@ -0,0 +1,22 @@
+import { sanitize, AnyFramework } from '@storybook/csf';
+
+import { ModuleExports, NormalizedComponentAnnotations } from '../types';
+import { normalizeInputTypes } from './normalizeInputTypes';
+
+export function normalizeComponentAnnotations(
+ defaultExport: ModuleExports['default'],
+ title: string = defaultExport.title,
+ importPath?: string
+): NormalizedComponentAnnotations {
+ const { id, argTypes } = defaultExport;
+ return {
+ id: sanitize(id || title),
+ ...defaultExport,
+ title,
+ ...(argTypes && { argTypes: normalizeInputTypes(argTypes) }),
+ parameters: {
+ fileName: importPath,
+ ...defaultExport.parameters,
+ },
+ };
+}
diff --git a/lib/store/src/csf/normalizeProjectAnnotations.ts b/lib/store/src/csf/normalizeProjectAnnotations.ts
index dd3eba96347..2b707c03f31 100644
--- a/lib/store/src/csf/normalizeProjectAnnotations.ts
+++ b/lib/store/src/csf/normalizeProjectAnnotations.ts
@@ -1,22 +1,25 @@
-import { sanitize, AnyFramework } from '@storybook/csf';
+import { AnyFramework, ProjectAnnotations } from '@storybook/csf';
+import { inferControls, NormalizedProjectAnnotations, normalizeInputTypes } from '..';
+import { inferArgTypes } from '../inferArgTypes';
-import { ModuleExports, NormalizedComponentAnnotations } from '../types';
-import { normalizeInputTypes } from './normalizeInputTypes';
-
-export function normalizeProjectAnnotations(
- defaultExport: ModuleExports['default'],
- title: string = defaultExport.title,
- importPath?: string
-): NormalizedComponentAnnotations {
- const { id, argTypes } = defaultExport;
+export function normalizeProjectAnnotations({
+ argTypes,
+ globalTypes,
+ argTypesEnhancers,
+ ...annotations
+}: ProjectAnnotations): NormalizedProjectAnnotations {
return {
- id: sanitize(id || title),
- ...defaultExport,
- title,
...(argTypes && { argTypes: normalizeInputTypes(argTypes) }),
- parameters: {
- fileName: importPath,
- ...defaultExport.parameters,
- },
+ ...(globalTypes && { globalTypes: normalizeInputTypes(globalTypes) }),
+ argTypesEnhancers: [
+ ...(argTypesEnhancers || []),
+ inferArgTypes,
+ // inferControls technically should only run if the user is using the controls addon,
+ // and so should be added by a preset there. However, as it seems some code relies on controls
+ // annotations (in particular the angular implementation's `cleanArgsDecorator`), for backwards
+ // compatibility reasons, we will leave this in the store until 7.0
+ inferControls,
+ ],
+ ...annotations,
};
}
diff --git a/lib/store/src/csf/processCSFFile.ts b/lib/store/src/csf/processCSFFile.ts
index 0e6d3207553..a905f49dc67 100644
--- a/lib/store/src/csf/processCSFFile.ts
+++ b/lib/store/src/csf/processCSFFile.ts
@@ -4,7 +4,7 @@ import { logger } from '@storybook/client-logger';
import { ModuleExports, CSFFile, NormalizedComponentAnnotations } from '../types';
import { normalizeStory } from './normalizeStory';
import { Path } from '..';
-import { normalizeProjectAnnotations } from './normalizeProjectAnnotations';
+import { normalizeComponentAnnotations } from './normalizeComponentAnnotations';
const checkGlobals = (parameters: Parameters) => {
const { globals, globalTypes } = parameters;
@@ -40,11 +40,8 @@ export function processCSFFile(
): CSFFile {
const { default: defaultExport, __namedExportsOrder, ...namedExports } = moduleExports;
- const meta: NormalizedComponentAnnotations = normalizeProjectAnnotations(
- defaultExport,
- title,
- importPath
- );
+ const meta: NormalizedComponentAnnotations =
+ normalizeComponentAnnotations(defaultExport, title, importPath);
checkDisallowedParameters(meta.parameters);
const csfFile: CSFFile = { meta, stories: {} };
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index acdbb1cf984..c1c2f4c534f 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -10,9 +10,9 @@ import {
import { prepareStory } from '../prepareStory';
import { normalizeStory } from '../normalizeStory';
-import { normalizeProjectAnnotations } from '../normalizeProjectAnnotations';
import { HooksContext } from '../../hooks';
-import { NormalizedProjectAnnotations } from '../..';
+import { normalizeComponentAnnotations } from '../normalizeComponentAnnotations';
+import type { NormalizedProjectAnnotations } from '../../types';
if (process.env.NODE_ENV === 'test') {
// eslint-disable-next-line global-require
@@ -42,7 +42,7 @@ export function composeStory<
throw new Error('Expected a story but received undefined.');
}
- const normalizedMeta = normalizeProjectAnnotations(meta);
+ const normalizedMeta = normalizeComponentAnnotations(meta);
const normalizedStory = normalizeStory(story.name, story, normalizedMeta);
From 6a8e6859ac07ecb123a751e757b6b150cc438803 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Mon, 24 Jan 2022 18:12:04 +0100
Subject: [PATCH 012/171] refactor: move setGlobalConfig to storybook/store
---
app/react/src/client/testing/index.ts | 39 ++++++++++++------------
app/react/src/client/testing/types.ts | 9 ------
lib/store/src/csf/testing-utils/index.ts | 26 +++++++++++++---
3 files changed, 41 insertions(+), 33 deletions(-)
diff --git a/app/react/src/client/testing/index.ts b/app/react/src/client/testing/index.ts
index 6ebec57206c..d92c24728c2 100644
--- a/app/react/src/client/testing/index.ts
+++ b/app/react/src/client/testing/index.ts
@@ -1,21 +1,13 @@
import {
composeStory as originalComposeStory,
composeStories as originalComposeStories,
+ setGlobalConfig as originalSetGlobalConfig,
} from '@storybook/store';
-// eslint-disable-next-line import/no-extraneous-dependencies
-import { composeConfigs } from '@storybook/preview-web';
+import { ProjectAnnotations } from '@storybook/csf';
+
import { render } from '../preview/render';
-
import type { Meta, ReactFramework } from '../preview/types-6-0';
-import type { StoriesWithPartialProps, GlobalConfig, StoryFile, TestingStory } from './types';
-
-const defaultGlobalConfig: GlobalConfig = {
- render,
-};
-
-let globalStorybookConfig = {
- ...defaultGlobalConfig,
-};
+import type { StoriesWithPartialProps, StoryFile, TestingStory } from './types';
/** Function that sets the globalConfig of your storybook. The global config is the preview module of your .storybook folder.
*
@@ -32,10 +24,14 @@ let globalStorybookConfig = {
*
* @param config - e.g. (import * as globalConfig from '../.storybook/preview')
*/
-export function setGlobalConfig(config: GlobalConfig) {
- globalStorybookConfig = composeConfigs([defaultGlobalConfig, config]) as GlobalConfig;
+export function setGlobalConfig(config: ProjectAnnotations) {
+ originalSetGlobalConfig(config);
}
+const defaultGlobalConfig: ProjectAnnotations = {
+ render,
+};
+
/**
* Function that will receive a story along with meta (e.g. a default export from a .stories file)
* and optionally a globalConfig e.g. (import * from '../.storybook/preview)
@@ -65,11 +61,14 @@ export function setGlobalConfig(config: GlobalConfig) {
export function composeStory(
story: TestingStory,
meta: Meta,
- globalConfig: GlobalConfig = globalStorybookConfig
+ globalConfig?: ProjectAnnotations
) {
- const projectAnnotations = { ...defaultGlobalConfig, ...globalConfig };
-
- return originalComposeStory(story, meta, projectAnnotations);
+ return originalComposeStory(
+ story,
+ meta,
+ globalConfig,
+ defaultGlobalConfig
+ );
}
/**
@@ -99,9 +98,9 @@ export function composeStory(
*/
export function composeStories(
storiesImport: TModule,
- globalConfig?: GlobalConfig
+ globalConfig?: ProjectAnnotations
) {
const composedStories = originalComposeStories(storiesImport, globalConfig, composeStory);
- return (composedStories as unknown) as Omit, keyof StoryFile>;
+ return composedStories as unknown as Omit, keyof StoryFile>;
}
diff --git a/app/react/src/client/testing/types.ts b/app/react/src/client/testing/types.ts
index ac8d58ce777..4c97e776c26 100644
--- a/app/react/src/client/testing/types.ts
+++ b/app/react/src/client/testing/types.ts
@@ -1,4 +1,3 @@
-import { NormalizedProjectAnnotations } from '@storybook/store';
import type {
StoryFn as OriginalStoryFn,
StoryObj,
@@ -8,14 +7,6 @@ import type {
ReactFramework,
} from '../preview/types-6-0';
-/**
- * Object representing the preview.ts module
- *
- * Used in storybook testing utilities.
- * @see [Unit testing with Storybook](https://storybook.js.org/docs/react/workflows/unit-testing)
- */
-export type GlobalConfig = NormalizedProjectAnnotations;
-
export type StoryFile = { default: Meta; __esModule?: boolean; __namedExportsOrder?: any };
export type TestingStoryPlayContext = Partial> &
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index c1c2f4c534f..fdb988e362c 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -4,6 +4,7 @@ import {
AnnotatedStoryFn,
StoryAnnotations,
ComponentAnnotations,
+ ProjectAnnotations,
Args,
StoryContext,
} from '@storybook/csf';
@@ -12,7 +13,7 @@ import { prepareStory } from '../prepareStory';
import { normalizeStory } from '../normalizeStory';
import { HooksContext } from '../../hooks';
import { normalizeComponentAnnotations } from '../normalizeComponentAnnotations';
-import type { NormalizedProjectAnnotations } from '../../types';
+import { normalizeProjectAnnotations } from '..';
if (process.env.NODE_ENV === 'test') {
// eslint-disable-next-line global-require
@@ -26,6 +27,14 @@ export type StoryFile = {
__namedExportsOrder?: string[];
};
+let GLOBAL_STORYBOOK_CONFIG = {};
+
+export function setGlobalConfig(
+ config: ProjectAnnotations
+) {
+ GLOBAL_STORYBOOK_CONFIG = config;
+}
+
type PartialPlayFn = (
context: Partial & Pick
) => Promise | void;
@@ -36,17 +45,26 @@ export function composeStory<
>(
story: AnnotatedStoryFn | StoryAnnotations,
meta: ComponentAnnotations,
- globalConfig: NormalizedProjectAnnotations = {}
+ globalConfig: ProjectAnnotations = GLOBAL_STORYBOOK_CONFIG,
+ defaultConfig: ProjectAnnotations = {}
) {
if (story === undefined) {
throw new Error('Expected a story but received undefined.');
}
+ const projectAnnotations = { ...defaultConfig, ...globalConfig };
+
const normalizedMeta = normalizeComponentAnnotations(meta);
const normalizedStory = normalizeStory(story.name, story, normalizedMeta);
- const preparedStory = prepareStory(normalizedStory, normalizedMeta, globalConfig);
+ const normalizedProjectAnnotations = normalizeProjectAnnotations(projectAnnotations);
+
+ const preparedStory = prepareStory(
+ normalizedStory,
+ normalizedMeta,
+ normalizedProjectAnnotations
+ );
const defaultGlobals = Object.entries(globalConfig.globalTypes || {}).reduce(
(acc, [arg, { defaultValue }]) => {
@@ -79,7 +97,7 @@ export function composeStory<
export function composeStories(
storiesImport: TModule,
- globalConfig: NormalizedProjectAnnotations,
+ globalConfig: ProjectAnnotations,
composeStoryFn: typeof composeStory
) {
const { default: meta, __esModule, __namedExportsOrder, ...stories } = storiesImport;
From e411f7296a58641b4446e0d34711140b9015061e Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Mon, 24 Jan 2022 18:40:16 +0100
Subject: [PATCH 013/171] refactor: dedupe testing types
---
app/react/src/client/testing/index.ts | 17 ++++++++--------
app/react/src/client/testing/types.ts | 23 +++++-----------------
lib/store/src/csf/testing-utils/index.ts | 17 +++++-----------
lib/store/src/csf/testing-utils/types.ts | 25 ++++++++++++++++++++++++
4 files changed, 44 insertions(+), 38 deletions(-)
create mode 100644 lib/store/src/csf/testing-utils/types.ts
diff --git a/app/react/src/client/testing/index.ts b/app/react/src/client/testing/index.ts
index d92c24728c2..00e28ba7a25 100644
--- a/app/react/src/client/testing/index.ts
+++ b/app/react/src/client/testing/index.ts
@@ -2,12 +2,13 @@ import {
composeStory as originalComposeStory,
composeStories as originalComposeStories,
setGlobalConfig as originalSetGlobalConfig,
+ CSFExports,
} from '@storybook/store';
-import { ProjectAnnotations } from '@storybook/csf';
+import { ProjectAnnotations, Args } from '@storybook/csf';
import { render } from '../preview/render';
import type { Meta, ReactFramework } from '../preview/types-6-0';
-import type { StoriesWithPartialProps, StoryFile, TestingStory } from './types';
+import type { StoriesWithPartialProps, TestingStory } from './types';
/** Function that sets the globalConfig of your storybook. The global config is the preview module of your .storybook folder.
*
@@ -58,12 +59,12 @@ const defaultGlobalConfig: ProjectAnnotations = {
* @param meta - e.g. (import Meta from './Button.stories')
* @param [globalConfig] - e.g. (import * as globalConfig from '../.storybook/preview') this can be applied automatically if you use `setGlobalConfig` in your setup files.
*/
-export function composeStory(
- story: TestingStory,
- meta: Meta,
+export function composeStory(
+ story: TestingStory,
+ meta: Meta,
globalConfig?: ProjectAnnotations
) {
- return originalComposeStory(
+ return originalComposeStory(
story,
meta,
globalConfig,
@@ -96,11 +97,11 @@ export function composeStory(
* @param storiesImport - e.g. (import * as stories from './Button.stories')
* @param [globalConfig] - e.g. (import * as globalConfig from '../.storybook/preview') this can be applied automatically if you use `setGlobalConfig` in your setup files.
*/
-export function composeStories(
+export function composeStories>(
storiesImport: TModule,
globalConfig?: ProjectAnnotations
) {
const composedStories = originalComposeStories(storiesImport, globalConfig, composeStory);
- return composedStories as unknown as Omit, keyof StoryFile>;
+ return composedStories as unknown as Omit, keyof CSFExports>;
}
diff --git a/app/react/src/client/testing/types.ts b/app/react/src/client/testing/types.ts
index 4c97e776c26..f20307e9578 100644
--- a/app/react/src/client/testing/types.ts
+++ b/app/react/src/client/testing/types.ts
@@ -1,24 +1,11 @@
import type {
StoryFn as OriginalStoryFn,
- StoryObj,
- Meta,
- Args,
- StoryContext,
- ReactFramework,
-} from '../preview/types-6-0';
+ TestingStory as OriginalTestingStory,
+} from '@storybook/store';
+import type { ReactFramework, Args } from '../preview/types-6-0';
-export type StoryFile = { default: Meta; __esModule?: boolean; __namedExportsOrder?: any };
-
-export type TestingStoryPlayContext = Partial> &
- Pick;
-
-export type TestingStoryPlayFn = (
- context: TestingStoryPlayContext
-) => Promise | void;
-
-export type StoryFn = OriginalStoryFn & { play: TestingStoryPlayFn };
-
-export type TestingStory = StoryFn | StoryObj;
+export type TestingStory = OriginalTestingStory;
+export type StoryFn = OriginalStoryFn;
/**
* T represents the whole ES module of a stories file. K of T means named exports (basically the Story type)
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index fdb988e362c..8137d4a10d2 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -14,6 +14,9 @@ import { normalizeStory } from '../normalizeStory';
import { HooksContext } from '../../hooks';
import { normalizeComponentAnnotations } from '../normalizeComponentAnnotations';
import { normalizeProjectAnnotations } from '..';
+import type { CSFExports, TestingStoryPlayFn } from './types';
+
+export * from './types';
if (process.env.NODE_ENV === 'test') {
// eslint-disable-next-line global-require
@@ -21,12 +24,6 @@ if (process.env.NODE_ENV === 'test') {
addons.setChannel(mockChannel());
}
-export type StoryFile = {
- default: Record;
- __esModule?: boolean;
- __namedExportsOrder?: string[];
-};
-
let GLOBAL_STORYBOOK_CONFIG = {};
export function setGlobalConfig(
@@ -35,10 +32,6 @@ export function setGlobalConfig(
GLOBAL_STORYBOOK_CONFIG = config;
}
-type PartialPlayFn = (
- context: Partial & Pick
-) => Promise | void;
-
export function composeStory<
TFramework extends AnyFramework = AnyFramework,
TArgs extends Args = Args
@@ -89,13 +82,13 @@ export function composeStory<
composedStory.storyName = story.storyName || story.name;
composedStory.args = preparedStory.initialArgs;
- composedStory.play = preparedStory.playFunction as PartialPlayFn;
+ composedStory.play = preparedStory.playFunction as TestingStoryPlayFn;
composedStory.parameters = preparedStory.parameters;
return composedStory;
}
-export function composeStories(
+export function composeStories(
storiesImport: TModule,
globalConfig: ProjectAnnotations,
composeStoryFn: typeof composeStory
diff --git a/lib/store/src/csf/testing-utils/types.ts b/lib/store/src/csf/testing-utils/types.ts
new file mode 100644
index 00000000000..4e0f5542582
--- /dev/null
+++ b/lib/store/src/csf/testing-utils/types.ts
@@ -0,0 +1,25 @@
+import type {
+ AnyFramework,
+ AnnotatedStoryFn,
+ StoryAnnotations,
+ ComponentAnnotations,
+ Args,
+ StoryContext,
+} from '@storybook/csf';
+
+export type CSFExports = {
+ default: ComponentAnnotations;
+ __esModule?: boolean;
+ __namedExportsOrder?: string[];
+};
+
+export type TestingStoryPlayContext = Partial & Pick;
+
+export type TestingStoryPlayFn = (context: TestingStoryPlayContext) => Promise | void;
+
+export type StoryFn =
+ AnnotatedStoryFn & { play: TestingStoryPlayFn };
+
+export type TestingStory =
+ | StoryFn
+ | StoryAnnotations;
From ef0bb7a90218fb62b4ecb9d6e69dead400629551 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Mon, 24 Jan 2022 18:51:26 +0100
Subject: [PATCH 014/171] refactor: move logic to getValuesFromArgTypes file
---
lib/store/src/GlobalsStore.ts | 6 ++----
lib/store/src/csf/getValuesFromArgTypes.ts | 9 +++++++++
lib/store/src/csf/index.ts | 1 +
lib/store/src/csf/prepareStory.ts | 12 +++---------
lib/store/src/csf/testing-utils/index.ts | 12 ++----------
5 files changed, 17 insertions(+), 23 deletions(-)
create mode 100644 lib/store/src/csf/getValuesFromArgTypes.ts
diff --git a/lib/store/src/GlobalsStore.ts b/lib/store/src/GlobalsStore.ts
index 800ed7558e9..4654c71dc2c 100644
--- a/lib/store/src/GlobalsStore.ts
+++ b/lib/store/src/GlobalsStore.ts
@@ -3,6 +3,7 @@ import dedent from 'ts-dedent';
import { Globals, GlobalTypes } from '@storybook/csf';
import { deepDiff, DEEPLY_EQUAL } from './args';
+import { getValuesFromArgTypes } from './csf/getValuesFromArgTypes';
const setUndeclaredWarning = deprecate(
() => {},
@@ -24,10 +25,7 @@ export class GlobalsStore {
this.allowedGlobalNames = new Set([...Object.keys(globals), ...Object.keys(globalTypes)]);
- const defaultGlobals = Object.entries(globalTypes).reduce((acc, [key, { defaultValue }]) => {
- if (defaultValue) acc[key] = defaultValue;
- return acc;
- }, {} as Globals);
+ const defaultGlobals: Globals = getValuesFromArgTypes(globalTypes);
this.initialGlobals = { ...defaultGlobals, ...globals };
this.globals = this.initialGlobals;
diff --git a/lib/store/src/csf/getValuesFromArgTypes.ts b/lib/store/src/csf/getValuesFromArgTypes.ts
new file mode 100644
index 00000000000..9a2a7b1aed9
--- /dev/null
+++ b/lib/store/src/csf/getValuesFromArgTypes.ts
@@ -0,0 +1,9 @@
+import { ArgTypes } from '@storybook/csf';
+
+export const getValuesFromArgTypes = (argTypes: ArgTypes = {}) =>
+ Object.entries(argTypes).reduce((acc, [arg, { defaultValue }]) => {
+ if (typeof defaultValue !== 'undefined') {
+ acc[arg] = defaultValue;
+ }
+ return acc;
+ }, {} as ArgTypes);
diff --git a/lib/store/src/csf/index.ts b/lib/store/src/csf/index.ts
index eb26845e410..215f85a9f71 100644
--- a/lib/store/src/csf/index.ts
+++ b/lib/store/src/csf/index.ts
@@ -4,4 +4,5 @@ export * from './processCSFFile';
export * from './prepareStory';
export * from './normalizeComponentAnnotations';
export * from './normalizeProjectAnnotations';
+export * from './getValuesFromArgTypes';
export * from './testing-utils';
diff --git a/lib/store/src/csf/prepareStory.ts b/lib/store/src/csf/prepareStory.ts
index 16d5919aa57..2e704309991 100644
--- a/lib/store/src/csf/prepareStory.ts
+++ b/lib/store/src/csf/prepareStory.ts
@@ -23,6 +23,7 @@ import { combineParameters } from '../parameters';
import { applyHooks } from '../hooks';
import { defaultDecorateStory } from '../decorators';
import { groupArgsByTarget, NO_TARGET_NAME } from '../args';
+import { getValuesFromArgTypes } from '..';
const argTypeDefaultValueWarning = deprecate(
() => {},
@@ -121,15 +122,8 @@ export function prepareStory(
// Add argTypes[X].defaultValue to initial args (note this deprecated)
// We need to do this *after* the argTypesEnhancers as they may add defaultValues
- const defaultArgs: Args = Object.entries(contextForEnhancers.argTypes).reduce(
- (acc, [arg, { defaultValue }]) => {
- if (typeof defaultValue !== 'undefined') {
- acc[arg] = defaultValue;
- }
- return acc;
- },
- {} as Args
- );
+ const defaultArgs = getValuesFromArgTypes(contextForEnhancers.argTypes);
+
if (Object.keys(defaultArgs).length > 0) {
argTypeDefaultValueWarning();
}
diff --git a/lib/store/src/csf/testing-utils/index.ts b/lib/store/src/csf/testing-utils/index.ts
index 8137d4a10d2..4769d355f7d 100644
--- a/lib/store/src/csf/testing-utils/index.ts
+++ b/lib/store/src/csf/testing-utils/index.ts
@@ -13,7 +13,7 @@ import { prepareStory } from '../prepareStory';
import { normalizeStory } from '../normalizeStory';
import { HooksContext } from '../../hooks';
import { normalizeComponentAnnotations } from '../normalizeComponentAnnotations';
-import { normalizeProjectAnnotations } from '..';
+import { getValuesFromArgTypes, normalizeProjectAnnotations } from '..';
import type { CSFExports, TestingStoryPlayFn } from './types';
export * from './types';
@@ -59,15 +59,7 @@ export function composeStory<
normalizedProjectAnnotations
);
- const defaultGlobals = Object.entries(globalConfig.globalTypes || {}).reduce(
- (acc, [arg, { defaultValue }]) => {
- if (defaultValue) {
- acc[arg] = defaultValue;
- }
- return acc;
- },
- {} as Record
- );
+ const defaultGlobals = getValuesFromArgTypes(globalConfig.globalTypes);
const composedStory = (extraArgs: Partial) => {
const context: Partial = {
From 800f17db238fa00b55ac9453e4fba33801683ff2 Mon Sep 17 00:00:00 2001
From: Yann Braga
Date: Mon, 24 Jan 2022 19:01:08 +0100
Subject: [PATCH 015/171] fix: ensure story has a name before calling
composeStory
---
.../__snapshots__/internals.test.tsx.snap | 34 +++++++++----------
lib/store/src/csf/testing-utils/index.ts | 26 ++++++++++----
2 files changed, 36 insertions(+), 24 deletions(-)
diff --git a/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap b/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
index dd0a3d4183f..dc838a93306 100644
--- a/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
+++ b/examples/cra-ts-essentials/src/stories/testing-react/components/__snapshots__/internals.test.tsx.snap
@@ -18,23 +18,6 @@ exports[`Renders CSF2Secondary story 1`] = `
-
-
- Locale:
- en
-
-
-
-
@@ -125,3 +108,20 @@ exports[`Renders CSF3Primary story 1`] = `
+
+
+ Locale:
+ en
+
+
+
+