mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 06:01:22 +08:00
feat(angular): add angular builder to start storybook
The builder allows to add a "architect" in angular.json to start storybook config ex : ``` "storybook": { "builder": "@storybook/angular:start-storybook", "options": { "browserTarget": "angular-cli:build", "port": 4400 } ``` cmd : `ng run angular-cli:storybook` With this solution it is possible to have several angular projects using different assets and style
This commit is contained in:
parent
c08e40f095
commit
1ce3a7494f
@ -22,6 +22,7 @@
|
||||
"main": "dist/ts3.9/client/index.js",
|
||||
"module": "dist/ts3.9/client/index.js",
|
||||
"types": "dist/ts3.9/client/index.d.ts",
|
||||
"builders": "dist/ts3.9/builders/builders.json",
|
||||
"typesVersions": {
|
||||
"<3.8": {
|
||||
"*": [
|
||||
|
9
app/angular/src/builders/builders.json
Normal file
9
app/angular/src/builders/builders.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"builders": {
|
||||
"start-storybook": {
|
||||
"implementation": "./start-storybook",
|
||||
"schema": "./start-storybook/schema.json",
|
||||
"description": "Start storybook"
|
||||
}
|
||||
}
|
||||
}
|
54
app/angular/src/builders/start-storybook/index.spec.ts
Normal file
54
app/angular/src/builders/start-storybook/index.spec.ts
Normal file
@ -0,0 +1,54 @@
|
||||
import { Architect } from '@angular-devkit/architect';
|
||||
import { TestingArchitectHost } from '@angular-devkit/architect/testing';
|
||||
import { schema } from '@angular-devkit/core';
|
||||
import * as path from 'path';
|
||||
|
||||
const buildStandaloneMock = jest.fn().mockImplementation((_options: unknown) => Promise.resolve());
|
||||
|
||||
jest.mock('@storybook/angular/standalone', () => buildStandaloneMock);
|
||||
|
||||
describe('Start Storybook Builder', () => {
|
||||
let architect: Architect;
|
||||
let architectHost: TestingArchitectHost;
|
||||
|
||||
beforeEach(async () => {
|
||||
const registry = new schema.CoreSchemaRegistry();
|
||||
registry.addPostTransform(schema.transforms.addUndefinedDefaults);
|
||||
|
||||
architectHost = new TestingArchitectHost();
|
||||
architect = new Architect(architectHost, registry);
|
||||
|
||||
// This will either take a Node package name, or a path to the directory
|
||||
// for the package.json file.
|
||||
await architectHost.addBuilderFromPackage(path.join(__dirname, '../../..'));
|
||||
});
|
||||
|
||||
it('should work', async () => {
|
||||
const run = await architect.scheduleBuilder('@storybook/angular:start-storybook', {
|
||||
browserTarget: 'angular-cli:build-2',
|
||||
port: 4400,
|
||||
});
|
||||
|
||||
const output = await run.result;
|
||||
|
||||
await run.stop();
|
||||
|
||||
expect(output.success).toBeTruthy();
|
||||
expect(buildStandaloneMock).toHaveBeenCalledWith({
|
||||
angularBrowserTarget: 'angular-cli:build-2',
|
||||
browserTarget: 'angular-cli:build-2',
|
||||
ci: false,
|
||||
configDir: '.storybook',
|
||||
docs: false,
|
||||
host: 'localhost',
|
||||
https: false,
|
||||
port: 4400,
|
||||
quiet: false,
|
||||
smokeTest: false,
|
||||
sslCa: undefined,
|
||||
sslCert: undefined,
|
||||
sslKey: undefined,
|
||||
staticDir: [],
|
||||
});
|
||||
});
|
||||
});
|
56
app/angular/src/builders/start-storybook/index.ts
Normal file
56
app/angular/src/builders/start-storybook/index.ts
Normal file
@ -0,0 +1,56 @@
|
||||
import { BuilderContext, BuilderOutput, createBuilder } from '@angular-devkit/architect';
|
||||
import { JsonObject } from '@angular-devkit/core';
|
||||
import { Observable, of } from 'rxjs';
|
||||
import { CLIOptions } from '@storybook/core-common';
|
||||
import { map, switchMap, tap } from 'rxjs/operators';
|
||||
|
||||
// TODO: find a better way 🤷♂️
|
||||
// eslint-disable-next-line import/no-extraneous-dependencies
|
||||
import buildStandalone, { StandaloneOptions } from '@storybook/angular/standalone';
|
||||
|
||||
export type StorybookBuilderOptions = JsonObject & {
|
||||
browserTarget: string;
|
||||
} & Pick<
|
||||
// makes sure the option exists
|
||||
CLIOptions,
|
||||
| 'port'
|
||||
| 'host'
|
||||
| 'staticDir'
|
||||
| 'configDir'
|
||||
| 'https'
|
||||
| 'sslCa'
|
||||
| 'sslCert'
|
||||
| 'sslKey'
|
||||
| 'smokeTest'
|
||||
| 'ci'
|
||||
| 'quiet'
|
||||
| 'docs'
|
||||
>;
|
||||
|
||||
export type StorybookBuilderOutput = JsonObject & BuilderOutput & {};
|
||||
|
||||
export default createBuilder(commandBuilder);
|
||||
|
||||
function commandBuilder(
|
||||
options: StorybookBuilderOptions,
|
||||
_context: BuilderContext
|
||||
): Observable<StorybookBuilderOutput> {
|
||||
return of({}).pipe(
|
||||
map(() => ({
|
||||
...options,
|
||||
angularBrowserTarget: options.browserTarget,
|
||||
})),
|
||||
switchMap((standaloneOptions) => runInstance(standaloneOptions)),
|
||||
map(() => {
|
||||
return { success: true };
|
||||
})
|
||||
);
|
||||
}
|
||||
|
||||
function runInstance(options: StandaloneOptions) {
|
||||
return new Observable<unknown>((obs) => {
|
||||
buildStandalone({ ...options })
|
||||
.then((sucess: unknown) => obs.next(sucess))
|
||||
.catch((err: unknown) => obs.error(err));
|
||||
});
|
||||
}
|
74
app/angular/src/builders/start-storybook/schema.json
Normal file
74
app/angular/src/builders/start-storybook/schema.json
Normal file
@ -0,0 +1,74 @@
|
||||
{
|
||||
"$schema": "http://json-schema.org/schema",
|
||||
"title": "Start Storybook",
|
||||
"description": "Serve up storybook in development mode.",
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"browserTarget": {
|
||||
"type": "string",
|
||||
"description": "Build target to be served in project-name:builder:config format. Should generally target on the builder: '@angular-devkit/build-angular:browser'. Useful for Storybook to use options (styles, assets, ...).",
|
||||
"pattern": "^[^:\\s]+:[^:\\s]+(:[^\\s]+)?$"
|
||||
},
|
||||
"port": {
|
||||
"type": "number",
|
||||
"description": "Port to listen on.",
|
||||
"default": 9009
|
||||
},
|
||||
"host": {
|
||||
"type": "string",
|
||||
"description": "Host to listen on.",
|
||||
"default": "localhost"
|
||||
},
|
||||
"staticDir": {
|
||||
"type": "array",
|
||||
"description": "Directory where to load static files from, array of strings.",
|
||||
"items": {
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
"configDir": {
|
||||
"type": "string",
|
||||
"description": "Directory where to load Storybook configurations from.",
|
||||
"default": ".storybook"
|
||||
},
|
||||
"https": {
|
||||
"type": "boolean",
|
||||
"description": "Serve Storybook over HTTPS. Note: You must provide your own certificate information.",
|
||||
"default": false
|
||||
},
|
||||
"sslCa": {
|
||||
"type": "string",
|
||||
"description": "Provide an SSL certificate authority. (Optional with --https, required if using a self-signed certificate)."
|
||||
},
|
||||
"sslCert": {
|
||||
"type": "string",
|
||||
"description": "Provide an SSL certificate. (Required with --https)."
|
||||
},
|
||||
"sslKey": {
|
||||
"type": "string",
|
||||
"description": "SSL key to use for serving HTTPS."
|
||||
},
|
||||
"smokeTest": {
|
||||
"type": "boolean",
|
||||
"description": "Exit after successful start.",
|
||||
"default": false
|
||||
},
|
||||
"ci": {
|
||||
"type": "boolean",
|
||||
"description": "CI mode (skip interactive prompts, don't open browser).",
|
||||
"default": false
|
||||
},
|
||||
"quiet": {
|
||||
"type": "boolean",
|
||||
"description": "Suppress verbose build output.",
|
||||
"default": false
|
||||
},
|
||||
"docs": {
|
||||
"type": "boolean",
|
||||
"description": "Starts Storybook in documentation mode. Learn more about it : https://storybook.js.org/docs/react/writing-docs/build-documentation#preview-storybooks-documentation.",
|
||||
"default": false
|
||||
}
|
||||
},
|
||||
"additionalProperties": false,
|
||||
"required": ["browserTarget"]
|
||||
}
|
13
app/angular/standalone.d.ts
vendored
Normal file
13
app/angular/standalone.d.ts
vendored
Normal file
@ -0,0 +1,13 @@
|
||||
import { CLIOptions, LoadOptions, BuilderOptions } from '@storybook/core-common';
|
||||
|
||||
export type StandaloneOptions = Partial<
|
||||
CLIOptions &
|
||||
LoadOptions &
|
||||
BuilderOptions & {
|
||||
angularBrowserTarget: string;
|
||||
}
|
||||
>;
|
||||
|
||||
declare module '@storybook/angular/standalone' {
|
||||
export default function buildStandalone(options: StandaloneOptions): Promise<unknown>;
|
||||
}
|
@ -6,5 +6,6 @@
|
||||
"types": ["webpack-env", "node"],
|
||||
"rootDir": "./src",
|
||||
"resolveJsonModule": true
|
||||
}
|
||||
},
|
||||
"include": ["src/**/*", "src/**/*.json"]
|
||||
}
|
||||
|
@ -74,6 +74,13 @@
|
||||
"scripts": [],
|
||||
"assets": ["src/favicon.ico", "src/assets"]
|
||||
}
|
||||
},
|
||||
"storybook": {
|
||||
"builder": "@storybook/angular:start-storybook",
|
||||
"options": {
|
||||
"browserTarget": "angular-cli:build",
|
||||
"port": 4400
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
|
Loading…
x
Reference in New Issue
Block a user