mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 16:11:33 +08:00
Merge branch 'next' into pr/Spy-Seth/6589
# Conflicts: # app/react/package.json # yarn.lock
This commit is contained in:
commit
d4ae753259
40
.babelrc.js
40
.babelrc.js
@ -1,6 +1,20 @@
|
||||
const withTests = {
|
||||
presets: [
|
||||
[
|
||||
'@babel/preset-env',
|
||||
{ shippedProposals: true, useBuiltIns: 'usage', corejs: '3', targets: { node: 'current' } },
|
||||
],
|
||||
],
|
||||
plugins: [
|
||||
'babel-plugin-require-context-hook',
|
||||
'babel-plugin-dynamic-import-node',
|
||||
'@babel/plugin-transform-runtime',
|
||||
],
|
||||
};
|
||||
|
||||
module.exports = {
|
||||
presets: [
|
||||
['@babel/preset-env', { shippedProposals: true, useBuiltIns: 'usage', corejs: '2' }],
|
||||
['@babel/preset-env', { shippedProposals: true, useBuiltIns: 'usage', corejs: '3' }],
|
||||
'@babel/preset-typescript',
|
||||
'@babel/preset-react',
|
||||
'@babel/preset-flow',
|
||||
@ -20,26 +34,20 @@ module.exports = {
|
||||
['emotion', { sourceMap: true, autoLabel: true }],
|
||||
],
|
||||
env: {
|
||||
test: {
|
||||
presets: [
|
||||
['@babel/preset-env', { shippedProposals: true, useBuiltIns: 'usage', corejs: '2' }],
|
||||
],
|
||||
plugins: [
|
||||
'babel-plugin-require-context-hook',
|
||||
'babel-plugin-dynamic-import-node',
|
||||
'@babel/plugin-transform-runtime',
|
||||
],
|
||||
},
|
||||
test: withTests,
|
||||
},
|
||||
overrides: [
|
||||
{
|
||||
test: './examples/vue-kitchen-sink',
|
||||
presets: ['babel-preset-vue'],
|
||||
env: {
|
||||
test: withTests,
|
||||
},
|
||||
},
|
||||
{
|
||||
test: './lib',
|
||||
presets: [
|
||||
['@babel/preset-env', { shippedProposals: true, useBuiltIns: 'usage', corejs: '2' }],
|
||||
['@babel/preset-env', { shippedProposals: true, useBuiltIns: 'usage', corejs: '3' }],
|
||||
'@babel/preset-react',
|
||||
],
|
||||
plugins: [
|
||||
@ -52,6 +60,9 @@ module.exports = {
|
||||
'@babel/plugin-transform-react-constant-elements',
|
||||
'babel-plugin-add-react-displayname',
|
||||
],
|
||||
env: {
|
||||
test: withTests,
|
||||
},
|
||||
},
|
||||
{
|
||||
test: [
|
||||
@ -72,7 +83,7 @@ module.exports = {
|
||||
targets: {
|
||||
node: '8.11',
|
||||
},
|
||||
corejs: '2',
|
||||
corejs: '3',
|
||||
},
|
||||
],
|
||||
],
|
||||
@ -83,6 +94,9 @@ module.exports = {
|
||||
'@babel/plugin-proposal-object-rest-spread',
|
||||
'@babel/plugin-proposal-export-default-from',
|
||||
],
|
||||
env: {
|
||||
test: withTests,
|
||||
},
|
||||
},
|
||||
],
|
||||
};
|
||||
|
@ -40,6 +40,15 @@ jobs:
|
||||
- addons
|
||||
- app
|
||||
- lib
|
||||
chromatic:
|
||||
<<: *defaults
|
||||
steps:
|
||||
- checkout
|
||||
- attach_workspace:
|
||||
at: .
|
||||
- run:
|
||||
name: Run chromatic on the pre-built storybook
|
||||
command: yarn chromatic -- -d ./storybook-static
|
||||
examples:
|
||||
<<: *defaults
|
||||
steps:
|
||||
@ -109,6 +118,11 @@ jobs:
|
||||
command: |
|
||||
cd examples/preact-kitchen-sink
|
||||
yarn build-storybook
|
||||
- run:
|
||||
name: Build cra react15
|
||||
command: |
|
||||
cd examples/cra-react15
|
||||
yarn build-storybook
|
||||
- run:
|
||||
name: Build official-storybook
|
||||
command: |
|
||||
@ -120,6 +134,14 @@ jobs:
|
||||
- store_artifacts:
|
||||
path: examples/official-storybook/image-snapshots/__image_snapshots__
|
||||
destination: official_storybook_image_snapshots
|
||||
- persist_to_workspace:
|
||||
root: .
|
||||
paths:
|
||||
- node_modules
|
||||
- examples
|
||||
- addons
|
||||
- app
|
||||
- lib
|
||||
smoke-tests:
|
||||
<<: *defaults
|
||||
steps:
|
||||
@ -186,6 +208,11 @@ jobs:
|
||||
command: |
|
||||
cd examples/preact-kitchen-sink
|
||||
yarn storybook --smoke-test --quiet
|
||||
- run:
|
||||
name: Run cra reac15 (smoke test)
|
||||
command: |
|
||||
cd examples/cra-react15
|
||||
yarn storybook --smoke-test --quiet
|
||||
native-smoke-tests:
|
||||
<<: *defaults
|
||||
steps:
|
||||
@ -337,3 +364,6 @@ workflows:
|
||||
- cli-test-latest-cra:
|
||||
requires:
|
||||
- build
|
||||
- chromatic:
|
||||
requires:
|
||||
- examples
|
||||
|
@ -7,10 +7,9 @@ docs/public
|
||||
storybook-static
|
||||
built-storybooks
|
||||
lib/cli/test
|
||||
scripts/storage
|
||||
*.bundle.js
|
||||
*.js.map
|
||||
*.ts
|
||||
*.tsx
|
||||
|
||||
!.remarkrc.js
|
||||
!.babelrc.js
|
||||
|
88
.eslintrc.js
88
.eslintrc.js
@ -1,26 +1,68 @@
|
||||
const error = 2;
|
||||
const warn = 1;
|
||||
const ignore = 0;
|
||||
|
||||
module.exports = {
|
||||
root: true,
|
||||
extends: [
|
||||
'airbnb',
|
||||
'plugin:jest/recommended',
|
||||
'plugin:import/react-native',
|
||||
'plugin:@typescript-eslint/recommended',
|
||||
'prettier',
|
||||
'prettier/react',
|
||||
'prettier/@typescript-eslint',
|
||||
],
|
||||
plugins: ['prettier', 'jest', 'import', 'react', 'jsx-a11y', 'json', 'html'],
|
||||
parser: 'babel-eslint',
|
||||
parserOptions: { ecmaVersion: 8, sourceType: 'module' },
|
||||
plugins: [
|
||||
'@typescript-eslint',
|
||||
'prettier',
|
||||
'jest',
|
||||
'import',
|
||||
'react',
|
||||
'jsx-a11y',
|
||||
'json',
|
||||
'html',
|
||||
],
|
||||
parser: '@typescript-eslint/parser',
|
||||
parserOptions: {
|
||||
ecmaVersion: 8,
|
||||
sourceType: 'module',
|
||||
ecmaFeatures: {
|
||||
jsx: true,
|
||||
},
|
||||
},
|
||||
env: { es6: true, node: true, 'jest/globals': true },
|
||||
settings: {
|
||||
'import/core-modules': ['enzyme'],
|
||||
'import/ignore': ['node_modules\\/(?!@storybook)'],
|
||||
'import/resolver': { node: { extensions: ['.js', '.ts'] } },
|
||||
'import/resolver': { node: { extensions: ['.js', '.ts', '.tsx', '.mjs'] } },
|
||||
'html/html-extensions': ['.html'],
|
||||
},
|
||||
rules: {
|
||||
'no-restricted-imports': [
|
||||
error,
|
||||
{
|
||||
paths: [
|
||||
{
|
||||
name: 'lodash.isequal',
|
||||
message:
|
||||
'Lodash modularised (and lodash < 4.17.11) have CVE vulnerabilities. Please use tree-shakeable imports like lodash/xxx instead',
|
||||
},
|
||||
{
|
||||
name: 'lodash.mergewith',
|
||||
message:
|
||||
'Lodash modularised (and lodash < 4.17.11) have CVE vulnerabilities. Please use tree-shakeable imports like lodash/xxx instead',
|
||||
},
|
||||
{
|
||||
name: 'lodash.pick',
|
||||
message:
|
||||
'Lodash modularised (and lodash < 4.17.11) have CVE vulnerabilities. Please use tree-shakeable imports like lodash/xxx instead',
|
||||
},
|
||||
],
|
||||
// catch-all for any lodash modularised. The CVE is listed against the entire family for lodash < 4.17.11
|
||||
patterns: ['lodash.*'],
|
||||
},
|
||||
],
|
||||
'prettier/prettier': [warn],
|
||||
'no-debugger': process.env.NODE_ENV === 'production' ? error : ignore,
|
||||
'class-methods-use-this': ignore,
|
||||
@ -30,6 +72,7 @@ module.exports = {
|
||||
{
|
||||
js: 'never',
|
||||
ts: 'never',
|
||||
tsx: 'never',
|
||||
mjs: 'never',
|
||||
},
|
||||
],
|
||||
@ -42,11 +85,11 @@ module.exports = {
|
||||
'**/example/**',
|
||||
'*.js',
|
||||
'**/*.test.js',
|
||||
'**/*.stories.js',
|
||||
'**/*.stories.*',
|
||||
'**/scripts/*.js',
|
||||
'**/stories/**/*.js',
|
||||
'**/__tests__/**/*.js',
|
||||
'**/.storybook/**/*.js',
|
||||
'**/.storybook/**/*.*',
|
||||
],
|
||||
peerDependencies: true,
|
||||
},
|
||||
@ -94,13 +137,21 @@ module.exports = {
|
||||
error,
|
||||
{ allow: ['__STORYBOOK_CLIENT_API__', '__STORYBOOK_ADDONS_CHANNEL__'] },
|
||||
],
|
||||
'@typescript-eslint/no-var-requires': ignore,
|
||||
'@typescript-eslint/camelcase': ignore,
|
||||
'@typescript-eslint/no-unused-vars': ignore,
|
||||
'@typescript-eslint/explicit-member-accessibility': ignore,
|
||||
'@typescript-eslint/explicit-function-return-type': ignore,
|
||||
'@typescript-eslint/no-explicit-any': ignore, // would prefer to enable this
|
||||
'@typescript-eslint/no-use-before-define': ignore, // this is duplicated
|
||||
'@typescript-eslint/interface-name-prefix': ignore, // I don't agree
|
||||
},
|
||||
overrides: [
|
||||
{
|
||||
files: [
|
||||
'**/__tests__/**',
|
||||
'**/*.test.js',
|
||||
'**/*.stories.js',
|
||||
'**/*.test.*',
|
||||
'**/*.stories.*',
|
||||
'**/storyshots/**/stories/**',
|
||||
'docs/src/new-components/lib/StoryLinkWrapper.js',
|
||||
'docs/src/stories/**',
|
||||
@ -110,5 +161,26 @@ module.exports = {
|
||||
},
|
||||
},
|
||||
{ files: '**/.storybook/config.js', rules: { 'global-require': ignore } },
|
||||
{
|
||||
files: ['**/*.stories.*'],
|
||||
rules: {
|
||||
'no-console': ignore,
|
||||
},
|
||||
},
|
||||
{
|
||||
files: ['**/*.tsx', '**/*.ts'],
|
||||
rules: {
|
||||
'react/prop-types': ignore, // we should use types
|
||||
'no-dupe-class-members': ignore, // this is called overloads in typescript
|
||||
},
|
||||
},
|
||||
{
|
||||
files: ['**/*.d.ts'],
|
||||
rules: {
|
||||
'vars-on-top': ignore,
|
||||
'no-var': ignore, // this is how typescript works
|
||||
'spaced-comment': ignore,
|
||||
},
|
||||
},
|
||||
],
|
||||
};
|
||||
|
5
.github/automention.yml
vendored
5
.github/automention.yml
vendored
@ -1,4 +1,4 @@
|
||||
'app: angular': ['kroeder', 'igor-dv']
|
||||
'app: angular': ['kroeder', 'igor-dv', 'MaximSagan']
|
||||
'app: ember': ['gabrielcsapo']
|
||||
'app: html': ['Hypnosphi']
|
||||
'app: marko': ['nm123github']
|
||||
@ -10,9 +10,10 @@
|
||||
'api: addons': ['ndelangen']
|
||||
'addon: a11y': ['CodeByAlex', 'Armanio', 'jsomsanith']
|
||||
'addon: contexts': ['leoyli']
|
||||
'addon: storysource': ['igor-dv', 'libetl']
|
||||
'addon: docs': ['shilman', 'elevatebart']
|
||||
'addon: info': ['shilman', 'elevatebart']
|
||||
'addon: knobs': ['leoyli', 'Armanio']
|
||||
'addon: storysource': ['igor-dv', 'libetl']
|
||||
typescript: ['kroeder', 'gaetanmaisse', 'ndelangen']
|
||||
theming: ['ndelangen', 'domyen']
|
||||
cra: ['mrmckeb']
|
||||
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -25,3 +25,5 @@ integration/__image_snapshots__/__diff_output__
|
||||
lib/*.jar
|
||||
lib/**/dll
|
||||
.expo/packager-info.json
|
||||
scripts/storage
|
||||
htpasswd
|
||||
|
@ -23,7 +23,6 @@ object Project : Project({
|
||||
buildType(OpenSourceProjects_Storybook_Examples)
|
||||
buildType(OpenSourceProjects_Storybook_Docs)
|
||||
buildType(OpenSourceProjects_Storybook_Build_2)
|
||||
buildType(OpenSourceProjects_Storybook_CliTest)
|
||||
buildType(OpenSourceProjects_Storybook_Test)
|
||||
buildType(OpenSourceProjects_Storybook_Lint)
|
||||
buildType(OpenSourceProjects_Storybook_Lint_Warnings)
|
||||
|
@ -111,10 +111,5 @@ object OpenSourceProjects_Storybook_Build_2 : BuildType({
|
||||
onDependencyCancel = FailureAction.ADD_PROBLEM
|
||||
}
|
||||
}
|
||||
dependency(OpenSourceProjects_Storybook.buildTypes.OpenSourceProjects_Storybook_CliTest) {
|
||||
snapshot {
|
||||
onDependencyCancel = FailureAction.ADD_PROBLEM
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
|
@ -26,10 +26,10 @@ object OpenSourceProjects_Storybook_Chromatic : BuildType({
|
||||
scriptContent = """
|
||||
#!/bin/sh
|
||||
|
||||
set -e -x
|
||||
|
||||
yarn
|
||||
yarn chromatic
|
||||
# set -e -x
|
||||
# yarn
|
||||
# yarn chromatic
|
||||
echo "chromatic moved to cirlce CI"
|
||||
""".trimIndent()
|
||||
dockerImage = "node:%docker.node.version%"
|
||||
}
|
||||
|
@ -1,63 +0,0 @@
|
||||
package OpenSourceProjects_Storybook.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.buildFeatures.commitStatusPublisher
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.buildSteps.script
|
||||
|
||||
object OpenSourceProjects_Storybook_CliTest : BuildType({
|
||||
uuid = "b1db1a3a-a4cf-46ea-8f55-98b86611f92e"
|
||||
id = "OpenSourceProjects_Storybook_CliTest"
|
||||
name = "CLI test"
|
||||
|
||||
vcs {
|
||||
root(OpenSourceProjects_Storybook.vcsRoots.OpenSourceProjects_Storybook_HttpsGithubComStorybooksStorybookRefsHeadsMaster)
|
||||
|
||||
}
|
||||
|
||||
steps {
|
||||
script {
|
||||
name = "Test"
|
||||
scriptContent = """
|
||||
#!/bin/sh
|
||||
|
||||
set -e -x
|
||||
|
||||
yarn
|
||||
yarn test --cli --teamcity
|
||||
""".trimIndent()
|
||||
dockerImage = "node:%docker.node.version%"
|
||||
}
|
||||
}
|
||||
|
||||
features {
|
||||
commitStatusPublisher {
|
||||
publisher = github {
|
||||
githubUrl = "https://api.github.com"
|
||||
authType = personalToken {
|
||||
token = "credentialsJSON:5ffe2d7e-531e-4f6f-b1fc-a41bfea26eaa"
|
||||
}
|
||||
}
|
||||
param("github_oauth_user", "Hypnosphi")
|
||||
}
|
||||
}
|
||||
|
||||
dependencies {
|
||||
dependency(OpenSourceProjects_Storybook.buildTypes.OpenSourceProjects_Storybook_Bootstrap) {
|
||||
snapshot {
|
||||
onDependencyFailure = FailureAction.FAIL_TO_START
|
||||
}
|
||||
|
||||
artifacts {
|
||||
artifactRules = "dist.zip!**"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
requirements {
|
||||
doesNotContain("env.OS", "Windows")
|
||||
}
|
||||
|
||||
cleanup {
|
||||
artifacts(days = 1)
|
||||
}
|
||||
})
|
@ -9,6 +9,7 @@ import jetbrains.buildServer.configs.kotlin.v2017_2.failureConditions.failOnMetr
|
||||
enum class StorybookApp(val appName: String, val exampleDir: String, val merged: Boolean = true) {
|
||||
CRA("CRA", "cra-kitchen-sink"),
|
||||
CRA_TS("CRA TS", "cra-ts-kitchen-sink"),
|
||||
CRA_REACT15("CRA REACT15", "cra-react15"),
|
||||
VUE("Vue", "vue-kitchen-sink"),
|
||||
ANGULAR("Angular", "angular-cli"),
|
||||
POLYMER("Polymer", "polymer-cli"),
|
||||
|
@ -5,13 +5,12 @@ import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = 'b1db1a3a-a4cf-46ea-8f55-98b86611f92e' (id = 'OpenSourceProjects_Storybook_CliTest')
|
||||
To apply the patch, change the buildType with uuid = '1bda59b5-d08d-4fd8-b317-953e7d79d881' (id = 'OpenSourceProjects_Storybook_Docs')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("b1db1a3a-a4cf-46ea-8f55-98b86611f92e") {
|
||||
params {
|
||||
add {
|
||||
param("docker.node.version", "latest")
|
||||
}
|
||||
changeBuildType("1bda59b5-d08d-4fd8-b317-953e7d79d881") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '1ea2b5bd-28f6-44f5-8ab3-6c659ce8fbd6' (id = 'OpenSourceProjects_Storybook_SmokeTests')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("1ea2b5bd-28f6-44f5-8ab3-6c659ce8fbd6") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '2b9c73e2-0a6e-47ca-95ae-729cac42be2b' (id = 'OpenSourceProjects_Storybook_Build_2')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("2b9c73e2-0a6e-47ca-95ae-729cac42be2b") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '42cfbb9a-f35b-4f96-afae-0b508927a737' (id = 'OpenSourceProjects_Storybook_Lint')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("42cfbb9a-f35b-4f96-afae-0b508927a737") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '42cfbb9a-f35b-4f96-afae-0b508927a738' (id = 'OpenSourceProjects_Storybook_Lint_Warnings')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("42cfbb9a-f35b-4f96-afae-0b508927a738") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-angular' (id = 'OpenSourceProjects_Storybook_Angular')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-angular") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-chromatic' (id = 'OpenSourceProjects_Storybook_Chromatic')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-chromatic") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-cra' (id = 'OpenSourceProjects_Storybook_CRA')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-cra") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-cra_react15' (id = 'OpenSourceProjects_Storybook_CRA_REACT15')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-cra_react15") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-cra_ts' (id = 'OpenSourceProjects_Storybook_CRA_TS')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-cra_ts") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-ember' (id = 'OpenSourceProjects_Storybook_Ember')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-ember") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-html' (id = 'OpenSourceProjects_Storybook_HTML')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-html") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-marko' (id = 'OpenSourceProjects_Storybook_Marko')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-marko") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-mithril' (id = 'OpenSourceProjects_Storybook_Mithril')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-mithril") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-polymer' (id = 'OpenSourceProjects_Storybook_Polymer')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-polymer") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-preact' (id = 'OpenSourceProjects_Storybook_Preact')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-preact") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-riot' (id = 'OpenSourceProjects_Storybook_Riot')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-riot") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-svelte' (id = 'OpenSourceProjects_Storybook_Svelte')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-svelte") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6-vue' (id = 'OpenSourceProjects_Storybook_Vue')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6-vue") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '8cc5f747-4ca7-4f0d-940d-b0c422f501a6' (id = 'OpenSourceProjects_Storybook_Examples')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("8cc5f747-4ca7-4f0d-940d-b0c422f501a6") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -9,6 +9,11 @@ To apply the patch, change the buildType with uuid = '9f9177e7-9ec9-4e2e-aabb-d3
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("9f9177e7-9ec9-4e2e-aabb-d304fd667711") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
|
||||
params {
|
||||
add {
|
||||
param("docker.node.version", "10.13")
|
||||
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = '9f9177e7-9ec9-4e2e-aabb-d304fd667712' (id = 'OpenSourceProjects_Storybook_Bootstrap')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("9f9177e7-9ec9-4e2e-aabb-d304fd667712") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
package OpenSourceProjects_Storybook.patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with uuid = 'd4320bd8-6094-4dd6-9bed-e13d6f0d12e2' (id = 'OpenSourceProjects_Storybook_CliTestLatestCra')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType("d4320bd8-6094-4dd6-9bed-e13d6f0d12e2") {
|
||||
check(paused == false) {
|
||||
"Unexpected paused: '$paused'"
|
||||
}
|
||||
paused = true
|
||||
}
|
17
.teamcity/OpenSourceProjects_Storybook/patches/projects/69382d9b-7791-418a-9ff6-1c83b86ed6b5.kts
vendored
Normal file
17
.teamcity/OpenSourceProjects_Storybook/patches/projects/69382d9b-7791-418a-9ff6-1c83b86ed6b5.kts
vendored
Normal file
@ -0,0 +1,17 @@
|
||||
package OpenSourceProjects_Storybook.patches.projects
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.Project
|
||||
import jetbrains.buildServer.configs.kotlin.v2017_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the project with uuid = '69382d9b-7791-418a-9ff6-1c83b86ed6b5' (id = 'OpenSourceProjects_Storybook')
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeProject("69382d9b-7791-418a-9ff6-1c83b86ed6b5") {
|
||||
check(archived == false) {
|
||||
"Unexpected archived: '$archived'"
|
||||
}
|
||||
archived = true
|
||||
}
|
@ -2,11 +2,11 @@
|
||||
|
||||
| | [React](app/react)|[React Native](app/react-native)|[Vue](app/vue)|[Angular](app/angular)| [Polymer](app/polymer)| [Mithril](app/mithril)| [HTML](app/html)| [Marko](app/marko)| [Svelte](app/svelte)| [Riot](app/riot)| [Ember](app/ember)| [Preact](app/preact)|
|
||||
| ----------- |:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:|
|
||||
|[a11y](addons/a11y) |+| |+|+|+|+|+|+| | |+|+|
|
||||
|[actions](addons/actions) |+|+|+|+|+|+|+|+|+|+|+|+|
|
||||
|[a11y](addons/a11y) |+| |+|+|+|+|+|+|+|+|+|+|
|
||||
|[actions](addons/actions) |+|+*|+|+|+|+|+|+|+|+|+|+|
|
||||
|[backgrounds](addons/backgrounds) |+|*|+|+|+|+|+|+|+|+|+|+|
|
||||
|[centered](addons/centered) |+| |+|+| |+|+| |+| |+|+|
|
||||
|[contexts](addons/contexts) |+| |+| | | | | | | | | |
|
||||
|[contexts](addons/contexts) |+| |+| | | | | | | | |+|
|
||||
|[events](addons/events) |+| |+|+|+|+|+|+| | |+|+|
|
||||
|[graphql](addons/graphql) |+| | | | | | | | | | | |
|
||||
|[google-analytics](addons/google-analytics) |+|+|+|+|+|+|+|+|+|+|+|+|
|
||||
|
193
CHANGELOG.md
193
CHANGELOG.md
@ -1,35 +1,210 @@
|
||||
## 5.1.0-rc.0 (May 21, 2019)
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* UI: Fix initial bottom panel size ([#6822](https://github.com/storybooks/storybook/pull/6822))
|
||||
* UI: Fix syntaxthighlighter themes ([#6814](https://github.com/storybooks/storybook/pull/6814))
|
||||
* Addon-knobs: Fix Boolean knob (#6366) ([#6830](https://github.com/storybooks/storybook/pull/6830))
|
||||
* Theming: Change lib/theming so it no longer depends on react-inspector ([#6818](https://github.com/storybooks/storybook/pull/6818))
|
||||
* Core: Handle loading `.storybook/babel.config.js` (#6633) ([#6634](https://github.com/storybooks/storybook/pull/6634))
|
||||
* CLI: Fix init in create-react-library projects ([#6815](https://github.com/storybooks/storybook/pull/6815))
|
||||
* HTML: support knobs for both cached and uncached nodes ([#6783](https://github.com/storybooks/storybook/pull/6783))
|
||||
* Uncorrupt yarn lock ([#6811](https://github.com/storybooks/storybook/pull/6811))
|
||||
* Core: set a better value for process in manager webpack config ([#6767](https://github.com/storybooks/storybook/pull/6767))
|
||||
|
||||
### Maintenance
|
||||
|
||||
* Typescript: Migrate addon-centered ([#6772](https://github.com/storybooks/storybook/pull/6772))
|
||||
* Add engine field to package.json in apps ([#6809](https://github.com/storybooks/storybook/pull/6809))
|
||||
* Fix required engine for apps ([#6810](https://github.com/storybooks/storybook/pull/6810))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
* Upgrade lodash to latest ([#6832](https://github.com/storybooks/storybook/pull/6832))
|
||||
* Bump svelte from 3.4.1 to 3.4.2 ([#6838](https://github.com/storybooks/storybook/pull/6838))
|
||||
* Misc upgrades ([#6820](https://github.com/storybooks/storybook/pull/6820))
|
||||
|
||||
## 5.1.0-beta.1 (May 16, 2019)
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* UI: Scrollbar supports theming again ([#6794](https://github.com/storybooks/storybook/pull/6794))
|
||||
* UI: Fix scrolling styling ([#6785](https://github.com/storybooks/storybook/pull/6785))
|
||||
* UI: Fix iframe refresh ([#6787](https://github.com/storybooks/storybook/pull/6787))
|
||||
* UI: Preserve dimensions on resizing for panel ([#6696](https://github.com/storybooks/storybook/pull/6696))
|
||||
|
||||
### Maintenance
|
||||
|
||||
* Move chromatic to circle ci ([#6752](https://github.com/storybooks/storybook/pull/6752))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
* Bump fs-extra from 7.0.1 to 8.0.1 ([#6776](https://github.com/storybooks/storybook/pull/6776))
|
||||
|
||||
## 5.1.0-beta.0 (May 10, 2019)
|
||||
|
||||
Welcome to the 5.1 beta! Feature development's done; `beta.0` kicks off the stabilization process for the 5.1 final release. 🚀
|
||||
|
||||
## 5.1.0-alpha.40 (May 8, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
- Svelte: Add svelte v3 support ([#6698](https://github.com/storybooks/storybook/pull/6698))
|
||||
- Angular: Disable production mode for debugging components ([#6215](https://github.com/storybooks/storybook/pull/6215))
|
||||
- Angular: Allow optional component declaration without additional configuration ([#6666](https://github.com/storybooks/storybook/pull/6666))
|
||||
- Core: Allow browsing to a kind and get the first story ([#6720](https://github.com/storybooks/storybook/pull/6720))
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- UI: Preserve dimensions on resizing for panel ([#6696](https://github.com/storybooks/storybook/pull/6696))
|
||||
|
||||
### Maintenance
|
||||
|
||||
- Add CRA React15 example to test back-compat ([#6475](https://github.com/storybooks/storybook/pull/6475))
|
||||
- Remove teamcity CLI tests ([#6707](https://github.com/storybooks/storybook/pull/6707))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
- Upgrade to core-js v3 ([#6655](https://github.com/storybooks/storybook/pull/6655))
|
||||
- Bump eslint-plugin-react from 7.12.4 to 7.13.0 ([#6728](https://github.com/storybooks/storybook/pull/6728))
|
||||
- Bump @types/react-native from 0.57.50 to 0.57.51 ([#6732](https://github.com/storybooks/storybook/pull/6732))
|
||||
- Bump @types/node from 11.13.7 to 12.0.0 ([#6730](https://github.com/storybooks/storybook/pull/6730))
|
||||
- Bump jest-cli from 24.7.1 to 24.8.0 ([#6729](https://github.com/storybooks/storybook/pull/6729))
|
||||
- Bump @babel/preset-env from 7.4.3 to 7.4.4 ([#6731](https://github.com/storybooks/storybook/pull/6731))
|
||||
- Bump raw-loader from 1.0.0 to 2.0.0 ([#6685](https://github.com/storybooks/storybook/pull/6685))
|
||||
- Bump react-color from 2.17.1 to 2.17.3 ([#6681](https://github.com/storybooks/storybook/pull/6681))
|
||||
- Bump @babel/plugin-proposal-class-properties from 7.4.0 to 7.4.4 ([#6686](https://github.com/storybooks/storybook/pull/6686))
|
||||
- Bump react-dev-utils from 8.0.0 to 9.0.0 ([#6682](https://github.com/storybooks/storybook/pull/6682))
|
||||
- Bump codelyzer from 5.0.0 to 5.0.1 ([#6687](https://github.com/storybooks/storybook/pull/6687))
|
||||
- Bump @types/react from 16.8.14 to 16.8.16 ([#6717](https://github.com/storybooks/storybook/pull/6717))
|
||||
- Bump react-redux from 7.0.2 to 7.0.3 ([#6684](https://github.com/storybooks/storybook/pull/6684))
|
||||
|
||||
## 5.1.0-alpha.39 (May 2, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
- React-native: Ondevice actions ([#6594](https://github.com/storybooks/storybook/pull/6594))
|
||||
- React-native: Use emotion to style RN UI ([#6603](https://github.com/storybooks/storybook/pull/6603))
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- API: Mimic PureComponent behavior for Consumer children ([#6412](https://github.com/storybooks/storybook/pull/6412))
|
||||
|
||||
## 5.1.0-alpha.38 (May 2, 2019)
|
||||
|
||||
Failed publish
|
||||
|
||||
## 5.1.0-alpha.37 (May 1, 2019)
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- Core: Fix regression with deep linking ([#6688](https://github.com/storybooks/storybook/pull/6688))
|
||||
- Addon-contexts: No cancel option in UI if the context have no param ([#6669](https://github.com/storybooks/storybook/pull/6669))
|
||||
- CLI: Fix `sb init` for projects with frozen lock files ([#6629](https://github.com/storybooks/storybook/pull/6629))
|
||||
|
||||
### Maintenance
|
||||
|
||||
- CLI: Refactor how we install dev dependencies in cli ([#6695](https://github.com/storybooks/storybook/pull/6695))
|
||||
|
||||
## 5.0.11 (April 28, 2019)
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- Polymer: Fix re-rendering lit-html elements after non-lit-html element ([#5868](https://github.com/storybooks/storybook/pull/5868))
|
||||
- Addon-knobs: Check color knob value before applying uppercase ([#6598](https://github.com/storybooks/storybook/pull/6598))
|
||||
- Angular: Fix sourceMap property of angulars webpack config ([#6535](https://github.com/storybooks/storybook/pull/6535))
|
||||
|
||||
### Maintenance
|
||||
|
||||
- UI: Add missing props in stories ([#6353](https://github.com/storybooks/storybook/pull/6353))
|
||||
|
||||
## 5.1.0-alpha.36 (April 27, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
- Addon-contexts: Preact support ([#6660](https://github.com/storybooks/storybook/pull/6660))
|
||||
- Angular: Allow optional component declaration ([#6346](https://github.com/storybooks/storybook/pull/6346))
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- CLI: Fix `sb init` for projects with frozen lock files ([#6629](https://github.com/storybooks/storybook/pull/6629))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
- [Snyk] Fix for 1 vulnerable dependencies ([#6647](https://github.com/storybooks/storybook/pull/6647))
|
||||
|
||||
## 5.1.0-alpha.35 (April 27, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
- Addon-notes: use @storybook/router <Link> to render links in notes ([#6398](https://github.com/storybooks/storybook/pull/6398))
|
||||
- Angular: Support default `storybook` project configuration ([#6484](https://github.com/storybooks/storybook/pull/6484))
|
||||
- Addon-contexts: Improve Vue integration ([#6632](https://github.com/storybooks/storybook/pull/6632))
|
||||
- Addon-a11y: Design enhancements ([#6563](https://github.com/storybooks/storybook/pull/6563))
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- UI: `active` PropTypes on MobileLayout ([#6241](https://github.com/storybooks/storybook/pull/6241))
|
||||
- Core: Fix css import when sideEffects is false ([#6650](https://github.com/storybooks/storybook/pull/6650))
|
||||
- Core: Fix infinite loop with special characters in kind names ([#6607](https://github.com/storybooks/storybook/pull/6607))
|
||||
- UI: Fix 'Escape' onKeyUp event doesn't work ([#6578](https://github.com/storybooks/storybook/pull/6578))
|
||||
|
||||
### Maintenance
|
||||
|
||||
- UI: Add missing props in stories ([#6353](https://github.com/storybooks/storybook/pull/6353))
|
||||
- Build: tslint, and use eslint for everything ([#6621](https://github.com/storybooks/storybook/pull/6621))
|
||||
- Build: deploy to local registry ([#6619](https://github.com/storybooks/storybook/pull/6619))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
- Bump ts-node from 8.0.3 to 8.1.0 ([#6585](https://github.com/storybooks/storybook/pull/6585))
|
||||
- Bump semver from 5.7.0 to 6.0.0 ([#6580](https://github.com/storybooks/storybook/pull/6580))
|
||||
- Bump react-color from 2.17.0 to 2.17.1 ([#6583](https://github.com/storybooks/storybook/pull/6583))
|
||||
|
||||
## 5.1.0-alpha.34 (April 24, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
- Addon-contexts: Add URL query param feature ([#6601](https://github.com/storybooks/storybook/pull/6601))
|
||||
- UI: Add classNames to sidebar nav elements ([#6571](https://github.com/storybooks/storybook/pull/6571))
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
- Addon-knobs: Check color knob value before applying uppercase ([#6598](https://github.com/storybooks/storybook/pull/6598))
|
||||
- React-native: Restore title in section header ([#6599](https://github.com/storybooks/storybook/pull/6599))
|
||||
|
||||
## 5.1.0-alpha.33 (April 23, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
* React: Add support for create-react-app@3.0.0 ([#6560](https://github.com/storybooks/storybook/pull/6560))
|
||||
- React: Add support for create-react-app@3.0.0 ([#6560](https://github.com/storybooks/storybook/pull/6560))
|
||||
|
||||
## 5.1.0-alpha.32 (April 22, 2019)
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* Addon-contexts: bug-fixing, testing, typing ([#6572](https://github.com/storybooks/storybook/pull/6572))
|
||||
- Addon-contexts: bug-fixing, testing, typing ([#6572](https://github.com/storybooks/storybook/pull/6572))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
* CHANGE opn to open ([#6567](https://github.com/storybooks/storybook/pull/6567))
|
||||
- CHANGE opn to open ([#6567](https://github.com/storybooks/storybook/pull/6567))
|
||||
|
||||
## 5.1.0-alpha.31 (April 19, 2019)
|
||||
|
||||
### Features
|
||||
|
||||
* Addon-backgrounds: Emit event on updating background ([#6561](https://github.com/storybooks/storybook/pull/6561))
|
||||
* Addon-contexts: Merge into monorepo ([#6559](https://github.com/storybooks/storybook/pull/6559))
|
||||
- Addon-backgrounds: Emit event on updating background ([#6561](https://github.com/storybooks/storybook/pull/6561))
|
||||
- Addon-contexts: Merge into monorepo ([#6559](https://github.com/storybooks/storybook/pull/6559))
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* Angular: Fix sourceMap property of angulars webpack config ([#6535](https://github.com/storybooks/storybook/pull/6535))
|
||||
* Addon-jest: Fix result display ([#6539](https://github.com/storybooks/storybook/pull/6539))
|
||||
- Angular: Fix sourceMap property of angulars webpack config ([#6535](https://github.com/storybooks/storybook/pull/6535))
|
||||
- Addon-jest: Fix result display ([#6539](https://github.com/storybooks/storybook/pull/6539))
|
||||
|
||||
### Dependency Upgrades
|
||||
|
||||
* Bump ember-source from 3.8.1 to 3.9.1 ([#6531](https://github.com/storybooks/storybook/pull/6531))
|
||||
* Bump typescript from 3.4.2 to 3.4.3 ([#6528](https://github.com/storybooks/storybook/pull/6528))
|
||||
- Bump ember-source from 3.8.1 to 3.9.1 ([#6531](https://github.com/storybooks/storybook/pull/6531))
|
||||
- Bump typescript from 3.4.2 to 3.4.3 ([#6528](https://github.com/storybooks/storybook/pull/6528))
|
||||
|
||||
## 5.0.10 (April 18, 2019)
|
||||
|
||||
|
121
CONTRIBUTING.md
121
CONTRIBUTING.md
@ -28,14 +28,16 @@ To test your project against the current latest version of storybook, you can cl
|
||||
git clone https://github.com/storybooks/storybook.git
|
||||
cd storybook
|
||||
yarn install
|
||||
yarn bootstrap --core
|
||||
yarn bootstrap
|
||||
```
|
||||
|
||||
The bootstrap command might ask which sections of the codebase you want to bootstrap. Unless you're going to work with ReactNative or the Documentation, you can keep the default.
|
||||
|
||||
You can also pick directly from CLI:
|
||||
|
||||
yarn bootstrap --core
|
||||
```sh
|
||||
yarn bootstrap --core
|
||||
```
|
||||
|
||||
#### 2a. Run unit tests
|
||||
|
||||
@ -67,7 +69,7 @@ Before the tests are run, the project must be bootstrapped with core. You can ac
|
||||
This option executes tests from `<rootdir>/examples/official-storybook`
|
||||
In order for the image snapshots to be correctly generated, you must have a static build of the storybook up-to-date :
|
||||
|
||||
```javascript
|
||||
```sh
|
||||
cd examples/official-storybook
|
||||
yarn build-storybook
|
||||
cd ../..
|
||||
@ -80,68 +82,118 @@ Puppeteer is used to launch and grab screenshots of example pages, while jest is
|
||||
|
||||
If you made any changes to the `lib/cli` package, the easiest way to verify that it doesn't break anything is to run e2e tests:
|
||||
|
||||
yarn test --cli
|
||||
```sh
|
||||
yarn test --cli
|
||||
```
|
||||
|
||||
This will run a bash script located at `lib/cli/test/run_tests.sh`. It will copy the contents of `fixtures` into a temporary `run` directory, run `getstorybook` in each of the subdirectories, and check that storybook starts successfully using `yarn storybook --smoke-test`.
|
||||
|
||||
After that, the `run` directory content will be compared with `snapshots`. You can update the snapshots by passing an `--update` flag:
|
||||
|
||||
yarn test --cli --update
|
||||
```sh
|
||||
yarn test --cli --update
|
||||
```
|
||||
|
||||
In that case, please check the git diff before committing to make sure it only contains the intended changes.
|
||||
|
||||
#### 2c. Link `storybook` and any other required dependencies
|
||||
#### 2c. Run Linter
|
||||
|
||||
If you want to test your own existing project using the GitHub version of storybook, you need to `link` the packages you use in your project.
|
||||
We use eslint as a linter for all code (including typescript code).
|
||||
|
||||
All you have to run is:
|
||||
|
||||
```sh
|
||||
cd app/react
|
||||
yarn link
|
||||
|
||||
cd <your-project>
|
||||
yarn link @storybook/react
|
||||
|
||||
# repeat with whichever other parts of the monorepo you are using.
|
||||
yarn lint
|
||||
```
|
||||
|
||||
It can be immensely helpful to get feedback in your editor, if you're using VsCode, you should install the `eslint` plugin and configure it with these settings:
|
||||
|
||||
```plaintext
|
||||
"eslint.autoFixOnSave": true,
|
||||
"eslint.packageManager": "yarn",
|
||||
"eslint.options": {
|
||||
"cache": true,
|
||||
"cacheLocation": ".cache/eslint",
|
||||
"extensions": [".js", ".jsx", ".mjs", ".json", ".ts", ".tsx"]
|
||||
},
|
||||
"eslint.validate": [
|
||||
"javascript",
|
||||
"javascriptreact",
|
||||
{"language": "typescript", "autoFix": true },
|
||||
{"language": "typescriptreact", "autoFix": true }
|
||||
],
|
||||
"eslint.alwaysShowStatus": true
|
||||
```
|
||||
|
||||
This should enable auto-fix for all source files, and give linting warnings and errors within your editor.
|
||||
|
||||
### Reproductions
|
||||
|
||||
#### In the monorepo
|
||||
|
||||
The best way to help figure out an issue you are having is to produce a minimal reproduction against the `master` branch.
|
||||
|
||||
A good way to do that is using the example `cra-kitchen-sink` app embedded in this repository:
|
||||
|
||||
```sh
|
||||
# Download and build this repository:
|
||||
git clone https://github.com/storybooks/storybook.git
|
||||
cd storybook
|
||||
yarn install
|
||||
yarn bootstrap --core
|
||||
# Download and build this repository:
|
||||
git clone https://github.com/storybooks/storybook.git
|
||||
cd storybook
|
||||
yarn install
|
||||
yarn bootstrap --core
|
||||
|
||||
# make changes to try and reproduce the problem, such as adding components + stories
|
||||
cd examples/cra-kitchen-sink
|
||||
yarn storybook
|
||||
# make changes to try and reproduce the problem, such as adding components + stories
|
||||
cd examples/cra-kitchen-sink
|
||||
yarn storybook
|
||||
|
||||
# see if you can see the problem, if so, commit it:
|
||||
git checkout "branch-describing-issue"
|
||||
git add -A
|
||||
git commit -m "reproduction for issue #123"
|
||||
# see if you can see the problem, if so, commit it:
|
||||
git checkout "branch-describing-issue"
|
||||
git add -A
|
||||
git commit -m "reproduction for issue #123"
|
||||
|
||||
# fork the storybook repo to your account, then add the resulting remote
|
||||
git remote add <your-username> https://github.com/<your-username>/storybook.git
|
||||
git push -u <your-username> master
|
||||
# fork the storybook repo to your account, then add the resulting remote
|
||||
git remote add <your-username> https://github.com/<your-username>/storybook.git
|
||||
git push -u <your-username> master
|
||||
```
|
||||
|
||||
If you follow that process, you can then link to the GitHub repository in the issue. See <https://github.com/storybooks/storybook/issues/708#issuecomment-290589886> for an example.
|
||||
|
||||
**NOTE**: If your issue involves a webpack config, create-react-app will prevent you from modifying the _app's_ webpack config, however, you can still modify storybook's to mirror your app's version of the storybook. Alternatively, use `yarn eject` in the CRA app to get a modifiable webpack config.
|
||||
|
||||
#### Outside the monorepo
|
||||
|
||||
Sometimes your storybook is deeply ingrained in your own setup and it's hard to create a minimal viable reproduction somewhere else.
|
||||
|
||||
Inside the storybook repo we have a script that allows you to test the packages inside this repo in your own seperate project.
|
||||
|
||||
You can use `npm link` on all packages, but npm linking is cumbersome and has subtle differences from what happens in a registry-based installation.
|
||||
So the way our script works is that it:
|
||||
|
||||
- sets up a npm registry running on your own local machine
|
||||
- changes your default registry to this local one
|
||||
- builds all packages in the storybook repo
|
||||
- publishes all packages as latest
|
||||
|
||||
Our script leaves the local registry running, for **as long as you keep it running** you can install storybook packages from this local registry.
|
||||
|
||||
- Navigate to your own project and then change `package.json` so the storybook packages match the version of the one you just published.
|
||||
- Then just do the normal install procedure using `yarn` or `npm`
|
||||
- Start using your storybook as normally.
|
||||
|
||||
If you've made a change to storybook's codebase and would want this change to be reflected in your app:
|
||||
|
||||
- Ensure the storybook packages are transpiled, by either having run `yarn dev` or `yarn bootstrap --core`.
|
||||
- Go to the terminal where the local regitry is running and press `<Enter>`. This will kick off a new publish.
|
||||
- Run the install procedure again in your local repo, (you may need to clean out node_modules first).
|
||||
- Restart your storybook.
|
||||
|
||||
### Updating Tests
|
||||
|
||||
Before any contributes are submitted in a PR, make sure to add or update meaningful tests. A PR that has failing tests will be regarded as a “Work in Progress” and will not be merged until all tests pass.
|
||||
When creating new unit test files, the tests should adhere to a particular folder structure and naming convention, as defined below.
|
||||
|
||||
```sh
|
||||
#Proper naming convention and structure for js tests files
|
||||
# Proper naming convention and structure for js tests files
|
||||
+-- parentFolder
|
||||
| +-- [filename].js
|
||||
| +-- [filename].test.js
|
||||
@ -149,14 +201,9 @@ When creating new unit test files, the tests should adhere to a particular folde
|
||||
|
||||
## Pull Requests (PRs)
|
||||
|
||||
We welcome your contributions. There are many ways you can help us. This is few of those ways:
|
||||
We welcome all contributions. There are many ways you can help us. This is few of those ways:
|
||||
|
||||
- Fix typos and add more [documentation](https://github.com/storybooks/storybook/labels/needs%20docs).
|
||||
- Try to fix some [bugs](https://github.com/storybooks/storybook/labels/bug).
|
||||
- Work on [API](https://github.com/storybooks/storybook/labels/enhancement%3A%20api), [Addons](https://github.com/storybooks/storybook/labels/enhancement%3A%20addons), [UI](https://github.com/storybooks/storybook/labels/enhancement%3A%20ui) or [Webpack](https://github.com/storybooks/storybook/labels/enhancement%3A%20webpack) use enhancements and new [features](https://github.com/storybooks/storybook/labels/feature%20request).
|
||||
- Add more [tests](https://codecov.io/gh/storybooks/storybook/tree/master/packages) (especially for the [UI](https://codecov.io/gh/storybooks/storybook/tree/master/packages/storybook-ui/src)).
|
||||
|
||||
Before you submit a new PR, make sure you run `yarn test`. Do not submit a PR if tests are failing. If you need any help, create an issue and ask.
|
||||
Before you submit a new PR, make sure you run `yarn test`. Do not submit a PR if tests are failing. If you need any help, the best way is to [join the discord server and ask in the maintenance channel](https://discord.gg/sMFvFsG).
|
||||
|
||||
### Reviewing PRs
|
||||
|
||||
|
15
MIGRATION.md
15
MIGRATION.md
@ -16,6 +16,7 @@
|
||||
- [Addon a11y uses parameters](#addon-a11y-uses-parameters-decorator-renamed)
|
||||
- [New keyboard shortcuts defaults](#new-keyboard-shortcuts-defaults)
|
||||
- [New URL structure](#new-url-structure)
|
||||
- [Vue integration](#vue-integration)
|
||||
- [From version 4.0.x to 4.1.x](#from-version-40x-to-41x)
|
||||
- [Private addon config](#private-addon-config)
|
||||
- [React 15.x](#react-15x)
|
||||
@ -107,9 +108,9 @@ module.exports = ({ config }) => ({
|
||||
...config.module,
|
||||
rules: [
|
||||
/* your own rules "..." here and/or some subset of config.module.rules */
|
||||
]
|
||||
}
|
||||
})
|
||||
],
|
||||
},
|
||||
});
|
||||
```
|
||||
|
||||
Please refer to the [current custom webpack documentation](https://github.com/storybooks/storybook/blob/next/docs/src/pages/configurations/custom-webpack-config/index.md) for more information on custom webpack config and to [Issue #6081](https://github.com/storybooks/storybook/issues/6081) for more information about the change.
|
||||
@ -216,7 +217,7 @@ Here is the mapping from old options to new:
|
||||
|
||||
Storybook v5 removes the search dialog box in favor of a quick search in the navigation view, so `showSearchBox` has been removed.
|
||||
|
||||
Storybook v5 introduce a new tool bar above the story view and you can show\hide it with the new `isToolshown` option.
|
||||
Storybook v5 introduce a new tool bar above the story view and you can show\hide it with the new `isToolshown` option.
|
||||
|
||||
## Individual story decorators
|
||||
|
||||
@ -395,6 +396,12 @@ You have to replace it with:
|
||||
start-storybook --https
|
||||
```
|
||||
|
||||
## Vue integration
|
||||
|
||||
The Vue integration was updated, so that every story returned from a story or decorator function is now being normalized with `Vue.extend` **and** is being wrapped by a functional component. Returning a string from a story or decorator function is still supported and is treated as a component with the returned string as the template.
|
||||
|
||||
Currently there is no recommended way of accessing the component options of a story inside a decorator.
|
||||
|
||||
## From version 4.0.x to 4.1.x
|
||||
|
||||
There are are a few migrations you should be aware of in 4.1, including one unintentionally breaking change for advanced addon usage.
|
||||
|
@ -4,7 +4,7 @@ This storybook addon can be helpful to make your UI components more accessible.
|
||||
|
||||
[Framework Support](https://github.com/storybooks/storybook/blob/master/ADDONS_SUPPORT.md)
|
||||
|
||||

|
||||

|
||||
|
||||
## Getting started
|
||||
|
||||
|
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@storybook/addon-a11y",
|
||||
"version": "5.1.0-alpha.33",
|
||||
"version": "5.1.0-rc.0",
|
||||
"description": "a11y addon for storybook",
|
||||
"keywords": [
|
||||
"a11y",
|
||||
@ -26,20 +26,21 @@
|
||||
"prepare": "node ../../scripts/prepare.js"
|
||||
},
|
||||
"dependencies": {
|
||||
"@storybook/addons": "5.1.0-alpha.33",
|
||||
"@storybook/api": "5.1.0-alpha.33",
|
||||
"@storybook/client-logger": "5.1.0-alpha.33",
|
||||
"@storybook/components": "5.1.0-alpha.33",
|
||||
"@storybook/core-events": "5.1.0-alpha.33",
|
||||
"@storybook/theming": "5.1.0-alpha.33",
|
||||
"@storybook/addons": "5.1.0-rc.0",
|
||||
"@storybook/api": "5.1.0-rc.0",
|
||||
"@storybook/client-logger": "5.1.0-rc.0",
|
||||
"@storybook/components": "5.1.0-rc.0",
|
||||
"@storybook/core-events": "5.1.0-rc.0",
|
||||
"@storybook/theming": "5.1.0-rc.0",
|
||||
"axe-core": "^3.2.2",
|
||||
"common-tags": "^1.8.0",
|
||||
"core-js": "^2.6.5",
|
||||
"core-js": "^3.0.1",
|
||||
"global": "^4.3.2",
|
||||
"hoist-non-react-statics": "^3.3.0",
|
||||
"memoizerific": "^1.11.3",
|
||||
"react": "^16.8.4",
|
||||
"react-redux": "^7.0.2",
|
||||
"react-sizeme": "^2.5.2",
|
||||
"redux": "^4.0.1",
|
||||
"util-deprecate": "^1.0.2"
|
||||
},
|
||||
|
@ -5,7 +5,7 @@ import { ThemeProvider, themes, convert } from '@storybook/theming';
|
||||
import { STORY_RENDERED } from '@storybook/core-events';
|
||||
import { ScrollArea } from '@storybook/components';
|
||||
|
||||
import { A11YPanel } from './A11YPanel.tsx';
|
||||
import { A11YPanel } from './A11YPanel';
|
||||
import { EVENTS } from '../constants';
|
||||
|
||||
function createApi() {
|
||||
|
@ -6,12 +6,12 @@ import { STORY_RENDERED } from '@storybook/core-events';
|
||||
import { ActionBar, Icons, ScrollArea } from '@storybook/components';
|
||||
|
||||
import { AxeResults, Result } from 'axe-core';
|
||||
import { API } from '@storybook/api';
|
||||
import { Provider } from 'react-redux';
|
||||
import { Report } from './Report';
|
||||
import { Tabs } from './Tabs';
|
||||
import { EVENTS } from '../constants';
|
||||
import { API } from '@storybook/api';
|
||||
|
||||
import { Provider } from 'react-redux';
|
||||
import store, { clearElements } from '../redux-config';
|
||||
|
||||
export enum RuleType {
|
||||
@ -48,7 +48,8 @@ const Incomplete = styled.span(({ theme }) => ({
|
||||
|
||||
const Loader = styled(({ className }) => (
|
||||
<div className={className}>
|
||||
<Icon inline icon="sync" status="running" /> Please wait while a11y scan is running ...
|
||||
<Icon inline icon="sync" status="running" /> Please wait while the accessibility scan is running
|
||||
...
|
||||
</div>
|
||||
))({
|
||||
display: 'flex',
|
||||
@ -175,10 +176,9 @@ export class A11YPanel extends Component<A11YPanelProps, A11YPanelState> {
|
||||
label: <Violations>{violations.length} Violations</Violations>,
|
||||
panel: (
|
||||
<Report
|
||||
passes={false}
|
||||
items={violations}
|
||||
type={RuleType.VIOLATION}
|
||||
empty="No a11y violations found."
|
||||
empty="No accessibility violations found."
|
||||
/>
|
||||
),
|
||||
items: violations,
|
||||
@ -188,10 +188,9 @@ export class A11YPanel extends Component<A11YPanelProps, A11YPanelState> {
|
||||
label: <Passes>{passes.length} Passes</Passes>,
|
||||
panel: (
|
||||
<Report
|
||||
passes
|
||||
items={passes}
|
||||
type={RuleType.PASS}
|
||||
empty="No a11y check passed."
|
||||
empty="No accessibility checks passed."
|
||||
/>
|
||||
),
|
||||
items: passes,
|
||||
@ -201,10 +200,9 @@ export class A11YPanel extends Component<A11YPanelProps, A11YPanelState> {
|
||||
label: <Incomplete>{incomplete.length} Incomplete</Incomplete>,
|
||||
panel: (
|
||||
<Report
|
||||
passes={false}
|
||||
items={incomplete}
|
||||
type={RuleType.INCOMPLETION}
|
||||
empty="No a11y incomplete found."
|
||||
empty="No accessibility checks incomplete."
|
||||
/>
|
||||
),
|
||||
items: incomplete,
|
||||
|
@ -34,7 +34,7 @@ const ColorIcon = styled.span(
|
||||
})
|
||||
);
|
||||
|
||||
// tslint:disable-next-line:no-empty-interface
|
||||
// eslint-disable-next-line @typescript-eslint/no-empty-interface
|
||||
interface ColorBlindnessProps {}
|
||||
|
||||
interface ColorBlindnessState {
|
||||
@ -63,7 +63,8 @@ export class ColorBlindness extends Component<ColorBlindnessProps, ColorBlindnes
|
||||
};
|
||||
|
||||
onVisibilityChange = (s: boolean) => {
|
||||
if (this.state.expanded !== s) {
|
||||
const { expanded } = this.state;
|
||||
if (expanded !== s) {
|
||||
this.setState({ expanded: s });
|
||||
}
|
||||
};
|
||||
@ -83,7 +84,7 @@ export class ColorBlindness extends Component<ColorBlindnessProps, ColorBlindnes
|
||||
'mono',
|
||||
].map(i => ({
|
||||
id: i,
|
||||
title: i,
|
||||
title: i.charAt(0).toUpperCase() + i.slice(1),
|
||||
onClick: () => {
|
||||
this.setFilter(i);
|
||||
},
|
||||
|
@ -14,25 +14,25 @@ const Item = styled.li({
|
||||
const ItemTitle = styled.span(({ theme }) => ({
|
||||
borderBottom: `1px solid ${theme.appBorderColor}`,
|
||||
width: '100%',
|
||||
display: 'inline-block',
|
||||
display: 'flex',
|
||||
paddingBottom: '6px',
|
||||
marginBottom: '6px',
|
||||
justifyContent: 'space-between',
|
||||
}));
|
||||
|
||||
const HighlightToggleElement = styled.span({
|
||||
fontWeight: 'normal',
|
||||
float: 'right',
|
||||
alignSelf: 'center',
|
||||
paddingRight: '15px',
|
||||
input: { margin: 0 },
|
||||
});
|
||||
|
||||
interface ElementProps {
|
||||
element: NodeResult;
|
||||
passes: boolean;
|
||||
type: RuleType;
|
||||
}
|
||||
|
||||
const Element: FunctionComponent<ElementProps> = ({ element, passes, type }) => {
|
||||
const Element: FunctionComponent<ElementProps> = ({ element, type }) => {
|
||||
const { any, all, none } = element;
|
||||
const rules = [...any, ...all, ...none];
|
||||
const highlightToggleId = `${type}-${element.target[0]}`;
|
||||
@ -51,21 +51,21 @@ const Element: FunctionComponent<ElementProps> = ({ element, passes, type }) =>
|
||||
/>
|
||||
</HighlightToggleElement>
|
||||
</ItemTitle>
|
||||
<Rules rules={rules} passes={passes} />
|
||||
<Rules rules={rules} />
|
||||
</Item>
|
||||
);
|
||||
};
|
||||
|
||||
interface ElementsProps {
|
||||
elements: NodeResult[];
|
||||
passes: boolean;
|
||||
type: RuleType;
|
||||
}
|
||||
|
||||
export const Elements: FunctionComponent<ElementsProps> = ({ elements, passes, type }) => (
|
||||
export const Elements: FunctionComponent<ElementsProps> = ({ elements, type }) => (
|
||||
<ol>
|
||||
{elements.map((element, index) => (
|
||||
<Element passes={passes} element={element} key={index} type={type} />
|
||||
// eslint-disable-next-line react/no-array-index-key
|
||||
<Element element={element} key={index} type={type} />
|
||||
))}
|
||||
</ol>
|
||||
);
|
||||
|
@ -2,8 +2,8 @@ import React from 'react';
|
||||
import { mount } from 'enzyme';
|
||||
import { Provider } from 'react-redux';
|
||||
import { ThemeProvider, themes, convert } from '@storybook/theming';
|
||||
import HighlightToggle from './HighlightToggle.tsx';
|
||||
import store from '../../redux-config.tsx';
|
||||
import HighlightToggle from './HighlightToggle';
|
||||
import store from '../../redux-config';
|
||||
|
||||
function ThemedHighlightToggle(props) {
|
||||
return (
|
||||
|
@ -1,3 +1,4 @@
|
||||
import { document } from 'global';
|
||||
import React, { Component } from 'react';
|
||||
import { connect } from 'react-redux';
|
||||
import { styled, themes, convert } from '@storybook/theming';
|
||||
@ -10,6 +11,7 @@ import { IFRAME } from '../../constants';
|
||||
|
||||
export class HighlightedElementData {
|
||||
originalOutline: string;
|
||||
|
||||
isHighlighted: boolean;
|
||||
}
|
||||
|
||||
@ -50,6 +52,7 @@ function getElementBySelectorPath(elementPath: string): HTMLElement {
|
||||
}
|
||||
|
||||
function setElementOutlineStyle(targetElement: HTMLElement, outlineStyle: string): void {
|
||||
// eslint-disable-next-line no-param-reassign
|
||||
targetElement.style.outline = outlineStyle;
|
||||
}
|
||||
|
||||
@ -65,6 +68,7 @@ function areAllRequiredElementsHighlighted(
|
||||
);
|
||||
}).length;
|
||||
|
||||
// eslint-disable-next-line no-nested-ternary
|
||||
return highlightedCount === 0
|
||||
? CheckBoxStates.UNCHECKED
|
||||
: highlightedCount === elementsToHighlight.length
|
||||
@ -99,58 +103,31 @@ class HighlightToggle extends Component<ToggleProps> {
|
||||
private checkBoxRef = React.createRef<HTMLInputElement>();
|
||||
|
||||
componentDidMount() {
|
||||
this.props.elementsToHighlight.forEach(element => {
|
||||
const { elementsToHighlight, highlightedElementsMap } = this.props;
|
||||
elementsToHighlight.forEach(element => {
|
||||
const targetElement = getElementBySelectorPath(element.target[0]);
|
||||
if (targetElement && !this.props.highlightedElementsMap.has(targetElement)) {
|
||||
if (targetElement && !highlightedElementsMap.has(targetElement)) {
|
||||
this.saveElementDataToMap(targetElement, false, targetElement.style.outline);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
componentDidUpdate(prevProps: Readonly<ToggleProps>): void {
|
||||
const { indeterminate } = this.props;
|
||||
if (this.checkBoxRef.current) {
|
||||
this.checkBoxRef.current.indeterminate = this.props.indeterminate;
|
||||
this.checkBoxRef.current.indeterminate = indeterminate;
|
||||
}
|
||||
}
|
||||
|
||||
highlightRuleLocation(targetElement: HTMLElement, addHighlight: boolean): void {
|
||||
if (!targetElement) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (addHighlight) {
|
||||
setElementOutlineStyle(targetElement, `${colorsByType[this.props.type]} dotted 1px`);
|
||||
return;
|
||||
}
|
||||
|
||||
if (this.props.highlightedElementsMap.has(targetElement)) {
|
||||
setElementOutlineStyle(
|
||||
targetElement,
|
||||
this.props.highlightedElementsMap.get(targetElement).originalOutline
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
saveElementDataToMap(
|
||||
targetElement: HTMLElement,
|
||||
isHighlighted: boolean,
|
||||
originalOutline: string
|
||||
): void {
|
||||
const data: HighlightedElementData = new HighlightedElementData();
|
||||
data.isHighlighted = isHighlighted;
|
||||
data.originalOutline = originalOutline;
|
||||
const payload = { element: targetElement, highlightedElementData: data };
|
||||
this.props.addElement(payload);
|
||||
}
|
||||
|
||||
onToggle = (): void => {
|
||||
this.props.elementsToHighlight.forEach(element => {
|
||||
const { elementsToHighlight, highlightedElementsMap } = this.props;
|
||||
elementsToHighlight.forEach(element => {
|
||||
const targetElement = getElementBySelectorPath(element.target[0]);
|
||||
if (!this.props.highlightedElementsMap.has(targetElement)) {
|
||||
if (!highlightedElementsMap.has(targetElement)) {
|
||||
return;
|
||||
}
|
||||
const originalOutline = this.props.highlightedElementsMap.get(targetElement).originalOutline;
|
||||
const { isHighlighted } = this.props.highlightedElementsMap.get(targetElement);
|
||||
const { originalOutline } = highlightedElementsMap.get(targetElement);
|
||||
const { isHighlighted } = highlightedElementsMap.get(targetElement);
|
||||
const { isToggledOn } = this.props;
|
||||
if ((isToggledOn && isHighlighted) || (!isToggledOn && !isHighlighted)) {
|
||||
const addHighlight = !isToggledOn && !isHighlighted;
|
||||
@ -160,16 +137,49 @@ class HighlightToggle extends Component<ToggleProps> {
|
||||
});
|
||||
};
|
||||
|
||||
highlightRuleLocation(targetElement: HTMLElement, addHighlight: boolean): void {
|
||||
const { highlightedElementsMap, type } = this.props;
|
||||
if (!targetElement) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (addHighlight) {
|
||||
setElementOutlineStyle(targetElement, `${colorsByType[type]} dotted 1px`);
|
||||
return;
|
||||
}
|
||||
|
||||
if (highlightedElementsMap.has(targetElement)) {
|
||||
setElementOutlineStyle(
|
||||
targetElement,
|
||||
highlightedElementsMap.get(targetElement).originalOutline
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
saveElementDataToMap(
|
||||
targetElement: HTMLElement,
|
||||
isHighlighted: boolean,
|
||||
originalOutline: string
|
||||
): void {
|
||||
const { addElement: localAddElement } = this.props;
|
||||
const data: HighlightedElementData = new HighlightedElementData();
|
||||
data.isHighlighted = isHighlighted;
|
||||
data.originalOutline = originalOutline;
|
||||
const payload = { element: targetElement, highlightedElementData: data };
|
||||
localAddElement(payload);
|
||||
}
|
||||
|
||||
render() {
|
||||
const { toggleId, elementsToHighlight, isToggledOn } = this.props;
|
||||
return (
|
||||
<Checkbox
|
||||
ref={this.checkBoxRef}
|
||||
id={this.props.toggleId}
|
||||
id={toggleId}
|
||||
type="checkbox"
|
||||
aria-label="Highlight result"
|
||||
disabled={!this.props.elementsToHighlight.length}
|
||||
disabled={!elementsToHighlight.length}
|
||||
onChange={this.onToggle}
|
||||
checked={this.props.isToggledOn}
|
||||
checked={isToggledOn}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
@ -12,6 +12,7 @@ import HighlightToggle from './HighlightToggle';
|
||||
|
||||
const Wrapper = styled.div(({ theme }) => ({
|
||||
display: 'flex',
|
||||
width: '100%',
|
||||
borderBottom: `1px solid ${theme.appBorderColor}`,
|
||||
'&:hover': {
|
||||
background: theme.background.hoverable,
|
||||
@ -49,14 +50,12 @@ const HighlightToggleElement = styled.span({
|
||||
fontWeight: 'normal',
|
||||
float: 'right',
|
||||
marginRight: '15px',
|
||||
marginTop: '10px',
|
||||
|
||||
alignSelf: 'center',
|
||||
input: { margin: 0 },
|
||||
});
|
||||
|
||||
interface ItemProps {
|
||||
item: Result;
|
||||
passes: boolean;
|
||||
type: RuleType;
|
||||
}
|
||||
|
||||
@ -75,7 +74,7 @@ export class Item extends Component<ItemProps, ItemState> {
|
||||
}));
|
||||
|
||||
render() {
|
||||
const { item, passes, type } = this.props;
|
||||
const { item, type } = this.props;
|
||||
const { open } = this.state;
|
||||
const highlightToggleId = `${type}-${item.id}`;
|
||||
|
||||
@ -104,7 +103,7 @@ export class Item extends Component<ItemProps, ItemState> {
|
||||
{open ? (
|
||||
<Fragment>
|
||||
<Info item={item} key="info" />
|
||||
<Elements elements={item.nodes} passes={passes} type={type} key="elements" />
|
||||
<Elements elements={item.nodes} type={type} key="elements" />
|
||||
<Tags tags={item.tags} key="tags" />
|
||||
</Fragment>
|
||||
) : null}
|
||||
|
@ -1,8 +1,9 @@
|
||||
import React, { FunctionComponent } from 'react';
|
||||
import { styled } from '@storybook/theming';
|
||||
|
||||
import { Icons } from '@storybook/components';
|
||||
import { Badge, Icons } from '@storybook/components';
|
||||
import { CheckResult } from 'axe-core';
|
||||
import { SizeMe } from 'react-sizeme';
|
||||
import { RuleType } from '../A11YPanel';
|
||||
|
||||
const impactColors = {
|
||||
minor: '#f1c40f',
|
||||
@ -15,18 +16,33 @@ const impactColors = {
|
||||
const List = styled.div({
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
padding: '4px',
|
||||
paddingBottom: '4px',
|
||||
paddingRight: '4px',
|
||||
paddingTop: '4px',
|
||||
fontWeight: '400',
|
||||
} as any);
|
||||
|
||||
const Item = styled.div({
|
||||
display: 'flex',
|
||||
flexDirection: 'row',
|
||||
marginBottom: '6px',
|
||||
const Item = styled.div(({ elementWidth }: { elementWidth: number }) => {
|
||||
const maxWidthBeforeBreak = 407;
|
||||
return {
|
||||
flexDirection: elementWidth > maxWidthBeforeBreak ? 'row' : 'inherit',
|
||||
marginBottom: elementWidth > maxWidthBeforeBreak ? '6px' : '12px',
|
||||
display: elementWidth > maxWidthBeforeBreak ? 'flex' : 'block',
|
||||
};
|
||||
});
|
||||
|
||||
const StyledBadge = styled(Badge)(({ status }: { status: string }) => ({
|
||||
padding: '2px 8px',
|
||||
marginBottom: '3px',
|
||||
minWidth: '65px',
|
||||
maxWidth: 'fit-content',
|
||||
width: '100%',
|
||||
textAlign: 'center',
|
||||
}));
|
||||
|
||||
const Message = styled.div({
|
||||
paddingLeft: '6px',
|
||||
paddingRight: '23px',
|
||||
});
|
||||
|
||||
const Status = styled.div(({ passes, impact }: { passes: boolean; impact: string }) => ({
|
||||
@ -40,30 +56,64 @@ const Status = styled.div(({ passes, impact }: { passes: boolean; impact: string
|
||||
},
|
||||
}));
|
||||
|
||||
interface RuleProps {
|
||||
rule: CheckResult;
|
||||
passes: boolean;
|
||||
export enum ImpactValue {
|
||||
MINOR = 'minor',
|
||||
MODERATE = 'moderate',
|
||||
SERIOUS = 'serious',
|
||||
CRITICAL = 'critical',
|
||||
}
|
||||
|
||||
const Rule: FunctionComponent<RuleProps> = ({ rule, passes }) => (
|
||||
<Item>
|
||||
<Status passes={passes || undefined} impact={rule.impact}>
|
||||
{passes ? <Icons icon="check" /> : <Icons icon="cross" />}
|
||||
</Status>
|
||||
<Message>{rule.message}</Message>
|
||||
</Item>
|
||||
);
|
||||
interface RuleProps {
|
||||
rule: CheckResult;
|
||||
}
|
||||
|
||||
const formatSeverityText = (severity: string) => {
|
||||
return severity
|
||||
.charAt(0)
|
||||
.toUpperCase()
|
||||
.concat(severity.slice(1));
|
||||
};
|
||||
|
||||
const Rule: FunctionComponent<RuleProps> = ({ rule }) => {
|
||||
let badgeType: any = null;
|
||||
switch (rule.impact) {
|
||||
case ImpactValue.CRITICAL:
|
||||
badgeType = 'critical';
|
||||
break;
|
||||
case ImpactValue.SERIOUS:
|
||||
badgeType = 'negative';
|
||||
break;
|
||||
case ImpactValue.MODERATE:
|
||||
badgeType = 'warning';
|
||||
break;
|
||||
case ImpactValue.MINOR:
|
||||
badgeType = 'neutral';
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
return (
|
||||
<SizeMe refreshMode="debounce">
|
||||
{({ size }: { size: any }) => (
|
||||
<Item elementWidth={size.width}>
|
||||
<StyledBadge status={badgeType}>{formatSeverityText(rule.impact)}</StyledBadge>
|
||||
<Message>{rule.message}</Message>
|
||||
</Item>
|
||||
)}
|
||||
</SizeMe>
|
||||
);
|
||||
};
|
||||
|
||||
interface RulesProps {
|
||||
rules: CheckResult[];
|
||||
passes: boolean;
|
||||
}
|
||||
|
||||
export const Rules: FunctionComponent<RulesProps> = ({ rules, passes }) => {
|
||||
export const Rules: FunctionComponent<RulesProps> = ({ rules }) => {
|
||||
return (
|
||||
<List>
|
||||
{rules.map((rule, index) => (
|
||||
<Rule passes={passes} rule={rule} key={index} />
|
||||
// eslint-disable-next-line react/no-array-index-key
|
||||
<Rule rule={rule} key={index} />
|
||||
))}
|
||||
</List>
|
||||
);
|
||||
|
@ -21,6 +21,7 @@ exports[`HighlightToggle component should match snapshot 1`] = `
|
||||
theme={
|
||||
Object {
|
||||
"addonActionsTheme": Object {
|
||||
"ARROW_ANIMATION_DURATION": "0",
|
||||
"ARROW_COLOR": "rgba(0,0,0,0.3)",
|
||||
"ARROW_FONT_SIZE": 8,
|
||||
"ARROW_MARGIN_RIGHT": 4,
|
||||
@ -37,8 +38,10 @@ exports[`HighlightToggle component should match snapshot 1`] = `
|
||||
"HTML_TAGNAME_TEXT_TRANSFORM": "lowercase",
|
||||
"HTML_TAG_COLOR": "rgb(168, 148, 166)",
|
||||
"OBJECT_NAME_COLOR": "rgb(136, 19, 145)",
|
||||
"OBJECT_PREVIEW_ARRAY_MAX_PROPERTIES": 10,
|
||||
"OBJECT_PREVIEW_OBJECT_MAX_PROPERTIES": 5,
|
||||
"OBJECT_VALUE_BOOLEAN_COLOR": "rgb(28, 0, 207)",
|
||||
"OBJECT_VALUE_FUNCTION_KEYWORD_COLOR": "rgb(170, 13, 145)",
|
||||
"OBJECT_VALUE_FUNCTION_PREFIX_COLOR": "rgb(13, 34, 170)",
|
||||
"OBJECT_VALUE_NULL_COLOR": "rgb(128, 128, 128)",
|
||||
"OBJECT_VALUE_NUMBER_COLOR": "rgb(28, 0, 207)",
|
||||
"OBJECT_VALUE_REGEXP_COLOR": "rgb(196, 26, 22)",
|
||||
@ -125,6 +128,7 @@ exports[`HighlightToggle component should match snapshot 1`] = `
|
||||
"app": "#F6F9FC",
|
||||
"bar": "#FFFFFF",
|
||||
"content": "#FFFFFF",
|
||||
"critical": "#FF4400",
|
||||
"gridCellSize": 10,
|
||||
"hoverable": "rgba(0,0,0,.05)",
|
||||
"negative": "#FEDED2",
|
||||
@ -256,6 +260,7 @@ exports[`HighlightToggle component should match snapshot 1`] = `
|
||||
"color": Object {
|
||||
"ancillary": "#22a699",
|
||||
"border": "rgba(0,0,0,.1)",
|
||||
"critical": "#FFFFFF",
|
||||
"dark": "#666666",
|
||||
"darker": "#444444",
|
||||
"darkest": "#333333",
|
||||
|
@ -7,14 +7,13 @@ import { RuleType } from '../A11YPanel';
|
||||
export interface ReportProps {
|
||||
items: Result[];
|
||||
empty: string;
|
||||
passes: boolean;
|
||||
type: RuleType;
|
||||
}
|
||||
|
||||
export const Report: FunctionComponent<ReportProps> = ({ items, empty, type, passes }) => (
|
||||
export const Report: FunctionComponent<ReportProps> = ({ items, empty, type }) => (
|
||||
<Fragment>
|
||||
{items && items.length ? (
|
||||
items.map(item => <Item passes={passes} item={item} key={`${type}:${item.id}`} type={type} />)
|
||||
items.map(item => <Item item={item} key={`${type}:${item.id}`} type={type} />)
|
||||
) : (
|
||||
<Placeholder key="placeholder">{empty}</Placeholder>
|
||||
)}
|
||||
|
@ -1,13 +1,13 @@
|
||||
import React, { Component, SyntheticEvent } from 'react';
|
||||
|
||||
import { styled } from '@storybook/theming';
|
||||
import { styled, themes } from '@storybook/theming';
|
||||
import { NodeResult, Result } from 'axe-core';
|
||||
import { SizeMe } from 'react-sizeme';
|
||||
import store, { clearElements } from '../redux-config';
|
||||
import HighlightToggle from './Report/HighlightToggle';
|
||||
import { NodeResult, Result } from 'axe-core';
|
||||
import { RuleType } from './A11YPanel';
|
||||
|
||||
// TODO: reuse the Tabs component from @storybook/theming instead
|
||||
// of re-building identical functionality
|
||||
// TODO: reuse the Tabs component from @storybook/theming instead of re-building identical functionality
|
||||
|
||||
const Container = styled.div({
|
||||
width: '100%',
|
||||
@ -18,26 +18,34 @@ const Container = styled.div({
|
||||
const HighlightToggleLabel = styled.label(({ theme }) => ({
|
||||
cursor: 'pointer',
|
||||
userSelect: 'none',
|
||||
marginBottom: '3px',
|
||||
marginRight: '3px',
|
||||
color: theme.color.dark,
|
||||
}));
|
||||
|
||||
const GlobalToggleWrapper = styled.div(({ theme }) => ({
|
||||
padding: '10px 15px 10px 0',
|
||||
cursor: 'pointer',
|
||||
fontSize: theme.typography.size.s2 - 1,
|
||||
height: 40,
|
||||
border: 'none',
|
||||
const GlobalToggle = styled.div(({ elementWidth }: { elementWidth: number }) => {
|
||||
const maxWidthBeforeBreak = 450;
|
||||
return {
|
||||
cursor: 'pointer',
|
||||
fontSize: '14px',
|
||||
padding: elementWidth > maxWidthBeforeBreak ? '12px 15px 10px 0' : '12px 0px 3px 12px',
|
||||
height: '40px',
|
||||
border: 'none',
|
||||
marginTop: elementWidth > maxWidthBeforeBreak ? '-40px' : '0px',
|
||||
float: elementWidth > maxWidthBeforeBreak ? 'right' : 'left',
|
||||
display: elementWidth > maxWidthBeforeBreak ? 'flex' : 'block',
|
||||
alignItems: 'center',
|
||||
width: elementWidth > maxWidthBeforeBreak ? 'auto' : '100%',
|
||||
borderBottom: elementWidth > maxWidthBeforeBreak ? 'none' : '1px solid rgba(0,0,0,.1)',
|
||||
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
|
||||
input: {
|
||||
marginLeft: 10,
|
||||
marginRight: 0,
|
||||
marginTop: 0,
|
||||
marginBottom: 0,
|
||||
},
|
||||
}));
|
||||
input: {
|
||||
marginLeft: '10',
|
||||
marginRight: '0',
|
||||
marginTop: '0',
|
||||
marginBottom: '0',
|
||||
},
|
||||
};
|
||||
});
|
||||
|
||||
const Item = styled.button(
|
||||
({ theme }) => ({
|
||||
@ -78,12 +86,12 @@ const List = styled.div(({ theme }) => ({
|
||||
}));
|
||||
|
||||
interface TabsProps {
|
||||
tabs: Array<{
|
||||
tabs: {
|
||||
label: JSX.Element;
|
||||
panel: JSX.Element;
|
||||
items: Result[];
|
||||
type: RuleType;
|
||||
}>;
|
||||
}[];
|
||||
}
|
||||
|
||||
interface TabsState {
|
||||
@ -113,34 +121,41 @@ export class Tabs extends Component<TabsProps, TabsState> {
|
||||
const highlightToggleId = `${tabs[active].type}-global-checkbox`;
|
||||
const highlightLabel = `Highlight results`;
|
||||
return (
|
||||
<Container>
|
||||
<List>
|
||||
<TabsWrapper>
|
||||
{tabs.map((tab, index) => (
|
||||
<Item
|
||||
key={index}
|
||||
data-index={index}
|
||||
active={active === index}
|
||||
onClick={this.onToggle}
|
||||
>
|
||||
{tab.label}
|
||||
</Item>
|
||||
))}
|
||||
</TabsWrapper>
|
||||
<GlobalToggleWrapper>
|
||||
<HighlightToggleLabel htmlFor={highlightToggleId}>
|
||||
{highlightLabel}
|
||||
</HighlightToggleLabel>
|
||||
<HighlightToggle
|
||||
toggleId={highlightToggleId}
|
||||
type={tabs[active].type}
|
||||
elementsToHighlight={retrieveAllNodesFromResults(tabs[active].items)}
|
||||
label={highlightLabel}
|
||||
/>
|
||||
</GlobalToggleWrapper>
|
||||
</List>
|
||||
{tabs[active].panel}
|
||||
</Container>
|
||||
<SizeMe refreshMode="debounce">
|
||||
{({ size }: { size: any }) => (
|
||||
<Container>
|
||||
<List>
|
||||
<TabsWrapper>
|
||||
{tabs.map((tab, index) => (
|
||||
<Item
|
||||
/* eslint-disable-next-line react/no-array-index-key */
|
||||
key={index}
|
||||
data-index={index}
|
||||
active={active === index}
|
||||
onClick={this.onToggle}
|
||||
>
|
||||
{tab.label}
|
||||
</Item>
|
||||
))}
|
||||
</TabsWrapper>
|
||||
</List>
|
||||
{tabs[active].items.length > 0 ? (
|
||||
<GlobalToggle elementWidth={size.width}>
|
||||
<HighlightToggleLabel htmlFor={highlightToggleId}>
|
||||
{highlightLabel}
|
||||
</HighlightToggleLabel>
|
||||
<HighlightToggle
|
||||
toggleId={highlightToggleId}
|
||||
type={tabs[active].type}
|
||||
elementsToHighlight={retrieveAllNodesFromResults(tabs[active].items)}
|
||||
label={highlightLabel}
|
||||
/>
|
||||
</GlobalToggle>
|
||||
) : null}
|
||||
{tabs[active].panel}
|
||||
</Container>
|
||||
)}
|
||||
</SizeMe>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -1,3 +1,4 @@
|
||||
/* eslint-disable @typescript-eslint/no-object-literal-type-assertion */
|
||||
import { document } from 'global';
|
||||
import axe, { AxeResults, ElementContext, RunOptions, Spec } from 'axe-core';
|
||||
import deprecate from 'util-deprecate';
|
||||
@ -35,7 +36,6 @@ const run = (element: ElementContext, config: Spec, options: RunOptions) => {
|
||||
.run(
|
||||
element || getElement(),
|
||||
options ||
|
||||
// tslint:disable-next-line:no-object-literal-type-assertion
|
||||
({
|
||||
restoreScroll: true,
|
||||
} as RunOptions) // cast to RunOptions is necessary because axe types are not up to date
|
||||
|
@ -28,8 +28,10 @@ function rootReducer(state = initialState, action: any) {
|
||||
action.payload.highlightedElementData
|
||||
),
|
||||
};
|
||||
} else if (action.type === CLEAR_ELEMENTS) {
|
||||
for (let key of Array.from(state.highlightedElementsMap.keys())) {
|
||||
}
|
||||
if (action.type === CLEAR_ELEMENTS) {
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const key of Array.from(state.highlightedElementsMap.keys())) {
|
||||
key.style.outline = state.highlightedElementsMap.get(key).originalOutline;
|
||||
state.highlightedElementsMap.delete(key);
|
||||
}
|
||||
|
@ -1,10 +1,10 @@
|
||||
import React, { Fragment, FunctionComponent } from 'react';
|
||||
import { styled } from '@storybook/theming';
|
||||
|
||||
import { addons, types } from '@storybook/addons';
|
||||
import { ADDON_ID, PANEL_ID } from './constants';
|
||||
import { ColorBlindness } from './components/ColorBlindness';
|
||||
import { A11YPanel } from './components/A11YPanel';
|
||||
import { addons, types } from '@storybook/addons';
|
||||
|
||||
const Hidden = styled.div(() => ({
|
||||
'&, & svg': {
|
||||
|
1
addons/a11y/src/typings.d.ts
vendored
1
addons/a11y/src/typings.d.ts
vendored
@ -1 +1,2 @@
|
||||
declare module 'global';
|
||||
declare module 'react-sizeme';
|
||||
|
@ -4,7 +4,7 @@ Storybook Addon Actions can be used to display data received by event handlers i
|
||||
|
||||
[Framework Support](https://github.com/storybooks/storybook/blob/master/ADDONS_SUPPORT.md)
|
||||
|
||||

|
||||

|
||||
|
||||
## Getting Started
|
||||
|
||||
|
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@storybook/addon-actions",
|
||||
"version": "5.1.0-alpha.33",
|
||||
"version": "5.1.0-rc.0",
|
||||
"description": "Action Logger addon for storybook",
|
||||
"keywords": [
|
||||
"storybook"
|
||||
@ -21,23 +21,23 @@
|
||||
"prepare": "node ../../scripts/prepare.js"
|
||||
},
|
||||
"dependencies": {
|
||||
"@storybook/addons": "5.1.0-alpha.33",
|
||||
"@storybook/api": "5.1.0-alpha.33",
|
||||
"@storybook/components": "5.1.0-alpha.33",
|
||||
"@storybook/core-events": "5.1.0-alpha.33",
|
||||
"@storybook/theming": "5.1.0-alpha.33",
|
||||
"core-js": "^2.6.5",
|
||||
"@storybook/addons": "5.1.0-rc.0",
|
||||
"@storybook/api": "5.1.0-rc.0",
|
||||
"@storybook/components": "5.1.0-rc.0",
|
||||
"@storybook/core-events": "5.1.0-rc.0",
|
||||
"@storybook/theming": "5.1.0-rc.0",
|
||||
"core-js": "^3.0.1",
|
||||
"fast-deep-equal": "^2.0.1",
|
||||
"global": "^4.3.2",
|
||||
"lodash": "^4.17.11",
|
||||
"polished": "^3.0.0",
|
||||
"polished": "^3.3.1",
|
||||
"prop-types": "^15.7.2",
|
||||
"react": "^16.8.4",
|
||||
"react-inspector": "^2.3.1",
|
||||
"react-inspector": "^3.0.2",
|
||||
"uuid": "^3.3.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/lodash": "^4.14.123",
|
||||
"@types/lodash": "^4.14.129",
|
||||
"@types/uuid": "^3.4.4"
|
||||
},
|
||||
"publishConfig": {
|
||||
|
@ -1,4 +1,5 @@
|
||||
export * from './constants';
|
||||
export * from './models';
|
||||
export * from './preview';
|
||||
|
||||
if (module && module.hot && module.hot.decline) {
|
||||
|
@ -8,8 +8,7 @@ export function action(name: string, options: ActionOptions = {}): HandlerFuncti
|
||||
...options,
|
||||
};
|
||||
|
||||
// tslint:disable-next-line:no-shadowed-variable
|
||||
const handler = function action(...args: any[]) {
|
||||
const handler = function actionHandler(...args: any[]) {
|
||||
const channel = addons.getChannel();
|
||||
const id = uuid();
|
||||
const minDepth = 5; // anything less is really just storybook internals
|
||||
|
@ -30,7 +30,6 @@ const hasMatchInAncestry = (element: any, selector: any): boolean => {
|
||||
const createHandlers = (actionsFn: (...arg: any[]) => object, ...args: any[]) => {
|
||||
const actionsObject = actionsFn(...args);
|
||||
return Object.entries(actionsObject).map(([key, action]) => {
|
||||
// eslint-disable-next-line no-unused-vars
|
||||
const [_, eventName, selector] = key.match(delegateEventSplitter);
|
||||
return {
|
||||
eventName,
|
||||
|
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@storybook/addon-backgrounds",
|
||||
"version": "5.1.0-alpha.33",
|
||||
"version": "5.1.0-rc.0",
|
||||
"description": "A storybook addon to show different backgrounds for your preview",
|
||||
"keywords": [
|
||||
"addon",
|
||||
@ -25,13 +25,13 @@
|
||||
"prepare": "node ../../scripts/prepare.js"
|
||||
},
|
||||
"dependencies": {
|
||||
"@storybook/addons": "5.1.0-alpha.33",
|
||||
"@storybook/api": "5.1.0-alpha.33",
|
||||
"@storybook/client-logger": "5.1.0-alpha.33",
|
||||
"@storybook/components": "5.1.0-alpha.33",
|
||||
"@storybook/core-events": "5.1.0-alpha.33",
|
||||
"@storybook/theming": "5.1.0-alpha.33",
|
||||
"core-js": "^2.6.5",
|
||||
"@storybook/addons": "5.1.0-rc.0",
|
||||
"@storybook/api": "5.1.0-rc.0",
|
||||
"@storybook/client-logger": "5.1.0-rc.0",
|
||||
"@storybook/components": "5.1.0-rc.0",
|
||||
"@storybook/core-events": "5.1.0-rc.0",
|
||||
"@storybook/theming": "5.1.0-rc.0",
|
||||
"core-js": "^3.0.1",
|
||||
"memoizerific": "^1.11.3",
|
||||
"react": "^16.8.4",
|
||||
"util-deprecate": "^1.0.2"
|
||||
|
@ -103,12 +103,14 @@ export class BackgroundSelector extends Component<{ api: API }, State> {
|
||||
};
|
||||
|
||||
change = ({ selected, name }: { selected: string; name: string }) => {
|
||||
this.props.api.emit(EVENTS.UPDATE, { selected, name });
|
||||
const { api } = this.props;
|
||||
api.emit(EVENTS.UPDATE, { selected, name });
|
||||
this.setState({ selected, expanded: false });
|
||||
};
|
||||
|
||||
onVisibilityChange = (s: boolean) => {
|
||||
if (this.state.expanded !== s) {
|
||||
const { expanded } = this.state;
|
||||
if (expanded !== s) {
|
||||
this.setState({ expanded: s });
|
||||
}
|
||||
};
|
||||
|
1
addons/centered/angular.d.ts
vendored
1
addons/centered/angular.d.ts
vendored
@ -1,4 +1,3 @@
|
||||
// tslint:disable-next-line:no-implicit-dependencies
|
||||
export interface ICollection {
|
||||
[p: string]: any;
|
||||
}
|
||||
|
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@storybook/addon-centered",
|
||||
"version": "5.1.0-alpha.33",
|
||||
"version": "5.1.0-rc.0",
|
||||
"description": "Storybook decorator to center components",
|
||||
"keywords": [
|
||||
"addon",
|
||||
@ -18,15 +18,23 @@
|
||||
"license": "MIT",
|
||||
"author": "Muhammed Thanish <mnmtanish@gmail.com>",
|
||||
"main": "dist/index.js",
|
||||
"jsnext:main": "src/index.js",
|
||||
"types": "dist/index.d.ts",
|
||||
"scripts": {
|
||||
"prepare": "node ../../scripts/prepare.js"
|
||||
},
|
||||
"dependencies": {
|
||||
"core-js": "^2.6.5",
|
||||
"core-js": "^3.0.1",
|
||||
"global": "^4.3.2",
|
||||
"util-deprecate": "^1.0.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/mithril": "^1.1.16"
|
||||
},
|
||||
"optionalDependencies": {
|
||||
"mithril": "*",
|
||||
"preact": "*",
|
||||
"react": "*"
|
||||
},
|
||||
"publishConfig": {
|
||||
"access": "public"
|
||||
}
|
||||
|
@ -1,11 +1,11 @@
|
||||
import styles from './styles';
|
||||
|
||||
function getComponentSelector(component) {
|
||||
function getComponentSelector(component: any) {
|
||||
// eslint-disable-next-line no-underscore-dangle
|
||||
return component.__annotations__[0].selector;
|
||||
}
|
||||
|
||||
function getTemplate(metadata) {
|
||||
function getTemplate(metadata: any) {
|
||||
let tpl = '';
|
||||
if (metadata.component) {
|
||||
const selector = getComponentSelector(metadata.component);
|
||||
@ -24,7 +24,7 @@ function getTemplate(metadata) {
|
||||
</div>`;
|
||||
}
|
||||
|
||||
function getModuleMetadata(metadata) {
|
||||
function getModuleMetadata(metadata: any) {
|
||||
const { moduleMetadata, component } = metadata;
|
||||
|
||||
if (component && !moduleMetadata) {
|
||||
@ -43,7 +43,7 @@ function getModuleMetadata(metadata) {
|
||||
return moduleMetadata;
|
||||
}
|
||||
|
||||
export default function(metadataFn) {
|
||||
export default function(metadataFn: any) {
|
||||
const metadata = metadataFn();
|
||||
|
||||
return {
|
@ -5,12 +5,6 @@
|
||||
</div>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
data() {
|
||||
return {
|
||||
style: '',
|
||||
innerStyle: ''
|
||||
};
|
||||
}
|
||||
};
|
||||
export let style = '';
|
||||
export let innerStyle = '';
|
||||
</script>
|
||||
|
@ -1,7 +1,7 @@
|
||||
import { document } from 'global';
|
||||
import styles from './styles';
|
||||
|
||||
export default function(storyFn) {
|
||||
export default function(storyFn: () => { template: any; context: any }) {
|
||||
const { template, context } = storyFn();
|
||||
|
||||
const element = document.createElement('div');
|
||||
@ -13,7 +13,7 @@ export default function(storyFn) {
|
||||
element.appendChild(innerElement);
|
||||
|
||||
// the inner element should append the parent
|
||||
innerElement.appendTo = function appendTo(el) {
|
||||
innerElement.appendTo = function appendTo(el: any) {
|
||||
el.appendChild(element);
|
||||
};
|
||||
|
@ -10,11 +10,11 @@ import { document } from 'global';
|
||||
* @returns {string}
|
||||
* @see https://stackoverflow.com/questions/38533544/jsx-css-to-inline-styles
|
||||
*/
|
||||
export default function jsonToCss(jsonStyles) {
|
||||
const frag = document.createElement('div');
|
||||
export default function jsonToCss(jsonStyles: Partial<CSSStyleDeclaration>) {
|
||||
const frag = document.createElement('div') as HTMLDivElement;
|
||||
|
||||
Object.keys(jsonStyles).forEach(key => {
|
||||
frag.style[key] = jsonStyles[key];
|
||||
(frag.style as any)[key] = (jsonStyles as any)[key];
|
||||
});
|
||||
|
||||
return frag.getAttribute('style');
|
@ -4,14 +4,14 @@ import styles from './styles';
|
||||
const INNER_ID = 'sb-addon-centered-inner';
|
||||
const WRAPPER_ID = 'sb-addon-centered-wrapper';
|
||||
|
||||
function getOrCreate(id, style) {
|
||||
function getOrCreate(id: string, style: Partial<CSSStyleDeclaration>): HTMLDivElement {
|
||||
const elementOnDom = document.getElementById(id);
|
||||
|
||||
if (elementOnDom) {
|
||||
return elementOnDom;
|
||||
}
|
||||
|
||||
const element = document.createElement('div');
|
||||
const element = document.createElement('div') as HTMLDivElement;
|
||||
element.setAttribute('id', id);
|
||||
Object.assign(element.style, style);
|
||||
|
||||
@ -26,7 +26,7 @@ function getWrapperDiv() {
|
||||
return getOrCreate(WRAPPER_ID, styles.style);
|
||||
}
|
||||
|
||||
export default function(storyFn) {
|
||||
export default function(storyFn: () => any) {
|
||||
const inner = getInnerDiv();
|
||||
const wrapper = getWrapperDiv();
|
||||
wrapper.appendChild(inner);
|
@ -1,10 +1,8 @@
|
||||
/** @jsx m */
|
||||
|
||||
// eslint-disable-next-line import/no-extraneous-dependencies
|
||||
import m from 'mithril';
|
||||
import m, { ComponentTypes } from 'mithril';
|
||||
import styles from './styles';
|
||||
|
||||
export default function(storyFn) {
|
||||
export default function(storyFn: () => ComponentTypes) {
|
||||
return {
|
||||
view: () => (
|
||||
<div style={styles.style}>
|
@ -1,9 +1,8 @@
|
||||
/** @jsx h */
|
||||
// eslint-disable-next-line import/no-extraneous-dependencies
|
||||
import { h } from 'preact';
|
||||
import { Component, h } from 'preact';
|
||||
import styles from './styles';
|
||||
|
||||
export default function(storyFn) {
|
||||
export default function(storyFn: () => Component) {
|
||||
return (
|
||||
<div style={styles.style}>
|
||||
<div style={styles.innerStyle}>{storyFn()}</div>
|
@ -1,8 +1,7 @@
|
||||
// eslint-disable-next-line import/no-extraneous-dependencies
|
||||
import React from 'react';
|
||||
import React, { ReactNode } from 'react';
|
||||
import styles from './styles';
|
||||
|
||||
export default function(storyFn) {
|
||||
export default function(storyFn: () => ReactNode) {
|
||||
return (
|
||||
<div style={styles.style}>
|
||||
<div style={styles.innerStyle}>{storyFn()}</div>
|
@ -1,10 +1,10 @@
|
||||
const styles = {
|
||||
style: {
|
||||
position: 'fixed',
|
||||
top: 0,
|
||||
left: 0,
|
||||
bottom: 0,
|
||||
right: 0,
|
||||
top: '0',
|
||||
left: '0',
|
||||
bottom: '0',
|
||||
right: '0',
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
overflow: 'auto',
|
||||
@ -14,6 +14,6 @@ const styles = {
|
||||
maxHeight: '100%', // Hack for centering correctly in IE11
|
||||
overflow: 'auto',
|
||||
},
|
||||
};
|
||||
} as const;
|
||||
|
||||
export default styles;
|
@ -18,10 +18,16 @@ const centeredStyles = {
|
||||
*
|
||||
* @see https://svelte.technology/guide#svelte-component
|
||||
*/
|
||||
export default function(storyFn) {
|
||||
const { Component: OriginalComponent, data, on } = storyFn();
|
||||
export default function(storyFn: () => any) {
|
||||
const { Component: OriginalComponent, props, on } = storyFn();
|
||||
|
||||
return { Component: OriginalComponent, data, on, Wrapper: Centered, WrapperData: centeredStyles };
|
||||
return {
|
||||
Component: OriginalComponent,
|
||||
props,
|
||||
on,
|
||||
Wrapper: Centered,
|
||||
WrapperData: centeredStyles,
|
||||
};
|
||||
}
|
||||
|
||||
if (module && module.hot && module.hot.decline) {
|
2
addons/centered/src/typings.d.ts
vendored
Normal file
2
addons/centered/src/typings.d.ts
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
declare module 'global';
|
||||
declare module '*.svelte';
|
13
addons/centered/tsconfig.json
Normal file
13
addons/centered/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"rootDir": "./src",
|
||||
"types": ["webpack-env"]
|
||||
},
|
||||
"include": [
|
||||
"src/**/*"
|
||||
],
|
||||
"exclude": [
|
||||
"src/__tests__/**/*"
|
||||
]
|
||||
}
|
@ -31,11 +31,13 @@ once then apply it everywhere**.
|
||||
use it to bridge with your favorite routing, state-management solutions, or even your own
|
||||
[React Context](https://reactjs.org/docs/context.html) provider.
|
||||
4. Offer chainable and granular configurations. It is even possible to fine-tune at per story level.
|
||||
5. Visual regression friendly. You can use this addon to driving the same story under different contexts to smoke
|
||||
testing important visual states.
|
||||
|
||||
## 🧰 Requirements
|
||||
|
||||
Make sure the version of your Storybook is above v5. Currently, this addon supports the following frameworks:
|
||||
**React**, and **Vue**. Other frameworks might get support in the near future (PRs are welcome!).
|
||||
Make sure the version of your Storybook is above v5. For the full list the current supported framework, see
|
||||
[Addon / Framework Support Table](../../ADDONS_SUPPORT.md).
|
||||
|
||||
## 🎬 Getting started
|
||||
|
||||
@ -56,8 +58,8 @@ To load your contextual setups for your stories globally, adding the following l
|
||||
see it near your `addon.js` file):
|
||||
|
||||
```js
|
||||
import { addDecorator } from '@storybook/react'; // or '@storybook/vue'
|
||||
import { withContexts } from '@storybook/addon-contexts/react'; // or '@storybook/addon-contexts/vue'
|
||||
import { addDecorator } from '@storybook/[framework]';
|
||||
import { withContexts } from '@storybook/addon-contexts/[framework]';
|
||||
import { contexts } from './configs/contexts'; // we will define the contextual setups later in API section
|
||||
|
||||
addDecorator(withContexts(contexts));
|
||||
@ -66,8 +68,8 @@ addDecorator(withContexts(contexts));
|
||||
Alternatively, just like other addons, you can use this addon only for a given set of stories:
|
||||
|
||||
```js
|
||||
import { storiesOf } from '@storybook/react'; // or '@storybook/vue'
|
||||
import { withContexts } from '@storybook/addon-contexts/react'; // or '@storybook/addon-contexts/vue'
|
||||
import { storiesOf } from '@storybook/[framework]';
|
||||
import { withContexts } from '@storybook/addon-contexts/[framework]';
|
||||
import { contexts } from './configs/contexts';
|
||||
|
||||
const story = storiesOf('Component With Contexts', module).addDecorator(withContexts(contexts)); // use this addon with a default contextual environment setups
|
||||
@ -227,6 +229,9 @@ be shown at first in the toolbar menu in your Storybook.
|
||||
4. The addon will persist the selected params (the addon state) between stories at run-time (similar to other
|
||||
addons). If the active param were gone after story switching, it fallback to the default then the first. As a
|
||||
rule of thumbs, whenever collisions made possible, always the first wins.
|
||||
5. Query parameters are supported for pre-selecting contexts param, which comes handy for visual regression testing.
|
||||
You can do this by appending `&contexts=[name of contexts]=[name of param]` in the URL under iframe mode. Use `,`
|
||||
to separate multiple contexts (e.g. `&contexts=Theme=Forests,Language=Fr`).
|
||||
|
||||
## 📖 License
|
||||
|
||||
|
@ -1,19 +1,20 @@
|
||||
{
|
||||
"name": "@storybook/addon-contexts",
|
||||
"version": "5.1.0-alpha.33",
|
||||
"version": "5.1.0-rc.0",
|
||||
"description": "Storybook Addon Contexts",
|
||||
"keywords": [
|
||||
"storybook",
|
||||
"preact",
|
||||
"react",
|
||||
"vue"
|
||||
],
|
||||
"author": "Leo Y. Li",
|
||||
"license": "MIT",
|
||||
"main": "dist/index.js",
|
||||
"types": "dist/index.d.ts",
|
||||
"main": "dist/register.js",
|
||||
"files": [
|
||||
"dist/**/*",
|
||||
"register.js",
|
||||
"preact.js",
|
||||
"react.js",
|
||||
"vue.js"
|
||||
],
|
||||
@ -23,15 +24,21 @@
|
||||
"directory": "addons/contexts"
|
||||
},
|
||||
"scripts": {
|
||||
"prepare": "node ../../scripts/prepare.js"
|
||||
"prepare": "node ../../scripts/prepare.js",
|
||||
"dev:check-types": "tsc --noEmit"
|
||||
},
|
||||
"dependencies": {
|
||||
"@storybook/addons": "5.1.0-alpha.33",
|
||||
"@storybook/channels": "5.1.0-alpha.33",
|
||||
"@storybook/components": "5.1.0-alpha.33",
|
||||
"@storybook/core-events": "5.1.0-alpha.33"
|
||||
"@storybook/addons": "5.1.0-rc.0",
|
||||
"@storybook/api": "5.1.0-rc.0",
|
||||
"@storybook/components": "5.1.0-rc.0",
|
||||
"@storybook/core-events": "5.1.0-rc.0"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"global": "*",
|
||||
"qs": "*"
|
||||
},
|
||||
"optionalDependencies": {
|
||||
"preact": "*",
|
||||
"react": "*",
|
||||
"vue": "*"
|
||||
},
|
||||
|
4
addons/contexts/preact.js
Normal file
4
addons/contexts/preact.js
Normal file
@ -0,0 +1,4 @@
|
||||
import { withContexts } from './dist/preview/frameworks/preact';
|
||||
|
||||
export { withContexts };
|
||||
export default withContexts;
|
@ -1,7 +1,38 @@
|
||||
import { withContexts } from './preview/frameworks/react';
|
||||
export { withContexts };
|
||||
export default withContexts;
|
||||
import { makeDecorator, StoryWrapper } from '@storybook/addons';
|
||||
import { ContextsPreviewAPI } from './preview/ContextsPreviewAPI';
|
||||
import { ID, PARAM } from './shared/constants';
|
||||
import { AddonSetting, AnyFunctionReturns, ContextNode, PropsMap } from './shared/types.d';
|
||||
|
||||
console.error(
|
||||
`[addon-contexts] Deprecation warning: "import { withContexts } from 'addon-contexts'" has been deprecated. Please import from 'addon-contexts/react' instead.`
|
||||
);
|
||||
/**
|
||||
* This file serves a idiomatic facade of a Storybook decorator.
|
||||
*
|
||||
* Wrapper function get called whenever the Storybook rerender the view. This reflow logic is
|
||||
* framework agnostic; on the other hand, the framework specific bindings are the implementation
|
||||
* details hidden behind the passed `render` function.
|
||||
*
|
||||
* Here, we need a dedicated singleton as a state manager for preview (the addon API, in vanilla)
|
||||
* who is also knowing how to communicate with the Storybook manager (in React) via the Storybook
|
||||
* event system.
|
||||
*
|
||||
* @param {Render} render - framework specific bindings
|
||||
*/
|
||||
export type Render<T> = (...args: [ContextNode[], PropsMap, AnyFunctionReturns<T>]) => T;
|
||||
type CreateAddonDecorator = <T>(render: Render<T>) => (contexts: AddonSetting[]) => unknown;
|
||||
|
||||
export const createAddonDecorator: CreateAddonDecorator = render => {
|
||||
const wrapper: StoryWrapper = (getStory, context, settings: any) => {
|
||||
const { getContextNodes, getSelectionState, getPropsMap } = ContextsPreviewAPI();
|
||||
const nodes = getContextNodes(settings);
|
||||
const state = getSelectionState();
|
||||
const props = getPropsMap(nodes, state);
|
||||
return render(nodes, props, () => getStory(context));
|
||||
};
|
||||
|
||||
return makeDecorator({
|
||||
name: ID,
|
||||
parameterName: PARAM,
|
||||
skipIfNoParametersOrOptions: true,
|
||||
allowDeprecatedUsage: false,
|
||||
wrapper,
|
||||
});
|
||||
};
|
||||
|
@ -1,31 +0,0 @@
|
||||
import React, { useEffect, useState, useCallback } from 'react';
|
||||
import { useChannel, Channel } from './libs/useChannel';
|
||||
import { ToolBar } from './ToolBar';
|
||||
import { REBOOT_MANAGER, UPDATE_MANAGER, UPDATE_PREVIEW } from '../constants';
|
||||
import { SelectionState, FCNoChildren } from '../types';
|
||||
|
||||
/**
|
||||
* Control addon states and addon-story interactions
|
||||
*/
|
||||
type AddonManager = FCNoChildren<{
|
||||
channel: Channel;
|
||||
}>;
|
||||
|
||||
export const AddonManager: AddonManager = ({ channel }) => {
|
||||
const [nodes, setNodes] = useState([]);
|
||||
const [state, setState] = useState<SelectionState>(undefined);
|
||||
const setSelected = useCallback(
|
||||
(nodeId, name) => setState((obj = {}) => ({ ...obj, [nodeId]: name })),
|
||||
[]
|
||||
);
|
||||
|
||||
// from preview
|
||||
useChannel(UPDATE_MANAGER, newNodes => setNodes(newNodes), []);
|
||||
useChannel(UPDATE_MANAGER, (_, newState) => newState && setState(newState), []);
|
||||
|
||||
// to preview
|
||||
useEffect(() => channel.emit(REBOOT_MANAGER), []);
|
||||
useEffect(() => state && channel.emit(UPDATE_PREVIEW, state), [state]);
|
||||
|
||||
return <ToolBar nodes={nodes} state={state || {}} setSelected={setSelected} />;
|
||||
};
|
32
addons/contexts/src/manager/ContextsManager.tsx
Normal file
32
addons/contexts/src/manager/ContextsManager.tsx
Normal file
@ -0,0 +1,32 @@
|
||||
import React, { useEffect, useState, useCallback } from 'react';
|
||||
import { useChannel } from './libs/useChannel';
|
||||
import { ToolBar } from './components/ToolBar';
|
||||
import { deserialize, serialize } from '../shared/serializers';
|
||||
import { PARAM, REBOOT_MANAGER, UPDATE_MANAGER, UPDATE_PREVIEW } from '../shared/constants';
|
||||
import { FCNoChildren, ManagerAPI } from '../shared/types.d';
|
||||
|
||||
/**
|
||||
* A smart component for handling manager-preview interactions.
|
||||
*/
|
||||
type ContextsManager = FCNoChildren<{
|
||||
api: ManagerAPI;
|
||||
}>;
|
||||
|
||||
export const ContextsManager: ContextsManager = ({ api }) => {
|
||||
const [nodes, setNodes] = useState([]);
|
||||
const [state, setState] = useState(deserialize(api.getQueryParam(PARAM)));
|
||||
const setSelected = useCallback(
|
||||
(nodeId, name) => setState(obj => ({ ...obj, [nodeId]: name })),
|
||||
[]
|
||||
);
|
||||
|
||||
// from preview
|
||||
useChannel(UPDATE_MANAGER, newNodes => setNodes(newNodes), []);
|
||||
|
||||
// to preview
|
||||
useEffect(() => api.emit(REBOOT_MANAGER), []);
|
||||
useEffect(() => api.emit(UPDATE_PREVIEW, state), [state]);
|
||||
useEffect(() => api.setQueryParams({ [PARAM]: serialize(state) }), [state]);
|
||||
|
||||
return <ToolBar nodes={nodes} state={state || {}} setSelected={setSelected} />;
|
||||
};
|
@ -1,7 +1,7 @@
|
||||
import React, { ComponentProps } from 'react';
|
||||
import { Separator } from '@storybook/components';
|
||||
import { ToolbarControl } from './ToolbarControl';
|
||||
import { ContextNode, FCNoChildren, SelectionState } from '../types';
|
||||
import { ContextNode, FCNoChildren, SelectionState } from '../../shared/types.d';
|
||||
|
||||
type ToolBar = FCNoChildren<{
|
||||
nodes: ContextNode[];
|
||||
@ -13,16 +13,14 @@ export const ToolBar: ToolBar = React.memo(({ nodes, state, setSelected }) =>
|
||||
nodes.length ? (
|
||||
<>
|
||||
<Separator />
|
||||
{nodes.map(({ components, ...forwardProps }) =>
|
||||
forwardProps.params.length > 1 ? (
|
||||
<ToolbarControl
|
||||
{...forwardProps}
|
||||
setSelected={setSelected}
|
||||
selected={state[forwardProps.nodeId]}
|
||||
key={forwardProps.nodeId}
|
||||
/>
|
||||
) : null
|
||||
)}
|
||||
{nodes.map(({ components, ...forwardProps }) => (
|
||||
<ToolbarControl
|
||||
{...forwardProps}
|
||||
setSelected={setSelected}
|
||||
selected={state[forwardProps.nodeId]}
|
||||
key={forwardProps.nodeId}
|
||||
/>
|
||||
))}
|
||||
</>
|
||||
) : null
|
||||
);
|
@ -1,10 +1,10 @@
|
||||
import React, { ComponentProps } from 'react';
|
||||
import { Icons, IconButton, WithTooltip } from '@storybook/components';
|
||||
import { ToolBarMenuOptions } from './ToolBarMenuOptions';
|
||||
import { ContextNode, FCNoChildren } from '../types';
|
||||
import { ContextNode, FCNoChildren } from '../../shared/types.d';
|
||||
|
||||
type ToolBarMenu = FCNoChildren<{
|
||||
icon: ContextNode['icon'];
|
||||
icon: ComponentProps<typeof Icons>['icon'];
|
||||
title: ContextNode['title'];
|
||||
active: boolean;
|
||||
expanded: boolean;
|
||||
@ -19,18 +19,17 @@ export const ToolBarMenu: ToolBarMenu = ({
|
||||
expanded,
|
||||
setExpanded,
|
||||
optionsProps,
|
||||
}) =>
|
||||
icon ? (
|
||||
<WithTooltip
|
||||
closeOnClick
|
||||
trigger="click"
|
||||
placement="top"
|
||||
tooltipShown={expanded}
|
||||
onVisibilityChange={setExpanded}
|
||||
tooltip={<ToolBarMenuOptions {...optionsProps} />}
|
||||
>
|
||||
<IconButton active={active} title={title}>
|
||||
<Icons icon={icon} />
|
||||
</IconButton>
|
||||
</WithTooltip>
|
||||
) : null;
|
||||
}) => (
|
||||
<WithTooltip
|
||||
closeOnClick
|
||||
trigger="click"
|
||||
placement="top"
|
||||
tooltipShown={expanded}
|
||||
onVisibilityChange={setExpanded}
|
||||
tooltip={<ToolBarMenuOptions {...optionsProps} />}
|
||||
>
|
||||
<IconButton active={active} title={title}>
|
||||
<Icons icon={icon} />
|
||||
</IconButton>
|
||||
</WithTooltip>
|
||||
);
|
@ -1,7 +1,7 @@
|
||||
import React from 'react';
|
||||
import { TooltipLinkList } from '@storybook/components';
|
||||
import { OPT_OUT } from '../constants';
|
||||
import { FCNoChildren } from '../types';
|
||||
import { OPT_OUT } from '../../shared/constants';
|
||||
import { FCNoChildren } from '../../shared/types.d';
|
||||
|
||||
type ToolBarMenuOptions = FCNoChildren<{
|
||||
activeName: string;
|
@ -1,7 +1,7 @@
|
||||
import React from 'react';
|
||||
import { ToolBarMenu } from './ToolBarMenu';
|
||||
import { OPT_OUT } from '../constants';
|
||||
import { ContextNode, FCNoChildren, Omit } from '../types';
|
||||
import { OPT_OUT } from '../../shared/constants';
|
||||
import { ContextNode, FCNoChildren, Omit } from '../../shared/types.d';
|
||||
|
||||
type ToolbarControl = FCNoChildren<
|
||||
Omit<
|
||||
@ -25,15 +25,14 @@ export const ToolbarControl: ToolbarControl = ({
|
||||
const [expanded, setExpanded] = React.useState(false);
|
||||
const paramNames = params.map(({ name }) => name);
|
||||
const activeName =
|
||||
// validate the selected name
|
||||
(paramNames.concat(OPT_OUT).includes(selected) && selected) ||
|
||||
// validate the integrity of the selected name
|
||||
([...paramNames, options.cancelable && OPT_OUT].includes(selected) && selected) ||
|
||||
// fallback to default
|
||||
(params.find(param => !!param.default) || { name: null }).name ||
|
||||
// fallback to the first
|
||||
params[0].name;
|
||||
const list = options.cancelable === false ? paramNames : [OPT_OUT, ...paramNames];
|
||||
const list = options.cancelable ? [OPT_OUT, ...paramNames] : paramNames;
|
||||
const props = {
|
||||
icon,
|
||||
title,
|
||||
active: activeName !== OPT_OUT,
|
||||
expanded,
|
||||
@ -48,5 +47,5 @@ export const ToolbarControl: ToolbarControl = ({
|
||||
},
|
||||
};
|
||||
|
||||
return options.disable || list.length < 2 ? null : <ToolBarMenu {...props} />;
|
||||
return icon && list.length && !options.disable ? <ToolBarMenu icon={icon} {...props} /> : null;
|
||||
};
|
@ -1,7 +1,6 @@
|
||||
export { Channel } from '@storybook/channels';
|
||||
import addons from '@storybook/addons';
|
||||
import { useEffect } from 'react';
|
||||
import { AnyFunctionReturns } from '../../types';
|
||||
import { AnyFunctionReturns } from '../../shared/types.d';
|
||||
|
||||
/**
|
||||
* The React hook version of Storybook Channel API.
|
||||
|
81
addons/contexts/src/preview/ContextsPreviewAPI.ts
Normal file
81
addons/contexts/src/preview/ContextsPreviewAPI.ts
Normal file
@ -0,0 +1,81 @@
|
||||
import addons from '@storybook/addons';
|
||||
import { window } from 'global';
|
||||
import { parse } from 'qs';
|
||||
import { getContextNodes, getPropsMap, getRendererFrom, singleton } from './libs';
|
||||
import { deserialize } from '../shared/serializers';
|
||||
import {
|
||||
PARAM,
|
||||
REBOOT_MANAGER,
|
||||
UPDATE_PREVIEW,
|
||||
UPDATE_MANAGER,
|
||||
FORCE_RE_RENDER,
|
||||
SET_CURRENT_STORY,
|
||||
} from '../shared/constants';
|
||||
import { ContextNode, PropsMap, SelectionState } from '../shared/types.d';
|
||||
|
||||
/**
|
||||
* A singleton for handling preview-manager and one-time-only side-effects.
|
||||
*/
|
||||
export const ContextsPreviewAPI = singleton(() => {
|
||||
const channel = addons.getChannel();
|
||||
let contextsNodesMemo: ContextNode[] | null = null;
|
||||
let selectionState: SelectionState = {};
|
||||
|
||||
/**
|
||||
* URL query param can be used to predetermine the contexts a story should render,
|
||||
* which is useful for performing image snapshot testing or URL sharing.
|
||||
*/
|
||||
if (window && window.location) {
|
||||
selectionState = deserialize(parse(window.location.search)[PARAM]) || {};
|
||||
}
|
||||
|
||||
/**
|
||||
* (Vue specific)
|
||||
* Vue will inject getter/setter watchers on the first rendering of the addon,
|
||||
* which is why we have to keep an internal reference and use `Object.assign` to notify the watcher.
|
||||
*/
|
||||
const reactivePropsMap = {};
|
||||
const updateReactiveSystem = (propsMap: PropsMap) => Object.assign(reactivePropsMap, propsMap);
|
||||
|
||||
/**
|
||||
* Preview-manager communications.
|
||||
*/
|
||||
// from manager
|
||||
channel.on(UPDATE_PREVIEW, state => {
|
||||
if (state) {
|
||||
selectionState = state;
|
||||
channel.emit(FORCE_RE_RENDER);
|
||||
}
|
||||
});
|
||||
channel.on(REBOOT_MANAGER, () => {
|
||||
channel.emit(UPDATE_MANAGER, contextsNodesMemo);
|
||||
});
|
||||
channel.on(SET_CURRENT_STORY, () => {
|
||||
// trash the memorization since the story-level setting may change (diffing it is much expensive)
|
||||
contextsNodesMemo = null;
|
||||
});
|
||||
|
||||
// to manager
|
||||
const getContextNodesWithSideEffects: typeof getContextNodes = (...arg) => {
|
||||
if (contextsNodesMemo === null) {
|
||||
contextsNodesMemo = getContextNodes(...arg);
|
||||
channel.emit(UPDATE_MANAGER, contextsNodesMemo);
|
||||
}
|
||||
return contextsNodesMemo;
|
||||
};
|
||||
|
||||
/**
|
||||
* @Public
|
||||
* Exposed interfaces
|
||||
*/
|
||||
return {
|
||||
// methods get called on Storybook event lifecycle
|
||||
getContextNodes: getContextNodesWithSideEffects,
|
||||
getSelectionState: () => selectionState,
|
||||
getPropsMap,
|
||||
|
||||
// methods for processing framework specific bindings
|
||||
getRendererFrom,
|
||||
updateReactiveSystem,
|
||||
};
|
||||
});
|
@ -1,52 +0,0 @@
|
||||
import addons from '@storybook/addons';
|
||||
import { FORCE_RE_RENDER, SET_CURRENT_STORY } from '@storybook/core-events';
|
||||
import { REBOOT_MANAGER, UPDATE_PREVIEW, UPDATE_MANAGER } from '../constants';
|
||||
import { getContextNodes, getPropsMap, getRendererFrom, singleton } from './libs';
|
||||
import { ContextNode, PropsMap } from '../types';
|
||||
|
||||
/**
|
||||
* @Public
|
||||
* A singleton for handling wrapper-manager side-effects
|
||||
*/
|
||||
export const addonContextsAPI = singleton(() => {
|
||||
const channel = addons.getChannel();
|
||||
let memorizedNodes: null | ContextNode[] = null;
|
||||
let selectionState = {};
|
||||
|
||||
// from manager
|
||||
channel.on(SET_CURRENT_STORY, () => (memorizedNodes = null));
|
||||
channel.on(REBOOT_MANAGER, () => channel.emit(UPDATE_MANAGER, memorizedNodes, selectionState));
|
||||
channel.on(UPDATE_PREVIEW, state => (selectionState = Object.freeze(state)));
|
||||
channel.on(UPDATE_PREVIEW, () => channel.emit(FORCE_RE_RENDER));
|
||||
|
||||
// to manager
|
||||
const getContextNodesWithSideEffects: typeof getContextNodes = (...arg) => {
|
||||
// we want to notify the manager only when the story changed since `parameter` can be changed
|
||||
if (memorizedNodes === null) {
|
||||
memorizedNodes = getContextNodes(...arg);
|
||||
channel.emit(UPDATE_MANAGER, memorizedNodes);
|
||||
}
|
||||
return memorizedNodes;
|
||||
};
|
||||
|
||||
/**
|
||||
* (Vue specific)
|
||||
* Vue will inject getter/setters on the first rendering of the addon,
|
||||
* which is the reason why we have to keep an internal reference and use `Object.assign` to update it.
|
||||
*/
|
||||
let reactivePropsMap = {};
|
||||
const updateReactiveSystem = (propsMap: PropsMap) =>
|
||||
/* tslint:disable:prefer-object-spread */
|
||||
Object.assign(reactivePropsMap, propsMap);
|
||||
|
||||
return {
|
||||
// methods get called on Storybook event lifecycle
|
||||
getContextNodes: getContextNodesWithSideEffects,
|
||||
getSelectionState: () => selectionState,
|
||||
getPropsMap,
|
||||
|
||||
// methods for processing framework specific bindings
|
||||
getRendererFrom,
|
||||
updateReactiveSystem,
|
||||
};
|
||||
});
|
14
addons/contexts/src/preview/frameworks/preact.ts
Normal file
14
addons/contexts/src/preview/frameworks/preact.ts
Normal file
@ -0,0 +1,14 @@
|
||||
import Preact from 'preact';
|
||||
import { createAddonDecorator, Render } from '../../index';
|
||||
import { ContextsPreviewAPI } from '../ContextsPreviewAPI';
|
||||
|
||||
/**
|
||||
* This is the framework specific bindings for Preact.
|
||||
* '@storybook/preact' expects the returning object from a decorator to be a 'Preact vNode'.
|
||||
*/
|
||||
export const renderPreact: Render<Preact.VNode> = (contextNodes, propsMap, getStoryVNode) => {
|
||||
const { getRendererFrom } = ContextsPreviewAPI();
|
||||
return getRendererFrom(Preact.h)(contextNodes, propsMap, getStoryVNode);
|
||||
};
|
||||
|
||||
export const withContexts = createAddonDecorator(renderPreact);
|
@ -1,13 +1,13 @@
|
||||
import React from 'react';
|
||||
import { createAddonDecorator, Render } from '../index';
|
||||
import { addonContextsAPI } from '../api';
|
||||
import { createAddonDecorator, Render } from '../../index';
|
||||
import { ContextsPreviewAPI } from '../ContextsPreviewAPI';
|
||||
|
||||
/**
|
||||
* This is the framework specific bindings for React.
|
||||
* '@storybook/react' expects the returning object from a decorator to be a 'React Element' (vNode).
|
||||
*/
|
||||
export const renderReact: Render<React.ReactElement> = (contextNodes, propsMap, getStoryVNode) => {
|
||||
const { getRendererFrom } = addonContextsAPI();
|
||||
const { getRendererFrom } = ContextsPreviewAPI();
|
||||
return getRendererFrom(React.createElement)(contextNodes, propsMap, getStoryVNode);
|
||||
};
|
||||
|
||||
|
@ -1,22 +1,27 @@
|
||||
import Vue from 'vue';
|
||||
import { createAddonDecorator, Render } from '../index';
|
||||
import { addonContextsAPI } from '../api';
|
||||
import { ID } from '../../constants';
|
||||
import { createAddonDecorator, Render } from '../../index';
|
||||
import { ContextsPreviewAPI } from '../ContextsPreviewAPI';
|
||||
import { ID } from '../../shared/constants';
|
||||
|
||||
/**
|
||||
* This is the framework specific bindings for Vue.
|
||||
* '@storybook/vue' expects the returning object from a decorator to be a 'VueComponent'.
|
||||
*/
|
||||
export const renderVue: Render<Vue.Component> = (contextNodes, propsMap, getStoryVNode) => {
|
||||
const { getRendererFrom, updateReactiveSystem } = addonContextsAPI();
|
||||
export const renderVue: Render<Vue.Component> = (contextNodes, propsMap, getStoryComponent) => {
|
||||
const { getRendererFrom, updateReactiveSystem } = ContextsPreviewAPI();
|
||||
const reactiveProps = updateReactiveSystem(propsMap);
|
||||
return Vue.extend({
|
||||
name: ID,
|
||||
data: () => reactiveProps,
|
||||
render: createElement =>
|
||||
getRendererFrom((component, props, children) =>
|
||||
createElement(component, { props }, [children])
|
||||
)(contextNodes, reactiveProps, () => createElement(getStoryVNode())),
|
||||
getRendererFrom((Component, props, children) => {
|
||||
const { key, ref, style, classNames, ...rest } = props || Object();
|
||||
const contextData =
|
||||
Component instanceof Object
|
||||
? { key, ref, style, class: classNames, props: rest } // component as a Vue object
|
||||
: { key, ref, style, class: classNames, attrs: rest }; // component as a HTML tag string
|
||||
return createElement(Component, contextData, [children]);
|
||||
})(contextNodes, reactiveProps, () => createElement(getStoryComponent())),
|
||||
});
|
||||
};
|
||||
|
||||
|
@ -1,38 +0,0 @@
|
||||
import { makeDecorator, StoryWrapper } from '@storybook/addons';
|
||||
import { addonContextsAPI } from './api';
|
||||
import { ID, PARAM } from '../constants';
|
||||
import { AddonSetting, AnyFunctionReturns, ContextNode, PropsMap } from '../types';
|
||||
|
||||
/**
|
||||
* This file serves a idiomatic facade of a Storybook decorator.
|
||||
*
|
||||
* Wrapper function get called whenever the Storybook rerender the view. This reflow logic is
|
||||
* framework agnostic; on the other hand, the framework specific bindings are the implementation
|
||||
* details hidden behind the passed `render` function.
|
||||
*
|
||||
* Here, we need a dedicated singleton as a state manager for preview (the addon API, in vanilla)
|
||||
* who is also knowing how to communicate with the Storybook manager (in React) via the Storybook
|
||||
* event system.
|
||||
*
|
||||
* @param {Render} render - framework specific bindings
|
||||
*/
|
||||
export type Render<T> = (...args: [ContextNode[], PropsMap, AnyFunctionReturns<T>]) => T;
|
||||
type CreateAddonDecorator = <T>(render: Render<T>) => (contexts: AddonSetting[]) => T;
|
||||
|
||||
export const createAddonDecorator: CreateAddonDecorator = render => {
|
||||
const wrapper: StoryWrapper = (getStory, context, settings: any) => {
|
||||
const { getContextNodes, getSelectionState, getPropsMap } = addonContextsAPI();
|
||||
const nodes = getContextNodes(settings);
|
||||
const state = getSelectionState();
|
||||
const props = getPropsMap(nodes, state);
|
||||
return render(nodes, props, () => getStory(context));
|
||||
};
|
||||
|
||||
return makeDecorator({
|
||||
name: ID,
|
||||
parameterName: PARAM,
|
||||
skipIfNoParametersOrOptions: true,
|
||||
allowDeprecatedUsage: false,
|
||||
wrapper,
|
||||
});
|
||||
};
|
@ -12,7 +12,7 @@ describe('Test on functional helpers: memorize', () => {
|
||||
const resultC = someFnMemo(1);
|
||||
|
||||
// assertion
|
||||
expect(someFn).toBeCalledTimes(2);
|
||||
expect(someFn).toHaveBeenCalledTimes(2);
|
||||
expect(resultA).toEqual(someFn(1));
|
||||
expect(resultA).not.toEqual(resultB);
|
||||
expect(resultA).toBe(resultC);
|
||||
@ -30,7 +30,7 @@ describe('Test on functional helpers: memorize', () => {
|
||||
const resultC = someFnMemo(1, 3);
|
||||
|
||||
// assertion
|
||||
expect(someFn).toBeCalledTimes(2);
|
||||
expect(someFn).toHaveBeenCalledTimes(2);
|
||||
expect(resultA).toEqual(someFn(1, 2));
|
||||
expect(resultA).toBe(resultB);
|
||||
expect(resultA).not.toEqual(resultC);
|
||||
@ -49,7 +49,7 @@ describe('Test on functional helpers: singleton', () => {
|
||||
const resultC = someFnSingleton(7, 8, 9);
|
||||
|
||||
// assertion
|
||||
expect(someFn).toBeCalledTimes(1);
|
||||
expect(someFn).toHaveBeenCalledTimes(1);
|
||||
expect(resultA).toEqual(someFn(1, 2, 3));
|
||||
expect(resultA).toBe(resultB);
|
||||
expect(resultA).toBe(resultC);
|
||||
|
@ -3,12 +3,12 @@
|
||||
* the default is to memorize its the first argument;
|
||||
* @return the memorized version of a function.
|
||||
*/
|
||||
type Memorize = <T, U extends any[]>(
|
||||
type memorize = <T, U extends any[]>(
|
||||
fn: (...args: U) => T,
|
||||
resolver?: (...args: U) => unknown
|
||||
) => (...args: U) => T;
|
||||
|
||||
export const memorize: Memorize = (fn, resolver) => {
|
||||
export const memorize: memorize = (fn, resolver) => {
|
||||
const memo = new Map();
|
||||
return (...arg) => {
|
||||
const key = resolver ? resolver(...arg) : arg[0];
|
||||
@ -21,6 +21,6 @@ export const memorize: Memorize = (fn, resolver) => {
|
||||
* the returned value is cached for resolving the subsequent calls.
|
||||
* @return the singleton version of a function.
|
||||
*/
|
||||
type Singleton = <T, U extends any[]>(fn: (...args: U) => T) => (...args: U) => T;
|
||||
type singleton = <T, U extends any[]>(fn: (...args: U) => T) => (...args: U) => T;
|
||||
|
||||
export const singleton: Singleton = fn => memorize(fn, () => 'singleton');
|
||||
export const singleton: singleton = fn => memorize(fn, () => 'singleton');
|
||||
|
@ -1,25 +1,25 @@
|
||||
import { AddonSetting, ContextNode, WrapperSettings } from '../../types';
|
||||
/* eslint-disable no-underscore-dangle */
|
||||
import { AddonSetting, ContextNode, WrapperSettings } from '../../shared/types.d';
|
||||
|
||||
/**
|
||||
* @private
|
||||
* Merge the top-level (global options) and the story-level (parameters) from a pair of setting;
|
||||
* @return the normalized definition for a contextual environment (i.e. a contextNode).
|
||||
*/
|
||||
type GetMergedSettings = (
|
||||
type _getMergedSettings = (
|
||||
topLevel: Partial<AddonSetting>,
|
||||
storyLevel: Partial<AddonSetting>
|
||||
) => ContextNode;
|
||||
|
||||
export const _getMergedSettings: GetMergedSettings = (topLevel, storyLevel) => ({
|
||||
nodeId: topLevel.title || storyLevel.title || '',
|
||||
export const _getMergedSettings: _getMergedSettings = (topLevel, storyLevel) => ({
|
||||
// strip out special characters reserved for serializing
|
||||
nodeId: (topLevel.title || storyLevel.title || '').replace(/[,+]/g, ''),
|
||||
icon: topLevel.icon || storyLevel.icon || '',
|
||||
title: topLevel.title || storyLevel.title || '',
|
||||
components: topLevel.components || storyLevel.components || [],
|
||||
params:
|
||||
topLevel.params || storyLevel.params
|
||||
? Array()
|
||||
.concat(topLevel.params, storyLevel.params)
|
||||
.filter(Boolean)
|
||||
? [...(topLevel.params || []), ...(storyLevel.params || [])].filter(Boolean)
|
||||
: [{ name: '', props: {} }],
|
||||
options: {
|
||||
deep: false,
|
||||
@ -35,11 +35,10 @@ export const _getMergedSettings: GetMergedSettings = (topLevel, storyLevel) => (
|
||||
* Pair up settings for merging normalizations to produce the contextual definitions (i.e. contextNodes);
|
||||
* it guarantee the adding order can be respected but not duplicated.
|
||||
*/
|
||||
type GetContextNodes = (settings: WrapperSettings) => ContextNode[];
|
||||
type getContextNodes = (settings: WrapperSettings) => ContextNode[];
|
||||
|
||||
export const getContextNodes: GetContextNodes = ({ options, parameters }) => {
|
||||
const titles = Array()
|
||||
.concat(options, parameters)
|
||||
export const getContextNodes: getContextNodes = ({ options, parameters }) => {
|
||||
const titles = [...(options || []), ...(parameters || [])]
|
||||
.filter(Boolean)
|
||||
.map(({ title }) => title);
|
||||
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user