forked from stolksdorf/vitreum
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathheadtags.js
74 lines (64 loc) · 2.08 KB
/
headtags.js
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
const React = require('react');
const obj2props = (obj)=>Object.entries(obj).map(([k,v])=>`${k}="${v}"`).join(' ');
const toStr = (chld)=>Array.isArray(chld) ? chld.join('') : chld;
const onServer = (typeof window === 'undefined');
const injectTag = require('./utils/injectTag.js');
let NamedTags = {};
let UnnamedTags = [];
const HeadComponents = {
Title({ children }){
if(onServer) NamedTags.title = `<title>${toStr(children)}</title>`;
React.useEffect(()=>{document.title = toStr(children)}, [children]);
return null;
},
Favicon({ type = 'image/png', href = '', rel='icon', id= 'favicon'}){
if(onServer) NamedTags.favicon = `<link rel='shortcut icon' type="${type}" id="${id}" href="${href}" />`
React.useEffect(()=>{document.getElementById(id).href=href}, [id, href]);
return null;
},
Description({ children }){
if(onServer) NamedTags.description = `<meta name='description' content='${toStr(children)}' />`
return null;
},
Noscript({ children }){
if(onServer) UnnamedTags.push(`<noscript>${toStr(children)}<\/noscript>`);
return null;
},
Script({ children=[], ...props }){
if(onServer) {
UnnamedTags.push(children.length
? `<script ${obj2props(props)}>${toStr(children)}\<\/script>`
: `<script ${obj2props(props)} />`
);
}
return null;
},
Meta(props) {
if (onServer) {}
let tag = `<meta ${obj2props(props)} />`;
props.property || props.name ? NamedTags[props.property || props.name] = tag : UnnamedTags.push(tag);
React.useEffect(() => {
document.getElementsByTagName('head')[0].insertAdjacentHTML('beforeend', Object.values(NamedTags).join('\n'));
},[NamedTags]);
return null;
},
Style({ children, type='text/css' }){
if(onServer) UnnamedTags.push(`<style type="${type}">${toStr(children)}</style>`);
return null;
}
};
const Inject = ({tag, children, ...props})=>{
React.useEffect(()=>{
injectTag(tag, props, children);
}, []);
return null;
};
module.exports = {
Inject,
...HeadComponents,
generate : ()=>Object.values(NamedTags).concat(UnnamedTags).join('\n'),
flush : ()=>{
NamedTags = {};
UnnamedTags = [];
}
};