-
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathsvg64.ts
93 lines (72 loc) · 2.13 KB
/
svg64.ts
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
81
82
83
84
85
86
87
88
89
90
91
92
93
// prettier-ignore
const CHARS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
const PREFIX = 'data:image/svg+xml;base64,';
export const utf8Encode = (input: string): string => {
input = input.replace(/\r\n/g, '\n');
let i = 0;
let output = '';
for (; i < input.length; i++) {
const c = input.charCodeAt(i);
if (c < 128) {
output += String.fromCharCode(c);
} else if (c > 127 && c < 2048) {
output += String.fromCharCode((c >> 6) | 192);
output += String.fromCharCode((c & 63) | 128);
} else {
output += String.fromCharCode((c >> 12) | 224);
output += String.fromCharCode(((c >> 6) & 63) | 128);
output += String.fromCharCode((c & 63) | 128);
}
}
return output;
};
export const encode = (input: string): string => {
let i = 0;
let chr1: number;
let chr2: number;
let chr3: number;
let enc1: number;
let enc2: number;
let enc3: number;
let enc4: number;
let output = '';
input = utf8Encode(input);
while (i < input.length) {
chr1 = input.charCodeAt(i++);
chr2 = input.charCodeAt(i++);
chr3 = input.charCodeAt(i++);
enc1 = chr1 >> 2;
enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
enc4 = chr3 & 63;
if (isNaN(chr2)) {
enc3 = enc4 = 64;
} else if (isNaN(chr3)) {
enc4 = 64;
}
output = output + CHARS.charAt(enc1) + CHARS.charAt(enc2) + CHARS.charAt(enc3) + CHARS.charAt(enc4);
}
return output;
};
export const detectInputType = (input: string | SVGElement): 'string' | 'element' | void => {
if (typeof input === 'string') {
return 'string';
}
if (typeof SVGElement !== 'undefined' && input instanceof SVGElement) {
return 'element';
}
};
export const getBase64 = (input: string) => PREFIX + encode(input);
export const convertElement = (input: SVGElement): string => getBase64(new XMLSerializer().serializeToString(input));
export const svg64 = (input: string | SVGElement): string => {
const type = detectInputType(input);
switch (type) {
case 'string':
return getBase64(input as string);
case 'element':
return convertElement(input as SVGElement);
default:
return input as string;
}
};
export default svg64;