source: src/theme/next-emotion-cache.tsx@ 5d6f37a

main
Last change on this file since 5d6f37a was 5d6f37a, checked in by Naum Shapkarovski <naumshapkarovski@…>, 7 weeks ago

add customer

  • Property mode set to 100644
File size: 2.9 KB
Line 
1'use client';
2
3import * as React from 'react';
4import createCache from '@emotion/cache';
5import { useServerInsertedHTML } from 'next/navigation';
6import { CacheProvider as DefaultCacheProvider } from '@emotion/react';
7import type { EmotionCache, Options as OptionsOfCreateCache } from '@emotion/cache';
8
9// ----------------------------------------------------------------------
10
11export 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
23export 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}
Note: See TracBrowser for help on using the repository browser.