1 | 'use client';
|
---|
2 |
|
---|
3 | import * as React from 'react';
|
---|
4 | import createCache from '@emotion/cache';
|
---|
5 | import { useServerInsertedHTML } from 'next/navigation';
|
---|
6 | import { CacheProvider as DefaultCacheProvider } from '@emotion/react';
|
---|
7 | import type { EmotionCache, Options as OptionsOfCreateCache } from '@emotion/cache';
|
---|
8 |
|
---|
9 | // ----------------------------------------------------------------------
|
---|
10 |
|
---|
11 | export type NextAppDirEmotionCacheProviderProps = {
|
---|
12 | /** This is the options passed to createCache() from 'import createCache from "@emotion/cache"' */
|
---|
13 | options: Omit<OptionsOfCreateCache, 'insertionPoint'>;
|
---|
14 | /** By default <CacheProvider /> from 'import { CacheProvider } from "@emotion/react"' */
|
---|
15 | CacheProvider?: (props: {
|
---|
16 | value: EmotionCache;
|
---|
17 | children: React.ReactNode;
|
---|
18 | }) => React.JSX.Element | null;
|
---|
19 | children: React.ReactNode;
|
---|
20 | };
|
---|
21 |
|
---|
22 | // Adapted from https://github.com/garronej/tss-react/blob/main/src/next/appDir.tsx
|
---|
23 | export default function NextAppDirEmotionCacheProvider(props: NextAppDirEmotionCacheProviderProps) {
|
---|
24 | const { options, CacheProvider = DefaultCacheProvider, children } = props;
|
---|
25 |
|
---|
26 | const [registry] = React.useState(() => {
|
---|
27 | const cache = createCache(options);
|
---|
28 | cache.compat = true;
|
---|
29 | const prevInsert = cache.insert;
|
---|
30 | let inserted: { name: string; isGlobal: boolean }[] = [];
|
---|
31 | cache.insert = (...args) => {
|
---|
32 | const [selector, serialized] = args;
|
---|
33 | if (cache.inserted[serialized.name] === undefined) {
|
---|
34 | inserted.push({
|
---|
35 | name: serialized.name,
|
---|
36 | isGlobal: !selector,
|
---|
37 | });
|
---|
38 | }
|
---|
39 | return prevInsert(...args);
|
---|
40 | };
|
---|
41 | const flush = () => {
|
---|
42 | const prevInserted = inserted;
|
---|
43 | inserted = [];
|
---|
44 | return prevInserted;
|
---|
45 | };
|
---|
46 | return { cache, flush };
|
---|
47 | });
|
---|
48 |
|
---|
49 | useServerInsertedHTML(() => {
|
---|
50 | const inserted = registry.flush();
|
---|
51 | if (inserted.length === 0) {
|
---|
52 | return null;
|
---|
53 | }
|
---|
54 | let styles = '';
|
---|
55 | let dataEmotionAttribute = registry.cache.key;
|
---|
56 |
|
---|
57 | const globals: {
|
---|
58 | name: string;
|
---|
59 | style: string;
|
---|
60 | }[] = [];
|
---|
61 |
|
---|
62 | inserted.forEach(({ name, isGlobal }) => {
|
---|
63 | const style = registry.cache.inserted[name];
|
---|
64 |
|
---|
65 | if (typeof style !== 'boolean') {
|
---|
66 | if (isGlobal) {
|
---|
67 | globals.push({ name, style });
|
---|
68 | } else {
|
---|
69 | styles += style;
|
---|
70 | dataEmotionAttribute += ` ${name}`;
|
---|
71 | }
|
---|
72 | }
|
---|
73 | });
|
---|
74 |
|
---|
75 | return (
|
---|
76 | <>
|
---|
77 | {globals.map(({ name, style }) => (
|
---|
78 | <style
|
---|
79 | key={name}
|
---|
80 | data-emotion={`${registry.cache.key}-global ${name}`}
|
---|
81 | // eslint-disable-next-line react/no-danger
|
---|
82 | dangerouslySetInnerHTML={{ __html: style }}
|
---|
83 | />
|
---|
84 | ))}
|
---|
85 | {styles && (
|
---|
86 | <style
|
---|
87 | data-emotion={dataEmotionAttribute}
|
---|
88 | // eslint-disable-next-line react/no-danger
|
---|
89 | dangerouslySetInnerHTML={{ __html: styles }}
|
---|
90 | />
|
---|
91 | )}
|
---|
92 | </>
|
---|
93 | );
|
---|
94 | });
|
---|
95 |
|
---|
96 | return <CacheProvider value={registry.cache}>{children}</CacheProvider>;
|
---|
97 | }
|
---|