mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-07 07:21:17 +08:00
fixes
This commit is contained in:
parent
6cde3ad106
commit
f5194da656
@ -80,6 +80,9 @@
|
|||||||
"ts-dedent": "^2.0.0",
|
"ts-dedent": "^2.0.0",
|
||||||
"util-deprecate": "^1.0.2"
|
"util-deprecate": "^1.0.2"
|
||||||
},
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@jest/globals": "^26.6.2"
|
||||||
|
},
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public"
|
"access": "public"
|
||||||
},
|
},
|
||||||
|
@ -18,7 +18,7 @@ describe('ClientApi', () => {
|
|||||||
};
|
};
|
||||||
clientApi.storyStore = store as any;
|
clientApi.storyStore = store as any;
|
||||||
|
|
||||||
let disposeCallback: () => void;
|
let disposeCallback: () => void = () => {};
|
||||||
const module1 = {
|
const module1 = {
|
||||||
id: 'file1',
|
id: 'file1',
|
||||||
hot: {
|
hot: {
|
||||||
|
@ -36,7 +36,7 @@ export function executeLoadable(loadable: Loadable) {
|
|||||||
typeof req.resolve === 'function' ? req.resolve(filename) : filename,
|
typeof req.resolve === 'function' ? req.resolve(filename) : filename,
|
||||||
fileExports
|
fileExports
|
||||||
);
|
);
|
||||||
} catch (error) {
|
} catch (error: any) {
|
||||||
const errorString =
|
const errorString =
|
||||||
error.message && error.stack ? `${error.message}\n ${error.stack}` : error.toString();
|
error.message && error.stack ? `${error.message}\n ${error.stack}` : error.toString();
|
||||||
logger.error(`Unexpected error while loading ${filename}: ${errorString}`);
|
logger.error(`Unexpected error while loading ${filename}: ${errorString}`);
|
||||||
@ -92,7 +92,12 @@ export function executeLoadableForChanges(loadable: Loadable, m?: NodeModule) {
|
|||||||
const removed = new Map<Path, Store_ModuleExports>();
|
const removed = new Map<Path, Store_ModuleExports>();
|
||||||
Array.from(lastExportsMap.keys())
|
Array.from(lastExportsMap.keys())
|
||||||
.filter((fileName) => !exportsMap.has(fileName))
|
.filter((fileName) => !exportsMap.has(fileName))
|
||||||
.forEach((fileName) => removed.set(fileName, lastExportsMap.get(fileName)));
|
.forEach((fileName) => {
|
||||||
|
const value = lastExportsMap.get(fileName);
|
||||||
|
if (value) {
|
||||||
|
removed.set(fileName, value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Save the value for the dispose() call above
|
// Save the value for the dispose() call above
|
||||||
lastExportsMap = exportsMap;
|
lastExportsMap = exportsMap;
|
||||||
|
@ -15,7 +15,6 @@ import {
|
|||||||
import { start as realStart } from './start';
|
import { start as realStart } from './start';
|
||||||
|
|
||||||
jest.mock('global', () => ({
|
jest.mock('global', () => ({
|
||||||
// @ts-expect-error (jest is not happy with this)
|
|
||||||
...jest.requireActual('global'),
|
...jest.requireActual('global'),
|
||||||
history: { replaceState: jest.fn() },
|
history: { replaceState: jest.fn() },
|
||||||
document: {
|
document: {
|
||||||
@ -122,7 +121,7 @@ describe('start', () => {
|
|||||||
|
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
|
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -277,9 +276,7 @@ describe('start', () => {
|
|||||||
await waitForRender();
|
await waitForRender();
|
||||||
expect(mockChannel.emit).toHaveBeenCalledWith(STORY_RENDERED, 'component-a--default');
|
expect(mockChannel.emit).toHaveBeenCalledWith(STORY_RENDERED, 'component-a--default');
|
||||||
|
|
||||||
const storiesOfData = mockChannel.emit.mock.calls.find(
|
const storiesOfData = mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1];
|
||||||
(call: any[]) => call[0] === SET_INDEX
|
|
||||||
)[1];
|
|
||||||
expect(Object.values(storiesOfData.entries).map((s: any) => s.parameters.fileName)).toEqual([
|
expect(Object.values(storiesOfData.entries).map((s: any) => s.parameters.fileName)).toEqual([
|
||||||
'file1',
|
'file1',
|
||||||
'file1-2',
|
'file1-2',
|
||||||
@ -410,7 +407,7 @@ describe('start', () => {
|
|||||||
.add('new', jest.fn());
|
.add('new', jest.fn());
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -481,7 +478,7 @@ describe('start', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -533,7 +530,7 @@ describe('start', () => {
|
|||||||
disposeCallback();
|
disposeCallback();
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -584,7 +581,7 @@ describe('start', () => {
|
|||||||
configure('test', () => [componentCExports]);
|
configure('test', () => [componentCExports]);
|
||||||
|
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -705,7 +702,7 @@ describe('start', () => {
|
|||||||
configure('test', () => [{ ...componentCExports, StoryThree: jest.fn() }], module as any);
|
configure('test', () => [{ ...componentCExports, StoryThree: jest.fn() }], module as any);
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -796,7 +793,7 @@ describe('start', () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -868,7 +865,7 @@ describe('start', () => {
|
|||||||
configure('test', () => [componentCExports], module as any);
|
configure('test', () => [componentCExports], module as any);
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -968,7 +965,7 @@ describe('start', () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -1014,7 +1011,7 @@ describe('start', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -1157,7 +1154,7 @@ describe('start', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
@ -1311,7 +1308,7 @@ describe('start', () => {
|
|||||||
configure('test', () => [componentDExports]);
|
configure('test', () => [componentDExports]);
|
||||||
|
|
||||||
await waitForEvents([SET_INDEX]);
|
await waitForEvents([SET_INDEX]);
|
||||||
expect(mockChannel.emit.mock.calls.find((call: [string, any]) => call[0] === SET_INDEX)[1])
|
expect(mockChannel.emit.mock.calls.find((call) => call[0] === SET_INDEX)?.[1])
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
Object {
|
Object {
|
||||||
"entries": Object {
|
"entries": Object {
|
||||||
|
@ -29,7 +29,6 @@ jest.mock('./WebView');
|
|||||||
|
|
||||||
const { window, document } = global;
|
const { window, document } = global;
|
||||||
jest.mock('global', () => ({
|
jest.mock('global', () => ({
|
||||||
// @ts-expect-error (Converted from ts-ignore)
|
|
||||||
...jest.requireActual('global'),
|
...jest.requireActual('global'),
|
||||||
history: { replaceState: jest.fn() },
|
history: { replaceState: jest.fn() },
|
||||||
document: {
|
document: {
|
||||||
|
@ -10,7 +10,11 @@ import {
|
|||||||
STORY_THREW_EXCEPTION,
|
STORY_THREW_EXCEPTION,
|
||||||
} from '@storybook/core-events';
|
} from '@storybook/core-events';
|
||||||
|
|
||||||
import type { Store_StoryIndex, Store_TeardownRenderToDOM } from '@storybook/types';
|
import type {
|
||||||
|
Store_ModuleImportFn,
|
||||||
|
Store_StoryIndex,
|
||||||
|
Store_TeardownRenderToDOM,
|
||||||
|
} from '@storybook/types';
|
||||||
import type { RenderPhase } from './render/StoryRender';
|
import type { RenderPhase } from './render/StoryRender';
|
||||||
|
|
||||||
export const componentOneExports = {
|
export const componentOneExports = {
|
||||||
@ -44,7 +48,7 @@ export const extraComponentOneExports = {
|
|||||||
},
|
},
|
||||||
e: {},
|
e: {},
|
||||||
};
|
};
|
||||||
export const importFn = jest.fn(
|
export const importFn: Store_ModuleImportFn = jest.fn(
|
||||||
async (path: string) =>
|
async (path: string) =>
|
||||||
({
|
({
|
||||||
'./src/ComponentOne.stories.js': componentOneExports,
|
'./src/ComponentOne.stories.js': componentOneExports,
|
||||||
@ -52,7 +56,7 @@ export const importFn = jest.fn(
|
|||||||
'./src/Introduction.mdx': standaloneDocsExports,
|
'./src/Introduction.mdx': standaloneDocsExports,
|
||||||
'./src/ExtraComponentOne.stories.js': extraComponentOneExports,
|
'./src/ExtraComponentOne.stories.js': extraComponentOneExports,
|
||||||
}[path])
|
}[path])
|
||||||
);
|
) as any as Store_ModuleImportFn;
|
||||||
|
|
||||||
export const docsRenderer = {
|
export const docsRenderer = {
|
||||||
render: jest.fn().mockImplementation((context, parameters, element, cb) => cb()),
|
render: jest.fn().mockImplementation((context, parameters, element, cb) => cb()),
|
||||||
|
@ -29,9 +29,13 @@ import {
|
|||||||
} from '@storybook/core-events';
|
} from '@storybook/core-events';
|
||||||
import { logger } from '@storybook/client-logger';
|
import { logger } from '@storybook/client-logger';
|
||||||
import { addons, mockChannel as createMockChannel } from '@storybook/addons';
|
import { addons, mockChannel as createMockChannel } from '@storybook/addons';
|
||||||
import type { AnyFramework } from '@storybook/types';
|
import type {
|
||||||
|
AnyFramework,
|
||||||
|
Store_ModuleImportFn,
|
||||||
|
Store_WebProjectAnnotations,
|
||||||
|
} from '@storybook/types';
|
||||||
import { mocked } from 'ts-jest/utils';
|
import { mocked } from 'ts-jest/utils';
|
||||||
import type { ModuleImportFn, WebProjectAnnotations } from '../../store';
|
// import type { ModuleImportFn, WebProjectAnnotations } from '../../store';
|
||||||
|
|
||||||
import { PreviewWeb } from './PreviewWeb';
|
import { PreviewWeb } from './PreviewWeb';
|
||||||
import {
|
import {
|
||||||
@ -57,7 +61,7 @@ const { history, document } = global;
|
|||||||
|
|
||||||
const mockStoryIndex = jest.fn(() => storyIndex);
|
const mockStoryIndex = jest.fn(() => storyIndex);
|
||||||
|
|
||||||
let mockFetchResult;
|
let mockFetchResult: any;
|
||||||
jest.mock('global', () => ({
|
jest.mock('global', () => ({
|
||||||
...(mockJest.requireActual('global') as any),
|
...(mockJest.requireActual('global') as any),
|
||||||
history: { replaceState: mockJest.fn() },
|
history: { replaceState: mockJest.fn() },
|
||||||
@ -109,8 +113,8 @@ async function createAndRenderPreview({
|
|||||||
importFn: inputImportFn = importFn,
|
importFn: inputImportFn = importFn,
|
||||||
getProjectAnnotations: inputGetProjectAnnotations = getProjectAnnotations,
|
getProjectAnnotations: inputGetProjectAnnotations = getProjectAnnotations,
|
||||||
}: {
|
}: {
|
||||||
importFn?: ModuleImportFn;
|
importFn?: Store_ModuleImportFn;
|
||||||
getProjectAnnotations?: () => WebProjectAnnotations<AnyFramework>;
|
getProjectAnnotations?: () => Store_WebProjectAnnotations<AnyFramework>;
|
||||||
} = {}) {
|
} = {}) {
|
||||||
const preview = new PreviewWeb();
|
const preview = new PreviewWeb();
|
||||||
await preview.initialize({
|
await preview.initialize({
|
||||||
@ -685,7 +689,7 @@ describe('PreviewWeb', () => {
|
|||||||
|
|
||||||
const context = docsRenderer.render.mock.calls[0][0];
|
const context = docsRenderer.render.mock.calls[0][0];
|
||||||
|
|
||||||
expect(context.componentStories().map((s) => s.id)).toEqual([
|
expect(context.componentStories().map((s: any) => s.id)).toEqual([
|
||||||
'component-one--a',
|
'component-one--a',
|
||||||
'component-one--b',
|
'component-one--b',
|
||||||
'component-one--e',
|
'component-one--e',
|
||||||
@ -1141,6 +1145,7 @@ describe('PreviewWeb', () => {
|
|||||||
const preview = await createAndRenderPreview();
|
const preview = await createAndRenderPreview();
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
|
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
importFn.mockClear();
|
importFn.mockClear();
|
||||||
await preview.onPreloadStories({ ids: ['component-two--c'] });
|
await preview.onPreloadStories({ ids: ['component-two--c'] });
|
||||||
expect(importFn).toHaveBeenCalledWith('./src/ComponentTwo.stories.js');
|
expect(importFn).toHaveBeenCalledWith('./src/ComponentTwo.stories.js');
|
||||||
@ -1151,6 +1156,7 @@ describe('PreviewWeb', () => {
|
|||||||
const preview = await createAndRenderPreview();
|
const preview = await createAndRenderPreview();
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
|
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
importFn.mockClear();
|
importFn.mockClear();
|
||||||
await preview.onPreloadStories({ ids: ['component-one--docs'] });
|
await preview.onPreloadStories({ ids: ['component-one--docs'] });
|
||||||
expect(importFn).toHaveBeenCalledWith('./src/ComponentOne.stories.js');
|
expect(importFn).toHaveBeenCalledWith('./src/ComponentOne.stories.js');
|
||||||
@ -1161,6 +1167,7 @@ describe('PreviewWeb', () => {
|
|||||||
const preview = await createAndRenderPreview();
|
const preview = await createAndRenderPreview();
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
|
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
importFn.mockClear();
|
importFn.mockClear();
|
||||||
await preview.onPreloadStories({ ids: ['introduction--docs'] });
|
await preview.onPreloadStories({ ids: ['introduction--docs'] });
|
||||||
expect(importFn).toHaveBeenCalledWith('./src/Introduction.mdx');
|
expect(importFn).toHaveBeenCalledWith('./src/Introduction.mdx');
|
||||||
@ -1170,6 +1177,7 @@ describe('PreviewWeb', () => {
|
|||||||
const preview = await createAndRenderPreview();
|
const preview = await createAndRenderPreview();
|
||||||
await waitForRender();
|
await waitForRender();
|
||||||
|
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
importFn.mockClear();
|
importFn.mockClear();
|
||||||
await preview.onPreloadStories({ ids: ['introduction--docs'] });
|
await preview.onPreloadStories({ ids: ['introduction--docs'] });
|
||||||
expect(importFn).toHaveBeenCalledWith('./src/ComponentTwo.stories.js');
|
expect(importFn).toHaveBeenCalledWith('./src/ComponentTwo.stories.js');
|
||||||
@ -1607,14 +1615,18 @@ describe('PreviewWeb', () => {
|
|||||||
const [gate, openGate] = createGate();
|
const [gate, openGate] = createGate();
|
||||||
const [importedGate, openImportedGate] = createGate();
|
const [importedGate, openImportedGate] = createGate();
|
||||||
importFn
|
importFn
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
.mockImplementationOnce(async (...args) => {
|
.mockImplementationOnce(async (...args) => {
|
||||||
await gate;
|
await gate;
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
return importFn(...args);
|
return importFn(...args);
|
||||||
})
|
})
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
.mockImplementationOnce(async (...args) => {
|
.mockImplementationOnce(async (...args) => {
|
||||||
// The second time we `import()` we open the "imported" gate
|
// The second time we `import()` we open the "imported" gate
|
||||||
openImportedGate();
|
openImportedGate();
|
||||||
await gate;
|
await gate;
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
return importFn(...args);
|
return importFn(...args);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -1652,14 +1664,18 @@ describe('PreviewWeb', () => {
|
|||||||
const [gate, openGate] = createGate();
|
const [gate, openGate] = createGate();
|
||||||
const [importedGate, openImportedGate] = createGate();
|
const [importedGate, openImportedGate] = createGate();
|
||||||
importFn
|
importFn
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
.mockImplementationOnce(async (...args) => {
|
.mockImplementationOnce(async (...args) => {
|
||||||
await gate;
|
await gate;
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
return importFn(...args);
|
return importFn(...args);
|
||||||
})
|
})
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
.mockImplementationOnce(async (...args) => {
|
.mockImplementationOnce(async (...args) => {
|
||||||
// The second time we `import()` we open the "imported" gate
|
// The second time we `import()` we open the "imported" gate
|
||||||
openImportedGate();
|
openImportedGate();
|
||||||
await gate;
|
await gate;
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
return importFn(...args);
|
return importFn(...args);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -1696,14 +1712,18 @@ describe('PreviewWeb', () => {
|
|||||||
const [gate, openGate] = createGate();
|
const [gate, openGate] = createGate();
|
||||||
const [importedGate, openImportedGate] = createGate();
|
const [importedGate, openImportedGate] = createGate();
|
||||||
importFn
|
importFn
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
.mockImplementationOnce(async (...args) => {
|
.mockImplementationOnce(async (...args) => {
|
||||||
await gate;
|
await gate;
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
return importFn(...args);
|
return importFn(...args);
|
||||||
})
|
})
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
.mockImplementationOnce(async (...args) => {
|
.mockImplementationOnce(async (...args) => {
|
||||||
// The second time we `import()` we open the "imported" gate
|
// The second time we `import()` we open the "imported" gate
|
||||||
openImportedGate();
|
openImportedGate();
|
||||||
await gate;
|
await gate;
|
||||||
|
// @ts-expect-error (jest mock)
|
||||||
return importFn(...args);
|
return importFn(...args);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -3072,7 +3092,7 @@ describe('PreviewWeb', () => {
|
|||||||
describe('when a standalone docs file changes', () => {
|
describe('when a standalone docs file changes', () => {
|
||||||
const newStandaloneDocsExports = { default: jest.fn() };
|
const newStandaloneDocsExports = { default: jest.fn() };
|
||||||
|
|
||||||
const newImportFn = jest.fn(async (path) => {
|
const newImportFn = jest.fn(async (path: string) => {
|
||||||
return path === './src/Introduction.mdx' ? newStandaloneDocsExports : importFn(path);
|
return path === './src/Introduction.mdx' ? newStandaloneDocsExports : importFn(path);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -3180,7 +3200,7 @@ describe('PreviewWeb', () => {
|
|||||||
expect(mockChannel.emit).toHaveBeenCalledWith(CONFIG_ERROR, err);
|
expect(mockChannel.emit).toHaveBeenCalledWith(CONFIG_ERROR, err);
|
||||||
});
|
});
|
||||||
|
|
||||||
const newGlobalDecorator = jest.fn((s) => s());
|
const newGlobalDecorator = jest.fn((s: any) => s());
|
||||||
const newGetProjectAnnotations = () => {
|
const newGetProjectAnnotations = () => {
|
||||||
return {
|
return {
|
||||||
...projectAnnotations,
|
...projectAnnotations,
|
||||||
@ -3388,7 +3408,7 @@ describe('PreviewWeb', () => {
|
|||||||
it('waits for stories to be cached', async () => {
|
it('waits for stories to be cached', async () => {
|
||||||
const [gate, openGate] = createGate();
|
const [gate, openGate] = createGate();
|
||||||
|
|
||||||
const gatedImportFn = async (path) => {
|
const gatedImportFn = async (path: string) => {
|
||||||
await gate;
|
await gate;
|
||||||
return importFn(path);
|
return importFn(path);
|
||||||
};
|
};
|
||||||
|
@ -20,7 +20,7 @@ describe('UrlStore', () => {
|
|||||||
expect(pathToId('/story/story--id')).toEqual('story--id');
|
expect(pathToId('/story/story--id')).toEqual('story--id');
|
||||||
});
|
});
|
||||||
it('should error on invalid ids', () => {
|
it('should error on invalid ids', () => {
|
||||||
[null, '', '/whatever/story/story--id'].forEach((path) => {
|
[null, '', '/whatever/story/story--id'].forEach((path: any) => {
|
||||||
expect(() => pathToId(path)).toThrow(/Invalid/);
|
expect(() => pathToId(path)).toThrow(/Invalid/);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -32,11 +32,11 @@ const storyIndex: Store_StoryIndex = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const makeStoryIndex = (titlesAndNames) => {
|
const makeStoryIndex = (titlesAndNames: any) => {
|
||||||
return {
|
return {
|
||||||
v: 4,
|
v: 4,
|
||||||
entries: Object.fromEntries(
|
entries: Object.fromEntries(
|
||||||
titlesAndNames.map(([title, name]) => {
|
titlesAndNames.map(([title, name]: any[]) => {
|
||||||
const id = `${title}--${name}`.replace('/', '-');
|
const id = `${title}--${name}`.replace('/', '-');
|
||||||
return [
|
return [
|
||||||
id,
|
id,
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
import type { AnyFramework, ProjectAnnotations } from '@storybook/types';
|
import type { AnyFramework, ProjectAnnotations, Store_StoryIndex } from '@storybook/types';
|
||||||
import global from 'global';
|
import global from 'global';
|
||||||
import { expect } from '@jest/globals';
|
import { expect } from '@jest/globals';
|
||||||
|
|
||||||
import { prepareStory } from './csf/prepareStory';
|
import { prepareStory } from './csf/prepareStory';
|
||||||
import { processCSFFile } from './csf/processCSFFile';
|
import { processCSFFile } from './csf/processCSFFile';
|
||||||
import { StoryStore } from './StoryStore';
|
import { StoryStore } from './StoryStore';
|
||||||
import type { StoryIndex } from './types';
|
import type { HooksContext } from '../..';
|
||||||
import type { HooksContext } from './hooks';
|
|
||||||
|
|
||||||
// Spy on prepareStory/processCSFFile
|
// Spy on prepareStory/processCSFFile
|
||||||
jest.mock('./csf/prepareStory', () => ({
|
jest.mock('./csf/prepareStory', () => ({
|
||||||
@ -43,7 +42,7 @@ const projectAnnotations: ProjectAnnotations<any> = {
|
|||||||
render: jest.fn(),
|
render: jest.fn(),
|
||||||
};
|
};
|
||||||
|
|
||||||
const storyIndex: StoryIndex = {
|
const storyIndex: Store_StoryIndex = {
|
||||||
v: 4,
|
v: 4,
|
||||||
entries: {
|
entries: {
|
||||||
'component-one--a': {
|
'component-one--a': {
|
||||||
@ -631,10 +630,11 @@ describe('StoryStore', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('does not include (modern) docs entries ever', async () => {
|
it('does not include (modern) docs entries ever', async () => {
|
||||||
const docsOnlyStoryIndex: StoryIndex = {
|
const docsOnlyStoryIndex: Store_StoryIndex = {
|
||||||
v: 4,
|
v: 4,
|
||||||
entries: {
|
entries: {
|
||||||
...storyIndex.entries,
|
...storyIndex.entries,
|
||||||
|
// @ts-expect-error (not strict)
|
||||||
'introduction--docs': {
|
'introduction--docs': {
|
||||||
type: 'docs',
|
type: 'docs',
|
||||||
id: 'introduction--docs',
|
id: 'introduction--docs',
|
||||||
|
@ -5,7 +5,7 @@ import { userOrAutoTitleFromSpecifier as userOrAuto } from './autoTitle';
|
|||||||
|
|
||||||
expect.addSnapshotSerializer({
|
expect.addSnapshotSerializer({
|
||||||
print: (val: any) => val,
|
print: (val: any) => val,
|
||||||
test: (val) => true,
|
test: (val: any) => true,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Make these two the same so `normalizeStoriesEntry` doesn't change anything
|
// Make these two the same so `normalizeStoriesEntry` doesn't change anything
|
||||||
|
@ -162,9 +162,9 @@ describe('composeConfigs', () => {
|
|||||||
const fn = jest.fn();
|
const fn = jest.fn();
|
||||||
|
|
||||||
const { runStep } = composeConfigs([
|
const { runStep } = composeConfigs([
|
||||||
{ runStep: (label, play, context) => fn(`${label}1`, play(context)) },
|
{ runStep: (label: any, play: any, context: any) => fn(`${label}1`, play(context)) },
|
||||||
{ runStep: (label, play, context) => fn(`${label}2`, play(context)) },
|
{ runStep: (label: any, play: any, context: any) => fn(`${label}2`, play(context)) },
|
||||||
{ runStep: (label, play, context) => fn(`${label}3`, play(context)) },
|
{ runStep: (label: any, play: any, context: any) => fn(`${label}3`, play(context)) },
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// @ts-expect-error We don't care about the context value here
|
// @ts-expect-error We don't care about the context value here
|
||||||
|
@ -31,6 +31,7 @@ describe('composeStory', () => {
|
|||||||
|
|
||||||
test('should throw an error if Story is undefined', () => {
|
test('should throw an error if Story is undefined', () => {
|
||||||
expect(() => {
|
expect(() => {
|
||||||
|
// @ts-expect-error (this is supposed to fail)
|
||||||
composeStory(undefined, meta);
|
composeStory(undefined, meta);
|
||||||
}).toThrow();
|
}).toThrow();
|
||||||
});
|
});
|
||||||
|
@ -18,9 +18,9 @@ describe('client-api.decorators', () => {
|
|||||||
it('calls decorators in out to in order', () => {
|
it('calls decorators in out to in order', () => {
|
||||||
const order: number[] = [];
|
const order: number[] = [];
|
||||||
const decorators = [
|
const decorators = [
|
||||||
(s) => order.push(1) && s(),
|
(s: any) => order.push(1) && s(),
|
||||||
(s) => order.push(2) && s(),
|
(s: any) => order.push(2) && s(),
|
||||||
(s) => order.push(3) && s(),
|
(s: any) => order.push(3) && s(),
|
||||||
];
|
];
|
||||||
const decorated = defaultDecorateStory(() => order.push(4), decorators);
|
const decorated = defaultDecorateStory(() => order.push(4), decorators);
|
||||||
|
|
||||||
@ -32,9 +32,9 @@ describe('client-api.decorators', () => {
|
|||||||
it('passes context through to sub decorators', () => {
|
it('passes context through to sub decorators', () => {
|
||||||
const contexts: StoryContext[] = [];
|
const contexts: StoryContext[] = [];
|
||||||
const decorators = [
|
const decorators = [
|
||||||
(s, c) => contexts.push(c) && s({ args: { k: 1 } }),
|
(s: any, c: any) => contexts.push(c) && s({ args: { k: 1 } }),
|
||||||
(s, c) => contexts.push(c) && s({ args: { k: 2 } }),
|
(s: any, c: any) => contexts.push(c) && s({ args: { k: 2 } }),
|
||||||
(s, c) => contexts.push(c) && s({ args: { k: 3 } }),
|
(s: any, c: any) => contexts.push(c) && s({ args: { k: 3 } }),
|
||||||
];
|
];
|
||||||
const decorated = defaultDecorateStory((c) => contexts.push(c), decorators);
|
const decorated = defaultDecorateStory((c) => contexts.push(c), decorators);
|
||||||
|
|
||||||
@ -46,8 +46,8 @@ describe('client-api.decorators', () => {
|
|||||||
it('passes context through to sub decorators additively', () => {
|
it('passes context through to sub decorators additively', () => {
|
||||||
const contexts: StoryContext[] = [];
|
const contexts: StoryContext[] = [];
|
||||||
const decorators = [
|
const decorators = [
|
||||||
(s, c) => contexts.push(c) && s({ args: { a: 1 } }),
|
(s: any, c: any) => contexts.push(c) && s({ args: { a: 1 } }),
|
||||||
(s, c) => contexts.push(c) && s({ globals: { g: 2 } }),
|
(s: any, c: any) => contexts.push(c) && s({ globals: { g: 2 } }),
|
||||||
];
|
];
|
||||||
const decorated = defaultDecorateStory((c) => contexts.push(c), decorators);
|
const decorated = defaultDecorateStory((c) => contexts.push(c), decorators);
|
||||||
|
|
||||||
@ -61,9 +61,9 @@ describe('client-api.decorators', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('does not recreate decorated story functions each time', () => {
|
it('does not recreate decorated story functions each time', () => {
|
||||||
const decoratedStories = [];
|
const decoratedStories: any[] = [];
|
||||||
const decorators = [
|
const decorators = [
|
||||||
(s, c) => {
|
(s: any, c: any) => {
|
||||||
decoratedStories.push = s;
|
decoratedStories.push = s;
|
||||||
return s();
|
return s();
|
||||||
},
|
},
|
||||||
@ -80,12 +80,12 @@ describe('client-api.decorators', () => {
|
|||||||
// the same story twice at the same time.
|
// the same story twice at the same time.
|
||||||
it('does not interleave contexts if two decorated stories are call simultaneously', async () => {
|
it('does not interleave contexts if two decorated stories are call simultaneously', async () => {
|
||||||
const contexts: StoryContext[] = [];
|
const contexts: StoryContext[] = [];
|
||||||
let resolve;
|
let resolve: any;
|
||||||
const fence = new Promise((r) => {
|
const fence = new Promise((r) => {
|
||||||
resolve = r;
|
resolve = r;
|
||||||
});
|
});
|
||||||
const decorators = [
|
const decorators = [
|
||||||
async (s, c) => {
|
async (s: any, c: any) => {
|
||||||
// The fence here simulates async-ness in react rendering an element (`<S />` doesn't run `S()` straight away)
|
// The fence here simulates async-ness in react rendering an element (`<S />` doesn't run `S()` straight away)
|
||||||
await fence;
|
await fence;
|
||||||
s();
|
s();
|
||||||
@ -107,7 +107,7 @@ describe('client-api.decorators', () => {
|
|||||||
it('DOES NOT merge core metadata or pass through core metadata keys in context', () => {
|
it('DOES NOT merge core metadata or pass through core metadata keys in context', () => {
|
||||||
const contexts: StoryContext[] = [];
|
const contexts: StoryContext[] = [];
|
||||||
const decorators = [
|
const decorators = [
|
||||||
(s, c) =>
|
(s: any, c: any) =>
|
||||||
contexts.push(c) &&
|
contexts.push(c) &&
|
||||||
s({ parameters: { c: 'd' }, id: 'notId', kind: 'notKind', name: 'notName' }),
|
s({ parameters: { c: 'd' }, id: 'notId', kind: 'notKind', name: 'notName' }),
|
||||||
];
|
];
|
||||||
|
@ -31,16 +31,16 @@ jest.mock('@storybook/client-logger', () => ({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
const SOME_EVENT = 'someEvent';
|
const SOME_EVENT = 'someEvent';
|
||||||
let mockChannel;
|
let mockChannel: any;
|
||||||
let hooks;
|
let hooks: any;
|
||||||
let onSomeEvent;
|
let onSomeEvent: any;
|
||||||
let removeSomeEventListener;
|
let removeSomeEventListener: any;
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
onSomeEvent = jest.fn();
|
onSomeEvent = jest.fn();
|
||||||
removeSomeEventListener = jest.fn();
|
removeSomeEventListener = jest.fn();
|
||||||
mockChannel = {
|
mockChannel = {
|
||||||
emit: jest.fn(),
|
emit: jest.fn(),
|
||||||
on(event, callback) {
|
on(event: any, callback: any) {
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case STORY_RENDERED:
|
case STORY_RENDERED:
|
||||||
callback();
|
callback();
|
||||||
@ -51,7 +51,7 @@ beforeEach(() => {
|
|||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
removeListener(event, callback) {
|
removeListener(event: any, callback: any) {
|
||||||
if (event === SOME_EVENT) {
|
if (event === SOME_EVENT) {
|
||||||
removeSomeEventListener(event, callback);
|
removeSomeEventListener(event, callback);
|
||||||
}
|
}
|
||||||
@ -63,7 +63,7 @@ beforeEach(() => {
|
|||||||
|
|
||||||
const decorateStory = applyHooks(defaultDecorateStory);
|
const decorateStory = applyHooks(defaultDecorateStory);
|
||||||
|
|
||||||
const run = (storyFn, decorators: DecoratorFunction[] = [], context = {}) =>
|
const run = (storyFn: any, decorators: DecoratorFunction[] = [], context = {}) =>
|
||||||
decorateStory(storyFn, decorators)({ ...context, hooks } as StoryContext);
|
decorateStory(storyFn, decorators)({ ...context, hooks } as StoryContext);
|
||||||
|
|
||||||
describe('Preview hooks', () => {
|
describe('Preview hooks', () => {
|
||||||
@ -116,7 +116,7 @@ describe('Preview hooks', () => {
|
|||||||
});
|
});
|
||||||
it("doesn't retrigger the effect from decorator if story has changed", () => {
|
it("doesn't retrigger the effect from decorator if story has changed", () => {
|
||||||
const effect = jest.fn();
|
const effect = jest.fn();
|
||||||
const decorator = (storyFn) => {
|
const decorator = (storyFn: any) => {
|
||||||
useEffect(effect, []);
|
useEffect(effect, []);
|
||||||
return storyFn();
|
return storyFn();
|
||||||
};
|
};
|
||||||
@ -126,7 +126,7 @@ describe('Preview hooks', () => {
|
|||||||
});
|
});
|
||||||
it("doesn't retrigger the effect from decorator if story has changed and story call comes before useEffect", () => {
|
it("doesn't retrigger the effect from decorator if story has changed and story call comes before useEffect", () => {
|
||||||
const effect = jest.fn();
|
const effect = jest.fn();
|
||||||
const decorator = (storyFn) => {
|
const decorator = (storyFn: any) => {
|
||||||
const story = storyFn();
|
const story = storyFn();
|
||||||
useEffect(effect, []);
|
useEffect(effect, []);
|
||||||
return story;
|
return story;
|
||||||
@ -140,7 +140,7 @@ describe('Preview hooks', () => {
|
|||||||
const story = () => {
|
const story = () => {
|
||||||
useEffect(effect, []);
|
useEffect(effect, []);
|
||||||
};
|
};
|
||||||
const decorator = (storyFn) => {
|
const decorator = (storyFn: any) => {
|
||||||
storyFn();
|
storyFn();
|
||||||
return storyFn();
|
return storyFn();
|
||||||
};
|
};
|
||||||
@ -348,7 +348,7 @@ describe('Preview hooks', () => {
|
|||||||
});
|
});
|
||||||
describe('useRef', () => {
|
describe('useRef', () => {
|
||||||
it('attaches initial value', () => {
|
it('attaches initial value', () => {
|
||||||
let ref;
|
let ref: any;
|
||||||
const storyFn = () => {
|
const storyFn = () => {
|
||||||
ref = useRef('foo');
|
ref = useRef('foo');
|
||||||
};
|
};
|
||||||
@ -430,8 +430,8 @@ describe('Preview hooks', () => {
|
|||||||
expect(state).toBe(3);
|
expect(state).toBe(3);
|
||||||
});
|
});
|
||||||
it('performs asynchronous state updates', () => {
|
it('performs asynchronous state updates', () => {
|
||||||
let state;
|
let state: any;
|
||||||
let setState;
|
let setState: any;
|
||||||
const storyFn = jest.fn(() => {
|
const storyFn = jest.fn(() => {
|
||||||
[state, setState] = useState('foo');
|
[state, setState] = useState('foo');
|
||||||
});
|
});
|
||||||
@ -484,8 +484,8 @@ describe('Preview hooks', () => {
|
|||||||
expect(state).toBe(3);
|
expect(state).toBe(3);
|
||||||
});
|
});
|
||||||
it('performs asynchronous state updates', () => {
|
it('performs asynchronous state updates', () => {
|
||||||
let state;
|
let state: any;
|
||||||
let dispatch;
|
let dispatch: any;
|
||||||
const storyFn = jest.fn(() => {
|
const storyFn = jest.fn(() => {
|
||||||
[state, dispatch] = useReducer((prevState, action) => {
|
[state, dispatch] = useReducer((prevState, action) => {
|
||||||
switch (action) {
|
switch (action) {
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
{
|
{
|
||||||
"extends": "../../tsconfig.json",
|
"extends": "../../tsconfig.json",
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"strict": true
|
"strict": true,
|
||||||
|
"skipLibCheck": true
|
||||||
},
|
},
|
||||||
"include": ["src/**/*"],
|
"include": ["src/**/*"],
|
||||||
"exclude": []
|
"exclude": []
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { IconButton, Icons } from '@storybook/components';
|
import { IconButton, Icons } from '@storybook/components';
|
||||||
import { Consumer, type Combo } from '@storybook/api';
|
import { Consumer, type Combo } from '@storybook/api';
|
||||||
import type { Addon } from '@storybook/preview-api';
|
import type { Addon } from '@storybook/addons';
|
||||||
|
|
||||||
const menuMapper = ({ api, state }: Combo) => ({
|
const menuMapper = ({ api, state }: Combo) => ({
|
||||||
isVisible: state.layout.showPanel,
|
isVisible: state.layout.showPanel,
|
||||||
|
@ -3,7 +3,7 @@ import React from 'react';
|
|||||||
import copy from 'copy-to-clipboard';
|
import copy from 'copy-to-clipboard';
|
||||||
import { getStoryHref, IconButton, Icons } from '@storybook/components';
|
import { getStoryHref, IconButton, Icons } from '@storybook/components';
|
||||||
import { Consumer, type Combo } from '@storybook/api';
|
import { Consumer, type Combo } from '@storybook/api';
|
||||||
import type { Addon } from '@storybook/preview-api';
|
import type { Addon } from '@storybook/addons';
|
||||||
|
|
||||||
const { PREVIEW_URL, document } = global;
|
const { PREVIEW_URL, document } = global;
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@ import global from 'global';
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { getStoryHref, IconButton, Icons } from '@storybook/components';
|
import { getStoryHref, IconButton, Icons } from '@storybook/components';
|
||||||
import { Consumer, type Combo } from '@storybook/api';
|
import { Consumer, type Combo } from '@storybook/api';
|
||||||
import type { Addon } from '@storybook/preview-api';
|
import type { Addon } from '@storybook/addons';
|
||||||
|
|
||||||
const { PREVIEW_URL } = global;
|
const { PREVIEW_URL } = global;
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { IconButton, Icons, Separator } from '@storybook/components';
|
import { IconButton, Icons, Separator } from '@storybook/components';
|
||||||
import { Consumer, type Combo } from '@storybook/api';
|
import { Consumer, type Combo } from '@storybook/api';
|
||||||
import type { Addon } from '@storybook/preview-api';
|
import type { Addon } from '@storybook/addons';
|
||||||
|
|
||||||
const menuMapper = ({ api, state }: Combo) => ({
|
const menuMapper = ({ api, state }: Combo) => ({
|
||||||
isVisible: state.layout.showNav,
|
isVisible: state.layout.showNav,
|
||||||
|
@ -2,7 +2,7 @@ import type { ComponentProps } from 'react';
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { IconButton, Icons } from '@storybook/components';
|
import { IconButton, Icons } from '@storybook/components';
|
||||||
import { Consumer, type Combo } from '@storybook/api';
|
import { Consumer, type Combo } from '@storybook/api';
|
||||||
import type { Addon } from '@storybook/preview-api';
|
import type { Addon } from '@storybook/addons';
|
||||||
import { styled } from '@storybook/theming';
|
import { styled } from '@storybook/theming';
|
||||||
import { FORCE_REMOUNT } from '@storybook/core-events';
|
import { FORCE_REMOUNT } from '@storybook/core-events';
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@ import type { SyntheticEvent, MouseEventHandler } from 'react';
|
|||||||
import React, { Component, useCallback } from 'react';
|
import React, { Component, useCallback } from 'react';
|
||||||
|
|
||||||
import { Icons, IconButton, Separator } from '@storybook/components';
|
import { Icons, IconButton, Separator } from '@storybook/components';
|
||||||
import type { Addon } from '@storybook/preview-api';
|
import type { Addon } from '@storybook/addons';
|
||||||
|
|
||||||
const initialZoom = 1 as const;
|
const initialZoom = 1 as const;
|
||||||
|
|
||||||
|
@ -7237,6 +7237,7 @@ __metadata:
|
|||||||
version: 0.0.0-use.local
|
version: 0.0.0-use.local
|
||||||
resolution: "@storybook/preview-api@workspace:lib/preview-api"
|
resolution: "@storybook/preview-api@workspace:lib/preview-api"
|
||||||
dependencies:
|
dependencies:
|
||||||
|
"@jest/globals": ^26.6.2
|
||||||
"@storybook/addons": 7.0.0-alpha.47
|
"@storybook/addons": 7.0.0-alpha.47
|
||||||
"@storybook/channels": next
|
"@storybook/channels": next
|
||||||
"@storybook/client-logger": 7.0.0-alpha.47
|
"@storybook/client-logger": 7.0.0-alpha.47
|
||||||
|
Loading…
x
Reference in New Issue
Block a user