From ff0f138157083ffa3a2f1bb99e9b9e7eb4c833a5 Mon Sep 17 00:00:00 2001 From: Michael Shilman Date: Tue, 2 Apr 2019 19:07:13 +0800 Subject: [PATCH 1/4] Refactor story_store - don't touch URL - only emit events, don't listen on them - add tests --- lib/client-api/src/story_store.js | 35 +--------- lib/client-api/src/story_store.test.js | 59 ----------------- lib/core/src/client/preview/start.js | 42 +++++++----- lib/core/src/client/preview/start.test.js | 39 ++++++++++- lib/core/src/client/preview/url.js | 39 +++++++++++ lib/core/src/client/preview/url.test.js | 79 +++++++++++++++++++++++ 6 files changed, 184 insertions(+), 109 deletions(-) create mode 100644 lib/core/src/client/preview/url.js create mode 100644 lib/core/src/client/preview/url.test.js diff --git a/lib/client-api/src/story_store.js b/lib/client-api/src/story_store.js index f6dbf7c47bd..13d25d71dbd 100644 --- a/lib/client-api/src/story_store.js +++ b/lib/client-api/src/story_store.js @@ -1,16 +1,11 @@ /* eslint no-underscore-dangle: 0 */ -import { history, document } from 'global'; import EventEmitter from 'eventemitter3'; -import qs from 'qs'; import memoize from 'memoizerific'; import debounce from 'lodash.debounce'; import { stripIndents } from 'common-tags'; import Events from '@storybook/core-events'; import { logger } from '@storybook/client-logger'; -import { toId } from '@storybook/router/utils'; - -import pathToId from './pathToId'; // TODO: these are copies from components/nav/lib // refactor to DRY @@ -49,9 +44,6 @@ const toExtracted = obj => return Object.assign(acc, { [key]: value }); }, {}); -const getIdFromLegacyQuery = ({ path, selectedKind, selectedStory }) => - (path && pathToId(path)) || (selectedKind && selectedStory && toId(selectedKind, selectedStory)); - export default class StoryStore extends EventEmitter { constructor(params) { super(); @@ -61,19 +53,6 @@ export default class StoryStore extends EventEmitter { this._revision = 0; this._selection = {}; this._channel = params.channel; - - this.on(Events.STORY_INIT, () => { - let storyId = this.getIdOnPath(); - if (!storyId) { - const query = qs.parse(document.location.search, { ignoreQueryPrefix: true }); - storyId = getIdFromLegacyQuery(query); - if (storyId) { - const { path, selectedKind, selectedStory, ...rest } = query; - this.setPath(storyId, rest); - } - } - this.setSelection(this.fromId(storyId)); - }); } setChannel = channel => { @@ -82,16 +61,6 @@ export default class StoryStore extends EventEmitter { // NEW apis - getIdOnPath = () => { - const { id } = qs.parse(document.location.search, { ignoreQueryPrefix: true }); - return id; - }; - - setPath = (storyId, params = {}) => { - const path = `${document.location.pathname}?${qs.stringify({ ...params, id: storyId })}`; - history.replaceState({}, '', path); - }; - fromId = id => { try { const data = this._data[id]; @@ -126,8 +95,8 @@ export default class StoryStore extends EventEmitter { ); } - setSelection = data => { - this._selection = data; + setSelection = ({ storyId }) => { + this._selection = { storyId }; setTimeout(() => this.emit(Events.STORY_RENDER), 1); }; diff --git a/lib/client-api/src/story_store.test.js b/lib/client-api/src/story_store.test.js index b68499539f2..e587e62027a 100644 --- a/lib/client-api/src/story_store.test.js +++ b/lib/client-api/src/story_store.test.js @@ -1,25 +1,9 @@ -import { history, document } from 'global'; import createChannel from '@storybook/channel-postmessage'; -import Events from '@storybook/core-events'; import { toId } from '@storybook/router/utils'; import StoryStore from './story_store'; import { defaultDecorateStory } from './client_api'; -jest.mock('global', () => ({ - history: { replaceState: jest.fn() }, - window: { - addEventListener: jest.fn(), - }, - document: { - location: { - pathname: 'pathname', - search: '', - }, - addEventListener: jest.fn(), - }, -})); - jest.mock('@storybook/node-logger', () => ({ logger: { info: jest.fn(), @@ -148,47 +132,4 @@ describe('preview.story_store', () => { }); }); }); - - describe('setPath', () => { - it('preserves custom URL params', () => { - const store = new StoryStore({ channel }); - - store.setPath('story--id', { foo: 'bar' }); - expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?foo=bar&id=story--id'); - }); - }); - - describe('STORY_INIT', () => { - const storyFn = () => 0; - - it('supports path params', () => { - document.location = { - pathname: 'pathname', - search: '?path=/story/kind--story&bar=baz', - }; - const store = new StoryStore({ channel }); - store.addStory(...make('kind', 'story', storyFn)); - store.setSelection = jest.fn(); - - store.emit(Events.STORY_INIT); - expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?bar=baz&id=kind--story'); - expect(store.setSelection).toHaveBeenCalled(); - expect(store.setSelection.mock.calls[0][0].getDecorated()).toEqual(storyFn); - }); - - it('supports story kind/name params', () => { - document.location = { - pathname: 'pathname', - search: '?selectedKind=kind&selectedStory=story&bar=baz', - }; - const store = new StoryStore({ channel }); - store.addStory(...make('kind', 'story', storyFn)); - store.setSelection = jest.fn(); - - store.emit(Events.STORY_INIT); - expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?bar=baz&id=kind--story'); - expect(store.setSelection).toHaveBeenCalled(); - expect(store.setSelection.mock.calls[0][0].getDecorated()).toEqual(storyFn); - }); - }); }); diff --git a/lib/core/src/client/preview/start.js b/lib/core/src/client/preview/start.js index deaa08b6774..6593ef2824a 100644 --- a/lib/core/src/client/preview/start.js +++ b/lib/core/src/client/preview/start.js @@ -6,6 +6,7 @@ import { toId } from '@storybook/router/utils'; import { logger } from '@storybook/client-logger'; import Events from '@storybook/core-events'; import deprecate from 'util-deprecate'; +import { initializePath, setPath } from './url'; const classes = { MAIN: 'sb-show-main', @@ -114,8 +115,19 @@ export default function start(render, { decorateStory } = {}) { const renderMain = forceRender => { const revision = storyStore.getRevision(); - const selection = storyStore.getSelection(); - const { kind, name, getDecorated, id } = selection || {}; + const { storyId } = storyStore.getSelection(); + + const data = storyStore.fromId(storyId); + + const { kind, name, getDecorated, id } = data || {}; + + const renderContext = { + ...context, + ...data, + selectedKind: kind, + selectedStory: name, + forceRender, + }; if (getDecorated) { // Render story only if selectedKind or selectedStory have changed. @@ -135,21 +147,16 @@ export default function start(render, { decorateStory } = {}) { addons.getChannel().emit(Events.STORY_CHANGED, id); } - render({ - ...context, - ...selection, - selectedKind: kind, - selectedStory: name, - forceRender, - }); + previousRevision = revision; + previousKind = kind; + previousStory = name; + + render(renderContext); addons.getChannel().emit(Events.STORY_RENDERED, id); } else { showNopreview(); addons.getChannel().emit(Events.STORY_MISSING, id); } - previousRevision = revision; - previousKind = kind; - previousStory = name; if (!forceRender) { document.documentElement.scrollTop = 0; @@ -187,10 +194,8 @@ export default function start(render, { decorateStory } = {}) { storyId = deprecatedToId(kind, name); } - const data = storyStore.fromId(storyId); - - storyStore.setSelection(data); - storyStore.setPath(storyId); + storyStore.setSelection({ storyId }); + setPath({ storyId }); }); // Handle keyboard shortcuts @@ -205,6 +210,11 @@ export default function start(render, { decorateStory } = {}) { }; } + storyStore.on(Events.STORY_INIT, () => { + const { storyId } = initializePath(); + storyStore.setSelection({ storyId }); + }); + storyStore.on(Events.STORY_RENDER, renderUI); if (typeof window !== 'undefined') { diff --git a/lib/core/src/client/preview/start.test.js b/lib/core/src/client/preview/start.test.js index 4b9270593e1..df34092ab60 100644 --- a/lib/core/src/client/preview/start.test.js +++ b/lib/core/src/client/preview/start.test.js @@ -1,9 +1,12 @@ -import { document } from 'global'; +/* eslint-disable no-underscore-dangle */ +import { history, document } from 'global'; +import Events from '@storybook/core-events'; import start from './start'; jest.mock('@storybook/client-logger'); jest.mock('global', () => ({ + history: { replaceState: jest.fn() }, navigator: { userAgent: 'browser', platform: '' }, window: { addEventListener: jest.fn(), @@ -85,3 +88,37 @@ it('emits an error and shows error when your framework calls showError', () => { expect(render).toHaveBeenCalled(); expect(document.body.classList.add).toHaveBeenCalledWith('sb-show-errordisplay'); }); + +describe('STORY_INIT', () => { + it('supports path params', () => { + document.location = { + pathname: 'pathname', + search: '?path=/story/kind--story&bar=baz', + }; + + const render = jest.fn(); + const { clientApi } = start(render); + const store = clientApi._storyStore; + store.setSelection = jest.fn(); + store.emit(Events.STORY_INIT); + + store.emit(); + expect(store.setSelection).toHaveBeenCalledWith({ storyId: 'kind--story' }); + }); + + it('supports story kind/name params', () => { + document.location = { + pathname: 'pathname', + search: '?selectedKind=kind&selectedStory=story&bar=baz', + }; + + const render = jest.fn(); + const { clientApi } = start(render); + const store = clientApi._storyStore; + store.setSelection = jest.fn(); + + store.emit(Events.STORY_INIT); + expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?bar=baz&id=kind--story'); + expect(store.setSelection).toHaveBeenCalledWith({ storyId: 'kind--story' }); + }); +}); diff --git a/lib/core/src/client/preview/url.js b/lib/core/src/client/preview/url.js new file mode 100644 index 00000000000..12c1c18bb75 --- /dev/null +++ b/lib/core/src/client/preview/url.js @@ -0,0 +1,39 @@ +import { history, document } from 'global'; +import qs from 'qs'; +import { toId } from '@storybook/router/utils'; + +export function pathToId(path) { + const match = (path || '').match(/^\/story\/(.+)/); + if (!match) { + throw new Error(`Invalid path '${path}', must start with '/story/'`); + } + return match[1]; +} + +export const setPath = ({ storyId }) => { + const { path, selectedKind, selectedStory, ...rest } = qs.parse(document.location.search, { + ignoreQueryPrefix: true, + }); + const newPath = `${document.location.pathname}?${qs.stringify({ ...rest, id: storyId })}`; + history.replaceState({}, '', newPath); +}; + +export const getIdFromLegacyQuery = ({ path, selectedKind, selectedStory }) => + (path && pathToId(path)) || (selectedKind && selectedStory && toId(selectedKind, selectedStory)); + +export const parseQueryParameters = search => { + const { id } = qs.parse(search, { ignoreQueryPrefix: true }); + return id; +}; + +export const initializePath = () => { + const query = qs.parse(document.location.search, { ignoreQueryPrefix: true }); + let { id: storyId } = query; + if (!storyId) { + storyId = getIdFromLegacyQuery(query); + if (storyId) { + setPath({ storyId }); + } + } + return { storyId }; +}; diff --git a/lib/core/src/client/preview/url.test.js b/lib/core/src/client/preview/url.test.js new file mode 100644 index 00000000000..0a33955e4c7 --- /dev/null +++ b/lib/core/src/client/preview/url.test.js @@ -0,0 +1,79 @@ +import { history, document } from 'global'; +import { + pathToId, + setPath, + getIdFromLegacyQuery, + parseQueryParameters, + initializePath, +} from './url'; + +jest.mock('global', () => ({ + history: { replaceState: jest.fn() }, + document: { + location: { + pathname: 'pathname', + search: '', + }, + }, +})); + +describe('url', () => { + describe('pathToId', () => { + it('should parse valid ids', () => { + expect(pathToId('/story/story--id')).toEqual('story--id'); + }); + it('should error on invalid ids', () => { + [null, '', '/whatever/story/story--id'].forEach(path => { + expect(() => pathToId(path)).toThrow(/Invalid/); + }); + }); + }); + + describe('setPath', () => { + it('should navigate to storyId', () => { + setPath({ storyId: 'story--id' }); + expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?id=story--id'); + }); + it('should replace legacy parameters but preserve others', () => { + document.location.search = 'foo=bar&selectedStory=selStory&selectedKind=selKind'; + setPath({ storyId: 'story--id' }); + expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?foo=bar&id=story--id'); + }); + }); + + describe('getIdFromLegacyQuery', () => { + it('should parse story paths', () => { + expect(getIdFromLegacyQuery({ path: '/story/story--id' })).toBe('story--id'); + }); + it('should parse legacy queries', () => { + expect( + getIdFromLegacyQuery({ path: null, selectedKind: 'kind', selectedStory: 'story' }) + ).toBe('kind--story'); + }); + it('should not parse non-queries', () => { + expect(getIdFromLegacyQuery({})).toBeUndefined(); + }); + }); + + describe('parseQueryParameters', () => { + it('should parse id', () => { + expect(parseQueryParameters('?foo=bar&id=story--id')).toBe('story--id'); + }); + it('should not parse non-ids', () => { + expect(parseQueryParameters('')).toBeUndefined(); + }); + }); + + describe('initializePath', () => { + it('should handle id queries', () => { + document.location.search = '?id=story--id'; + expect(initializePath()).toEqual({ storyId: 'story--id' }); + expect(history.replaceState).not.toHaveBeenCalled(); + }); + it('should redirect legacy queries', () => { + document.location.search = '?selectedKind=kind&selectedStory=story'; + expect(initializePath()).toEqual({ storyId: 'kind--story' }); + expect(history.replaceState).toHaveBeenCalledWith({}, '', 'pathname?id=kind--story'); + }); + }); +}); From f979ecadca38961d579387cefd9a9950fd855cf4 Mon Sep 17 00:00:00 2001 From: Michael Shilman Date: Fri, 7 Jun 2019 12:50:32 -0700 Subject: [PATCH 2/4] Add SCSS to presets gallery --- docs/src/pages/presets/preset-gallery/index.md | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/docs/src/pages/presets/preset-gallery/index.md b/docs/src/pages/presets/preset-gallery/index.md index d993c54d65f..450d98a61ee 100644 --- a/docs/src/pages/presets/preset-gallery/index.md +++ b/docs/src/pages/presets/preset-gallery/index.md @@ -11,7 +11,11 @@ Storybook-maintained presets are available in the [Presets repo](https://github. ### [Typescript](https://github.com/storybookjs/presets/tree/master/packages/preset-typescript) -Write your stories in typescript with a single line of configuration. +One-line Typescript w/ docgen configuration for storybook. + +### [SCSS](https://github.com/storybookjs/presets/tree/master/packages/preset-scss) + +One-line SCSS configuration for storybook. ## Community presets From 1e81ea4fb4aafb5942e4a663e26f750c4defe65e Mon Sep 17 00:00:00 2001 From: Michael Shilman Date: Mon, 17 Jun 2019 18:07:42 +0800 Subject: [PATCH 3/4] Clean up comments --- lib/client-api/src/story_store.js | 17 ----------------- 1 file changed, 17 deletions(-) diff --git a/lib/client-api/src/story_store.js b/lib/client-api/src/story_store.js index b048c662ba3..9a7edd00635 100644 --- a/lib/client-api/src/story_store.js +++ b/lib/client-api/src/story_store.js @@ -6,10 +6,6 @@ import { stripIndents } from 'common-tags'; import Events from '@storybook/core-events'; import { logger } from '@storybook/client-logger'; -// import { toId } from '@storybook/router/utils'; - -// import pathToId from './pathToId'; -// import { getQueryParams } from './queryparams'; // TODO: these are copies from components/nav/lib // refactor to DRY @@ -46,19 +42,6 @@ export default class StoryStore extends EventEmitter { this._revision = 0; this._selection = {}; this._channel = params.channel; - - // this.on(Events.STORY_INIT, () => { - // let storyId = this.getIdOnPath(); - // if (!storyId) { - // const query = getQueryParams(); - // storyId = getIdFromLegacyQuery(query); - // if (storyId) { - // const { path, selectedKind, selectedStory, ...rest } = query; - // this.setPath(storyId, rest); - // } - // } - // this.setSelection(this.fromId(storyId)); - // }); } setChannel = channel => { From cad915bce78a7c33c65fd2885b4b48e378eb88ad Mon Sep 17 00:00:00 2001 From: "dependabot-preview[bot]" Date: Tue, 18 Jun 2019 06:56:41 +0000 Subject: [PATCH 4/4] Bump css-loader from 2.1.1 to 3.0.0 Bumps [css-loader](https://github.com/webpack-contrib/css-loader) from 2.1.1 to 3.0.0. - [Release notes](https://github.com/webpack-contrib/css-loader/releases) - [Changelog](https://github.com/webpack-contrib/css-loader/blob/master/CHANGELOG.md) - [Commits](https://github.com/webpack-contrib/css-loader/compare/v2.1.1...v3.0.0) Signed-off-by: dependabot-preview[bot] --- lib/core/package.json | 2 +- yarn.lock | 53 ++++++++++++++++++++++++++++++++++++++----- 2 files changed, 48 insertions(+), 7 deletions(-) diff --git a/lib/core/package.json b/lib/core/package.json index a5c0959ebac..88f05dbb1e2 100644 --- a/lib/core/package.json +++ b/lib/core/package.json @@ -48,7 +48,7 @@ "common-tags": "^1.8.0", "core-js": "^3.0.1", "corejs-upgrade-webpack-plugin": "^2.0.0", - "css-loader": "^2.1.1", + "css-loader": "^3.0.0", "detect-port": "^1.3.0", "dotenv-webpack": "^1.7.0", "ejs": "^2.6.1", diff --git a/yarn.lock b/yarn.lock index 7197f04ca62..de3157fb024 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9117,6 +9117,24 @@ css-loader@2.1.1, css-loader@^2.1.1: postcss-value-parser "^3.3.0" schema-utils "^1.0.0" +css-loader@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/css-loader/-/css-loader-3.0.0.tgz#bdd48a4921eefedf1f0a55266585944d4e5efc63" + integrity sha512-WR6KZuCkFbnMhRrGPlkwAA7SSCtwqPwpyXJAPhotYkYsc0mKU9n/fu5wufy4jl2WhBw9Ia8gUQMIp/1w98DuPw== + dependencies: + camelcase "^5.3.1" + cssesc "^3.0.0" + icss-utils "^4.1.1" + loader-utils "^1.2.3" + normalize-path "^3.0.0" + postcss "^7.0.17" + postcss-modules-extract-imports "^2.0.0" + postcss-modules-local-by-default "^3.0.2" + postcss-modules-scope "^2.1.0" + postcss-modules-values "^3.0.0" + postcss-value-parser "^4.0.0" + schema-utils "^1.0.0" + css-parse@1.7.x: version "1.7.0" resolved "https://registry.yarnpkg.com/css-parse/-/css-parse-1.7.0.tgz#321f6cf73782a6ff751111390fc05e2c657d8c9b" @@ -14160,10 +14178,10 @@ icss-replace-symbols@^1.1.0: resolved "https://registry.yarnpkg.com/icss-replace-symbols/-/icss-replace-symbols-1.1.0.tgz#06ea6f83679a7749e386cfe1fe812ae5db223ded" integrity sha1-Bupvg2ead0njhs/h/oEq5dsiPe0= -icss-utils@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/icss-utils/-/icss-utils-4.1.0.tgz#339dbbffb9f8729a243b701e1c29d4cc58c52f0e" - integrity sha512-3DEun4VOeMvSczifM3F2cKQrDQ5Pj6WKhkOq6HD4QTnDUAq8MQRxy5TX6Sy1iY6WPBe4gQ3p5vTECjbIkglkkQ== +icss-utils@^4.0.0, icss-utils@^4.1.0, icss-utils@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/icss-utils/-/icss-utils-4.1.1.tgz#21170b53789ee27447c2f47dd683081403f9a467" + integrity sha512-4aFq7wvWyMHKgxsH8QQtGpvbASCf+eM3wPRLI6R+MgAnTCZ6STYsRvttLvRWK0Nfif5piF394St3HeJDaljGPA== dependencies: postcss "^7.0.14" @@ -21009,6 +21027,16 @@ postcss-modules-local-by-default@^2.0.6: postcss-selector-parser "^6.0.0" postcss-value-parser "^3.3.1" +postcss-modules-local-by-default@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/postcss-modules-local-by-default/-/postcss-modules-local-by-default-3.0.2.tgz#e8a6561be914aaf3c052876377524ca90dbb7915" + integrity sha512-jM/V8eqM4oJ/22j0gx4jrp63GSvDH6v86OqyTHHUvk4/k1vceipZsaymiZ5PvocqZOl5SFHiFJqjs3la0wnfIQ== + dependencies: + icss-utils "^4.1.1" + postcss "^7.0.16" + postcss-selector-parser "^6.0.2" + postcss-value-parser "^4.0.0" + postcss-modules-scope@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/postcss-modules-scope/-/postcss-modules-scope-2.1.0.tgz#ad3f5bf7856114f6fcab901b0502e2a2bc39d4eb" @@ -21025,6 +21053,14 @@ postcss-modules-values@^2.0.0: icss-replace-symbols "^1.1.0" postcss "^7.0.6" +postcss-modules-values@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/postcss-modules-values/-/postcss-modules-values-3.0.0.tgz#5b5000d6ebae29b4255301b4a3a54574423e7f10" + integrity sha512-1//E5jCBrZ9DmRX+zCtmQtRSV6PV42Ix7Bzj9GbwJceduuf7IqP8MgeTXuRDHOWj2m0VzZD5+roFWDuU8RQjcg== + dependencies: + icss-utils "^4.0.0" + postcss "^7.0.6" + postcss-nesting@^7.0.0: version "7.0.0" resolved "https://registry.yarnpkg.com/postcss-nesting/-/postcss-nesting-7.0.0.tgz#6e26a770a0c8fcba33782a6b6f350845e1a448f6" @@ -21273,7 +21309,7 @@ postcss-selector-parser@^5.0.0, postcss-selector-parser@^5.0.0-rc.3, postcss-sel indexes-of "^1.0.1" uniq "^1.0.1" -postcss-selector-parser@^6.0.0: +postcss-selector-parser@^6.0.0, postcss-selector-parser@^6.0.2: version "6.0.2" resolved "https://registry.yarnpkg.com/postcss-selector-parser/-/postcss-selector-parser-6.0.2.tgz#934cf799d016c83411859e09dcecade01286ec5c" integrity sha512-36P2QR59jDTOAiIkqEprfJDsoNrvwFei3eCqKd1Y0tUsBimsq39BLp7RD+JWny3WgB1zGhJX8XVePwm9k4wdBg== @@ -21306,6 +21342,11 @@ postcss-value-parser@^3.0.0, postcss-value-parser@^3.2.3, postcss-value-parser@^ resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz#9ff822547e2893213cf1c30efa51ac5fd1ba8281" integrity sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ== +postcss-value-parser@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-4.0.0.tgz#99a983d365f7b2ad8d0f9b8c3094926eab4b936d" + integrity sha512-ESPktioptiSUchCKgggAkzdmkgzKfmp0EU8jXH+5kbIUB+unr0Y4CY9SRMvibuvYUBjNh1ACLbxqYNpdTQOteQ== + postcss-values-parser@^2.0.0, postcss-values-parser@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/postcss-values-parser/-/postcss-values-parser-2.0.1.tgz#da8b472d901da1e205b47bdc98637b9e9e550e5f" @@ -21333,7 +21374,7 @@ postcss@^6.0.9: source-map "^0.6.1" supports-color "^5.4.0" -postcss@^7.0.0, postcss@^7.0.1, postcss@^7.0.13, postcss@^7.0.14, postcss@^7.0.16, postcss@^7.0.2, postcss@^7.0.5, postcss@^7.0.6: +postcss@^7.0.0, postcss@^7.0.1, postcss@^7.0.13, postcss@^7.0.14, postcss@^7.0.16, postcss@^7.0.17, postcss@^7.0.2, postcss@^7.0.5, postcss@^7.0.6: version "7.0.17" resolved "https://registry.yarnpkg.com/postcss/-/postcss-7.0.17.tgz#4da1bdff5322d4a0acaab4d87f3e782436bad31f" integrity sha512-546ZowA+KZ3OasvQZHsbuEpysvwTZNGJv9EfyCQdsIDltPSWHAeTQ5fQy/Npi2ZDtLI3zs7Ps/p6wThErhm9fQ==