[5d6f37a] | 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 | }
|
---|