-
Notifications
You must be signed in to change notification settings - Fork 1
/
iconBase.tsx
80 lines (72 loc) · 1.87 KB
/
iconBase.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
import * as React from "react";
import { IconContext, DefaultContext } from "./iconContext";
export interface IconTree {
tag: string;
attr: { [key: string]: string };
child: IconTree[];
}
function Tree2Element(tree: IconTree[]): React.ReactElement[] {
return (
tree &&
tree.map((node, i) =>
React.createElement(
node.tag,
{ key: i, ...node.attr },
Tree2Element(node.child)
)
)
);
}
export function GenIcon(data: IconTree) {
// eslint-disable-next-line react/display-name
return (props: IconBaseProps) => (
<IconBase attr={{ ...data.attr }} {...props}>
{Tree2Element(data.child)}
</IconBase>
);
}
export interface IconBaseProps extends React.SVGAttributes<SVGElement> {
children?: React.ReactNode;
size?: string | number;
color?: string;
title?: string;
}
export type IconType = (props: IconBaseProps) => JSX.Element;
export function IconBase(
props: IconBaseProps & { attr?: Record<string, string> }
): JSX.Element {
const elem = (conf: IconContext) => {
const { attr, title, ...svgProps } = props;
let className;
if (conf.className) className = conf.className;
if (props.className)
className = (className ? className + " " : "") + props.className;
return (
<svg
stroke="currentColor"
fill="currentColor"
strokeWidth="0"
{...conf.attr}
{...attr}
{...svgProps}
className={className}
style={{
color: props.color || conf.color,
...conf.style,
...props.style,
}}
xmlns="http://www.w3.org/2000/svg"
>
{title && <title>{title}</title>}
{props.children}
</svg>
);
};
return IconContext !== undefined ? (
<IconContext.Consumer>
{(conf: IconContext) => elem(conf)}
</IconContext.Consumer>
) : (
elem(DefaultContext)
);
}