Options: support array value paths for panel options (#39499)

This commit is contained in:
Ryan McKinley 2021-09-23 09:28:16 -07:00 committed by GitHub
parent 518a0d0458
commit 01deae2105
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 53 additions and 7 deletions

View File

@ -182,6 +182,27 @@ describe('PanelPlugin', () => {
expect(panel.fieldConfigDefaults.defaults.custom).toEqual(expectedDefaults);
});
test('throw error with array fieldConfigs', () => {
const panel = new PanelPlugin(() => {
return <div>Panel</div>;
});
panel.useFieldConfig({
useCustomConfig: (builder) => {
builder.addCustomEditor({
id: 'somethingUnique',
path: 'numericOption[0]',
name: 'Option editor',
description: 'Option editor description',
defaultValue: 10,
} as any);
},
});
expect(() => panel.fieldConfigRegistry).toThrowErrorMatchingInlineSnapshot(
`"[undefined] Field config paths do not support arrays: custom.somethingUnique"`
);
});
test('default values for nested paths', () => {
const panel = new PanelPlugin(() => {
return <div>Panel</div>;

View File

@ -75,6 +75,13 @@ export function createFieldConfigRegistry<TFieldConfigOptions>(
}
}
// assert that field configs do not use array path syntax
for (const item of registry.list()) {
if (item.path.indexOf('[') > 0) {
throw new Error(`[${pluginName}] Field config paths do not support arrays: ${item.id}`);
}
}
return registry;
}

View File

@ -89,13 +89,16 @@ describe('updateDefaultFieldConfigValue', () => {
describe('setOptionImmutably', () => {
it.each`
source | path | value | expected
${{}} | ${'a'} | ${1} | ${{ a: 1 }}
${{}} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } } }}
${{ a: {} }} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } } }}
${{ b: {} }} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } }, b: {} }}
${{ a: { b: { c: 3 } } }} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } } }}
`('numeric-like text mapping, value:${value', ({ source, path, value, expected }) => {
source | path | value | expected
${{}} | ${'a'} | ${1} | ${{ a: 1 }}
${{}} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } } }}
${{ a: {} }} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } } }}
${{ b: {} }} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } }, b: {} }}
${{ a: { b: { c: 3 } } }} | ${'a.b.c'} | ${[1, 2]} | ${{ a: { b: { c: [1, 2] } } }}
${{}} | ${'a.b[2]'} | ${'x'} | ${{ a: { b: [undefined, undefined, 'x'] } }}
${{}} | ${'a[0]'} | ${1} | ${{ a: [1] }}
${{}} | ${'a[0].b.c'} | ${1} | ${{ a: [{ b: { c: 1 } }] }}
`('property value:${value', ({ source, path, value, expected }) => {
expect(setOptionImmutably(source, path, value)).toEqual(expected);
});

View File

@ -68,6 +68,21 @@ export function setOptionImmutably<T extends object>(options: T, path: string |
const splat = !Array.isArray(path) ? path.split('.') : path;
const key = splat.shift()!;
if (key.endsWith(']')) {
const idx = key.lastIndexOf('[');
const index = +key.substring(idx + 1, key.length - 1);
const propKey = key.substr(0, idx);
let current = (options as Record<string, any>)[propKey];
const arr = Array.isArray(current) ? [...current] : [];
if (splat.length) {
if (current == null || typeof current !== 'object') {
current = {};
}
value = setOptionImmutably(current, splat, value);
}
arr[index] = value;
return { ...options, [propKey]: arr };
}
if (!splat.length) {
return { ...options, [key]: value };