mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 19:01:05 +08:00
Core: Fix default sorting of docs-only stories (#9504)
Core: Fix default sorting of docs-only stories
This commit is contained in:
commit
3a4f1877d0
5
examples/official-storybook/intro.stories.mdx
Normal file
5
examples/official-storybook/intro.stories.mdx
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
<Meta title="About/Intro" />
|
||||||
|
|
||||||
|
# Official-storybook
|
||||||
|
|
||||||
|
Welcome to `official-storybook`, a collection of test cases and demos for `@storybook/react` and all its addons.
|
@ -1,5 +1,6 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
stories: [
|
stories: [
|
||||||
|
// FIXME: Breaks e2e tests './intro.stories.mdx',
|
||||||
'../../lib/ui/src/**/*.stories.(js|tsx|mdx)',
|
'../../lib/ui/src/**/*.stories.(js|tsx|mdx)',
|
||||||
'../../lib/components/src/**/*.stories.(js|tsx|mdx)',
|
'../../lib/components/src/**/*.stories.(js|tsx|mdx)',
|
||||||
'./stories/**/*.stories.(js|tsx|mdx)',
|
'./stories/**/*.stories.(js|tsx|mdx)',
|
||||||
|
@ -514,21 +514,23 @@ describe('preview.client_api', () => {
|
|||||||
clientApi: { storiesOf, getStorybook },
|
clientApi: { storiesOf, getStorybook },
|
||||||
channel,
|
channel,
|
||||||
} = getContext(undefined);
|
} = getContext(undefined);
|
||||||
|
const module0 = new MockModule();
|
||||||
const module1 = new MockModule();
|
const module1 = new MockModule();
|
||||||
const module2 = new MockModule();
|
const module2 = new MockModule();
|
||||||
channel.emit = jest.fn();
|
channel.emit = jest.fn();
|
||||||
|
|
||||||
expect(getStorybook()).toEqual([]);
|
expect(getStorybook()).toEqual([]);
|
||||||
|
|
||||||
|
storiesOf('kind0', module0).add('story0-docs-only', jest.fn(), { docsOnly: true });
|
||||||
storiesOf('kind1', module1).add('story1', jest.fn());
|
storiesOf('kind1', module1).add('story1', jest.fn());
|
||||||
storiesOf('kind2', module2).add('story2', jest.fn());
|
storiesOf('kind2', module2).add('story2', jest.fn());
|
||||||
|
|
||||||
// storyStore debounces so we need to wait for the next tick
|
// storyStore debounces so we need to wait for the next tick
|
||||||
await new Promise(r => setTimeout(r, 0));
|
await new Promise(r => setTimeout(r, 0));
|
||||||
|
|
||||||
let [event, storiesHash] = channel.emit.mock.calls[0];
|
let [event, args] = channel.emit.mock.calls[0];
|
||||||
expect(event).toEqual(Events.SET_STORIES);
|
expect(event).toEqual(Events.SET_STORIES);
|
||||||
expect(Object.values(storiesHash.stories).map(v => v.kind)).toEqual(['kind1', 'kind2']);
|
expect(Object.values(args.stories).map(v => v.kind)).toEqual(['kind0', 'kind1', 'kind2']);
|
||||||
expect(getStorybook().map(story => story.kind)).toEqual(['kind1', 'kind2']);
|
expect(getStorybook().map(story => story.kind)).toEqual(['kind1', 'kind2']);
|
||||||
|
|
||||||
channel.emit.mockClear();
|
channel.emit.mockClear();
|
||||||
@ -540,10 +542,10 @@ describe('preview.client_api', () => {
|
|||||||
|
|
||||||
await new Promise(r => setTimeout(r, 0));
|
await new Promise(r => setTimeout(r, 0));
|
||||||
// eslint-disable-next-line prefer-destructuring
|
// eslint-disable-next-line prefer-destructuring
|
||||||
[event, storiesHash] = channel.emit.mock.calls[0];
|
[event, args] = channel.emit.mock.calls[0];
|
||||||
|
|
||||||
expect(event).toEqual(Events.SET_STORIES);
|
expect(event).toEqual(Events.SET_STORIES);
|
||||||
expect(Object.values(storiesHash.stories).map(v => v.kind)).toEqual(['kind1', 'kind2']);
|
expect(Object.values(args.stories).map(v => v.kind)).toEqual(['kind0', 'kind1', 'kind2']);
|
||||||
expect(getStorybook().map(story => story.kind)).toEqual(['kind1', 'kind2']);
|
expect(getStorybook().map(story => story.kind)).toEqual(['kind1', 'kind2']);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -55,6 +55,8 @@ interface StoryOptions {
|
|||||||
includeDocsOnly?: boolean;
|
includeDocsOnly?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type KindOrder = Record<string, number>;
|
||||||
|
|
||||||
const isStoryDocsOnly = (parameters?: Parameters) => {
|
const isStoryDocsOnly = (parameters?: Parameters) => {
|
||||||
return parameters && parameters.docsOnly;
|
return parameters && parameters.docsOnly;
|
||||||
};
|
};
|
||||||
@ -81,6 +83,8 @@ export default class StoryStore extends EventEmitter {
|
|||||||
|
|
||||||
_selection: Selection;
|
_selection: Selection;
|
||||||
|
|
||||||
|
_kindOrder: KindOrder;
|
||||||
|
|
||||||
constructor(params: { channel: Channel }) {
|
constructor(params: { channel: Channel }) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
@ -90,6 +94,7 @@ export default class StoryStore extends EventEmitter {
|
|||||||
this._selection = {} as any;
|
this._selection = {} as any;
|
||||||
this._channel = params.channel;
|
this._channel = params.channel;
|
||||||
this._error = undefined;
|
this._error = undefined;
|
||||||
|
this._kindOrder = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
setChannel = (channel: Channel) => {
|
setChannel = (channel: Channel) => {
|
||||||
@ -133,17 +138,11 @@ export default class StoryStore extends EventEmitter {
|
|||||||
stable.inplace(stories, sortFn);
|
stable.inplace(stories, sortFn);
|
||||||
} else {
|
} else {
|
||||||
// NOTE: when kinds are HMR'ed they get temporarily removed from the `_data` array
|
// NOTE: when kinds are HMR'ed they get temporarily removed from the `_data` array
|
||||||
// and thus lose order. However in `_legacydata` they just get zeroed out, meaning
|
// and thus lose order. However `_kindOrder` preservers the original load order
|
||||||
// that the order is preserved. Here we can use this to preserve the load
|
stable.inplace(
|
||||||
// order if there is no sort function, although it's a hack.
|
stories,
|
||||||
const kindOrder = Object.values(this._legacydata).reduce(
|
(s1, s2) => this._kindOrder[s1[1].kind] - this._kindOrder[s2[1].kind]
|
||||||
(acc: Record<string, number>, { kind }: any, idx: number) => {
|
|
||||||
acc[kind] = idx;
|
|
||||||
return acc;
|
|
||||||
},
|
|
||||||
{}
|
|
||||||
);
|
);
|
||||||
stable.inplace(stories, (s1, s2) => kindOrder[s1[1].kind] - kindOrder[s2[1].kind]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// removes function values from all stories so they are safe to transport over the channel
|
// removes function values from all stories so they are safe to transport over the channel
|
||||||
@ -252,11 +251,18 @@ export default class StoryStore extends EventEmitter {
|
|||||||
parameters,
|
parameters,
|
||||||
};
|
};
|
||||||
|
|
||||||
// LEGACY DATA
|
// Don't store docs-only stories in legacy data because
|
||||||
|
// existing clients (at the time?!), e.g. storyshots/chromatic
|
||||||
|
// are not necessarily equipped to process them
|
||||||
if (!isStoryDocsOnly(parameters)) {
|
if (!isStoryDocsOnly(parameters)) {
|
||||||
this.addLegacyStory({ kind, name, storyFn, parameters });
|
this.addLegacyStory({ kind, name, storyFn, parameters });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Store 1-based order of kind loading to preserve sorting on HMR
|
||||||
|
if (!this._kindOrder[kind]) {
|
||||||
|
this._kindOrder[kind] = 1 + Object.keys(this._kindOrder).length;
|
||||||
|
}
|
||||||
|
|
||||||
// LET'S SEND IT TO THE MANAGER
|
// LET'S SEND IT TO THE MANAGER
|
||||||
this.pushToManager();
|
this.pushToManager();
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user