diff --git a/addons/notes/README.md b/addons/notes/README.md index 368c2a834e0..432ee9f6969 100644 --- a/addons/notes/README.md +++ b/addons/notes/README.md @@ -7,7 +7,7 @@ [![Storybook Slack](https://now-examples-slackin-rrirkqohko.now.sh/badge.svg)](https://now-examples-slackin-rrirkqohko.now.sh/) [![Backers on Open Collective](https://opencollective.com/storybook/backers/badge.svg)](#backers) [![Sponsors on Open Collective](https://opencollective.com/storybook/sponsors/badge.svg)](#sponsors) -* * * +--- Storybook Addon Notes allows you to write notes (text or HTML) for your stories in [Storybook](https://storybook.js.org). @@ -29,16 +29,25 @@ Add following content to it: import '@storybook/addon-notes/register'; ``` -Then write your stories like this: +Then add the `withNotes` decorator to all stories in your `config.js`: + +```js +// Import from @storybook/X where X is your framework +import { configure, addDecorator } from '@storybook/react'; +import { withNotes } from '@storybook/addon-notes'; + +addDecorator(withNotes); +``` + +You can use the `notes` parameter to add a note to each story: ```js import { storiesOf } from '@storybook/react'; -import { withNotes } from '@storybook/addon-notes'; import Component from './Component'; storiesOf('Component', module) - .add('with some emoji', withNotes('A very simple component')(() => )); + .add('with some emoji', () => , { notes: 'A very simple component' }); ``` #### Using Markdown @@ -47,25 +56,27 @@ To use markdown in your notes simply import a markdown file and use that in your ```js import { storiesOf } from '@storybook/react'; -import { withNotes } from '@storybook/addon-notes'; import Component from './Component'; import someMarkdownText from './someMarkdownText.md'; -storiesOf('Component', module) - .add('With Markdown', withNotes(someMarkdownText)(() => )); - +storiesOf('Component', module).add( + 'With Markdown', + () => + { notes: someMarkdownText } +); ``` -If you want to use Github flavored markdown inline, use `withMarkdownNotes`: +If you want to use Github flavored markdown inline, use `notes: { markdownText: 'your md' }`: ```js import { storiesOf } from '@storybook/react'; -import { withMarkdownNotes } from '@storybook/addon-notes'; import Component from './Component'; -storiesOf('Component', module) - .add('With Markdown', withMarkdownNotes(` -# Hello World +storiesOf('Component', module).add( + 'With Markdown', + () => + { notes: { markdown: ` + # Hello World This is some code showing usage of the component and other inline documentation @@ -75,20 +86,6 @@ This is some code showing usage of the component and other inline documentation ~~~ - `)(() => )); - -``` - -### Deprecated API -This API is slated for removal in 4.0 - -```js -import { WithNotes } from '@storybook/addon-notes'; - -storiesOf('Addon Notes', module) - .add('using deprecated API', () => ( - - - - )); +`} } +); ``` diff --git a/addons/notes/src/__tests__/index.js b/addons/notes/src/__tests__/index.js index a21aec2a7f2..611a8a2bb66 100644 --- a/addons/notes/src/__tests__/index.js +++ b/addons/notes/src/__tests__/index.js @@ -4,12 +4,39 @@ import { withNotes } from '..'; jest.mock('@storybook/addons'); describe('Storybook Addon Notes', () => { - it('should inject info', () => { + it('should inject text from `notes` parameter', () => { const channel = { emit: jest.fn() }; addons.getChannel.mockReturnValue(channel); const getStory = jest.fn(); - const context = {}; + const context = { parameters: { notes: 'hello' } }; + + withNotes(getStory, context); + expect(channel.emit).toHaveBeenCalledWith('storybook/notes/add_notes', 'hello'); + expect(getStory).toHaveBeenCalledWith(context); + }); + + it('should inject markdown from `notes.markdown` parameter', () => { + const channel = { emit: jest.fn() }; + addons.getChannel.mockReturnValue(channel); + + const getStory = jest.fn(); + const context = { parameters: { notes: { markdown: '# hello' } } }; + + withNotes(getStory, context); + expect(channel.emit).toHaveBeenCalledWith( + 'storybook/notes/add_notes', + expect.stringContaining('

hello

') + ); + expect(getStory).toHaveBeenCalledWith(context); + }); + + it('should inject info (deprecated API)', () => { + const channel = { emit: jest.fn() }; + addons.getChannel.mockReturnValue(channel); + + const getStory = jest.fn(); + const context = { parameters: {} }; const decoratedStory = withNotes('hello')(getStory); decoratedStory(context); diff --git a/addons/notes/src/index.js b/addons/notes/src/index.js index 8e709f0b20f..01e90aec273 100644 --- a/addons/notes/src/index.js +++ b/addons/notes/src/index.js @@ -1,34 +1,46 @@ -import deprecate from 'util-deprecate'; import addons from '@storybook/addons'; import marked from 'marked'; -import { WithNotes as ReactWithNotes } from './react'; -export const withMarkdownNotes = (text, options) => { +function renderMarkdown(text, options) { + marked.setOptions({ ...marked.defaults, options }); + return marked(text); +} + +const decorator = options => { const channel = addons.getChannel(); - return getStory => context => { - marked.setOptions({ ...marked.defaults, options }); - // send the notes to the channel before the story is rendered - channel.emit('storybook/notes/add_notes', marked(text)); + return (getStory, context) => { + const { parameters: { notes } } = context; + const storyOptions = notes || options; + + if (storyOptions) { + const { text, markdown, markdownOptions } = + typeof storyOptions === 'string' ? { text: storyOptions } : storyOptions; + + if (!text && !markdown) { + throw new Error('You must set of one of `text` or `markdown` on the `notes` parameter'); + } + + channel.emit('storybook/notes/add_notes', text || renderMarkdown(markdown, markdownOptions)); + } + return getStory(context); }; }; -export const withNotes = textOrOptions => { - const channel = addons.getChannel(); - const options = typeof textOrOptions === 'string' ? { text: textOrOptions } : textOrOptions; +const hoc = options => story => context => decorator(options)(story, context); - return getStory => context => { - // send the notes to the channel before the story is rendered - channel.emit('storybook/notes/add_notes', options.text); - return getStory(context); - }; +export const withMarkdownNotes = (text, options) => + hoc({ + markdown: text, + markdownOptions: options, + }); + +export const withNotes = (...args) => { + // Used without options as .addDecorator(withNotes) + if (typeof args[0] === 'function') { + return decorator()(...args); + } + + // Input are options, ala .add('name', withNotes('note')(() => )) + return hoc(args[0]); }; - -Object.defineProperty(exports, 'WithNotes', { - configurable: true, - enumerable: true, - get: deprecate( - () => ReactWithNotes, - '@storybook/addon-notes WithNotes Component is deprecated, use withNotes() instead. See https://github.com/storybooks/storybook/tree/master/addons/notes' - ), -}); diff --git a/app/mithril/src/client/preview/render.js b/app/mithril/src/client/preview/render.js index 1c6272a6411..6f45dfbf88d 100644 --- a/app/mithril/src/client/preview/render.js +++ b/app/mithril/src/client/preview/render.js @@ -44,7 +44,7 @@ export function renderMain(data, storyStore, forceRender) { const noPreview = ; const { selectedKind, selectedStory } = data; - const story = storyStore.getStory(selectedKind, selectedStory); + const story = storyStore.getStoryWithContext(selectedKind, selectedStory); if (!story) { m.mount(rootEl, { view: () => noPreview }); return; @@ -57,12 +57,7 @@ export function renderMain(data, storyStore, forceRender) { previousKind = selectedKind; previousStory = selectedStory; - const context = { - kind: selectedKind, - story: selectedStory, - }; - - const element = story(context); + const element = story(); if (!element) { const error = { diff --git a/examples/angular-cli/src/stories/addon-notes.stories.ts b/examples/angular-cli/src/stories/addon-notes.stories.ts index c47e63d39cb..f83cf704cf9 100644 --- a/examples/angular-cli/src/stories/addon-notes.stories.ts +++ b/examples/angular-cli/src/stories/addon-notes.stories.ts @@ -3,31 +3,34 @@ import { withNotes } from '@storybook/addon-notes'; import { Button } from '@storybook/angular/demo'; storiesOf('Addon|Notes', module) + .addDecorator(withNotes) .add( 'Simple note', - withNotes({ text: 'My notes on some button' })(() => ({ + () => ({ component: Button, props: { text: 'Notes on some Button', onClick: () => {}, }, - })) + }), + { notes: 'My notes on some button' } ) .add( 'Note with HTML', - withNotes({ - text: ` + () => ({ + component: Button, + props: { + text: 'Notes with HTML', + onClick: () => {}, + }, + }), + { + notes: `

My notes on emojis

It's not all that important to be honest, but.. Emojis are great, I love emojis, in fact I like using them in my Component notes too! 😇 `, - })(() => ({ - component: Button, - props: { - text: 'Notes with HTML', - onClick: () => {}, - }, - })) + } ); diff --git a/examples/cra-kitchen-sink/src/stories/index.stories.js b/examples/cra-kitchen-sink/src/stories/index.stories.js index a13abed14f9..b7c5b7c2a07 100644 --- a/examples/cra-kitchen-sink/src/stories/index.stories.js +++ b/examples/cra-kitchen-sink/src/stories/index.stories.js @@ -3,8 +3,7 @@ import React from 'react'; import { storiesOf } from '@storybook/react'; import { setOptions } from '@storybook/addon-options'; import { action } from '@storybook/addon-actions'; -// eslint-disable-next-line import/named -import { withNotes, WithNotes } from '@storybook/addon-notes'; +import { withNotes } from '@storybook/addon-notes'; import centered from '@storybook/addon-centered'; import { withInfo } from '@storybook/addon-info'; import { Button } from '@storybook/react/demo'; @@ -32,6 +31,7 @@ const InfoButton = () => ( ); storiesOf('Button', module) + .addDecorator(withNotes) .add('with text', () => ( )) - .add('with notes', () => ( - // deprecated usage - + .add( + 'with notes', + () => ( - - )) + ), + { notes: 'A very simple button' } + ) .addWithInfo( 'with some info', 'Use the [info addon](https://github.com/storybooks/storybook/tree/master/addons/info) with its painful API.', diff --git a/examples/mithril-kitchen-sink/src/stories/addon-notes.stories.js b/examples/mithril-kitchen-sink/src/stories/addon-notes.stories.js index ba1edfa822f..ff9030b4270 100644 --- a/examples/mithril-kitchen-sink/src/stories/addon-notes.stories.js +++ b/examples/mithril-kitchen-sink/src/stories/addon-notes.stories.js @@ -7,9 +7,10 @@ import { storiesOf } from '@storybook/mithril'; import { withNotes } from '@storybook/addon-notes'; storiesOf('Addons|Notes', module) + .addDecorator(withNotes) .add( 'Simple note', - withNotes({ text: 'My notes on some bold text' })(() => ({ + () => ({ view: () => (

@@ -22,19 +23,12 @@ storiesOf('Addons|Notes', module)

), - })) + }), + { notes: 'My notes on some bold text' } ) .add( 'Note with HTML', - withNotes({ - text: ` -

My notes on emojies

- - It's not all that important to be honest, but.. - - Emojis are great, I love emojis, in fact I like using them in my Component notes too! 😇 - `, - })(() => ({ + () => ({ view: () => (

🤔😳😯😮 @@ -44,5 +38,14 @@ storiesOf('Addons|Notes', module) 🤓😑😶😊

), - })) + }), + { + notes: ` +

My notes on emojies

+ + It's not all that important to be honest, but.. + + Emojis are great, I love emojis, in fact I like using them in my Component notes too! 😇 + `, + } ); diff --git a/examples/official-storybook/stories/__snapshots__/addon-notes.stories.storyshot b/examples/official-storybook/stories/__snapshots__/addon-notes.stories.storyshot index 32623432519..ed6cc4ddcbe 100644 --- a/examples/official-storybook/stories/__snapshots__/addon-notes.stories.storyshot +++ b/examples/official-storybook/stories/__snapshots__/addon-notes.stories.storyshot @@ -1,8 +1,14 @@ // Jest Snapshot v1, https://goo.gl/fbAQLP -exports[`Storyshots Addons|Notes using deprecated API 1`] = ` +exports[`Storyshots Addons|Notes using decorator arguments, withMarkdownNotes 1`] = ` +`; + +exports[`Storyshots Addons|Notes using decorator arguments, withNotes 1`] = ` + `; @@ -20,6 +26,6 @@ exports[`Storyshots Addons|Notes withNotes rendering imported markdown 1`] = ` exports[`Storyshots Addons|Notes withNotes rendering inline, github-flavored markdown 1`] = ` `; diff --git a/examples/official-storybook/stories/addon-notes.stories.js b/examples/official-storybook/stories/addon-notes.stories.js index 62dabca8379..54d70f09f8c 100644 --- a/examples/official-storybook/stories/addon-notes.stories.js +++ b/examples/official-storybook/stories/addon-notes.stories.js @@ -1,28 +1,15 @@ import React from 'react'; import { storiesOf } from '@storybook/react'; -// eslint-disable-next-line import/named -import { withNotes, WithNotes, withMarkdownNotes } from '@storybook/addon-notes'; -import { action } from '@storybook/addon-actions'; +import { withNotes, withMarkdownNotes } from '@storybook/addon-notes'; import BaseButton from '../components/BaseButton'; import markdownNotes from './notes/notes.md'; -storiesOf('Addons|Notes', module) - .add( - 'withNotes', - withNotes( - 'This is the notes for a button. This is helpful for adding details about a story in a separate panel.' - )(() => ) - ) - .add( - 'withNotes rendering imported markdown', - withNotes(markdownNotes)(() => ( - - )) - ) - .add( - 'withNotes rendering inline, github-flavored markdown', - withMarkdownNotes(` +const baseStory = () => ( + +); + +const markdownString = ` # Documentation This is inline github-flavored markdown! @@ -37,12 +24,20 @@ storiesOf('Addons|Notes', module) )) ) ~~~ - `)(() => ( - - )) - ) - .add('using deprecated API', () => ( - - - - )); + `; + +storiesOf('Addons|Notes', module) + .addDecorator(withNotes) + .add('withNotes', baseStory, { + notes: + 'This is the notes for a button. This is helpful for adding details about a story in a separate panel.', + }) + .add('withNotes rendering imported markdown', baseStory, { notes: markdownNotes }) + .add('withNotes rendering inline, github-flavored markdown', baseStory, { + notes: { markdown: markdownString }, + }) + .add('using decorator arguments, withNotes', withNotes('Notes into withNotes')(baseStory)) + .add( + 'using decorator arguments, withMarkdownNotes', + withMarkdownNotes(markdownString)(baseStory) + ); diff --git a/examples/polymer-cli/src/stories/addon-notes.stories.js b/examples/polymer-cli/src/stories/addon-notes.stories.js index cab392df79f..789eac67b41 100644 --- a/examples/polymer-cli/src/stories/addon-notes.stories.js +++ b/examples/polymer-cli/src/stories/addon-notes.stories.js @@ -2,22 +2,21 @@ import { storiesOf } from '@storybook/polymer'; import { withNotes } from '@storybook/addon-notes'; storiesOf('Addon|Notes', module) + .addDecorator(withNotes) .add( 'Simple note', - withNotes({ text: 'My notes on some bold text' })( - () => - '

Etiam vulputate elit eu venenatis eleifend. Duis nec lectus augue. Morbi egestas diam sed vulputate mollis. Fusce egestas pretium vehicula. Integer sed neque diam. Donec consectetur velit vitae enim varius, ut placerat arcu imperdiet. Praesent sed faucibus arcu. Nullam sit amet nibh a enim eleifend rhoncus. Donec pretium elementum leo at fermentum. Nulla sollicitudin, mauris quis semper tempus, sem metus tristique diam, efficitur pulvinar mi urna id urna.

' - ) + () => + '

Etiam vulputate elit eu venenatis eleifend. Duis nec lectus augue. Morbi egestas diam sed vulputate mollis. Fusce egestas pretium vehicula. Integer sed neque diam. Donec consectetur velit vitae enim varius, ut placerat arcu imperdiet. Praesent sed faucibus arcu. Nullam sit amet nibh a enim eleifend rhoncus. Donec pretium elementum leo at fermentum. Nulla sollicitudin, mauris quis semper tempus, sem metus tristique diam, efficitur pulvinar mi urna id urna.

', + { + notes: 'My notes on some bold text', + } ) - .add( - 'Note with HTML', - withNotes({ - text: ` + .add('Note with HTML', () => '

🤔😳😯😮
😄😩😓😱
🤓😑😶😊

', { + notes: `

My notes on emojies

It's not all that important to be honest, but.. Emojis are great, I love emojis, in fact I like using them in my Component notes too! 😇 `, - })(() => '

🤔😳😯😮
😄😩😓😱
🤓😑😶😊

') - ); + }); diff --git a/examples/vue-kitchen-sink/src/stories/addon-notes.stories.js b/examples/vue-kitchen-sink/src/stories/addon-notes.stories.js index a35213b8ff2..9a3f3a3b395 100644 --- a/examples/vue-kitchen-sink/src/stories/addon-notes.stories.js +++ b/examples/vue-kitchen-sink/src/stories/addon-notes.stories.js @@ -2,24 +2,27 @@ import { storiesOf } from '@storybook/vue'; import { withNotes } from '@storybook/addon-notes'; storiesOf('Addon|Notes', module) + .addDecorator(withNotes) .add( 'Simple note', - withNotes({ text: 'My notes on some bold text' })(() => ({ + () => ({ template: '

Etiam vulputate elit eu venenatis eleifend. Duis nec lectus augue. Morbi egestas diam sed vulputate mollis. Fusce egestas pretium vehicula. Integer sed neque diam. Donec consectetur velit vitae enim varius, ut placerat arcu imperdiet. Praesent sed faucibus arcu. Nullam sit amet nibh a enim eleifend rhoncus. Donec pretium elementum leo at fermentum. Nulla sollicitudin, mauris quis semper tempus, sem metus tristique diam, efficitur pulvinar mi urna id urna.

', - })) + }), + { notes: 'My notes on some bold text' } ) .add( 'Note with HTML', - withNotes({ - text: ` + () => ({ + template: '

🤔😳😯😮
😄😩😓😱
🤓😑😶😊

', + }), + { + notes: `

My notes on emojies

It's not all that important to be honest, but.. Emojis are great, I love emojis, in fact I like using them in my Component notes too! 😇 `, - })(() => ({ - template: '

🤔😳😯😮
😄😩😓😱
🤓😑😶😊

', - })) + } );