2
0
mirror of https://github.com/iconify/iconify.git synced 2025-01-22 14:48:24 +00:00

Rewite icon set functions to allow deep aliases tree

This commit is contained in:
Vjacheslav Trushkin 2022-06-20 20:00:56 +03:00
parent e6336fe3da
commit 89a2def126
9 changed files with 156 additions and 172 deletions

View File

@ -11,10 +11,11 @@ import {
*/ */
export function mergeCustomisations<T extends FullIconCustomisations>( export function mergeCustomisations<T extends FullIconCustomisations>(
defaults: T, defaults: T,
item: IconifyIconCustomisations item: IconifyIconCustomisations,
keepOtherProps = true
): T { ): T {
// Merge transformations // Merge transformations
const result = mergeIconTransformations(defaults, item); const result = mergeIconTransformations(defaults, item, keepOtherProps);
// Merge dimensions // Merge dimensions
for (const key in defaultIconSizeCustomisations) { for (const key in defaultIconSizeCustomisations) {

View File

@ -1,7 +1,8 @@
import type { IconifyDimenisons, IconifyJSON } from '@iconify/types'; import type { IconifyJSON } from '@iconify/types';
import { defaultIconProps, defaultIconDimensions } from '../icon/defaults'; import { defaultIconProps } from '../icon/defaults';
import type { IconifyIcon, FullIconifyIcon } from '../icon/defaults'; import type { IconifyIcon, FullIconifyIcon } from '../icon/defaults';
import { mergeIconData } from '../icon/merge'; import { mergeIconData } from '../icon/merge';
import { getIconsTree } from './tree';
/** /**
* Get data for icon * Get data for icon
@ -21,54 +22,43 @@ export function getIconData(
name: string, name: string,
full = false full = false
): FullIconifyIcon | IconifyIcon | null { ): FullIconifyIcon | IconifyIcon | null {
function getIcon(name: string, iteration: number): IconifyIcon | null { const icons = data.icons;
if (data.icons[name] !== void 0) { const aliases = data.aliases || {};
// Return icon
return Object.assign({}, data.icons[name]); let currentProps = {} as IconifyIcon;
// Parse parent item
function parse(name: string) {
currentProps = mergeIconData(
icons[name] || aliases[name],
currentProps,
false
);
} }
// Check loop const icon = icons[name];
if (iteration > 5) { if (icon) {
// Parse only icon
parse(name);
} else {
// Resolve tree
const tree = getIconsTree(data, [name])[name];
if (!tree) {
return null; return null;
} }
parse(name);
// Check if alias exists tree.forEach(parse);
const aliases = data.aliases;
if (aliases && aliases[name] !== void 0) {
const item = aliases[name];
const result = getIcon(item.parent, iteration + 1);
if (result) {
return mergeIconData(result, item);
}
return result;
} }
// Check if character exists // Add default values
const chars = data.chars; currentProps = mergeIconData(
if (!iteration && chars && chars[name] !== void 0) { data,
return getIcon(chars[name], iteration + 1); currentProps,
} false
) as unknown as IconifyIcon;
return null;
}
const result = getIcon(name, 0);
// Add default properties
if (result) {
for (const key in defaultIconDimensions) {
if (
result[key as keyof IconifyDimenisons] === void 0 &&
data[key as keyof IconifyDimenisons] !== void 0
) {
(result as unknown as Record<string, unknown>)[key] =
data[key as keyof IconifyDimenisons];
}
}
}
// Return icon // Return icon
return result && full return full
? Object.assign({}, defaultIconProps, result) ? Object.assign({}, defaultIconProps, currentProps)
: result; : currentProps;
} }

View File

@ -1,11 +1,11 @@
/* eslint-disable @typescript-eslint/no-non-null-assertion */ import type { IconifyAliases, IconifyJSON } from '@iconify/types';
import type { IconifyJSON } from '@iconify/types'; import { defaultIconDimensions } from '../icon/defaults';
import { defaultIconProps } from '../icon/defaults'; import { getIconsTree } from './tree';
/** /**
* Optional properties that must be copied when copying icon set * Optional properties that must be copied when copying icon set
*/ */
export const propsToCopy = Object.keys(defaultIconProps).concat([ export const propsToCopy = Object.keys(defaultIconDimensions).concat([
'provider', 'provider',
]) as (keyof IconifyJSON)[]; ]) as (keyof IconifyJSON)[];
@ -14,61 +14,43 @@ export const propsToCopy = Object.keys(defaultIconProps).concat([
*/ */
export function getIcons( export function getIcons(
data: IconifyJSON, data: IconifyJSON,
icons: string[], names: string[],
not_found?: boolean not_found?: boolean
): IconifyJSON | null { ): IconifyJSON | null {
const icons = Object.create(null) as IconifyJSON['icons'];
const aliases = Object.create(null) as IconifyAliases;
const result: IconifyJSON = { const result: IconifyJSON = {
prefix: data.prefix, prefix: data.prefix,
icons: Object.create(null) as never, icons,
}; };
const tested: Set<string> = new Set();
const sourceIcons = data.icons;
const sourceAliases = data.aliases || {};
const tree = getIconsTree(data, names);
let empty = true; let empty = true;
function copy(name: string, iteration: number): boolean { // Copy all icons
if (iteration > 5 || tested.has(name)) { for (const name in tree) {
// Already copied or too much nesting if (!tree[name]) {
return true; // Failed
if (not_found && names.indexOf(name) !== -1) {
// Add to not_found
(result.not_found || (result.not_found = [])).push(name);
} }
tested.add(name); } else if (sourceIcons[name]) {
// Icon
// Check for icon icons[name] = {
if (data.icons[name] !== void 0) { ...sourceIcons[name],
empty = false;
result.icons[name] = { ...data.icons[name] };
return true;
}
// Check for alias
const aliases = data.aliases;
if (aliases && aliases[name] !== void 0) {
const copied = copy(aliases[name].parent, iteration + 1);
if (copied) {
if (result.aliases === void 0) {
result.aliases = Object.create(null) as never;
}
result.aliases[name] = { ...aliases[name] };
}
return copied;
}
// Check for character, return as alias
const chars = data.chars;
if (chars && chars[name] !== void 0) {
const parent = chars[name];
const copied = copy(parent, iteration + 1);
if (copied) {
if (result.aliases === void 0) {
result.aliases = Object.create(null) as never;
}
result.aliases[name] = {
parent,
}; };
empty = false;
} else {
// Alias
aliases[name] = {
...sourceAliases[name],
};
result.aliases = aliases;
} }
return copied;
}
// Not found
return false;
} }
// Copy common properties // Copy common properties
@ -78,15 +60,5 @@ export function getIcons(
} }
}); });
// Copy all icons
icons.forEach((name) => {
if (!copy(name, 0) && not_found === true) {
if (result.not_found === void 0) {
result.not_found = [];
}
result.not_found.push(name);
}
});
return empty && not_found !== true ? null : result; return empty && not_found !== true ? null : result;
} }

View File

@ -11,18 +11,20 @@ export type ParentIconsTree = Record<string, ParentIconsList | null>;
* *
* Returns parent icon for each icon * Returns parent icon for each icon
*/ */
export function getIconsTree(data: IconifyJSON): ParentIconsTree { export function getIconsTree(
const resolved = Object.create(null) as ParentIconsTree; data: IconifyJSON,
names?: string[]
// Add all icons ): ParentIconsTree {
for (const key in data.icons) { const icons = data.icons;
resolved[key] = [];
}
// Add all aliases
const aliases = data.aliases || {}; const aliases = data.aliases || {};
function resolveAlias(name: string): ParentIconsList | null { const resolved = Object.create(null) as ParentIconsTree;
function resolve(name: string): ParentIconsList | null {
if (icons[name]) {
return (resolved[name] = []);
}
if (resolved[name] === void 0) { if (resolved[name] === void 0) {
// Mark as failed if parent alias points to this icon to avoid infinite loop // Mark as failed if parent alias points to this icon to avoid infinite loop
resolved[name] = null; resolved[name] = null;
@ -31,7 +33,7 @@ export function getIconsTree(data: IconifyJSON): ParentIconsTree {
const parent = aliases[name] && aliases[name].parent; const parent = aliases[name] && aliases[name].parent;
// Get value for parent // Get value for parent
const value = parent && resolveAlias(parent); const value = parent && resolve(parent);
if (value) { if (value) {
resolved[name] = [parent].concat(value); resolved[name] = [parent].concat(value);
} }
@ -40,9 +42,8 @@ export function getIconsTree(data: IconifyJSON): ParentIconsTree {
return resolved[name]; return resolved[name];
} }
for (const name in aliases) { // Resolve only required icons
resolveAlias(name); (names || Object.keys(aliases).concat(Object.keys(icons))).forEach(resolve);
}
return resolved; return resolved;
} }

View File

@ -1,22 +1,43 @@
import type { IconifyDimenisons, IconifyOptional } from '@iconify/types'; import type {
IconifyDimenisons,
IconifyIcon,
IconifyOptional,
IconifyTransformations,
} from '@iconify/types';
import { defaultIconDimensions } from './defaults'; import { defaultIconDimensions } from './defaults';
import { mergeIconTransformations } from './transformations'; import { mergeIconTransformations } from './transformations';
// Props to copy: all icon properties, except transformations
type PropsToCopy = Omit<IconifyIcon, keyof IconifyTransformations>;
const propsToMerge: Required<PropsToCopy> = {
...defaultIconDimensions,
body: '',
};
/** /**
* Merge icon and alias * Merge icon and alias
*
* Can also be used to merge default values and icon
*/ */
export function mergeIconData<T extends IconifyOptional>( export function mergeIconData<T extends IconifyOptional>(
icon: T, parent: T,
alias: IconifyOptional child: IconifyOptional | IconifyIcon,
keepOtherParentProps = true
): T { ): T {
// Merge transformations, while keeping other props // Merge transformations
const result = mergeIconTransformations(icon, alias); const result = mergeIconTransformations(
parent,
child,
keepOtherParentProps
);
// Merge dimensions // Merge icon properties that aren't transformations
for (const key in defaultIconDimensions) { for (const key in propsToMerge) {
const prop = key as keyof IconifyDimenisons; const prop = key as keyof IconifyDimenisons;
if (alias[prop] !== void 0) { if (child[prop] !== void 0) {
result[prop] = alias[prop]; result[prop] = child[prop];
} else if (parent[prop] !== void 0) {
result[prop] = parent[prop];
} }
} }

View File

@ -1,23 +1,22 @@
import type { IconifyTransformations } from '@iconify/types'; import type { IconifyTransformations } from '@iconify/types';
/** /**
* Merge transformations. Also copies other properties from first parameter * Merge transformations
*/ */
export function mergeIconTransformations<T extends IconifyTransformations>( export function mergeIconTransformations<T extends IconifyTransformations>(
obj1: T, obj1: T,
obj2: IconifyTransformations obj2: IconifyTransformations,
keepOtherProps = true
): T { ): T {
const result = { const result = keepOtherProps ? { ...obj1 } : ({} as T);
...obj1, if (obj1.hFlip || obj2.hFlip) {
}; result.hFlip = obj1.hFlip !== obj2.hFlip;
if (obj2.hFlip) {
result.hFlip = !result.hFlip;
} }
if (obj2.vFlip) { if (obj1.vFlip || obj2.vFlip) {
result.vFlip = !result.vFlip; result.vFlip = obj1.vFlip !== obj2.vFlip;
} }
if (obj2.rotate) { if (obj1.rotate || obj2.rotate) {
result.rotate = ((result.rotate || 0) + obj2.rotate) % 4; result.rotate = ((obj1.rotate || 0) + (obj2.rotate || 0)) % 4;
} }
return result; return result;
} }

View File

@ -33,3 +33,20 @@ export function unmergeObjects<T extends Record<string, unknown>>(
} }
return result; return result;
} }
/**
* Get common properties in 2 objects
*/
export function commonObjectProps<T extends Record<string, unknown>>(
item: unknown,
reference: T
): Partial<T> {
const result = {} as T;
for (const key in reference) {
const value = (item as T)[key];
if (value !== void 0) {
result[key] = value;
}
}
return result;
}

View File

@ -137,38 +137,10 @@ describe('Testing retrieving icons from icon set', () => {
}); });
// Character // Character
expect(getIcons(data, ['f00'])).toEqual({ expect(getIcons(data, ['f00'])).toBeNull();
prefix: 'foo',
icons: {
bar2: {
body: '<g />',
},
},
aliases: {
f00: {
parent: 'bar2',
},
},
});
// Character that points to alias // Character that points to alias
expect(getIcons(data, ['f02'])).toEqual({ expect(getIcons(data, ['f02'])).toBeNull();
prefix: 'foo',
icons: {
bar: {
body: '<g />',
},
},
aliases: {
f02: {
parent: 'foo',
},
foo: {
parent: 'bar',
hFlip: true,
},
},
});
// Bad character // Bad character
expect(getIcons(data, ['f04'])).toBeNull(); expect(getIcons(data, ['f04'])).toBeNull();

View File

@ -173,6 +173,7 @@ describe('Testing parsing icon set', () => {
'alias2z4', 'alias2z4',
'alias2z5', 'alias2z5',
'alias2z6', 'alias2z6',
'alias2z7',
]; ];
const namesCopy = names.slice(0); const namesCopy = names.slice(0);
@ -275,6 +276,17 @@ describe('Testing parsing icon set', () => {
vFlip: true, vFlip: true,
rotate: 1, rotate: 1,
}, },
alias2z7: {
// alias of alias2z6
body: '<path d="icon2" />',
width: 21,
height: 24,
top: 0,
left: 0,
hFlip: false,
vFlip: true,
rotate: 1,
},
}; };
// Do stuff // Do stuff
@ -333,7 +345,6 @@ describe('Testing parsing icon set', () => {
}, },
alias2z7: { alias2z7: {
// 7 parents: alias2z6, alias2z5, alias2z4, alias2z3, alias2z, alias2f, icon2 // 7 parents: alias2z6, alias2z5, alias2z4, alias2z3, alias2z, alias2f, icon2
// nesting is too deep and should not be parsed
parent: 'alias2z6', parent: 'alias2z6',
}, },
alias3: { alias3: {