mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 08:01:20 +08:00
Omit nested unchanged properties and array values
This commit is contained in:
parent
8c32cfbeb7
commit
0e849d7d3b
@ -152,10 +152,7 @@ export const init: ModuleFn = ({ store, navigate, state, provider, fullAPI, ...r
|
||||
const updateArgsParam = (args?: Story['args']) => {
|
||||
const currentStory = fullAPI.getCurrentStoryData();
|
||||
const initialArgs = (isStory(currentStory) && currentStory.initialArgs) || {};
|
||||
const customizedArgs = Object.entries(args || {}).reduce((acc, [key, value]) => {
|
||||
return deepEqual(value, initialArgs[key]) ? acc : Object.assign(acc, { [key]: value });
|
||||
}, {} as Story['args']);
|
||||
const argsString = buildArgsParam(customizedArgs);
|
||||
const argsString = buildArgsParam(initialArgs, args);
|
||||
const argsParam = argsString.length ? `&args=${argsString}` : '';
|
||||
queryNavigate(`${fullAPI.getUrlState().path}${argsParam}`, { replace: true });
|
||||
api.setQueryParams({ args: argsString });
|
||||
|
@ -15,13 +15,14 @@ const validateArgs = (key = '', value: any = ''): boolean => {
|
||||
const QS_OPTIONS = {
|
||||
delimiter: ';', // we're parsing a single query param
|
||||
allowDots: true, // objects are encoded using dot notation
|
||||
allowSparse: true, // arrays will be merged on top of their initial value
|
||||
};
|
||||
export const parseArgsParam = (argsString: string): Args => {
|
||||
const parts = argsString.split(';').map((part) => part.replace('=', '~').replace(':', '='));
|
||||
return Object.entries(qs.parse(parts.join(';'), QS_OPTIONS)).reduce((acc, [key, value]) => {
|
||||
if (validateArgs(key, value)) return Object.assign(acc, { [key]: value });
|
||||
once.warn(
|
||||
'Cannot safely apply some args from the URL. See https://storybook.js.org/docs/react/writing-stories/args#setting-args-through-the-url'
|
||||
'Omitted potentially unsafe URL args.\n\nMore info: https://storybook.js.org/docs/react/writing-stories/args#setting-args-through-the-url'
|
||||
);
|
||||
return acc;
|
||||
}, {} as Args);
|
||||
|
@ -40,6 +40,7 @@
|
||||
"@storybook/client-logger": "6.2.0-alpha.25",
|
||||
"@types/reach__router": "^1.3.7",
|
||||
"core-js": "^3.8.2",
|
||||
"fast-deep-equal": "^3.1.3",
|
||||
"global": "^4.4.0",
|
||||
"memoizerific": "^1.11.3",
|
||||
"qs": "^6.9.5"
|
||||
|
@ -78,52 +78,77 @@ describe('parsePath', () => {
|
||||
|
||||
describe('buildArgsParam', () => {
|
||||
it('builds a simple key-value pair', () => {
|
||||
const param = buildArgsParam({ key: 'val' });
|
||||
const param = buildArgsParam({}, { key: 'val' });
|
||||
expect(param).toEqual('key:val');
|
||||
});
|
||||
|
||||
it('builds multiple values', () => {
|
||||
const param = buildArgsParam({ one: '1', two: '2', three: '3' });
|
||||
const param = buildArgsParam({}, { one: '1', two: '2', three: '3' });
|
||||
expect(param).toEqual('one:1;two:2;three:3');
|
||||
});
|
||||
|
||||
it('builds booleans', () => {
|
||||
const param = buildArgsParam({ yes: true, no: false });
|
||||
const param = buildArgsParam({}, { yes: true, no: false });
|
||||
expect(param).toEqual('yes:true;no:false');
|
||||
});
|
||||
|
||||
it('builds arrays', () => {
|
||||
const param = buildArgsParam({ arr: ['1', '2', '3'] });
|
||||
expect(param).toEqual('arr[]:1;arr[]:2;arr[]:3');
|
||||
const param = buildArgsParam({}, { arr: ['1', '2', '3'] });
|
||||
expect(param).toEqual('arr[0]:1;arr[1]:2;arr[2]:3');
|
||||
});
|
||||
|
||||
it('builds simple objects', () => {
|
||||
const param = buildArgsParam({ obj: { one: '1', two: '2' } });
|
||||
const param = buildArgsParam({}, { obj: { one: '1', two: '2' } });
|
||||
expect(param).toEqual('obj.one:1;obj.two:2');
|
||||
});
|
||||
|
||||
it('builds nested objects', () => {
|
||||
const param = buildArgsParam({ obj: { foo: { one: '1', two: '2' }, bar: { one: '1' } } });
|
||||
const param = buildArgsParam({}, { obj: { foo: { one: '1', two: '2' }, bar: { one: '1' } } });
|
||||
expect(param).toEqual('obj.foo.one:1;obj.foo.two:2;obj.bar.one:1');
|
||||
});
|
||||
|
||||
it('builds arrays in objects', () => {
|
||||
const param = buildArgsParam({ obj: { foo: ['1', '2'] } });
|
||||
expect(param).toEqual('obj.foo[]:1;obj.foo[]:2');
|
||||
const param = buildArgsParam({}, { obj: { foo: ['1', '2'] } });
|
||||
expect(param).toEqual('obj.foo[0]:1;obj.foo[1]:2');
|
||||
});
|
||||
|
||||
it('builds single object in array', () => {
|
||||
const param = buildArgsParam({ arr: [{ one: '1', two: '2' }] });
|
||||
expect(param).toEqual('arr[].one:1;arr[].two:2');
|
||||
const param = buildArgsParam({}, { arr: [{ one: '1', two: '2' }] });
|
||||
expect(param).toEqual('arr[0].one:1;arr[0].two:2');
|
||||
});
|
||||
|
||||
it('builds multiple objects in array', () => {
|
||||
const param = buildArgsParam({ arr: [{ one: '1' }, { two: '2' }] });
|
||||
expect(param).toEqual('arr[].one:1;arr[].two:2');
|
||||
const param = buildArgsParam({}, { arr: [{ one: '1' }, { two: '2' }] });
|
||||
expect(param).toEqual('arr[0].one:1;arr[1].two:2');
|
||||
});
|
||||
|
||||
it('builds nested object in array', () => {
|
||||
const param = buildArgsParam({ arr: [{ foo: { bar: 'val' } }] });
|
||||
expect(param).toEqual('arr[].foo.bar:val');
|
||||
const param = buildArgsParam({}, { arr: [{ foo: { bar: 'val' } }] });
|
||||
expect(param).toEqual('arr[0].foo.bar:val');
|
||||
});
|
||||
|
||||
describe('with initial state', () => {
|
||||
it('omits unchanged values', () => {
|
||||
const param = buildArgsParam({ one: 1 }, { one: 1, two: 2 });
|
||||
expect(param).toEqual('two:2');
|
||||
});
|
||||
|
||||
it('omits unchanged object properties', () => {
|
||||
const param = buildArgsParam({ obj: { one: 1 } }, { obj: { one: 1, two: 2 } });
|
||||
expect(param).toEqual('obj.two:2');
|
||||
});
|
||||
|
||||
it('omits unchanged array values (yielding sparse arrays)', () => {
|
||||
const param = buildArgsParam({ arr: [1, 2, 3] }, { arr: [1, 3, 4] });
|
||||
expect(param).toEqual('arr[1]:3;arr[2]:4');
|
||||
});
|
||||
|
||||
it('omits nested unchanged object properties and array values', () => {
|
||||
const param = buildArgsParam(
|
||||
{ obj: { nested: [{ one: 1 }, { two: 2 }] } },
|
||||
{ obj: { nested: [{ one: 1 }, { two: 2, three: 3 }] } }
|
||||
);
|
||||
expect(param).toEqual('obj.nested[1].three:3');
|
||||
});
|
||||
});
|
||||
});
|
||||
|
@ -1,3 +1,4 @@
|
||||
import deepEqual from 'fast-deep-equal';
|
||||
import qs from 'qs';
|
||||
import memoize from 'memoizerific';
|
||||
import { once } from '@storybook/client-logger';
|
||||
@ -36,6 +37,22 @@ interface Args {
|
||||
[key: string]: any;
|
||||
}
|
||||
|
||||
const deepDiff = (value: any, update: any): any => {
|
||||
if (deepEqual(value, update)) return undefined;
|
||||
if (typeof value !== typeof update) return update;
|
||||
if (Array.isArray(value)) {
|
||||
if (!Array.isArray(update)) return update;
|
||||
return update.map((upd, index) => deepDiff(value[index], upd));
|
||||
}
|
||||
if (typeof update === 'object') {
|
||||
return Object.keys(update).reduce((acc, key) => {
|
||||
const diff = deepDiff(value[key], update[key]);
|
||||
return diff === undefined ? acc : Object.assign(acc, { [key]: diff });
|
||||
}, {});
|
||||
}
|
||||
return update;
|
||||
};
|
||||
|
||||
// Keep this in sync with validateArgs in @storybook/core
|
||||
const VALIDATION_REGEXP = /^[a-zA-Z0-9 _-]*$/;
|
||||
const validateArgs = (key = '', value: any = ''): boolean => {
|
||||
@ -50,19 +67,25 @@ const QS_OPTIONS = {
|
||||
encode: false, // we handle URL encoding ourselves
|
||||
delimiter: ';', // we don't actually create multiple query params
|
||||
allowDots: true, // encode objects using dot notation: obj.key=val
|
||||
arrayFormat: 'brackets', // encode arrays using brackets without indices: arr[]=one&arr[]=two
|
||||
format: 'RFC1738', // encode spaces using the + sign
|
||||
};
|
||||
export const buildArgsParam = (args: Args) => {
|
||||
const object = Object.entries(args).reduce((acc, [key, value]) => {
|
||||
export const buildArgsParam = (initialArgs: Args, args: Args): string => {
|
||||
const update = deepDiff(initialArgs, args);
|
||||
if (!update) return '';
|
||||
|
||||
const object = Object.entries(update).reduce((acc, [key, value]) => {
|
||||
if (validateArgs(key, value)) return Object.assign(acc, { [key]: value });
|
||||
once.warn(
|
||||
'Some args cannot be safely serialized to the URL. See https://storybook.js.org/docs/react/writing-stories/args#setting-args-through-the-url'
|
||||
'Omitted potentially unsafe URL args.\n\nMore info: https://storybook.js.org/docs/react/writing-stories/args#setting-args-through-the-url'
|
||||
);
|
||||
return acc;
|
||||
}, {} as Args);
|
||||
const parts = qs.stringify(object, QS_OPTIONS).split(';');
|
||||
return parts.map((part: string) => part.replace('=', ':')).join(';');
|
||||
|
||||
return qs
|
||||
.stringify(object, QS_OPTIONS)
|
||||
.split(';')
|
||||
.map((part: string) => part.replace('=', ':'))
|
||||
.join(';');
|
||||
};
|
||||
|
||||
interface Query {
|
||||
|
Loading…
x
Reference in New Issue
Block a user