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:
parent
e6336fe3da
commit
89a2def126
@ -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) {
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
|
}
|
||||||
|
@ -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();
|
||||||
|
@ -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: {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user