mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 07:21:16 +08:00
Merge remote-tracking branch 'origin/next' into nested-addons
This commit is contained in:
commit
a291e26f62
@ -80,7 +80,16 @@ commands:
|
|||||||
echo "🏁 The PR isn't labelled with '<< parameters.label >>' so this job will end at the current step."
|
echo "🏁 The PR isn't labelled with '<< parameters.label >>' so this job will end at the current step."
|
||||||
circleci-agent step halt
|
circleci-agent step halt
|
||||||
fi
|
fi
|
||||||
|
cancel-workflow-on-failure:
|
||||||
|
description: 'Cancels the entire workflow in case the previous step has failed'
|
||||||
|
steps:
|
||||||
|
- run:
|
||||||
|
name: Cancel current workflow
|
||||||
|
when: on_fail
|
||||||
|
command: |
|
||||||
|
echo "Canceling workflow as previous step resulted in failure."
|
||||||
|
echo "To execute all checks locally, please run yarn ci-tests"
|
||||||
|
curl -X POST --header "Content-Type: application/json" "https://circleci.com/api/v2/workflow/${CIRCLE_WORKFLOW_ID}/cancel?circle-token=${WORKFLOW_CANCELER}"
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
executor:
|
executor:
|
||||||
@ -98,6 +107,11 @@ jobs:
|
|||||||
command: |
|
command: |
|
||||||
yarn task --task compile --start-from=auto --no-link --debug
|
yarn task --task compile --start-from=auto --no-link --debug
|
||||||
git diff --exit-code
|
git diff --exit-code
|
||||||
|
- run:
|
||||||
|
name: Publish to Verdaccio
|
||||||
|
command: |
|
||||||
|
cd code
|
||||||
|
yarn local-registry --publish
|
||||||
- save_cache:
|
- save_cache:
|
||||||
name: Save Yarn cache
|
name: Save Yarn cache
|
||||||
key: build-yarn-2-cache-v4--{{ checksum "code/yarn.lock" }}--{{ checksum "scripts/yarn.lock" }}
|
key: build-yarn-2-cache-v4--{{ checksum "code/yarn.lock" }}--{{ checksum "scripts/yarn.lock" }}
|
||||||
@ -116,23 +130,6 @@ jobs:
|
|||||||
- code/ui
|
- code/ui
|
||||||
- code/renderers
|
- code/renderers
|
||||||
- code/presets
|
- code/presets
|
||||||
publish:
|
|
||||||
executor:
|
|
||||||
class: small
|
|
||||||
name: sb_node_16_classic
|
|
||||||
steps:
|
|
||||||
- git-shallow-clone/checkout_advanced:
|
|
||||||
clone_options: '--depth 1 --verbose'
|
|
||||||
- attach_workspace:
|
|
||||||
at: .
|
|
||||||
- run:
|
|
||||||
name: running local registry
|
|
||||||
command: |
|
|
||||||
cd code
|
|
||||||
yarn local-registry --publish
|
|
||||||
- persist_to_workspace:
|
|
||||||
root: .
|
|
||||||
paths:
|
|
||||||
- .verdaccio-cache
|
- .verdaccio-cache
|
||||||
cra-bench:
|
cra-bench:
|
||||||
executor:
|
executor:
|
||||||
@ -228,6 +225,7 @@ jobs:
|
|||||||
command: |
|
command: |
|
||||||
cd code
|
cd code
|
||||||
yarn lint
|
yarn lint
|
||||||
|
- cancel-workflow-on-failure
|
||||||
check:
|
check:
|
||||||
executor:
|
executor:
|
||||||
class: xlarge
|
class: xlarge
|
||||||
@ -242,6 +240,7 @@ jobs:
|
|||||||
command: |
|
command: |
|
||||||
yarn task --task check --start-from=auto --no-link --debug
|
yarn task --task check --start-from=auto --no-link --debug
|
||||||
git diff --exit-code
|
git diff --exit-code
|
||||||
|
- cancel-workflow-on-failure
|
||||||
script-unit-tests:
|
script-unit-tests:
|
||||||
executor: sb_node_16_browsers
|
executor: sb_node_16_browsers
|
||||||
steps:
|
steps:
|
||||||
@ -256,6 +255,7 @@ jobs:
|
|||||||
yarn test --coverage --ci
|
yarn test --coverage --ci
|
||||||
- store_test_results:
|
- store_test_results:
|
||||||
path: scripts/junit.xml
|
path: scripts/junit.xml
|
||||||
|
- cancel-workflow-on-failure
|
||||||
unit-tests:
|
unit-tests:
|
||||||
executor:
|
executor:
|
||||||
class: xlarge
|
class: xlarge
|
||||||
@ -276,6 +276,7 @@ jobs:
|
|||||||
root: .
|
root: .
|
||||||
paths:
|
paths:
|
||||||
- code/coverage
|
- code/coverage
|
||||||
|
- cancel-workflow-on-failure
|
||||||
coverage:
|
coverage:
|
||||||
executor:
|
executor:
|
||||||
class: small
|
class: small
|
||||||
@ -463,14 +464,11 @@ workflows:
|
|||||||
equal: [ daily-tests, << pipeline.parameters.workflow >> ]
|
equal: [ daily-tests, << pipeline.parameters.workflow >> ]
|
||||||
jobs:
|
jobs:
|
||||||
- build
|
- build
|
||||||
- publish:
|
|
||||||
requires:
|
|
||||||
- build
|
|
||||||
- create-sandboxes:
|
- create-sandboxes:
|
||||||
parallelism: 24
|
parallelism: 24
|
||||||
cadence: "daily"
|
cadence: "daily"
|
||||||
requires:
|
requires:
|
||||||
- publish
|
- build
|
||||||
# - smoke-test-sandboxes: # disabled for now
|
# - smoke-test-sandboxes: # disabled for now
|
||||||
# requires:
|
# requires:
|
||||||
# - create-sandboxes
|
# - create-sandboxes
|
||||||
@ -517,19 +515,16 @@ workflows:
|
|||||||
- coverage:
|
- coverage:
|
||||||
requires:
|
requires:
|
||||||
- unit-tests
|
- unit-tests
|
||||||
- publish:
|
|
||||||
requires:
|
|
||||||
- build
|
|
||||||
- cra-bench:
|
- cra-bench:
|
||||||
requires:
|
requires:
|
||||||
- publish
|
- build
|
||||||
- react-vite-bench:
|
- react-vite-bench:
|
||||||
requires:
|
requires:
|
||||||
- publish
|
- build
|
||||||
## new workflow
|
## new workflow
|
||||||
- create-sandboxes:
|
- create-sandboxes:
|
||||||
requires:
|
requires:
|
||||||
- publish
|
- build
|
||||||
# - smoke-test-sandboxes: # disabled for now
|
# - smoke-test-sandboxes: # disabled for now
|
||||||
# requires:
|
# requires:
|
||||||
# - create-sandboxes
|
# - create-sandboxes
|
||||||
|
@ -39,6 +39,7 @@
|
|||||||
- [MDX2 upgrade](#mdx2-upgrade)
|
- [MDX2 upgrade](#mdx2-upgrade)
|
||||||
- [Dropped source loader / storiesOf static snippets](#dropped-source-loader--storiesof-static-snippets)
|
- [Dropped source loader / storiesOf static snippets](#dropped-source-loader--storiesof-static-snippets)
|
||||||
- [Dropped addon-docs manual configuration](#dropped-addon-docs-manual-configuration)
|
- [Dropped addon-docs manual configuration](#dropped-addon-docs-manual-configuration)
|
||||||
|
- [Autoplay in docs](#autoplay-in-docs)
|
||||||
- [7.0 Deprecations](#70-deprecations)
|
- [7.0 Deprecations](#70-deprecations)
|
||||||
- [`Story` type deprecated](#story-type-deprecated)
|
- [`Story` type deprecated](#story-type-deprecated)
|
||||||
- [`ComponentStory`, `ComponentStoryObj`, `ComponentStoryFn` and `ComponentMeta` types are deprecated](#componentstory-componentstoryobj-componentstoryfn-and-componentmeta-types-are-deprecated)
|
- [`ComponentStory`, `ComponentStoryObj`, `ComponentStoryFn` and `ComponentMeta` types are deprecated](#componentstory-componentstoryobj-componentstoryfn-and-componentmeta-types-are-deprecated)
|
||||||
@ -780,6 +781,12 @@ module.exports = {
|
|||||||
|
|
||||||
Storybook Docs 5.x shipped with instructions for how to manually configure webpack and storybook without the use of Storybook's "presets" feature. Over time, these docs went out of sync. Now in Storybook 7 we have removed support for manual configuration entirely.
|
Storybook Docs 5.x shipped with instructions for how to manually configure webpack and storybook without the use of Storybook's "presets" feature. Over time, these docs went out of sync. Now in Storybook 7 we have removed support for manual configuration entirely.
|
||||||
|
|
||||||
|
#### Autoplay in docs
|
||||||
|
|
||||||
|
Running play functions in docs is generally tricky, as they can steal focus and cause the window to scroll. Consequently, we've disabled play functions in docs by default.
|
||||||
|
|
||||||
|
If your story depends on a play function to render correctly, _and_ you are confident the function autoplaying won't mess up your docs, you can set `parameters.docs.autoplay = true` to have it auto play.
|
||||||
|
|
||||||
### 7.0 Deprecations
|
### 7.0 Deprecations
|
||||||
|
|
||||||
#### `Story` type deprecated
|
#### `Story` type deprecated
|
||||||
|
33
code/addons/docs/template/stories/docspage/autoplay.stories.ts
vendored
Normal file
33
code/addons/docs/template/stories/docspage/autoplay.stories.ts
vendored
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import globalThis from 'global';
|
||||||
|
import { expect } from '@storybook/jest';
|
||||||
|
import { within } from '@storybook/testing-library';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
component: globalThis.Components.Pre,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
args: { text: 'Play has not run' },
|
||||||
|
parameters: { chromatic: { disable: true } },
|
||||||
|
};
|
||||||
|
|
||||||
|
// Should not autoplay
|
||||||
|
export const NoAutoplay = {
|
||||||
|
play: async ({ viewMode, canvasElement }) => {
|
||||||
|
const pre = await within(canvasElement).findByText('Play has not run');
|
||||||
|
if (viewMode === 'docs') {
|
||||||
|
pre.innerText = 'Play should not have run!';
|
||||||
|
// Sort of pointless
|
||||||
|
expect(viewMode).not.toBe('docs');
|
||||||
|
} else {
|
||||||
|
pre.innerText = 'Play has run';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// Should autoplay
|
||||||
|
export const Autoplay = {
|
||||||
|
parameters: { docs: { autoplay: true } },
|
||||||
|
play: async ({ canvasElement }) => {
|
||||||
|
const pre = await within(canvasElement).findByText('Play has not run');
|
||||||
|
pre.innerText = 'Play has run';
|
||||||
|
},
|
||||||
|
};
|
@ -40,4 +40,16 @@ test.describe('addon-docs', () => {
|
|||||||
await expect(text).not.toMatch(/^\(args\) => /);
|
await expect(text).not.toMatch(/^\(args\) => /);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should not run autoplay stories without parameter', async ({ page }) => {
|
||||||
|
const sbPage = new SbPage(page);
|
||||||
|
await sbPage.navigateToStory('addons/docs/docspage/autoplay', 'docs');
|
||||||
|
|
||||||
|
const root = sbPage.previewRoot();
|
||||||
|
const autoplayPre = root.locator('#story--addons-docs-docspage-autoplay--autoplay pre');
|
||||||
|
await expect(autoplayPre).toHaveText('Play has run');
|
||||||
|
|
||||||
|
const noAutoplayPre = root.locator('#story--addons-docs-docspage-autoplay--no-autoplay pre');
|
||||||
|
await expect(noAutoplayPre).toHaveText('Play has not run');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -7,6 +7,7 @@ import { storyPropsProvider } from './StorybookProvider';
|
|||||||
import { isComponentAlreadyDeclaredInModules } from './utils/NgModulesAnalyzer';
|
import { isComponentAlreadyDeclaredInModules } from './utils/NgModulesAnalyzer';
|
||||||
import { isDeclarable, isStandaloneComponent } from './utils/NgComponentAnalyzer';
|
import { isDeclarable, isStandaloneComponent } from './utils/NgComponentAnalyzer';
|
||||||
import { createStorybookWrapperComponent } from './StorybookWrapperComponent';
|
import { createStorybookWrapperComponent } from './StorybookWrapperComponent';
|
||||||
|
import { computesTemplateFromComponent } from './ComputesTemplateFromComponent';
|
||||||
|
|
||||||
export const getStorybookModuleMetadata = (
|
export const getStorybookModuleMetadata = (
|
||||||
{
|
{
|
||||||
@ -21,7 +22,12 @@ export const getStorybookModuleMetadata = (
|
|||||||
storyProps$: Subject<ICollection>
|
storyProps$: Subject<ICollection>
|
||||||
): NgModule => {
|
): NgModule => {
|
||||||
const { props, styles, moduleMetadata = {} } = storyFnAngular;
|
const { props, styles, moduleMetadata = {} } = storyFnAngular;
|
||||||
const { template } = storyFnAngular;
|
let { template } = storyFnAngular;
|
||||||
|
|
||||||
|
const hasTemplate = !hasNoTemplate(template);
|
||||||
|
if (!hasTemplate && component) {
|
||||||
|
template = computesTemplateFromComponent(component, props, '');
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a component that wraps generated template and gives it props
|
* Create a component that wraps generated template and gives it props
|
||||||
@ -68,3 +74,7 @@ export const createStorybookModule = (ngModule: NgModule): Type<unknown> => {
|
|||||||
class StorybookModule {}
|
class StorybookModule {}
|
||||||
return StorybookModule;
|
return StorybookModule;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function hasNoTemplate(template: string | null | undefined): template is undefined {
|
||||||
|
return template === null || template === undefined;
|
||||||
|
}
|
||||||
|
@ -31,6 +31,7 @@ import type {
|
|||||||
} from '@storybook/types';
|
} from '@storybook/types';
|
||||||
import { StoryStore } from '@storybook/store';
|
import { StoryStore } from '@storybook/store';
|
||||||
|
|
||||||
|
import type { StoryRenderOptions } from './render/StoryRender';
|
||||||
import { StoryRender } from './render/StoryRender';
|
import { StoryRender } from './render/StoryRender';
|
||||||
import type { TemplateDocsRender } from './render/TemplateDocsRender';
|
import type { TemplateDocsRender } from './render/TemplateDocsRender';
|
||||||
import type { StandaloneDocsRender } from './render/StandaloneDocsRender';
|
import type { StandaloneDocsRender } from './render/StandaloneDocsRender';
|
||||||
@ -304,7 +305,11 @@ export class Preview<TFramework extends AnyFramework> {
|
|||||||
// main to be consistent with the previous behaviour. In the future,
|
// main to be consistent with the previous behaviour. In the future,
|
||||||
// we will change it to go ahead and load the story, which will end up being
|
// we will change it to go ahead and load the story, which will end up being
|
||||||
// "instant", although async.
|
// "instant", although async.
|
||||||
renderStoryToElement(story: Store_Story<TFramework>, element: HTMLElement) {
|
renderStoryToElement(
|
||||||
|
story: Store_Story<TFramework>,
|
||||||
|
element: HTMLElement,
|
||||||
|
options: StoryRenderOptions
|
||||||
|
) {
|
||||||
if (!this.renderToDOM)
|
if (!this.renderToDOM)
|
||||||
throw new Error(`Cannot call renderStoryToElement before initialization`);
|
throw new Error(`Cannot call renderStoryToElement before initialization`);
|
||||||
|
|
||||||
@ -315,6 +320,7 @@ export class Preview<TFramework extends AnyFramework> {
|
|||||||
this.inlineStoryCallbacks(story.id),
|
this.inlineStoryCallbacks(story.id),
|
||||||
story.id,
|
story.id,
|
||||||
'docs',
|
'docs',
|
||||||
|
options,
|
||||||
story
|
story
|
||||||
);
|
);
|
||||||
render.renderToElement(element);
|
render.renderToElement(element);
|
||||||
|
@ -8,6 +8,7 @@ import type {
|
|||||||
StoryName,
|
StoryName,
|
||||||
} from '@storybook/types';
|
} from '@storybook/types';
|
||||||
import type { Channel } from '@storybook/channels';
|
import type { Channel } from '@storybook/channels';
|
||||||
|
import type { StoryRenderOptions } from '../render/StoryRender';
|
||||||
|
|
||||||
export interface DocsContextProps<TFramework extends AnyFramework = AnyFramework> {
|
export interface DocsContextProps<TFramework extends AnyFramework = AnyFramework> {
|
||||||
/**
|
/**
|
||||||
@ -54,7 +55,8 @@ export interface DocsContextProps<TFramework extends AnyFramework = AnyFramework
|
|||||||
*/
|
*/
|
||||||
renderStoryToElement: (
|
renderStoryToElement: (
|
||||||
story: Store_Story<TFramework>,
|
story: Store_Story<TFramework>,
|
||||||
element: HTMLElement
|
element: HTMLElement,
|
||||||
|
options: StoryRenderOptions
|
||||||
) => () => Promise<void>;
|
) => () => Promise<void>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import type { StoryId, AnyFramework } from '@storybook/types';
|
import type { StoryId, AnyFramework } from '@storybook/types';
|
||||||
|
import type { StoryRenderOptions } from './StoryRender';
|
||||||
|
|
||||||
export type RenderType = 'story' | 'docs';
|
export type RenderType = 'story' | 'docs';
|
||||||
|
|
||||||
@ -17,7 +18,11 @@ export interface Render<TFramework extends AnyFramework> {
|
|||||||
disableKeyListeners: boolean;
|
disableKeyListeners: boolean;
|
||||||
teardown?: (options: { viewModeChanged: boolean }) => Promise<void>;
|
teardown?: (options: { viewModeChanged: boolean }) => Promise<void>;
|
||||||
torndown: boolean;
|
torndown: boolean;
|
||||||
renderToElement: (canvasElement: HTMLElement, renderStoryToElement?: any) => Promise<void>;
|
renderToElement: (
|
||||||
|
canvasElement: HTMLElement,
|
||||||
|
renderStoryToElement?: any,
|
||||||
|
options?: StoryRenderOptions
|
||||||
|
) => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const PREPARE_ABORTED = new Error('prepareAborted');
|
export const PREPARE_ABORTED = new Error('prepareAborted');
|
||||||
|
@ -50,4 +50,56 @@ describe('StoryRender', () => {
|
|||||||
|
|
||||||
await expect(preparePromise).rejects.toThrowError(PREPARE_ABORTED);
|
await expect(preparePromise).rejects.toThrowError(PREPARE_ABORTED);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('does run play function if passed autoplay=true', async () => {
|
||||||
|
const story = {
|
||||||
|
id: 'id',
|
||||||
|
title: 'title',
|
||||||
|
name: 'name',
|
||||||
|
tags: [],
|
||||||
|
applyLoaders: jest.fn(),
|
||||||
|
unboundStoryFn: jest.fn(),
|
||||||
|
playFunction: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const render = new StoryRender(
|
||||||
|
new Channel(),
|
||||||
|
{ getStoryContext: () => ({}) } as any,
|
||||||
|
jest.fn() as any,
|
||||||
|
{} as any,
|
||||||
|
entry.id,
|
||||||
|
'story',
|
||||||
|
{ autoplay: true },
|
||||||
|
story as any
|
||||||
|
);
|
||||||
|
|
||||||
|
await render.renderToElement({} as any);
|
||||||
|
expect(story.playFunction).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not run play function if passed autoplay=false', async () => {
|
||||||
|
const story = {
|
||||||
|
id: 'id',
|
||||||
|
title: 'title',
|
||||||
|
name: 'name',
|
||||||
|
tags: [],
|
||||||
|
applyLoaders: jest.fn(),
|
||||||
|
unboundStoryFn: jest.fn(),
|
||||||
|
playFunction: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const render = new StoryRender(
|
||||||
|
new Channel(),
|
||||||
|
{ getStoryContext: () => ({}) } as any,
|
||||||
|
jest.fn() as any,
|
||||||
|
{} as any,
|
||||||
|
entry.id,
|
||||||
|
'story',
|
||||||
|
{ autoplay: false },
|
||||||
|
story as any
|
||||||
|
);
|
||||||
|
|
||||||
|
await render.renderToElement({} as any);
|
||||||
|
expect(story.playFunction).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -47,6 +47,10 @@ export type RenderContextCallbacks<TFramework extends AnyFramework> = Pick<
|
|||||||
'showMain' | 'showError' | 'showException'
|
'showMain' | 'showError' | 'showException'
|
||||||
>;
|
>;
|
||||||
|
|
||||||
|
export type StoryRenderOptions = {
|
||||||
|
autoplay?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
export class StoryRender<TFramework extends AnyFramework> implements Render<TFramework> {
|
export class StoryRender<TFramework extends AnyFramework> implements Render<TFramework> {
|
||||||
public type: RenderType = 'story';
|
public type: RenderType = 'story';
|
||||||
|
|
||||||
@ -73,6 +77,7 @@ export class StoryRender<TFramework extends AnyFramework> implements Render<TFra
|
|||||||
private callbacks: RenderContextCallbacks<TFramework>,
|
private callbacks: RenderContextCallbacks<TFramework>,
|
||||||
public id: StoryId,
|
public id: StoryId,
|
||||||
public viewMode: ViewMode,
|
public viewMode: ViewMode,
|
||||||
|
public renderOptions: StoryRenderOptions = { autoplay: true },
|
||||||
story?: Store_Story<TFramework>
|
story?: Store_Story<TFramework>
|
||||||
) {
|
) {
|
||||||
this.abortController = new AbortController();
|
this.abortController = new AbortController();
|
||||||
@ -220,7 +225,7 @@ export class StoryRender<TFramework extends AnyFramework> implements Render<TFra
|
|||||||
if (abortSignal.aborted) return;
|
if (abortSignal.aborted) return;
|
||||||
|
|
||||||
// The phase should be 'rendering' but it might be set to 'aborted' by another render cycle
|
// The phase should be 'rendering' but it might be set to 'aborted' by another render cycle
|
||||||
if (forceRemount && playFunction && this.phase !== 'errored') {
|
if (this.renderOptions.autoplay && forceRemount && playFunction && this.phase !== 'errored') {
|
||||||
this.disableKeyListeners = true;
|
this.disableKeyListeners = true;
|
||||||
try {
|
try {
|
||||||
await this.runPhase(abortSignal, 'playing', async () => {
|
await this.runPhase(abortSignal, 'playing', async () => {
|
||||||
|
@ -31,13 +31,9 @@ export const ChangeArgs = {
|
|||||||
await expect(button).toHaveFocus();
|
await expect(button).toHaveFocus();
|
||||||
|
|
||||||
// Vue3: https://github.com/storybookjs/storybook/issues/13913
|
// Vue3: https://github.com/storybookjs/storybook/issues/13913
|
||||||
// Svelte: https://github.com/storybookjs/storybook/issues/19205
|
|
||||||
// Web-components: https://github.com/storybookjs/storybook/issues/19415
|
// Web-components: https://github.com/storybookjs/storybook/issues/19415
|
||||||
// Preact: https://github.com/storybookjs/storybook/issues/19504
|
// Preact: https://github.com/storybookjs/storybook/issues/19504
|
||||||
if (
|
if (['vue3', 'web-components', 'html', 'preact'].includes(globalThis.storybookRenderer)) return;
|
||||||
['vue3', 'svelte', 'web-components', 'html', 'preact'].includes(globalThis.storybookRenderer)
|
|
||||||
)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// When we change the args to the button, it should not rerender
|
// When we change the args to the button, it should not rerender
|
||||||
await channel.emit('updateStoryArgs', { storyId: id, updatedArgs: { label: 'New Text' } });
|
await channel.emit('updateStoryArgs', { storyId: id, updatedArgs: { label: 'New Text' } });
|
||||||
|
@ -1,5 +1,3 @@
|
|||||||
import global from 'global';
|
|
||||||
|
|
||||||
import type { Store_RenderContext, ArgsStoryFn } from '@storybook/types';
|
import type { Store_RenderContext, ArgsStoryFn } from '@storybook/types';
|
||||||
import type { SvelteComponentTyped } from 'svelte';
|
import type { SvelteComponentTyped } from 'svelte';
|
||||||
// eslint-disable-next-line import/no-extraneous-dependencies
|
// eslint-disable-next-line import/no-extraneous-dependencies
|
||||||
@ -7,40 +5,66 @@ import PreviewRender from '@storybook/svelte/templates/PreviewRender.svelte';
|
|||||||
|
|
||||||
import type { SvelteFramework } from './types';
|
import type { SvelteFramework } from './types';
|
||||||
|
|
||||||
const { document } = global;
|
const componentsByDomElement = new Map<Element, SvelteComponentTyped>();
|
||||||
|
|
||||||
let previousComponent: SvelteComponentTyped | null = null;
|
function teardown(domElement: Element) {
|
||||||
|
if (!componentsByDomElement.has(domElement)) {
|
||||||
function cleanUpPreviousStory() {
|
|
||||||
if (!previousComponent) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
previousComponent.$destroy();
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- we know it exists because we just checked
|
||||||
previousComponent = null;
|
componentsByDomElement.get(domElement)!.$destroy();
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-param-reassign -- this is on purpose
|
||||||
|
domElement.innerHTML = '';
|
||||||
|
componentsByDomElement.delete(domElement);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function renderToDOM(
|
export function renderToDOM(
|
||||||
{ storyFn, kind, name, showMain, showError, storyContext }: Store_RenderContext<SvelteFramework>,
|
{
|
||||||
|
storyFn,
|
||||||
|
kind,
|
||||||
|
name,
|
||||||
|
showMain,
|
||||||
|
showError,
|
||||||
|
storyContext,
|
||||||
|
forceRemount,
|
||||||
|
}: Store_RenderContext<SvelteFramework>,
|
||||||
domElement: Element
|
domElement: Element
|
||||||
) {
|
) {
|
||||||
cleanUpPreviousStory();
|
const existingComponent = componentsByDomElement.get(domElement);
|
||||||
|
|
||||||
const target = domElement || document.getElementById('storybook-root');
|
if (forceRemount) {
|
||||||
|
teardown(domElement);
|
||||||
|
}
|
||||||
|
|
||||||
target.innerHTML = '';
|
if (!existingComponent || forceRemount) {
|
||||||
|
const createdComponent = new PreviewRender({
|
||||||
previousComponent = new PreviewRender({
|
target: domElement,
|
||||||
target,
|
props: {
|
||||||
props: {
|
storyFn,
|
||||||
|
storyContext,
|
||||||
|
name,
|
||||||
|
kind,
|
||||||
|
showError,
|
||||||
|
},
|
||||||
|
}) as SvelteComponentTyped;
|
||||||
|
componentsByDomElement.set(domElement, createdComponent);
|
||||||
|
} else {
|
||||||
|
existingComponent.$set({
|
||||||
storyFn,
|
storyFn,
|
||||||
storyContext,
|
storyContext,
|
||||||
name,
|
name,
|
||||||
kind,
|
kind,
|
||||||
showError,
|
showError,
|
||||||
},
|
});
|
||||||
});
|
}
|
||||||
|
|
||||||
showMain();
|
showMain();
|
||||||
|
|
||||||
|
// teardown the component when the story changes
|
||||||
|
return () => {
|
||||||
|
teardown(domElement);
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export const render: ArgsStoryFn<SvelteFramework> = (args, context) => {
|
export const render: ArgsStoryFn<SvelteFramework> = (args, context) => {
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
/**
|
/**
|
||||||
* What background color to use
|
* What background color to use
|
||||||
*/
|
*/
|
||||||
export let backgroundColor;
|
export let backgroundColor = undefined;
|
||||||
/**
|
/**
|
||||||
* How large should the button be?
|
* How large should the button be?
|
||||||
*/
|
*/
|
||||||
@ -19,9 +19,9 @@
|
|||||||
*/
|
*/
|
||||||
export let label = '';
|
export let label = '';
|
||||||
|
|
||||||
let mode = primary ? 'storybook-button--primary' : 'storybook-button--secondary';
|
$: mode = primary ? 'storybook-button--primary' : 'storybook-button--secondary';
|
||||||
|
|
||||||
let style = backgroundColor ? `background-color: ${backgroundColor}` : '';
|
$: style = backgroundColor ? `background-color: ${backgroundColor}` : '';
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
@ -21,9 +21,9 @@
|
|||||||
*/
|
*/
|
||||||
export let label = '';
|
export let label = '';
|
||||||
|
|
||||||
let mode = primary ? 'storybook-button--primary' : 'storybook-button--secondary';
|
$: mode = primary ? 'storybook-button--primary' : 'storybook-button--secondary';
|
||||||
|
|
||||||
let style = backgroundColor ? `background-color: ${backgroundColor}` : '';
|
$: style = backgroundColor ? `background-color: ${backgroundColor}` : '';
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
export let showError;
|
export let showError;
|
||||||
export let storyContext;
|
export let storyContext;
|
||||||
|
|
||||||
const {
|
let {
|
||||||
/** @type {SvelteComponent} */
|
/** @type {SvelteComponent} */
|
||||||
Component,
|
Component,
|
||||||
/** @type {any} */
|
/** @type {any} */
|
||||||
@ -19,11 +19,16 @@
|
|||||||
WrapperData = {},
|
WrapperData = {},
|
||||||
} = storyFn();
|
} = storyFn();
|
||||||
|
|
||||||
const eventsFromArgTypes = Object.fromEntries(Object.entries(storyContext.argTypes)
|
// reactive, re-render on storyFn change
|
||||||
.filter(([k, v]) => v.action && props[k] != null)
|
$: ({ Component, props = {}, on, Wrapper, WrapperData = {} } = storyFn());
|
||||||
.map(([k, v]) => [v.action, props[k]]));
|
|
||||||
|
|
||||||
const events = {...eventsFromArgTypes, ...on};
|
const eventsFromArgTypes = Object.fromEntries(
|
||||||
|
Object.entries(storyContext.argTypes)
|
||||||
|
.filter(([k, v]) => v.action && props[k] != null)
|
||||||
|
.map(([k, v]) => [v.action, props[k]])
|
||||||
|
);
|
||||||
|
|
||||||
|
const events = { ...eventsFromArgTypes, ...on };
|
||||||
|
|
||||||
if (!Component) {
|
if (!Component) {
|
||||||
showError({
|
showError({
|
||||||
@ -36,9 +41,11 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<SlotDecorator
|
<SlotDecorator
|
||||||
decorator={Wrapper}
|
decorator={Wrapper}
|
||||||
decoratorProps={WrapperData}
|
decoratorProps={WrapperData}
|
||||||
component={Component}
|
component={Component}
|
||||||
props={props}
|
{props}
|
||||||
on={events}/>
|
on={events}
|
||||||
|
/>
|
||||||
|
@ -21,10 +21,11 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if decorator}
|
{#if decorator}
|
||||||
<svelte:component this={decorator} {...decoratorProps} bind:this={decoratorInstance}>
|
<svelte:component this={decorator} {...decoratorProps} bind:this={decoratorInstance}>
|
||||||
<svelte:component this={component} {...props} bind:this={instance}/>
|
<svelte:component this={component} {...props} bind:this={instance} />
|
||||||
</svelte:component>
|
</svelte:component>
|
||||||
{:else}
|
{:else}
|
||||||
<svelte:component this={component} {...props} bind:this={instance}/>
|
<svelte:component this={component} {...props} bind:this={instance} />
|
||||||
{/if}
|
{/if}
|
||||||
|
@ -17,7 +17,7 @@ const allStories = [
|
|||||||
titlePrefix: '@storybook-blocks',
|
titlePrefix: '@storybook-blocks',
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
const blocksOnlyStories = ['../blocks/src/**/*.stories.@(js|jsx|ts|tsx|mdx)'];
|
const blocksOnlyStories = ['../blocks/src/@(blocks|controls)/**/*.@(mdx|stories.@(tsx|ts|jsx|js))'];
|
||||||
|
|
||||||
const config: StorybookConfig = {
|
const config: StorybookConfig = {
|
||||||
stories: isBlocksOnly ? blocksOnlyStories : allStories,
|
stories: isBlocksOnly ? blocksOnlyStories : allStories,
|
||||||
@ -36,6 +36,7 @@ const config: StorybookConfig = {
|
|||||||
viteFinal: (vite) => ({
|
viteFinal: (vite) => ({
|
||||||
...vite,
|
...vite,
|
||||||
plugins: [...(vite.plugins || []), csfPlugin({})],
|
plugins: [...(vite.plugins || []), csfPlugin({})],
|
||||||
|
optimizeDeps: { ...vite.optimizeDeps, force: true },
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -10,7 +10,14 @@ import {
|
|||||||
styled,
|
styled,
|
||||||
useTheme,
|
useTheme,
|
||||||
} from '@storybook/theming';
|
} from '@storybook/theming';
|
||||||
|
import { useArgs } from '@storybook/addons';
|
||||||
import { Symbols } from '@storybook/components';
|
import { Symbols } from '@storybook/components';
|
||||||
|
import type { PreviewWeb } from '@storybook/preview-web';
|
||||||
|
import { DocsContext } from '@storybook/preview-web';
|
||||||
|
import type { ReactFramework } from '@storybook/react';
|
||||||
|
import type { Channel } from '@storybook/channels';
|
||||||
|
|
||||||
|
import { DocsContainer } from '../blocks/src/blocks/DocsContainer';
|
||||||
|
|
||||||
const { document } = global;
|
const { document } = global;
|
||||||
|
|
||||||
@ -86,7 +93,49 @@ const ThemedSetRoot = () => {
|
|||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-underscore-dangle
|
||||||
|
const preview = (window as any).__STORYBOOK_PREVIEW__ as PreviewWeb<ReactFramework>;
|
||||||
|
const channel = (window as any).__STORYBOOK_ADDONS_CHANNEL__ as Channel;
|
||||||
|
export const loaders = [
|
||||||
|
async () => ({ globalValue: 1 }),
|
||||||
|
|
||||||
|
async ({ parameters: { relativeCsfPaths } }) => {
|
||||||
|
if (!relativeCsfPaths) return {};
|
||||||
|
|
||||||
|
const csfFiles = await Promise.all(
|
||||||
|
(relativeCsfPaths as string[]).map(async (relativePath) => {
|
||||||
|
const webpackPath = `./ui/blocks/src/${relativePath.replace(/^..\//, '')}.tsx`;
|
||||||
|
const entry = preview.storyStore.storyIndex!.importPathToEntry(webpackPath);
|
||||||
|
|
||||||
|
if (!entry) {
|
||||||
|
throw new Error(`Couldn't find story file at ${webpackPath} (passed as ${relativePath})`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return preview.storyStore.loadCSFFileByStoryId(entry.id);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
docsContext: new DocsContext(
|
||||||
|
channel,
|
||||||
|
preview.storyStore,
|
||||||
|
preview.renderStoryToElement.bind(preview),
|
||||||
|
csfFiles,
|
||||||
|
false
|
||||||
|
),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
export const decorators = [
|
export const decorators = [
|
||||||
|
(Story, { loaded: { docsContext } }) =>
|
||||||
|
docsContext ? (
|
||||||
|
<DocsContainer context={docsContext}>
|
||||||
|
<Story />
|
||||||
|
</DocsContainer>
|
||||||
|
) : (
|
||||||
|
<Story />
|
||||||
|
),
|
||||||
(StoryFn, { globals, parameters, playFunction }) => {
|
(StoryFn, { globals, parameters, playFunction }) => {
|
||||||
const defaultTheme = isChromatic() && !playFunction ? 'stacked' : 'light';
|
const defaultTheme = isChromatic() && !playFunction ? 'stacked' : 'light';
|
||||||
const theme = globals.theme || parameters.theme || defaultTheme;
|
const theme = globals.theme || parameters.theme || defaultTheme;
|
||||||
@ -150,6 +199,37 @@ export const decorators = [
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* This decorator shows the current state of the arg named in the
|
||||||
|
* parameters.withRawArg property, by updating the arg in the onChange function
|
||||||
|
* this also means that the arg will sync with the control panel
|
||||||
|
*
|
||||||
|
* If parameters.withRawArg is not set, this decorator will do nothing
|
||||||
|
*/
|
||||||
|
(StoryFn, { parameters, args, hooks }) => {
|
||||||
|
const [, updateArgs] = useArgs();
|
||||||
|
if (!parameters.withRawArg) {
|
||||||
|
return <StoryFn />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<StoryFn
|
||||||
|
args={{
|
||||||
|
...args,
|
||||||
|
onChange: (newValue) => {
|
||||||
|
updateArgs({ [parameters.withRawArg]: newValue });
|
||||||
|
args.onChange?.(newValue);
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<div style={{ marginTop: '1rem' }}>
|
||||||
|
Current <code>{parameters.withRawArg}</code>:{' '}
|
||||||
|
<pre>{JSON.stringify(args[parameters.withRawArg], null, 2) || 'undefined'}</pre>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export const parameters = {
|
export const parameters = {
|
||||||
@ -242,7 +322,5 @@ export const globalTypes = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export const loaders = [async () => ({ globalValue: 1 })];
|
|
||||||
|
|
||||||
export const argTypes = { color: { control: 'color' } };
|
export const argTypes = { color: { control: 'color' } };
|
||||||
export const args = { color: 'red' };
|
export const args = { color: 'red' };
|
15
code/ui/blocks/src/blocks/Anchor.stories.tsx
Normal file
15
code/ui/blocks/src/blocks/Anchor.stories.tsx
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { Anchor } from './Anchor';
|
||||||
|
|
||||||
|
const meta = {
|
||||||
|
component: Anchor,
|
||||||
|
} as Meta<typeof Anchor>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
export const Default: StoryObj<typeof Anchor> = {
|
||||||
|
args: {
|
||||||
|
children: 'This is an anchor for storyId: "default"',
|
||||||
|
storyId: 'default',
|
||||||
|
},
|
||||||
|
};
|
20
code/ui/blocks/src/blocks/Story.stories.tsx
Normal file
20
code/ui/blocks/src/blocks/Story.stories.tsx
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
|
||||||
|
import { Story as StoryComponent } from './Story';
|
||||||
|
import * as BooleanStories from '../controls/Boolean.stories';
|
||||||
|
|
||||||
|
const meta: Meta<typeof StoryComponent> = {
|
||||||
|
component: StoryComponent,
|
||||||
|
parameters: {
|
||||||
|
relativeCsfPaths: ['../controls/Boolean.stories'],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
type Story = StoryObj<typeof meta>;
|
||||||
|
|
||||||
|
export const BasicOf: Story = {
|
||||||
|
args: {
|
||||||
|
of: BooleanStories.Undefined,
|
||||||
|
},
|
||||||
|
};
|
@ -90,11 +90,12 @@ const Story: FC<StoryProps> = (props) => {
|
|||||||
let cleanup: () => void;
|
let cleanup: () => void;
|
||||||
if (story && storyRef.current) {
|
if (story && storyRef.current) {
|
||||||
const element = storyRef.current as HTMLElement;
|
const element = storyRef.current as HTMLElement;
|
||||||
cleanup = context.renderStoryToElement(story, element);
|
const { autoplay } = story.parameters.docs || {};
|
||||||
|
cleanup = context.renderStoryToElement(story, element, { autoplay });
|
||||||
setShowLoader(false);
|
setShowLoader(false);
|
||||||
}
|
}
|
||||||
return () => cleanup && cleanup();
|
return () => cleanup && cleanup();
|
||||||
}, [story]);
|
}, [context, story]);
|
||||||
|
|
||||||
if (!story) {
|
if (!story) {
|
||||||
return <StorySkeleton />;
|
return <StorySkeleton />;
|
||||||
|
@ -1,27 +1,28 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { BooleanControl } from './Boolean';
|
import { BooleanControl } from './Boolean';
|
||||||
|
|
||||||
export default {
|
const meta = {
|
||||||
title: 'Controls/Boolean',
|
|
||||||
component: BooleanControl,
|
component: BooleanControl,
|
||||||
tags: ['docsPage'],
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value'] } },
|
||||||
|
args: { name: 'boolean' },
|
||||||
|
} as Meta<typeof BooleanControl>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
export const True: StoryObj<typeof BooleanControl> = {
|
||||||
|
args: {
|
||||||
|
value: true,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
export const False: StoryObj<typeof BooleanControl> = {
|
||||||
|
args: {
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const Template = (initialValue?: boolean) => {
|
export const Undefined: StoryObj<typeof BooleanControl> = {
|
||||||
const [value, setValue] = useState(initialValue);
|
args: {
|
||||||
return (
|
value: undefined,
|
||||||
<>
|
},
|
||||||
<BooleanControl name="boolean" value={value} onChange={(newVal) => setValue(newVal)} />
|
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const True = () => Template(true);
|
|
||||||
|
|
||||||
export const False = () => Template(false);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* When no value is set on the control
|
|
||||||
*/
|
|
||||||
export const Undefined = () => Template(undefined);
|
|
||||||
|
@ -1,57 +1,65 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { ColorControl } from './Color';
|
import { ColorControl } from './Color';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: 'Controls/Color',
|
|
||||||
component: ColorControl,
|
component: ColorControl,
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value', 'startOpen'] } },
|
||||||
|
tags: ['docsPage'],
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
control: {
|
||||||
|
type: 'color',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
args: { name: 'color' },
|
||||||
|
} as Meta<typeof ColorControl>;
|
||||||
|
|
||||||
|
export const Basic: StoryObj<typeof ColorControl> = {
|
||||||
|
args: {
|
||||||
|
value: '#ff00ff',
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const Template = (
|
export const Undefined: StoryObj<typeof ColorControl> = {
|
||||||
initialValue?: string,
|
args: {
|
||||||
presetColors?: Array<string | { color: string; title?: string }>
|
value: undefined,
|
||||||
) => {
|
},
|
||||||
const [value, setValue] = useState(initialValue);
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<ColorControl
|
|
||||||
name="Color"
|
|
||||||
value={value}
|
|
||||||
onChange={(newVal) => setValue(newVal)}
|
|
||||||
presetColors={presetColors}
|
|
||||||
startOpen
|
|
||||||
/>
|
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Basic = () => Template('#ff0');
|
export const WithPresetColors: StoryObj<typeof ColorControl> = {
|
||||||
|
args: {
|
||||||
|
value: '#00ffff',
|
||||||
|
presetColors: [
|
||||||
|
{ color: '#ff4785', title: 'Coral' },
|
||||||
|
{ color: '#1EA7FD', title: 'Ocean' },
|
||||||
|
{ color: 'rgb(252, 82, 31)', title: 'Orange' },
|
||||||
|
{ color: 'RGBA(255, 174, 0, 0.5)', title: 'Gold' },
|
||||||
|
{ color: 'hsl(101, 52%, 49%)', title: 'Green' },
|
||||||
|
{ color: 'HSLA(179,65%,53%,0.5)', title: 'Seafoam' },
|
||||||
|
{ color: '#6F2CAC', title: 'Purple' },
|
||||||
|
{ color: '#2A0481', title: 'Ultraviolet' },
|
||||||
|
{ color: 'black' },
|
||||||
|
{ color: '#333', title: 'Darkest' },
|
||||||
|
{ color: '#444', title: 'Darker' },
|
||||||
|
{ color: '#666', title: 'Dark' },
|
||||||
|
{ color: '#999', title: 'Mediumdark' },
|
||||||
|
{ color: '#ddd', title: 'Medium' },
|
||||||
|
{ color: '#EEE', title: 'Mediumlight' },
|
||||||
|
{ color: '#F3F3F3', title: 'Light' },
|
||||||
|
{ color: '#F8F8F8', title: 'Lighter' },
|
||||||
|
{ color: '#FFFFFF', title: 'Lightest' },
|
||||||
|
'#fe4a49',
|
||||||
|
'#FED766',
|
||||||
|
'rgba(0, 159, 183, 1)',
|
||||||
|
'HSLA(240,11%,91%,0.5)',
|
||||||
|
'slategray',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
export const Undefined = () => Template(undefined);
|
export const StartOpen: StoryObj<typeof ColorControl> = {
|
||||||
|
args: {
|
||||||
export const WithPresetColors = () =>
|
startOpen: true,
|
||||||
Template('tan', [
|
},
|
||||||
{ color: '#ff4785', title: 'Coral' },
|
};
|
||||||
{ color: '#1EA7FD', title: 'Ocean' },
|
|
||||||
{ color: 'rgb(252, 82, 31)', title: 'Orange' },
|
|
||||||
{ color: 'RGBA(255, 174, 0, 0.5)', title: 'Gold' },
|
|
||||||
{ color: 'hsl(101, 52%, 49%)', title: 'Green' },
|
|
||||||
{ color: 'HSLA(179,65%,53%,0.5)', title: 'Seafoam' },
|
|
||||||
{ color: '#6F2CAC', title: 'Purple' },
|
|
||||||
{ color: '#2A0481', title: 'Ultraviolet' },
|
|
||||||
{ color: 'black' },
|
|
||||||
{ color: '#333', title: 'Darkest' },
|
|
||||||
{ color: '#444', title: 'Darker' },
|
|
||||||
{ color: '#666', title: 'Dark' },
|
|
||||||
{ color: '#999', title: 'Mediumdark' },
|
|
||||||
{ color: '#ddd', title: 'Medium' },
|
|
||||||
{ color: '#EEE', title: 'Mediumlight' },
|
|
||||||
{ color: '#F3F3F3', title: 'Light' },
|
|
||||||
{ color: '#F8F8F8', title: 'Lighter' },
|
|
||||||
{ color: '#FFFFFF', title: 'Lightest' },
|
|
||||||
'#fe4a49',
|
|
||||||
'#FED766',
|
|
||||||
'rgba(0, 159, 183, 1)',
|
|
||||||
'HSLA(240,11%,91%,0.5)',
|
|
||||||
'slategray',
|
|
||||||
]);
|
|
||||||
|
@ -306,7 +306,7 @@ export const ColorControl: FC<ColorControlProps> = ({
|
|||||||
onFocus,
|
onFocus,
|
||||||
onBlur,
|
onBlur,
|
||||||
presetColors,
|
presetColors,
|
||||||
startOpen,
|
startOpen = false,
|
||||||
}) => {
|
}) => {
|
||||||
const throttledOnChange = useCallback(throttle(onChange, 200), [onChange]);
|
const throttledOnChange = useCallback(throttle(onChange, 200), [onChange]);
|
||||||
const { value, realValue, updateValue, color, colorSpace, cycleColorSpace } = useColorInput(
|
const { value, realValue, updateValue, color, colorSpace, cycleColorSpace } = useColorInput(
|
||||||
|
@ -1,21 +1,22 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { DateControl } from './Date';
|
import { DateControl } from './Date';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: 'Controls/Date',
|
|
||||||
component: DateControl,
|
component: DateControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value'] } },
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
description: 'The date',
|
||||||
|
control: { type: 'date' },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
args: { name: 'date' },
|
||||||
|
} as Meta<typeof DateControl>;
|
||||||
|
|
||||||
|
export const Basic: StoryObj<typeof DateControl> = {
|
||||||
|
args: { value: new Date('2020-10-20T09:30:02') },
|
||||||
};
|
};
|
||||||
|
export const Undefined: StoryObj<typeof DateControl> = {
|
||||||
const Template = (initialValue) => {
|
args: { value: undefined },
|
||||||
const [value, setValue] = useState(initialValue);
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<DateControl name="date" value={value} onChange={(newVal) => setValue(newVal)} />
|
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Basic = () => Template(new Date(2020, 4, 20));
|
|
||||||
|
|
||||||
export const Undefined = () => Template(undefined);
|
|
||||||
|
29
code/ui/blocks/src/controls/Files.stories.tsx
Normal file
29
code/ui/blocks/src/controls/Files.stories.tsx
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { FilesControl } from './Files';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
component: FilesControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value', 'accept'] } },
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
description: 'Selected file',
|
||||||
|
control: { type: 'file' },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
args: { name: 'files' },
|
||||||
|
} as Meta<typeof FilesControl>;
|
||||||
|
|
||||||
|
export const Undefined: StoryObj<typeof FilesControl> = {
|
||||||
|
args: { value: undefined },
|
||||||
|
};
|
||||||
|
// for security reasons a file input field cannot have an initial value, so it doesn't make sense to have stories for it
|
||||||
|
|
||||||
|
export const AcceptAnything: StoryObj<typeof FilesControl> = {
|
||||||
|
args: { accept: '*/*' },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const AcceptPDFs: StoryObj<typeof FilesControl> = {
|
||||||
|
name: 'Accept PDFs',
|
||||||
|
args: { accept: '.pdf' },
|
||||||
|
};
|
@ -7,6 +7,17 @@ import type { ControlProps } from './types';
|
|||||||
import { getControlId } from './helpers';
|
import { getControlId } from './helpers';
|
||||||
|
|
||||||
export interface FilesControlProps extends ControlProps<string[]> {
|
export interface FilesControlProps extends ControlProps<string[]> {
|
||||||
|
/**
|
||||||
|
* The accept attribute value is a string that defines the file types the file input should accept. This string is a comma-separated list of unique file type specifiers.
|
||||||
|
* @example
|
||||||
|
* *\/*
|
||||||
|
* @example
|
||||||
|
* .webm,video/webm
|
||||||
|
* @example
|
||||||
|
* .doc,.docx,application/msword
|
||||||
|
* @defaultValue `image/*`
|
||||||
|
* @see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept
|
||||||
|
*/
|
||||||
accept?: string;
|
accept?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,23 +1,41 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { NumberControl } from './Number';
|
import { NumberControl } from './Number';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: 'Controls/Number',
|
|
||||||
component: NumberControl,
|
component: NumberControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value', 'min', 'max', 'step'] } },
|
||||||
|
args: { name: 'number' },
|
||||||
|
} as Meta<typeof NumberControl>;
|
||||||
|
|
||||||
|
export const Undefined: StoryObj<typeof NumberControl> = {
|
||||||
|
args: { value: undefined },
|
||||||
|
};
|
||||||
|
// for security reasons a file input field cannot have an initial value, so it doesn't make sense to have stories for it
|
||||||
|
|
||||||
|
export const Ten: StoryObj<typeof NumberControl> = {
|
||||||
|
args: { value: 10 },
|
||||||
|
};
|
||||||
|
export const Zero: StoryObj<typeof NumberControl> = {
|
||||||
|
args: { value: 0 },
|
||||||
};
|
};
|
||||||
|
|
||||||
const Template = (initialValue) => {
|
export const WithMin: StoryObj<typeof NumberControl> = {
|
||||||
const [value, setValue] = useState(initialValue);
|
args: { min: 1, value: 3 },
|
||||||
return (
|
};
|
||||||
<>
|
export const WithMax: StoryObj<typeof NumberControl> = {
|
||||||
<NumberControl name="number" value={value} onChange={(newVal) => setValue(newVal)} />
|
args: { max: 7, value: 3 },
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
};
|
||||||
</>
|
export const WithMinAndMax: StoryObj<typeof NumberControl> = {
|
||||||
);
|
args: { min: -2, max: 5, value: 3 },
|
||||||
|
};
|
||||||
|
export const LessThanMin: StoryObj<typeof NumberControl> = {
|
||||||
|
args: { min: 3, value: 1 },
|
||||||
|
};
|
||||||
|
export const MoreThanMax: StoryObj<typeof NumberControl> = {
|
||||||
|
args: { max: 3, value: 6 },
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Basic = () => Template(10);
|
export const WithStep: StoryObj<typeof NumberControl> = {
|
||||||
|
args: { step: 5, value: 3 },
|
||||||
export const Zero = () => Template(0);
|
};
|
||||||
|
|
||||||
export const Undefined = () => Template(undefined);
|
|
||||||
|
@ -1,40 +1,55 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { ObjectControl } from './Object';
|
import { ObjectControl } from './Object';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: 'Controls/Object',
|
|
||||||
component: ObjectControl,
|
component: ObjectControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value'] } },
|
||||||
|
args: { name: 'object' },
|
||||||
|
} as Meta<typeof ObjectControl>;
|
||||||
|
|
||||||
|
export const Object: StoryObj<typeof ObjectControl> = {
|
||||||
|
args: {
|
||||||
|
value: {
|
||||||
|
name: 'Michael',
|
||||||
|
someDate: new Date('2022-10-30T12:31:11'),
|
||||||
|
nested: { someBool: true, someNumber: 22 },
|
||||||
|
},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const Template = (initialValue: any) => {
|
export const Array: StoryObj<typeof ObjectControl> = {
|
||||||
const [value, setValue] = useState(initialValue);
|
args: {
|
||||||
return (
|
value: [
|
||||||
<>
|
'someString',
|
||||||
<ObjectControl name="object" value={value} onChange={(newVal) => setValue(newVal)} />
|
22,
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
true,
|
||||||
</>
|
new Date('2022-10-30T12:31:11'),
|
||||||
);
|
{ someBool: true, someNumber: 22 },
|
||||||
|
],
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Basic = () => Template({ name: 'Michael', nested: { something: true } });
|
export const EmptyObject: StoryObj<typeof ObjectControl> = {
|
||||||
|
args: {
|
||||||
export const Empty = () => Template({});
|
value: {},
|
||||||
|
},
|
||||||
export const Null = () => Template(null);
|
};
|
||||||
|
|
||||||
export const Undefined = () => Template(undefined);
|
export const EmptyArray: StoryObj<typeof ObjectControl> = {
|
||||||
|
args: {
|
||||||
export const ValidatedAsArray = () => {
|
value: {},
|
||||||
const [value, setValue] = useState([]);
|
},
|
||||||
return (
|
};
|
||||||
<>
|
|
||||||
<ObjectControl
|
export const Null: StoryObj<typeof ObjectControl> = {
|
||||||
name="object"
|
args: {
|
||||||
argType={{ type: { name: 'array' } }}
|
value: null,
|
||||||
value={value}
|
},
|
||||||
onChange={(newVal) => setValue(newVal)}
|
};
|
||||||
/>
|
|
||||||
<p>{value && JSON.stringify(value)}</p>
|
export const Undefined: StoryObj<typeof ObjectControl> = {
|
||||||
</>
|
args: {
|
||||||
);
|
value: undefined,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
@ -1,43 +1,81 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { RangeControl } from './Range';
|
import { RangeControl } from './Range';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: 'Controls/Range',
|
|
||||||
component: RangeControl,
|
component: RangeControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value', 'min', 'max', 'step'] } },
|
||||||
|
args: { name: 'range' },
|
||||||
|
} as Meta<typeof RangeControl>;
|
||||||
|
|
||||||
|
export const Undefined: StoryObj<typeof RangeControl> = {
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const Template = ({
|
export const Zero: StoryObj<typeof RangeControl> = {
|
||||||
initialValue,
|
args: {
|
||||||
step,
|
value: 0,
|
||||||
max,
|
},
|
||||||
}: {
|
};
|
||||||
initialValue?: number;
|
export const WithMin: StoryObj<typeof RangeControl> = {
|
||||||
step?: number;
|
args: {
|
||||||
max?: number;
|
min: 5,
|
||||||
}) => {
|
value: 20,
|
||||||
const [value, setValue] = useState(initialValue);
|
},
|
||||||
return (
|
};
|
||||||
<>
|
export const WithMax: StoryObj<typeof RangeControl> = {
|
||||||
<RangeControl
|
args: {
|
||||||
name="range"
|
max: 50,
|
||||||
value={value}
|
value: 20,
|
||||||
onChange={(newVal) => setValue(newVal)}
|
},
|
||||||
min={0}
|
};
|
||||||
max={max}
|
export const WithBigMax: StoryObj<typeof RangeControl> = {
|
||||||
step={step}
|
args: {
|
||||||
/>
|
max: 10000000000,
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
value: 20,
|
||||||
</>
|
},
|
||||||
);
|
};
|
||||||
|
export const WithMinAndMax: StoryObj<typeof RangeControl> = {
|
||||||
|
args: {
|
||||||
|
min: 10,
|
||||||
|
max: 50,
|
||||||
|
value: 20,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Basic = () => Template({ initialValue: 10, max: 20, step: 2 });
|
export const LessThanMin: StoryObj<typeof RangeControl> = {
|
||||||
|
args: {
|
||||||
|
min: 10,
|
||||||
|
value: 5,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
export const Zero = () => Template({ initialValue: 0, max: 20, step: 2 });
|
export const MoreThanMax: StoryObj<typeof RangeControl> = {
|
||||||
|
args: {
|
||||||
|
max: 20,
|
||||||
|
value: 50,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
export const Decimals = () =>
|
export const WithSteps: StoryObj<typeof RangeControl> = {
|
||||||
Template({ step: 0.000000000002, initialValue: 1989.123123123123, max: 2000 });
|
args: {
|
||||||
|
step: 5,
|
||||||
|
value: 50,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
export const BigMaxValue = () => Template({ step: 1000, initialValue: 15, max: 10000000000 });
|
export const Decimals: StoryObj<typeof RangeControl> = {
|
||||||
|
args: {
|
||||||
export const Undefined = () => Template({});
|
step: 0.000000000002,
|
||||||
|
value: 989.123123123123,
|
||||||
|
max: 2000,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
export const WithInfiniteMax: StoryObj<typeof RangeControl> = {
|
||||||
|
args: {
|
||||||
|
max: Infinity,
|
||||||
|
value: 50,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
@ -1,23 +1,34 @@
|
|||||||
import React, { useState } from 'react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { TextControl } from './Text';
|
import { TextControl } from './Text';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: 'Controls/Text',
|
|
||||||
component: TextControl,
|
component: TextControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: { withRawArg: 'value', controls: { include: ['value', 'maxLength'] } },
|
||||||
|
args: { name: 'text' },
|
||||||
|
} as Meta<typeof TextControl>;
|
||||||
|
|
||||||
|
export const Basic: StoryObj<typeof TextControl> = {
|
||||||
|
args: {
|
||||||
|
value: 'Storybook says hi. 👋',
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const Template = (initialValue?: string) => {
|
export const Empty: StoryObj<typeof TextControl> = {
|
||||||
const [value, setValue] = useState(initialValue);
|
args: {
|
||||||
return (
|
value: '',
|
||||||
<>
|
},
|
||||||
<TextControl name="Text" value={value} onChange={(newVal) => setValue(newVal)} />
|
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Basic = () => Template('Hello text');
|
export const Undefined: StoryObj<typeof TextControl> = {
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
export const Empty = () => Template('');
|
export const WithMaxLength: StoryObj<typeof TextControl> = {
|
||||||
|
args: {
|
||||||
export const Undefined = () => Template(undefined);
|
value: "You can't finish this sente",
|
||||||
|
maxLength: 28,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
99
code/ui/blocks/src/controls/options/CheckOptions.stories.tsx
Normal file
99
code/ui/blocks/src/controls/options/CheckOptions.stories.tsx
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { OptionsControl } from './Options';
|
||||||
|
|
||||||
|
const arrayOptions = ['Bat', 'Cat', 'Rat'];
|
||||||
|
const labels = {
|
||||||
|
Bat: 'Batwoman',
|
||||||
|
Cat: 'Catwoman',
|
||||||
|
Rat: 'Ratwoman',
|
||||||
|
};
|
||||||
|
const objectOptions = {
|
||||||
|
A: { id: 'Aardvark' },
|
||||||
|
B: { id: 'Bat' },
|
||||||
|
C: { id: 'Cat' },
|
||||||
|
};
|
||||||
|
|
||||||
|
const meta = {
|
||||||
|
title: 'Controls/Options/Check',
|
||||||
|
component: OptionsControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: {
|
||||||
|
withRawArg: 'value',
|
||||||
|
controls: { include: ['argType', 'type', 'value', 'labels'] },
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
name: 'check',
|
||||||
|
type: 'check',
|
||||||
|
argType: { options: arrayOptions },
|
||||||
|
},
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
control: { type: 'check' },
|
||||||
|
options: arrayOptions,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
} as Meta<typeof OptionsControl>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
export const Array: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: [arrayOptions[0]],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayInline: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'inline-check',
|
||||||
|
value: [arrayOptions[1], arrayOptions[2]],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayLabels: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: [arrayOptions[0]],
|
||||||
|
labels,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayInlineLabels: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'inline-check',
|
||||||
|
value: [arrayOptions[1], arrayOptions[2]],
|
||||||
|
labels,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Object: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object',
|
||||||
|
args: {
|
||||||
|
value: [objectOptions.B],
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectInline: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Inline',
|
||||||
|
args: {
|
||||||
|
type: 'inline-check',
|
||||||
|
value: [objectOptions.A, objectOptions.C],
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Undefined',
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
@ -1,66 +0,0 @@
|
|||||||
import React, { useState } from 'react';
|
|
||||||
import { OptionsControl } from './Options';
|
|
||||||
|
|
||||||
export default {
|
|
||||||
title: 'Controls/Options',
|
|
||||||
component: OptionsControl,
|
|
||||||
};
|
|
||||||
|
|
||||||
const arrayOptions = ['Bat', 'Cat', 'Rat'];
|
|
||||||
const objectOptions = {
|
|
||||||
A: { id: 'Aardvark' },
|
|
||||||
B: { id: 'Bat' },
|
|
||||||
C: { id: 'Cat' },
|
|
||||||
};
|
|
||||||
|
|
||||||
const rawOptionsHelper = (options, type, isMulti, initial) => {
|
|
||||||
const [value, setValue] = useState(isMulti ? [initial] : initial);
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<OptionsControl
|
|
||||||
name="options"
|
|
||||||
labels={{}}
|
|
||||||
argType={{ options }}
|
|
||||||
value={value}
|
|
||||||
type={type}
|
|
||||||
onChange={(newVal) => setValue(newVal)}
|
|
||||||
/>
|
|
||||||
<pre>{JSON.stringify(value) || 'undefined'}</pre>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const optionsHelper = (options, type, isMulti) =>
|
|
||||||
rawOptionsHelper(options, type, isMulti, Array.isArray(options) ? options[1] : options.B);
|
|
||||||
|
|
||||||
// Check
|
|
||||||
export const ArrayCheck = () => optionsHelper(arrayOptions, 'check', true);
|
|
||||||
export const ArrayInlineCheck = () => optionsHelper(arrayOptions, 'inline-check', true);
|
|
||||||
export const ObjectCheck = () => optionsHelper(objectOptions, 'check', true);
|
|
||||||
export const ObjectInlineCheck = () => optionsHelper(objectOptions, 'inline-check', true);
|
|
||||||
export const ArrayCheckUndefined = () => rawOptionsHelper(arrayOptions, 'check', false, undefined);
|
|
||||||
export const ObjectCheckUndefined = () =>
|
|
||||||
rawOptionsHelper(objectOptions, 'check', false, undefined);
|
|
||||||
|
|
||||||
// Radio
|
|
||||||
export const ArrayRadio = () => optionsHelper(arrayOptions, 'radio', false);
|
|
||||||
export const ArrayInlineRadio = () => optionsHelper(arrayOptions, 'inline-radio', false);
|
|
||||||
export const ObjectRadio = () => optionsHelper(objectOptions, 'radio', false);
|
|
||||||
export const ObjectInlineRadio = () => optionsHelper(objectOptions, 'inline-radio', false);
|
|
||||||
export const ArrayRadioUndefined = () => rawOptionsHelper(arrayOptions, 'radio', false, undefined);
|
|
||||||
export const ObjectRadioUndefined = () =>
|
|
||||||
rawOptionsHelper(objectOptions, 'radio', false, undefined);
|
|
||||||
|
|
||||||
// Select
|
|
||||||
export const ArraySelect = () => optionsHelper(arrayOptions, 'select', false);
|
|
||||||
export const ArrayMultiSelect = () => optionsHelper(arrayOptions, 'multi-select', true);
|
|
||||||
export const ObjectSelect = () => optionsHelper(objectOptions, 'select', false);
|
|
||||||
export const ObjectMultiSelect = () => optionsHelper(objectOptions, 'multi-select', true);
|
|
||||||
export const ArraySelectUndefined = () =>
|
|
||||||
rawOptionsHelper(arrayOptions, 'select', false, undefined);
|
|
||||||
export const ObjectSelectUndefined = () =>
|
|
||||||
rawOptionsHelper(objectOptions, 'select', false, undefined);
|
|
||||||
export const ArrayMultiSelectUndefined = () =>
|
|
||||||
rawOptionsHelper(arrayOptions, 'multi-select', false, undefined);
|
|
||||||
export const ObjectMultiSelectUndefined = () =>
|
|
||||||
rawOptionsHelper(objectOptions, 'multi-select', false, undefined);
|
|
99
code/ui/blocks/src/controls/options/RadioOptions.stories.tsx
Normal file
99
code/ui/blocks/src/controls/options/RadioOptions.stories.tsx
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { OptionsControl } from './Options';
|
||||||
|
|
||||||
|
const arrayOptions = ['Bat', 'Cat', 'Rat'];
|
||||||
|
const labels = {
|
||||||
|
Bat: 'Batwoman',
|
||||||
|
Cat: 'Catwoman',
|
||||||
|
Rat: 'Ratwoman',
|
||||||
|
};
|
||||||
|
const objectOptions = {
|
||||||
|
A: { id: 'Aardvark' },
|
||||||
|
B: { id: 'Bat' },
|
||||||
|
C: { id: 'Cat' },
|
||||||
|
};
|
||||||
|
|
||||||
|
const meta = {
|
||||||
|
title: 'Controls/Options/Radio',
|
||||||
|
component: OptionsControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: {
|
||||||
|
withRawArg: 'value',
|
||||||
|
controls: { include: ['argType', 'type', 'value', 'labels'] },
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
name: 'radio',
|
||||||
|
type: 'radio',
|
||||||
|
argType: { options: arrayOptions },
|
||||||
|
},
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
control: { type: 'radio' },
|
||||||
|
options: arrayOptions,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
} as Meta<typeof OptionsControl>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
export const Array: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: arrayOptions[0],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayInline: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'inline-radio',
|
||||||
|
value: arrayOptions[1],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayLabels: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: arrayOptions[0],
|
||||||
|
labels,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayInlineLabels: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'inline-radio',
|
||||||
|
value: arrayOptions[1],
|
||||||
|
labels,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Object: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object',
|
||||||
|
args: {
|
||||||
|
value: objectOptions.B,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectInline: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Inline',
|
||||||
|
args: {
|
||||||
|
type: 'inline-radio',
|
||||||
|
value: objectOptions.A,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Undefined',
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
127
code/ui/blocks/src/controls/options/SelectOptions.stories.tsx
Normal file
127
code/ui/blocks/src/controls/options/SelectOptions.stories.tsx
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { OptionsControl } from './Options';
|
||||||
|
|
||||||
|
const arrayOptions = ['Bat', 'Cat', 'Rat'];
|
||||||
|
const objectOptions = {
|
||||||
|
A: { id: 'Aardvark' },
|
||||||
|
B: { id: 'Bat' },
|
||||||
|
C: { id: 'Cat' },
|
||||||
|
};
|
||||||
|
const labels = {
|
||||||
|
Bat: 'Batwoman',
|
||||||
|
Cat: 'Catwoman',
|
||||||
|
Rat: 'Ratwoman',
|
||||||
|
};
|
||||||
|
const argTypeMultiSelect = {
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
control: { type: 'multi-select' },
|
||||||
|
options: arrayOptions,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const meta = {
|
||||||
|
title: 'Controls/Options/Select',
|
||||||
|
component: OptionsControl,
|
||||||
|
tags: ['docsPage'],
|
||||||
|
parameters: {
|
||||||
|
withRawArg: 'value',
|
||||||
|
controls: { include: ['argType', 'type', 'value', 'labels'] },
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
name: 'select',
|
||||||
|
type: 'select',
|
||||||
|
argType: { options: arrayOptions },
|
||||||
|
},
|
||||||
|
argTypes: {
|
||||||
|
value: {
|
||||||
|
control: { type: 'select' },
|
||||||
|
options: arrayOptions,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
} as Meta<typeof OptionsControl>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
export const Array: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: arrayOptions[0],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayMulti: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'multi-select',
|
||||||
|
value: [arrayOptions[1], arrayOptions[2]],
|
||||||
|
},
|
||||||
|
...argTypeMultiSelect,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayMultiUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'multi-select',
|
||||||
|
value: undefined,
|
||||||
|
},
|
||||||
|
...argTypeMultiSelect,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayLabels: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
value: arrayOptions[0],
|
||||||
|
labels,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ArrayMultiLabels: StoryObj<typeof OptionsControl> = {
|
||||||
|
args: {
|
||||||
|
type: 'multi-select',
|
||||||
|
value: [arrayOptions[1], arrayOptions[2]],
|
||||||
|
labels,
|
||||||
|
},
|
||||||
|
...argTypeMultiSelect,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Object: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object',
|
||||||
|
args: {
|
||||||
|
value: objectOptions.B,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectMulti: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Multi',
|
||||||
|
args: {
|
||||||
|
type: 'multi-select',
|
||||||
|
value: [objectOptions.A, objectOptions.B],
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Undefined',
|
||||||
|
args: {
|
||||||
|
value: undefined,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ObjectMultiUndefined: StoryObj<typeof OptionsControl> = {
|
||||||
|
name: 'DEPRECATED: Object Multi Undefined',
|
||||||
|
args: {
|
||||||
|
type: 'multi-select',
|
||||||
|
value: undefined,
|
||||||
|
argType: { options: objectOptions },
|
||||||
|
},
|
||||||
|
argTypes: { value: { control: { type: 'object' } } },
|
||||||
|
};
|
@ -19,6 +19,10 @@ export type ColorValue = string;
|
|||||||
export type PresetColor = ColorValue | { color: ColorValue; title?: string };
|
export type PresetColor = ColorValue | { color: ColorValue; title?: string };
|
||||||
export interface ColorConfig {
|
export interface ColorConfig {
|
||||||
presetColors?: PresetColor[];
|
presetColors?: PresetColor[];
|
||||||
|
/**
|
||||||
|
* Whether the color picker should be open by default when rendered.
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
startOpen?: boolean;
|
startOpen?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,8 +124,8 @@ const WithToolTipState: FC<
|
|||||||
WithTooltipPureProps & {
|
WithTooltipPureProps & {
|
||||||
startOpen?: boolean;
|
startOpen?: boolean;
|
||||||
}
|
}
|
||||||
> = ({ startOpen, onVisibilityChange: onChange, ...rest }) => {
|
> = ({ startOpen = false, onVisibilityChange: onChange, ...rest }) => {
|
||||||
const [tooltipShown, setTooltipShown] = useState(startOpen || false);
|
const [tooltipShown, setTooltipShown] = useState(startOpen);
|
||||||
const onVisibilityChange: (visibility: boolean) => void = useCallback(
|
const onVisibilityChange: (visibility: boolean) => void = useCallback(
|
||||||
(visibility) => {
|
(visibility) => {
|
||||||
if (onChange && onChange(visibility) === false) return;
|
if (onChange && onChange(visibility) === false) return;
|
||||||
|
@ -181,11 +181,6 @@
|
|||||||
"root": "lib/core-server",
|
"root": "lib/core-server",
|
||||||
"type": "library"
|
"type": "library"
|
||||||
},
|
},
|
||||||
"@storybook/core-vite": {
|
|
||||||
"implicitDependencies": [],
|
|
||||||
"root": "lib/core-vite",
|
|
||||||
"type": "library"
|
|
||||||
},
|
|
||||||
"@storybook/core-webpack": {
|
"@storybook/core-webpack": {
|
||||||
"implicitDependencies": [],
|
"implicitDependencies": [],
|
||||||
"root": "lib/core-webpack",
|
"root": "lib/core-webpack",
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
/**
|
/**
|
||||||
* What background color to use
|
* What background color to use
|
||||||
*/
|
*/
|
||||||
export let backgroundColor;
|
export let backgroundColor = undefined;
|
||||||
/**
|
/**
|
||||||
* How large should the button be?
|
* How large should the button be?
|
||||||
*/
|
*/
|
||||||
@ -34,4 +34,4 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<button type="button" {style} on:click="{onClick}">{label}</button>
|
<button type="button" {style} on:click="{onClick}">{label}</button>
|
||||||
```
|
```
|
||||||
|
Loading…
x
Reference in New Issue
Block a user