Skip to content

Commit

Permalink
fix(theme): merge partial with empty initial partial (#1452)
Browse files Browse the repository at this point in the history
  • Loading branch information
nickofthyme authored Nov 2, 2021
1 parent 23567c4 commit d1e690a
Show file tree
Hide file tree
Showing 5 changed files with 96 additions and 24 deletions.
2 changes: 1 addition & 1 deletion packages/charts/api/charts.api.md
Original file line number Diff line number Diff line change
Expand Up @@ -1350,7 +1350,7 @@ export function mergeWithDefaultAnnotationLine(config?: Partial<LineAnnotationSt
// @public (undocumented)
export function mergeWithDefaultAnnotationRect(config?: Partial<RectAnnotationStyle>): RectAnnotationStyle;

// @public
// @public @deprecated
export function mergeWithDefaultTheme(theme: PartialTheme, defaultTheme?: Theme, auxiliaryThemes?: PartialTheme[]): Theme;

// @public (undocumented)
Expand Down
8 changes: 4 additions & 4 deletions packages/charts/src/state/selectors/get_chart_theme.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,8 @@
* Side Public License, v 1.
*/

import { mergePartial } from '../../utils/common';
import { LIGHT_THEME } from '../../utils/themes/light_theme';
import { mergeWithDefaultTheme } from '../../utils/themes/merge_utils';
import { PartialTheme, Theme } from '../../utils/themes/theme';
import { createCustomCachedSelector } from '../create_selector';
import { getSettingsSpecSelector } from './get_settings_specs';
Expand All @@ -19,12 +19,12 @@ export const getChartThemeSelector = createCustomCachedSelector(
);

function getTheme(baseTheme?: Theme, theme?: PartialTheme | PartialTheme[]): Theme {
const base = baseTheme || LIGHT_THEME;
const base = baseTheme ?? LIGHT_THEME;

if (Array.isArray(theme)) {
const [firstTheme, ...axillaryThemes] = theme;
return mergeWithDefaultTheme(firstTheme, base, axillaryThemes);
return mergePartial(base, firstTheme, {}, axillaryThemes);
}

return theme ? mergeWithDefaultTheme(theme, base) : base;
return theme ? mergePartial(base, theme) : base;
}
78 changes: 74 additions & 4 deletions packages/charts/src/utils/common.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -105,28 +105,46 @@ describe('common utilities', () => {
key6: 6,
};

it('should allow undefined object', () => {
const result = getAllKeys(undefined, [object1]);

expect(result).toEqual(new Set(['key1', 'key2']));
});

it('should return allow undefined objects', () => {
const result = getAllKeys(undefined, undefined);

expect(result).toEqual(new Set([]));
});

it('should remove duplicates', () => {
const result = getAllKeys(object1, [object1]);

expect(result).toEqual(new Set(['key1', 'key2']));
});

it('should return all keys from single object', () => {
const result = getAllKeys(object1);

expect(result).toEqual(['key1', 'key2']);
expect(result).toEqual(new Set(['key1', 'key2']));
});

it('should return all keys from all objects x 2', () => {
const result = getAllKeys(object1, [object2]);

expect(result).toEqual(['key1', 'key2', 'key3', 'key4']);
expect(result).toEqual(new Set(['key1', 'key2', 'key3', 'key4']));
});

it('should return all keys from single objects x 3', () => {
const result = getAllKeys(object1, [object2, object3]);

expect(result).toEqual(['key1', 'key2', 'key3', 'key4', 'key5', 'key6']);
expect(result).toEqual(new Set(['key1', 'key2', 'key3', 'key4', 'key5', 'key6']));
});

it('should return all keys from only defined objects', () => {
const result = getAllKeys(object1, [null, object2, {}, undefined]);

expect(result).toEqual(['key1', 'key2', 'key3', 'key4']);
expect(result).toEqual(new Set(['key1', 'key2', 'key3', 'key4']));
});
});

Expand Down Expand Up @@ -726,6 +744,58 @@ describe('common utilities', () => {
},
});
});

test('should traverse all partial keys', () => {
interface Test {
crosshair: {
line: {
dash?: number[];
strokeWidth: number;
};
band?: {
visible: boolean;
};
};
}
const customBase: Test = {
crosshair: {
line: {
strokeWidth: 1,
},
},
};
const partials: RecursivePartial<Test>[] = [
{
crosshair: {
line: {
dash: [4, 4],
},
},
},
{
crosshair: {
line: {
strokeWidth: 10,
},
band: {
visible: true,
},
},
},
];
const newBase = mergePartial(customBase, {}, {}, partials);
expect(newBase).toEqual({
crosshair: {
line: {
strokeWidth: 10,
dash: [4, 4],
},
band: {
visible: true,
},
},
});
});
});

describe('MergeOptions', () => {
Expand Down
31 changes: 16 additions & 15 deletions packages/charts/src/utils/common.ts
Original file line number Diff line number Diff line change
Expand Up @@ -255,17 +255,17 @@ export function getPartialValue<T>(base: T, partial?: RecursivePartial<T>, parti
* @param objects
* @internal
*/
export function getAllKeys(object: any, objects: any[] = []): string[] {
const initialKeys = object instanceof Map ? [...object.keys()] : Object.keys(object);

return objects.reduce((keys: any[], obj) => {
if (obj && typeof obj === 'object') {
const newKeys = obj instanceof Map ? obj.keys() : Object.keys(obj);
keys.push(...newKeys);
}
export function getAllKeys(object?: any, objects: any[] = []): Set<any> {
return new Set(
[object, ...objects].filter(Boolean).reduce((keys: any[], obj) => {
if (obj && typeof obj === 'object') {
const newKeys = obj ? (obj instanceof Map ? obj.keys() : Object.keys(obj)) : [];
keys.push(...newKeys);
}

return keys;
}, initialKeys);
return keys;
}, []),
);
}

/** @internal */
Expand Down Expand Up @@ -357,8 +357,9 @@ export function mergePartial<T>(

if (hasPartialObjectToMerge(base, partial, additionalPartials)) {
const mapCondition = !(baseClone instanceof Map) || options.mergeMaps;
if (partial !== undefined && (options.mergeOptionalPartialValues ?? true) && mapCondition) {
getAllKeys(partial, additionalPartials).forEach((key) => {
const partialKeys = getAllKeys(partial, additionalPartials);
if (partialKeys.size > 0 && (options.mergeOptionalPartialValues ?? true) && mapCondition) {
partialKeys.forEach((key) => {
if (baseClone instanceof Map) {
if (!baseClone.has(key)) {
baseClone.set(
Expand All @@ -370,9 +371,9 @@ export function mergePartial<T>(
}
} else if (!(key in baseClone)) {
baseClone[key] =
(partial as any)[key] !== undefined
(partial as any)?.[key] !== undefined
? (partial as any)[key]
: (additionalPartials.find((v: any) => v[key] !== undefined) || ({} as any))[key];
: (additionalPartials.find((v: any) => v?.[key] !== undefined) ?? ({} as any))[key];
}
});
}
Expand Down Expand Up @@ -404,7 +405,7 @@ export function mergePartial<T>(
return baseClone as any;
}

return Object.keys(base).reduce((newBase, key) => {
return Object.keys(baseClone).reduce((newBase, key) => {
const partialValue = partial && (partial as any)[key];
const partialValues = additionalPartials.map((v) => (typeof v === 'object' ? (v as any)[key] : undefined));
const baseValue = (base as any)[key];
Expand Down
1 change: 1 addition & 0 deletions packages/charts/src/utils/themes/merge_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ export function mergeWithDefaultAnnotationRect(config?: Partial<RectAnnotationSt
* @param auxiliaryThemes - additional themes to be merged
*
* @public
* @deprecated - Please use `baseTheme` and `theme` on Settings instead
*/
export function mergeWithDefaultTheme(
theme: PartialTheme,
Expand Down

0 comments on commit d1e690a

Please sign in to comment.