From 5fca6018aa18bc3c86129a02e9b790966d96adb1 Mon Sep 17 00:00:00 2001 From: Ben Elan Date: Wed, 22 Nov 2023 11:15:45 -0800 Subject: [PATCH] fix(react)!: disable includeImportCustomElements to resolve initial render issues (#8248) **Related Issue:** #8143 ## Summary The `ReferenceError: navigator is not defined` error is still present in Stencil v4 when removing the patch. Disabled [`includeImportCustomElements`](https://stenciljs.com/docs/react#includeimportcustomelements) for now while we investigate an alternative fix that doesn't cause initial render issues. BREAKING CHANGE: Disabled `includeImportCustomElements`. Make sure to import components from `@esri/calcite-components` in addition to the react wrappers. For example, the first code snippet in #7185 is now required, or else the custom elements will not be defined on the window. --- .../calcite-components-react/package.json | 3 +- .../support/patchSSR.ts | 65 ------------------- packages/calcite-components/stencil.config.ts | 1 - 3 files changed, 1 insertion(+), 68 deletions(-) delete mode 100644 packages/calcite-components-react/support/patchSSR.ts diff --git a/packages/calcite-components-react/package.json b/packages/calcite-components-react/package.json index c24f273cd8a..aefa17afdfd 100644 --- a/packages/calcite-components-react/package.json +++ b/packages/calcite-components-react/package.json @@ -7,11 +7,10 @@ "license": "SEE LICENSE.md", "scripts": { "build": "rimraf dist && npm run compile", - "prebuild": "npm run patch:ssr && npm run patch:jsx-import", + "prebuild": "npm run patch:jsx-import", "clean": "rimraf dist node_modules .turbo", "compile": "npm run tsc", "patch:jsx-import": "ts-node support/patchJSXImport.ts", - "patch:ssr": "ts-node support/patchSSR.ts", "tsc": "tsc" }, "main": "./dist/index.js", diff --git a/packages/calcite-components-react/support/patchSSR.ts b/packages/calcite-components-react/support/patchSSR.ts deleted file mode 100644 index 35de55a951d..00000000000 --- a/packages/calcite-components-react/support/patchSSR.ts +++ /dev/null @@ -1,65 +0,0 @@ -// patch needed due to Stencil executing client side code on the server -// when using the includeImportCustomElements option -// https://stenciljs.com/docs/react#includeimportcustomelements -// https://github.com/Esri/calcite-design-system/issues/7486 - -const { - promises: { readFile, writeFile }, -} = require("fs"); -const { resolve } = require("path"); - -// Matches imports of defineCustomElement from calcite-component's custom-elements output target. -// Importing defineCustomElement on the server throws errors due to ESM/CJS conflicts and -// attempting to use browser APIs, which don't exist on the server. -const defineCustomElementImports = /import { defineCustomElement as defineCalcite.*(\r\n|\r|\n)/gm; - -// The removed imports are replaced with utils that check the environment -// to make sure the components are only defined on the browser. -const browserCheckUtils = ` -// CodeSandbox exposes 'process', which makes it look like NodeJS. The only way to determine it should be -// be treated as the browser is the non-standard value they use for 'process.platform'. -// https://nodejs.org/api/process.html#processplatform -type CodeSandboxWorkaround = NodeJS.Platform | "browser"; - -// https://github.com/flexdinesh/browser-or-node/blob/master/src/index.js -const isBrowser = typeof window !== "undefined" && typeof window.document !== "undefined"; -const isNode = - typeof process !== "undefined" && - process.versions != null && - process.versions.node != null && - (process?.platform as CodeSandboxWorkaround) !== "browser";`; - -// Matches createReactComponent exports to replace the defineCustomElement -// arguments with inlined, anonymous functions that dynamically import the -// components when in the browser. The regex creates capture groups for the -// component name and other arguments that shouldn't be replaced/removed. -const reactWrapperExports = /createReactComponent<(.*)>.*\('([\w|-]*)'(.*)(defineCalcite\w*)\)/g; - -// The patched version of the createReactComponent export using capture groups to fill in the blanks. -const patchedReactWrapperExports = - "createReactComponent<$1>('$2'$3isBrowser && !isNode ? async () => (await import('@esri/calcite-components/dist/components/$2.js')).defineCustomElement() : undefined)"; - -// The isBrowser and isNode utils are placed below this line -const jsxTypeImport = "import type { JSX } from '@esri/calcite-components/dist/components';"; - -(async () => { - try { - const filePath = resolve(`${__dirname}/../src/components.ts`); - const contents = await readFile(filePath, { encoding: "utf8" }); - - if (contents.includes("isBrowser")) { - console.log("SSR patch: skipping, components.ts is already patched"); - return; - } - - const patchedContents = contents - .replace(jsxTypeImport, `$&\n${browserCheckUtils}`) - .replace(defineCustomElementImports, "") - .replace(reactWrapperExports, patchedReactWrapperExports); - - await writeFile(filePath, patchedContents); - } catch (err) { - console.error(err); - process.exit(1); - } -})(); diff --git a/packages/calcite-components/stencil.config.ts b/packages/calcite-components/stencil.config.ts index 2b47301b142..7f16a3ef5b6 100644 --- a/packages/calcite-components/stencil.config.ts +++ b/packages/calcite-components/stencil.config.ts @@ -97,7 +97,6 @@ export const create: () => Config = () => ({ proxiesFile: "../calcite-components-react/src/components.ts", excludeComponents: ["context-consumer"], customElementsDir: "dist/components", - includeImportCustomElements: true, }), { type: "dist-hydrate-script" }, { type: "dist-custom-elements", customElementsExportBehavior: "auto-define-custom-elements" },