@luxfi/core 4.4.15 → 4.4.16
Sign up to get free protection for your applications and to get access to all the features.
- package/components/access-code-input.tsx +71 -71
- package/components/auth/auth-listener.tsx +29 -29
- package/components/auth/auth-token/clear-auth-token.tsx +12 -12
- package/components/auth/auth-token/set-auth-token.tsx +16 -16
- package/components/auth/common-auth-domains.ts +16 -16
- package/components/auth/login-panel.tsx +103 -100
- package/components/chat-widget.tsx +77 -77
- package/components/commerce/bag-button.tsx +67 -67
- package/components/commerce/checkout-panel/close-button.tsx +26 -26
- package/components/commerce/checkout-panel/dt-bag-carousel.tsx +36 -36
- package/components/commerce/checkout-panel/dt-checkout-panel.tsx +66 -66
- package/components/commerce/checkout-panel/index.tsx +123 -123
- package/components/commerce/checkout-panel/links-row.tsx +21 -21
- package/components/commerce/checkout-panel/mb-checkout-panel.tsx +55 -55
- package/components/commerce/checkout-panel/steps-indicator.tsx +39 -39
- package/components/commerce/checkout-panel/thank-you.tsx +18 -18
- package/components/commerce/desktop-bag-popup.tsx +78 -78
- package/components/commerce/mobile-bag-drawer.tsx +51 -51
- package/components/commerce/mobile-menu-toggle-button.tsx +35 -35
- package/components/commerce/mobile-nav-menu.tsx +64 -64
- package/components/contact-dialog/contact-form.tsx +112 -112
- package/components/contact-dialog/disclaimer.tsx +13 -13
- package/components/contact-dialog/index.tsx +64 -64
- package/components/copyright.tsx +21 -21
- package/components/footer.tsx +77 -77
- package/components/header/desktop.tsx +54 -54
- package/components/header/index.tsx +26 -26
- package/components/header/mobile.tsx +161 -161
- package/components/header/theme-toggle.tsx +26 -26
- package/components/icons/bag-icon.tsx +10 -10
- package/components/icons/github.tsx +14 -14
- package/components/icons/index.tsx +35 -35
- package/components/icons/lux-logo.tsx +10 -10
- package/components/icons/secure-delivery.tsx +13 -13
- package/components/icons/social-icon.tsx +35 -35
- package/components/icons/social-svg.css +3 -3
- package/components/icons/youtube-logo.tsx +59 -59
- package/components/index.ts +27 -27
- package/components/logo.tsx +81 -81
- package/components/mini-chart/index.tsx +7 -7
- package/components/mini-chart/mini-chart-props.ts +43 -43
- package/components/mini-chart/mini-chart.tsx +85 -85
- package/components/mini-chart/wrapper.tsx +23 -23
- package/components/not-found/index.tsx +27 -27
- package/components/not-found/not-found-content.mdx +5 -5
- package/components/root-layout.tsx +71 -71
- package/components/scripts.tsx +23 -23
- package/conf/index.ts +50 -50
- package/environment.d.ts +5 -5
- package/next/analytics/fpixel.ts +15 -15
- package/next/analytics/google-analytics.ts +13 -13
- package/next/analytics/index.ts +3 -3
- package/next/analytics/pixel-analytics.tsx +54 -54
- package/next/determine-device-mw.ts +16 -16
- package/next/font/get-app-router-font-classes.ts +12 -12
- package/next/font/load-and-return-lux-next-fonts-on-import.ts +68 -68
- package/next/font/next-font-desc.ts +27 -27
- package/next/font/pages-router-font-vars.tsx +18 -18
- package/next/head-metadata/from-next/metadata-types.ts +158 -158
- package/next/head-metadata/from-next/opengraph-types.ts +267 -267
- package/next/head-metadata/from-next/twitter-types.ts +92 -92
- package/next/head-metadata/index.tsx +208 -208
- package/package.json +73 -73
- package/server-actions/firebase-app.ts +14 -14
- package/server-actions/index.ts +5 -5
- package/server-actions/store-contact.ts +51 -51
- package/site-def/footer/community.tsx +67 -67
- package/site-def/footer/company.ts +37 -37
- package/site-def/footer/ecosystem.ts +37 -37
- package/site-def/footer/index.tsx +26 -26
- package/site-def/footer/legal.ts +28 -28
- package/site-def/footer/network.ts +45 -45
- package/site-def/footer/svg/warpcast-logo.svg +11 -11
- package/site-def/index.ts +2 -2
- package/site-def/main-nav.ts +35 -35
- package/site-def/site-def.ts +36 -36
- package/style/lux-colors.css +85 -85
- package/style/lux-global.css +30 -30
- package/tailwind/fontFamily.tailwind.lux.ts +18 -18
- package/tailwind/index.ts +2 -2
- package/tailwind/lux-tw-fonts.ts +39 -39
- package/tailwind/tailwind.config.lux-preset.ts +10 -10
- package/tsconfig.json +10 -10
- package/types/contact-info.ts +10 -10
@@ -1,92 +1,92 @@
|
|
1
|
-
// Reference: https://developer.twitter.com/en/docs/twitter-for-websites/cards/overview/markup
|
2
|
-
|
3
|
-
import type { AbsoluteTemplateString, TemplateString } from './metadata-types'
|
4
|
-
|
5
|
-
export type Twitter =
|
6
|
-
| TwitterSummary
|
7
|
-
| TwitterSummaryLargeImage
|
8
|
-
| TwitterPlayer
|
9
|
-
| TwitterApp
|
10
|
-
| TwitterMetadata
|
11
|
-
|
12
|
-
type TwitterMetadata = {
|
13
|
-
// defaults to card="summary"
|
14
|
-
site?: string // username for account associated to the site itself
|
15
|
-
siteId?: string // id for account associated to the site itself
|
16
|
-
creator?: string // username for the account associated to the creator of the content on the site
|
17
|
-
creatorId?: string // id for the account associated to the creator of the content on the site
|
18
|
-
description?: string
|
19
|
-
title?: string | TemplateString
|
20
|
-
images?: TwitterImage | Array<TwitterImage>
|
21
|
-
}
|
22
|
-
type TwitterSummary = TwitterMetadata & {
|
23
|
-
card: 'summary'
|
24
|
-
}
|
25
|
-
type TwitterSummaryLargeImage = TwitterMetadata & {
|
26
|
-
card: 'summary_large_image'
|
27
|
-
}
|
28
|
-
type TwitterPlayer = TwitterMetadata & {
|
29
|
-
card: 'player'
|
30
|
-
players: TwitterPlayerDescriptor | Array<TwitterPlayerDescriptor>
|
31
|
-
}
|
32
|
-
type TwitterApp = TwitterMetadata & {
|
33
|
-
card: 'app'
|
34
|
-
app: TwitterAppDescriptor
|
35
|
-
}
|
36
|
-
export type TwitterAppDescriptor = {
|
37
|
-
id: {
|
38
|
-
iphone?: string | number
|
39
|
-
ipad?: string | number
|
40
|
-
googleplay?: string
|
41
|
-
}
|
42
|
-
url?: {
|
43
|
-
iphone?: string | URL
|
44
|
-
ipad?: string | URL
|
45
|
-
googleplay?: string | URL
|
46
|
-
}
|
47
|
-
name?: string
|
48
|
-
}
|
49
|
-
|
50
|
-
export type TwitterImage = string | TwitterImageDescriptor | URL
|
51
|
-
type TwitterImageDescriptor = {
|
52
|
-
url: string | URL
|
53
|
-
alt?: string
|
54
|
-
secureUrl?: string | URL
|
55
|
-
type?: string
|
56
|
-
width?: string | number
|
57
|
-
height?: string | number
|
58
|
-
}
|
59
|
-
type TwitterPlayerDescriptor = {
|
60
|
-
playerUrl: string | URL
|
61
|
-
streamUrl: string | URL
|
62
|
-
width: number
|
63
|
-
height: number
|
64
|
-
}
|
65
|
-
|
66
|
-
type ResolvedTwitterImage = {
|
67
|
-
url: string | URL
|
68
|
-
alt?: string
|
69
|
-
secureUrl?: string | URL
|
70
|
-
type?: string
|
71
|
-
width?: string | number
|
72
|
-
height?: string | number
|
73
|
-
}
|
74
|
-
type ResolvedTwitterSummary = {
|
75
|
-
site: string | null
|
76
|
-
siteId: string | null
|
77
|
-
creator: string | null
|
78
|
-
creatorId: string | null
|
79
|
-
description: string | null
|
80
|
-
title: AbsoluteTemplateString
|
81
|
-
images?: Array<ResolvedTwitterImage>
|
82
|
-
}
|
83
|
-
type ResolvedTwitterPlayer = ResolvedTwitterSummary & {
|
84
|
-
players: Array<TwitterPlayerDescriptor>
|
85
|
-
}
|
86
|
-
type ResolvedTwitterApp = ResolvedTwitterSummary & { app: TwitterAppDescriptor }
|
87
|
-
|
88
|
-
export type ResolvedTwitterMetadata =
|
89
|
-
| ({ card: 'summary' } & ResolvedTwitterSummary)
|
90
|
-
| ({ card: 'summary_large_image' } & ResolvedTwitterSummary)
|
91
|
-
| ({ card: 'player' } & ResolvedTwitterPlayer)
|
92
|
-
| ({ card: 'app' } & ResolvedTwitterApp)
|
1
|
+
// Reference: https://developer.twitter.com/en/docs/twitter-for-websites/cards/overview/markup
|
2
|
+
|
3
|
+
import type { AbsoluteTemplateString, TemplateString } from './metadata-types'
|
4
|
+
|
5
|
+
export type Twitter =
|
6
|
+
| TwitterSummary
|
7
|
+
| TwitterSummaryLargeImage
|
8
|
+
| TwitterPlayer
|
9
|
+
| TwitterApp
|
10
|
+
| TwitterMetadata
|
11
|
+
|
12
|
+
type TwitterMetadata = {
|
13
|
+
// defaults to card="summary"
|
14
|
+
site?: string // username for account associated to the site itself
|
15
|
+
siteId?: string // id for account associated to the site itself
|
16
|
+
creator?: string // username for the account associated to the creator of the content on the site
|
17
|
+
creatorId?: string // id for the account associated to the creator of the content on the site
|
18
|
+
description?: string
|
19
|
+
title?: string | TemplateString
|
20
|
+
images?: TwitterImage | Array<TwitterImage>
|
21
|
+
}
|
22
|
+
type TwitterSummary = TwitterMetadata & {
|
23
|
+
card: 'summary'
|
24
|
+
}
|
25
|
+
type TwitterSummaryLargeImage = TwitterMetadata & {
|
26
|
+
card: 'summary_large_image'
|
27
|
+
}
|
28
|
+
type TwitterPlayer = TwitterMetadata & {
|
29
|
+
card: 'player'
|
30
|
+
players: TwitterPlayerDescriptor | Array<TwitterPlayerDescriptor>
|
31
|
+
}
|
32
|
+
type TwitterApp = TwitterMetadata & {
|
33
|
+
card: 'app'
|
34
|
+
app: TwitterAppDescriptor
|
35
|
+
}
|
36
|
+
export type TwitterAppDescriptor = {
|
37
|
+
id: {
|
38
|
+
iphone?: string | number
|
39
|
+
ipad?: string | number
|
40
|
+
googleplay?: string
|
41
|
+
}
|
42
|
+
url?: {
|
43
|
+
iphone?: string | URL
|
44
|
+
ipad?: string | URL
|
45
|
+
googleplay?: string | URL
|
46
|
+
}
|
47
|
+
name?: string
|
48
|
+
}
|
49
|
+
|
50
|
+
export type TwitterImage = string | TwitterImageDescriptor | URL
|
51
|
+
type TwitterImageDescriptor = {
|
52
|
+
url: string | URL
|
53
|
+
alt?: string
|
54
|
+
secureUrl?: string | URL
|
55
|
+
type?: string
|
56
|
+
width?: string | number
|
57
|
+
height?: string | number
|
58
|
+
}
|
59
|
+
type TwitterPlayerDescriptor = {
|
60
|
+
playerUrl: string | URL
|
61
|
+
streamUrl: string | URL
|
62
|
+
width: number
|
63
|
+
height: number
|
64
|
+
}
|
65
|
+
|
66
|
+
type ResolvedTwitterImage = {
|
67
|
+
url: string | URL
|
68
|
+
alt?: string
|
69
|
+
secureUrl?: string | URL
|
70
|
+
type?: string
|
71
|
+
width?: string | number
|
72
|
+
height?: string | number
|
73
|
+
}
|
74
|
+
type ResolvedTwitterSummary = {
|
75
|
+
site: string | null
|
76
|
+
siteId: string | null
|
77
|
+
creator: string | null
|
78
|
+
creatorId: string | null
|
79
|
+
description: string | null
|
80
|
+
title: AbsoluteTemplateString
|
81
|
+
images?: Array<ResolvedTwitterImage>
|
82
|
+
}
|
83
|
+
type ResolvedTwitterPlayer = ResolvedTwitterSummary & {
|
84
|
+
players: Array<TwitterPlayerDescriptor>
|
85
|
+
}
|
86
|
+
type ResolvedTwitterApp = ResolvedTwitterSummary & { app: TwitterAppDescriptor }
|
87
|
+
|
88
|
+
export type ResolvedTwitterMetadata =
|
89
|
+
| ({ card: 'summary' } & ResolvedTwitterSummary)
|
90
|
+
| ({ card: 'summary_large_image' } & ResolvedTwitterSummary)
|
91
|
+
| ({ card: 'player' } & ResolvedTwitterPlayer)
|
92
|
+
| ({ card: 'app' } & ResolvedTwitterApp)
|
@@ -1,208 +1,208 @@
|
|
1
|
-
import React from 'react'
|
2
|
-
|
3
|
-
import type { Metadata } from 'next'
|
4
|
-
import Head from "next/head"
|
5
|
-
|
6
|
-
import type {
|
7
|
-
IconDescriptor,
|
8
|
-
TemplateString,
|
9
|
-
Author,
|
10
|
-
ThemeColorDescriptor
|
11
|
-
} from './from-next/metadata-types'
|
12
|
-
|
13
|
-
import type { OpenGraph, OGImage } from './from-next/opengraph-types'
|
14
|
-
import type { Twitter, TwitterImage } from './from-next/twitter-types'
|
15
|
-
|
16
|
-
/*
|
17
|
-
NOTE: This is ONLY for sites that use the pages router in next.
|
18
|
-
The app router does this automatically
|
19
|
-
*/
|
20
|
-
|
21
|
-
const getURLasString = (url: string | URL) => {
|
22
|
-
return (
|
23
|
-
(typeof url === 'string') ? (url as string) : (url.href)
|
24
|
-
)
|
25
|
-
}
|
26
|
-
|
27
|
-
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
28
|
-
const Icons: React.FC<{icons: IconDescriptor[]}> = ({
|
29
|
-
icons
|
30
|
-
}) => {
|
31
|
-
return <Head>
|
32
|
-
{icons.map(({url, ...rest}: IconDescriptor, index) => (
|
33
|
-
<link {...rest} href={getURLasString(url)} key={`icons-${index}`}/>
|
34
|
-
))}
|
35
|
-
</Head>
|
36
|
-
}
|
37
|
-
|
38
|
-
export const getTitleFromTemplateString = (title: string | TemplateString | null | undefined): string | null => {
|
39
|
-
|
40
|
-
if (!title) {
|
41
|
-
return null
|
42
|
-
}
|
43
|
-
if (typeof title === 'object') {
|
44
|
-
if ('default' in title) {
|
45
|
-
return title.default
|
46
|
-
}
|
47
|
-
else if ('absolute' in title) {
|
48
|
-
return title.absolute
|
49
|
-
}
|
50
|
-
}
|
51
|
-
return title as string
|
52
|
-
}
|
53
|
-
|
54
|
-
const Authors: React.FC<{
|
55
|
-
authors: null | undefined | Author | Array<Author>
|
56
|
-
}> = ({
|
57
|
-
authors
|
58
|
-
}) => {
|
59
|
-
|
60
|
-
const Author: React.FC<{author: Author}> = ({author}) => (<>
|
61
|
-
{author.name && <meta name="author" content={author.name} />}
|
62
|
-
{author.url && <link rel="author" href={getURLasString(author.url)}/>}
|
63
|
-
</>)
|
64
|
-
|
65
|
-
if (!authors) {
|
66
|
-
return null
|
67
|
-
}
|
68
|
-
|
69
|
-
if (Array.isArray(authors)) {
|
70
|
-
return (<>
|
71
|
-
{authors.map((el: Author, index) => (
|
72
|
-
<Author author={el} key={`authors-${index}`} />
|
73
|
-
))}
|
74
|
-
</>)
|
75
|
-
}
|
76
|
-
return (<Author author={authors as Author} />)
|
77
|
-
}
|
78
|
-
|
79
|
-
const Keywords: React.FC<{keywords: undefined | null | string | Array<string>}> = ({
|
80
|
-
keywords
|
81
|
-
}) => {
|
82
|
-
if (!keywords) return null
|
83
|
-
const content = (Array.isArray(keywords) ? keywords.join(', ') : keywords as string)
|
84
|
-
return (<meta name="keywords" content={content} />)
|
85
|
-
}
|
86
|
-
|
87
|
-
const ThemeColor: React.FC<{
|
88
|
-
thColors: null | undefined | string | ThemeColorDescriptor | ThemeColorDescriptor[]
|
89
|
-
}> = ({
|
90
|
-
thColors
|
91
|
-
}) => {
|
92
|
-
|
93
|
-
const ThColor: React.FC<{thColor: ThemeColorDescriptor}> = ({thColor}) => {
|
94
|
-
const toSpread: any = {
|
95
|
-
content: thColor.color
|
96
|
-
}
|
97
|
-
|
98
|
-
if ('media' in thColor) {
|
99
|
-
toSpread.media = thColor.media
|
100
|
-
}
|
101
|
-
|
102
|
-
return <meta name="theme-color" {...toSpread}/>
|
103
|
-
}
|
104
|
-
|
105
|
-
if (!thColors) {
|
106
|
-
return null
|
107
|
-
}
|
108
|
-
|
109
|
-
if (Array.isArray(thColors)) {
|
110
|
-
return (<>
|
111
|
-
{thColors.map((el: ThemeColorDescriptor, index) => (
|
112
|
-
<ThColor thColor={el} key={`theme-colors-${index}`} />
|
113
|
-
))}
|
114
|
-
</>)
|
115
|
-
}
|
116
|
-
else if (typeof thColors === 'string') {
|
117
|
-
<meta name="theme-color" content={thColors as string}/>
|
118
|
-
}
|
119
|
-
return (<ThColor thColor={thColors as ThemeColorDescriptor} />)
|
120
|
-
}
|
121
|
-
|
122
|
-
const Manifest: React.FC<{
|
123
|
-
manifest: undefined | null | string | URL
|
124
|
-
}> = ({
|
125
|
-
manifest
|
126
|
-
}) => (
|
127
|
-
manifest && (<link rel="manifest" href={getURLasString(manifest)}/>)
|
128
|
-
)
|
129
|
-
|
130
|
-
const getOGImageURL = (img: OGImage | undefined): string | null => {
|
131
|
-
|
132
|
-
if (!img) {
|
133
|
-
return null
|
134
|
-
}
|
135
|
-
if (typeof img === 'object' && 'url' in img) { // this is a OGImageDescriptor
|
136
|
-
return getURLasString(img.url)
|
137
|
-
}
|
138
|
-
return getURLasString(img) // this is a URL or string
|
139
|
-
}
|
140
|
-
|
141
|
-
const getTwitterImageURL = (img: TwitterImage | undefined): string | null => {
|
142
|
-
|
143
|
-
if (!img) {
|
144
|
-
return null
|
145
|
-
}
|
146
|
-
if (typeof img === 'object' && 'url' in img) { // this is a TwitterImageDescriptor
|
147
|
-
return getURLasString(img.url)
|
148
|
-
}
|
149
|
-
return getURLasString(img) // this is a URL or string
|
150
|
-
}
|
151
|
-
|
152
|
-
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
153
|
-
const OpenGraphComponent: React.FC<{
|
154
|
-
og: OpenGraph | undefined | null
|
155
|
-
}> = ({
|
156
|
-
og
|
157
|
-
}) => (og && (<Head>
|
158
|
-
{og.url && (<meta property="og:url" content={(typeof og.url === 'string') ? (og.url as string) : (og.url.href)} />)}
|
159
|
-
{(og as any).type && (<meta property="og:type" content={(og as any).type} />)}
|
160
|
-
{og.title && (<meta property="og:title" content={getTitleFromTemplateString(og.title)!} />)}
|
161
|
-
{og.description && (<meta property="og:description" content={og.description} />)}
|
162
|
-
{og.images && (<meta property="og:image" content={getOGImageURL(Array.isArray(og.images) ? og.images[0] : og.images)!} />)}
|
163
|
-
</Head>))
|
164
|
-
|
165
|
-
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
166
|
-
export const TwitterComponent: React.FC<{
|
167
|
-
tw: Twitter | undefined | null
|
168
|
-
}> = ({
|
169
|
-
tw
|
170
|
-
}) => (tw && (<Head>
|
171
|
-
{(tw as any).card && (<meta name="twitter:card" content={(tw as any).card} />)}
|
172
|
-
{tw.title && (<meta name="twitter:title" content={getTitleFromTemplateString(tw.title)!} />)}
|
173
|
-
{tw.description && (<meta name="twitter:description" content={tw.description} />)}
|
174
|
-
{tw.images && (<meta name="twitter:image" content={getTwitterImageURL(Array.isArray(tw.images) ? tw.images[0] : tw.images)!} />)}
|
175
|
-
{tw.site && (<meta name="twitter:site" content={tw.site} />)}
|
176
|
-
</Head>))
|
177
|
-
|
178
|
-
/* See NOTE at top of file! */
|
179
|
-
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
180
|
-
const HeadMetadataComponent: React.FC<{
|
181
|
-
metadata: Metadata
|
182
|
-
}> = ({
|
183
|
-
metadata
|
184
|
-
}) => {
|
185
|
-
const mainTitle = getTitleFromTemplateString(metadata.title)
|
186
|
-
|
187
|
-
return (<>
|
188
|
-
<Head>
|
189
|
-
{mainTitle && (<title>{mainTitle}</title>) /* must be here, directly under Head component */}
|
190
|
-
{metadata.description && (
|
191
|
-
<meta name="description" content={metadata.description} />
|
192
|
-
)}
|
193
|
-
{metadata.applicationName && (
|
194
|
-
<meta name="application-name" content={metadata.applicationName} />
|
195
|
-
)}
|
196
|
-
<Authors authors={metadata.authors} />
|
197
|
-
<Keywords keywords={metadata.keywords} />
|
198
|
-
<ThemeColor thColors={metadata.themeColor} />
|
199
|
-
<Manifest manifest={metadata.manifest} />
|
200
|
-
</Head>
|
201
|
-
{/* Icons: We only support this format for now */}
|
202
|
-
<Icons icons={metadata.icons as IconDescriptor[]} />
|
203
|
-
<OpenGraphComponent og={metadata.openGraph} />
|
204
|
-
<TwitterComponent tw={metadata.twitter} />
|
205
|
-
</>)
|
206
|
-
}
|
207
|
-
|
208
|
-
export default HeadMetadataComponent
|
1
|
+
import React from 'react'
|
2
|
+
|
3
|
+
import type { Metadata } from 'next'
|
4
|
+
import Head from "next/head"
|
5
|
+
|
6
|
+
import type {
|
7
|
+
IconDescriptor,
|
8
|
+
TemplateString,
|
9
|
+
Author,
|
10
|
+
ThemeColorDescriptor
|
11
|
+
} from './from-next/metadata-types'
|
12
|
+
|
13
|
+
import type { OpenGraph, OGImage } from './from-next/opengraph-types'
|
14
|
+
import type { Twitter, TwitterImage } from './from-next/twitter-types'
|
15
|
+
|
16
|
+
/*
|
17
|
+
NOTE: This is ONLY for sites that use the pages router in next.
|
18
|
+
The app router does this automatically
|
19
|
+
*/
|
20
|
+
|
21
|
+
const getURLasString = (url: string | URL) => {
|
22
|
+
return (
|
23
|
+
(typeof url === 'string') ? (url as string) : (url.href)
|
24
|
+
)
|
25
|
+
}
|
26
|
+
|
27
|
+
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
28
|
+
const Icons: React.FC<{icons: IconDescriptor[]}> = ({
|
29
|
+
icons
|
30
|
+
}) => {
|
31
|
+
return <Head>
|
32
|
+
{icons.map(({url, ...rest}: IconDescriptor, index) => (
|
33
|
+
<link {...rest} href={getURLasString(url)} key={`icons-${index}`}/>
|
34
|
+
))}
|
35
|
+
</Head>
|
36
|
+
}
|
37
|
+
|
38
|
+
export const getTitleFromTemplateString = (title: string | TemplateString | null | undefined): string | null => {
|
39
|
+
|
40
|
+
if (!title) {
|
41
|
+
return null
|
42
|
+
}
|
43
|
+
if (typeof title === 'object') {
|
44
|
+
if ('default' in title) {
|
45
|
+
return title.default
|
46
|
+
}
|
47
|
+
else if ('absolute' in title) {
|
48
|
+
return title.absolute
|
49
|
+
}
|
50
|
+
}
|
51
|
+
return title as string
|
52
|
+
}
|
53
|
+
|
54
|
+
const Authors: React.FC<{
|
55
|
+
authors: null | undefined | Author | Array<Author>
|
56
|
+
}> = ({
|
57
|
+
authors
|
58
|
+
}) => {
|
59
|
+
|
60
|
+
const Author: React.FC<{author: Author}> = ({author}) => (<>
|
61
|
+
{author.name && <meta name="author" content={author.name} />}
|
62
|
+
{author.url && <link rel="author" href={getURLasString(author.url)}/>}
|
63
|
+
</>)
|
64
|
+
|
65
|
+
if (!authors) {
|
66
|
+
return null
|
67
|
+
}
|
68
|
+
|
69
|
+
if (Array.isArray(authors)) {
|
70
|
+
return (<>
|
71
|
+
{authors.map((el: Author, index) => (
|
72
|
+
<Author author={el} key={`authors-${index}`} />
|
73
|
+
))}
|
74
|
+
</>)
|
75
|
+
}
|
76
|
+
return (<Author author={authors as Author} />)
|
77
|
+
}
|
78
|
+
|
79
|
+
const Keywords: React.FC<{keywords: undefined | null | string | Array<string>}> = ({
|
80
|
+
keywords
|
81
|
+
}) => {
|
82
|
+
if (!keywords) return null
|
83
|
+
const content = (Array.isArray(keywords) ? keywords.join(', ') : keywords as string)
|
84
|
+
return (<meta name="keywords" content={content} />)
|
85
|
+
}
|
86
|
+
|
87
|
+
const ThemeColor: React.FC<{
|
88
|
+
thColors: null | undefined | string | ThemeColorDescriptor | ThemeColorDescriptor[]
|
89
|
+
}> = ({
|
90
|
+
thColors
|
91
|
+
}) => {
|
92
|
+
|
93
|
+
const ThColor: React.FC<{thColor: ThemeColorDescriptor}> = ({thColor}) => {
|
94
|
+
const toSpread: any = {
|
95
|
+
content: thColor.color
|
96
|
+
}
|
97
|
+
|
98
|
+
if ('media' in thColor) {
|
99
|
+
toSpread.media = thColor.media
|
100
|
+
}
|
101
|
+
|
102
|
+
return <meta name="theme-color" {...toSpread}/>
|
103
|
+
}
|
104
|
+
|
105
|
+
if (!thColors) {
|
106
|
+
return null
|
107
|
+
}
|
108
|
+
|
109
|
+
if (Array.isArray(thColors)) {
|
110
|
+
return (<>
|
111
|
+
{thColors.map((el: ThemeColorDescriptor, index) => (
|
112
|
+
<ThColor thColor={el} key={`theme-colors-${index}`} />
|
113
|
+
))}
|
114
|
+
</>)
|
115
|
+
}
|
116
|
+
else if (typeof thColors === 'string') {
|
117
|
+
<meta name="theme-color" content={thColors as string}/>
|
118
|
+
}
|
119
|
+
return (<ThColor thColor={thColors as ThemeColorDescriptor} />)
|
120
|
+
}
|
121
|
+
|
122
|
+
const Manifest: React.FC<{
|
123
|
+
manifest: undefined | null | string | URL
|
124
|
+
}> = ({
|
125
|
+
manifest
|
126
|
+
}) => (
|
127
|
+
manifest && (<link rel="manifest" href={getURLasString(manifest)}/>)
|
128
|
+
)
|
129
|
+
|
130
|
+
const getOGImageURL = (img: OGImage | undefined): string | null => {
|
131
|
+
|
132
|
+
if (!img) {
|
133
|
+
return null
|
134
|
+
}
|
135
|
+
if (typeof img === 'object' && 'url' in img) { // this is a OGImageDescriptor
|
136
|
+
return getURLasString(img.url)
|
137
|
+
}
|
138
|
+
return getURLasString(img) // this is a URL or string
|
139
|
+
}
|
140
|
+
|
141
|
+
const getTwitterImageURL = (img: TwitterImage | undefined): string | null => {
|
142
|
+
|
143
|
+
if (!img) {
|
144
|
+
return null
|
145
|
+
}
|
146
|
+
if (typeof img === 'object' && 'url' in img) { // this is a TwitterImageDescriptor
|
147
|
+
return getURLasString(img.url)
|
148
|
+
}
|
149
|
+
return getURLasString(img) // this is a URL or string
|
150
|
+
}
|
151
|
+
|
152
|
+
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
153
|
+
const OpenGraphComponent: React.FC<{
|
154
|
+
og: OpenGraph | undefined | null
|
155
|
+
}> = ({
|
156
|
+
og
|
157
|
+
}) => (og && (<Head>
|
158
|
+
{og.url && (<meta property="og:url" content={(typeof og.url === 'string') ? (og.url as string) : (og.url.href)} />)}
|
159
|
+
{(og as any).type && (<meta property="og:type" content={(og as any).type} />)}
|
160
|
+
{og.title && (<meta property="og:title" content={getTitleFromTemplateString(og.title)!} />)}
|
161
|
+
{og.description && (<meta property="og:description" content={og.description} />)}
|
162
|
+
{og.images && (<meta property="og:image" content={getOGImageURL(Array.isArray(og.images) ? og.images[0] : og.images)!} />)}
|
163
|
+
</Head>))
|
164
|
+
|
165
|
+
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
166
|
+
export const TwitterComponent: React.FC<{
|
167
|
+
tw: Twitter | undefined | null
|
168
|
+
}> = ({
|
169
|
+
tw
|
170
|
+
}) => (tw && (<Head>
|
171
|
+
{(tw as any).card && (<meta name="twitter:card" content={(tw as any).card} />)}
|
172
|
+
{tw.title && (<meta name="twitter:title" content={getTitleFromTemplateString(tw.title)!} />)}
|
173
|
+
{tw.description && (<meta name="twitter:description" content={tw.description} />)}
|
174
|
+
{tw.images && (<meta name="twitter:image" content={getTwitterImageURL(Array.isArray(tw.images) ? tw.images[0] : tw.images)!} />)}
|
175
|
+
{tw.site && (<meta name="twitter:site" content={tw.site} />)}
|
176
|
+
</Head>))
|
177
|
+
|
178
|
+
/* See NOTE at top of file! */
|
179
|
+
// https://stackoverflow.com/questions/68746228/next-head-wont-render-meta-tags-inside-of-fragment
|
180
|
+
const HeadMetadataComponent: React.FC<{
|
181
|
+
metadata: Metadata
|
182
|
+
}> = ({
|
183
|
+
metadata
|
184
|
+
}) => {
|
185
|
+
const mainTitle = getTitleFromTemplateString(metadata.title)
|
186
|
+
|
187
|
+
return (<>
|
188
|
+
<Head>
|
189
|
+
{mainTitle && (<title>{mainTitle}</title>) /* must be here, directly under Head component */}
|
190
|
+
{metadata.description && (
|
191
|
+
<meta name="description" content={metadata.description} />
|
192
|
+
)}
|
193
|
+
{metadata.applicationName && (
|
194
|
+
<meta name="application-name" content={metadata.applicationName} />
|
195
|
+
)}
|
196
|
+
<Authors authors={metadata.authors} />
|
197
|
+
<Keywords keywords={metadata.keywords} />
|
198
|
+
<ThemeColor thColors={metadata.themeColor} />
|
199
|
+
<Manifest manifest={metadata.manifest} />
|
200
|
+
</Head>
|
201
|
+
{/* Icons: We only support this format for now */}
|
202
|
+
<Icons icons={metadata.icons as IconDescriptor[]} />
|
203
|
+
<OpenGraphComponent og={metadata.openGraph} />
|
204
|
+
<TwitterComponent tw={metadata.twitter} />
|
205
|
+
</>)
|
206
|
+
}
|
207
|
+
|
208
|
+
export default HeadMetadataComponent
|