diff --git a/.changeset/gold-windows-fly.md b/.changeset/gold-windows-fly.md
new file mode 100644
index 000000000000..ca08419767ec
--- /dev/null
+++ b/.changeset/gold-windows-fly.md
@@ -0,0 +1,8 @@
+---
+'@astrojs/lit': major
+---
+
+Update to use `@lit-labs/ssr@^3`
+**[BREAKING]** DOM shim required for Lit SSR has been greatly reduced. `window`, `document`, and other objects are no longer available in global. Most SSR-ready component code should not be affected but, if so, they can be fixed with optional chaining or by using the `isServer` environment checker from the `lit` package. See [lit.dev docs on authoring components for SSR].(https://lit.dev/docs/ssr/authoring/#browser-only-code)
+**[BREAKING]** Adds compatibility with `lit@2.7.0` hydration behavior. Do not update if you're using `lit@2.6.1` or lower.
+Includes support for template[shadowrootmode] support.
diff --git a/examples/framework-lit/package.json b/examples/framework-lit/package.json
index 517d370f9e8b..d6b19909839d 100644
--- a/examples/framework-lit/package.json
+++ b/examples/framework-lit/package.json
@@ -12,8 +12,8 @@
},
"dependencies": {
"astro": "^2.1.8",
- "lit": "^2.2.5",
+ "lit": "^2.7.0",
"@astrojs/lit": "^1.3.0",
- "@webcomponents/template-shadowroot": "^0.1.0"
+ "@webcomponents/template-shadowroot": "^0.2.1"
}
}
diff --git a/packages/astro/e2e/fixtures/lit-component/package.json b/packages/astro/e2e/fixtures/lit-component/package.json
index 8781654d775d..06e374a42cc5 100644
--- a/packages/astro/e2e/fixtures/lit-component/package.json
+++ b/packages/astro/e2e/fixtures/lit-component/package.json
@@ -4,8 +4,8 @@
"private": true,
"dependencies": {
"@astrojs/lit": "workspace:*",
- "@webcomponents/template-shadowroot": "^0.1.0",
+ "@webcomponents/template-shadowroot": "^0.2.1",
"astro": "workspace:*",
- "lit": "^2.2.5"
+ "lit": "^2.7.0"
}
}
diff --git a/packages/astro/e2e/fixtures/multiple-frameworks/package.json b/packages/astro/e2e/fixtures/multiple-frameworks/package.json
index 36723d2fc281..91f95a97a498 100644
--- a/packages/astro/e2e/fixtures/multiple-frameworks/package.json
+++ b/packages/astro/e2e/fixtures/multiple-frameworks/package.json
@@ -12,8 +12,8 @@
"astro": "workspace:*"
},
"dependencies": {
- "@webcomponents/template-shadowroot": "^0.1.0",
- "lit": "^2.2.5",
+ "@webcomponents/template-shadowroot": "^0.2.1",
+ "lit": "^2.7.0",
"preact": "^10.7.3",
"react": "^18.1.0",
"react-dom": "^18.1.0",
diff --git a/packages/astro/e2e/lit-component.test.js b/packages/astro/e2e/lit-component.test.js
index d73005b278f7..37f8d9eed852 100644
--- a/packages/astro/e2e/lit-component.test.js
+++ b/packages/astro/e2e/lit-component.test.js
@@ -8,10 +8,6 @@ const test = testFactory({
// TODO: configure playwright to handle web component APIs
// https://github.com/microsoft/playwright/issues/14241
test.describe('Lit components', () => {
- test.beforeAll(() => {
- delete globalThis.window;
- });
-
test.describe('Development', () => {
let devServer;
const t = test.extend({});
@@ -158,7 +154,6 @@ test.describe('Lit components', () => {
const t = test.extend({});
t.beforeAll(async ({ astro }) => {
- delete globalThis.window;
// Playwright's Node version doesn't have these functions, so stub them.
process.stdout.clearLine = () => {};
process.stdout.cursorTo = () => {};
diff --git a/packages/astro/test/fixtures/lit-element/package.json b/packages/astro/test/fixtures/lit-element/package.json
index 5a84713a9217..2b3b252ac795 100644
--- a/packages/astro/test/fixtures/lit-element/package.json
+++ b/packages/astro/test/fixtures/lit-element/package.json
@@ -4,8 +4,8 @@
"private": true,
"dependencies": {
"@astrojs/lit": "workspace:*",
- "@webcomponents/template-shadowroot": "^0.1.0",
+ "@webcomponents/template-shadowroot": "^0.2.1",
"astro": "workspace:*",
- "lit": "^2.2.5"
+ "lit": "^2.7.0"
}
}
diff --git a/packages/astro/test/fixtures/lit-element/src/components/my-element.ts b/packages/astro/test/fixtures/lit-element/src/components/my-element.ts
index c4c1b92b408f..b4c98b010104 100644
--- a/packages/astro/test/fixtures/lit-element/src/components/my-element.ts
+++ b/packages/astro/test/fixtures/lit-element/src/components/my-element.ts
@@ -26,13 +26,11 @@ export class MyElement extends LitElement {
this.reflectedStr = 'default reflected string';
}
render() {
- let typeofwindow = typeof window.Window;
return html`
Testing...
${this.bool ? 'A' : 'B'}
${this.str}
data: ${this.obj.data}
- ${typeofwindow}
diff --git a/packages/astro/test/ssr-lit.test.js b/packages/astro/test/ssr-lit.test.js
index 98d58b395acd..6615e644402c 100644
--- a/packages/astro/test/ssr-lit.test.js
+++ b/packages/astro/test/ssr-lit.test.js
@@ -25,9 +25,8 @@ describe('Lit integration in SSR', () => {
}
it('Is able to load', async () => {
- delete globalThis.window; // On Windows this results in `ReferenceError: window is not defined`
const html = await fetchHTML('/');
const $ = cheerioLoad(html);
- expect($('#win').text()).to.equal('function');
+ expect($('#str').text()).to.equal('initialized');
});
});
diff --git a/packages/integrations/lit/package.json b/packages/integrations/lit/package.json
index 8ad1a527911b..19892743d9a8 100644
--- a/packages/integrations/lit/package.json
+++ b/packages/integrations/lit/package.json
@@ -34,7 +34,8 @@
"test": "mocha"
},
"dependencies": {
- "@lit-labs/ssr": "^2.2.0",
+ "@lit-labs/ssr": "^3.1.0",
+ "@lit-labs/ssr-dom-shim": "^1.1.0",
"parse5": "^7.1.2"
},
"devDependencies": {
@@ -42,12 +43,12 @@
"astro-scripts": "workspace:*",
"chai": "^4.3.6",
"cheerio": "^1.0.0-rc.11",
- "lit": "^2.2.5",
+ "lit": "^2.7.0",
"mocha": "^9.2.2",
"sass": "^1.52.2"
},
"peerDependencies": {
- "@webcomponents/template-shadowroot": "^0.1.0",
- "lit": "^2.1.3"
+ "@webcomponents/template-shadowroot": "^0.2.1",
+ "lit": "^2.7.0"
}
}
diff --git a/packages/integrations/lit/server-shim.js b/packages/integrations/lit/server-shim.js
index 3f4a8df4fbd2..ed371f89a32f 100644
--- a/packages/integrations/lit/server-shim.js
+++ b/packages/integrations/lit/server-shim.js
@@ -1,20 +1,35 @@
-import { installWindowOnGlobal } from '@lit-labs/ssr/lib/dom-shim.js';
+import { customElements as litCE, HTMLElement as litShimHTMLElement } from '@lit-labs/ssr-dom-shim';
-if (typeof fetch === 'function') {
- const _fetch = fetch;
- installWindowOnGlobal();
- globalThis.fetch = window.fetch = _fetch;
-} else {
- installWindowOnGlobal();
+// Something at build time injects document.currentScript = undefined instead of
+// document.currentScript = null. This causes Sass build to fail because it
+// seems to be expecting `=== null`. This set to `undefined` doesn't seem to be
+// caused by Lit and only happens at build / test time, but not in dev or
+// preview time.
+if (globalThis.document) {
+ document.currentScript = null;
}
-window.global = window;
-document.getElementsByTagName = () => [];
-// See https://github.com/lit/lit/issues/2393
-document.currentScript = null;
+if (globalThis.HTMLElement) {
+ // Seems Astro's Element shim does nothing when `.setAttribute` is called
+ // and subsequently `.getAttribute` is called. Causes Lit to not SSR attrs
+ globalThis.HTMLElement = litShimHTMLElement;
+}
+
+// Astro seems to have a DOM shim and the only real difference that we need out
+// of the Lit DOM shim is that the Lit DOM shim reads
+// `HTMLElement.observedAttributes` which is meant to trigger
+// `ReactiveElement.finalize()`. So this is the only thing we will re-shim since
+// Lit will try to respect other global DOM shims.
+globalThis.customElements = litCE;
+
+const litCeDefine = customElements.define;
-const ceDefine = customElements.define;
+// We need to patch customElements.define to keep track of the tagName on the
+// class itself so that we can transform JSX custom element class definintion to
+// a DSD string on the server, because there is no way to get the tagName from a
+// CE class otherwise. Not an issue on client:only because the browser supports
+// appending a class instance directly to the DOM.
customElements.define = function (tagName, Ctr) {
Ctr[Symbol.for('tagName')] = tagName;
- return ceDefine.call(this, tagName, Ctr);
+ return litCeDefine.call(this, tagName, Ctr);
};
diff --git a/packages/integrations/lit/server.js b/packages/integrations/lit/server.js
index ae89bd610f86..762c77844759 100644
--- a/packages/integrations/lit/server.js
+++ b/packages/integrations/lit/server.js
@@ -1,5 +1,4 @@
import './server-shim.js';
-import '@lit-labs/ssr/lib/render-lit-html.js';
import { LitElementRenderer } from '@lit-labs/ssr/lib/lit-element-renderer.js';
import * as parse5 from 'parse5';
diff --git a/packages/integrations/lit/test/server.test.js b/packages/integrations/lit/test/server.test.js
index 315937401829..51e083241b9d 100644
--- a/packages/integrations/lit/test/server.test.js
+++ b/packages/integrations/lit/test/server.test.js
@@ -1,6 +1,8 @@
import { expect } from 'chai';
-import server from '../server.js';
import { LitElement, html } from 'lit';
+// Must come after lit import because @lit/reactive-element defines
+// globalThis.customElements which the server shim expects to be defined.
+import server from '../server.js';
import * as cheerio from 'cheerio';
const { check, renderToStaticMarkup } = server;
@@ -12,6 +14,10 @@ describe('check', () => {
it('should be false with a registered non-lit component', async () => {
const tagName = 'non-lit-component';
+ // Lit no longer shims HTMLElement globally, so we need to do it ourselves.
+ if (!globalThis.HTMLElement) {
+ globalThis.HTMLElement = class {};
+ }
customElements.define(tagName, class TestComponent extends HTMLElement {});
expect(await check(tagName)).to.equal(false);
});
@@ -85,7 +91,7 @@ describe('renderToStaticMarkup', () => {
});
it('should render DSD attributes based on shadowRootOptions', async () => {
- const tagName = 'lit-component';
+ const tagName = 'shadow-root-options-component';
customElements.define(
tagName,
class extends LitElement {
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index d5adbfa94a77..26472d0c48ad 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -181,14 +181,14 @@ importers:
examples/framework-lit:
specifiers:
'@astrojs/lit': ^1.3.0
- '@webcomponents/template-shadowroot': ^0.1.0
+ '@webcomponents/template-shadowroot': ^0.2.1
astro: ^2.1.8
- lit: ^2.2.5
+ lit: ^2.7.0
dependencies:
'@astrojs/lit': link:../../packages/integrations/lit
- '@webcomponents/template-shadowroot': 0.1.0
+ '@webcomponents/template-shadowroot': 0.2.1
astro: link:../../packages/astro
- lit: 2.6.1
+ lit: 2.7.0
examples/framework-multiple:
specifiers:
@@ -785,14 +785,14 @@ importers:
packages/astro/e2e/fixtures/lit-component:
specifiers:
'@astrojs/lit': workspace:*
- '@webcomponents/template-shadowroot': ^0.1.0
+ '@webcomponents/template-shadowroot': ^0.2.1
astro: workspace:*
- lit: ^2.2.5
+ lit: ^2.7.0
dependencies:
'@astrojs/lit': link:../../../../integrations/lit
- '@webcomponents/template-shadowroot': 0.1.0
+ '@webcomponents/template-shadowroot': 0.2.1
astro: link:../../..
- lit: 2.6.1
+ lit: 2.7.0
packages/astro/e2e/fixtures/multiple-frameworks:
specifiers:
@@ -802,9 +802,9 @@ importers:
'@astrojs/solid-js': workspace:*
'@astrojs/svelte': workspace:*
'@astrojs/vue': workspace:*
- '@webcomponents/template-shadowroot': ^0.1.0
+ '@webcomponents/template-shadowroot': ^0.2.1
astro: workspace:*
- lit: ^2.2.5
+ lit: ^2.7.0
preact: ^10.7.3
react: ^18.1.0
react-dom: ^18.1.0
@@ -812,8 +812,8 @@ importers:
svelte: ^3.48.0
vue: ^3.2.37
dependencies:
- '@webcomponents/template-shadowroot': 0.1.0
- lit: 2.6.1
+ '@webcomponents/template-shadowroot': 0.2.1
+ lit: 2.7.0
preact: 10.12.0
react: 18.2.0
react-dom: 18.2.0_react@18.2.0
@@ -2169,14 +2169,14 @@ importers:
packages/astro/test/fixtures/lit-element:
specifiers:
'@astrojs/lit': workspace:*
- '@webcomponents/template-shadowroot': ^0.1.0
+ '@webcomponents/template-shadowroot': ^0.2.1
astro: workspace:*
- lit: ^2.2.5
+ lit: ^2.7.0
dependencies:
'@astrojs/lit': link:../../../../integrations/lit
- '@webcomponents/template-shadowroot': 0.1.0
+ '@webcomponents/template-shadowroot': 0.2.1
astro: link:../../..
- lit: 2.6.1
+ lit: 2.7.0
packages/astro/test/fixtures/markdown:
specifiers:
@@ -3060,24 +3060,26 @@ importers:
packages/integrations/lit:
specifiers:
- '@lit-labs/ssr': ^2.2.0
+ '@lit-labs/ssr': ^3.1.0
+ '@lit-labs/ssr-dom-shim': ^1.1.0
astro: workspace:*
astro-scripts: workspace:*
chai: ^4.3.6
cheerio: ^1.0.0-rc.11
- lit: ^2.2.5
+ lit: ^2.7.0
mocha: ^9.2.2
parse5: ^7.1.2
sass: ^1.52.2
dependencies:
- '@lit-labs/ssr': 2.3.0
+ '@lit-labs/ssr': 3.1.0
+ '@lit-labs/ssr-dom-shim': 1.1.0
parse5: 7.1.2
devDependencies:
astro: link:../../astro
astro-scripts: link:../../../scripts
chai: 4.3.7
cheerio: 1.0.0-rc.12
- lit: 2.6.1
+ lit: 2.7.0
mocha: 9.2.2
sass: 1.58.0
@@ -4260,15 +4262,14 @@ packages:
- react
dev: false
- /@astrojs/markdown-remark/2.1.0_astro@packages+astro:
- resolution: {integrity: sha512-w9T5o3UWQIfMcCkM2nLWrlfVQazh/7mw+2N/85QGcSUkZy6oNJoyy8Xz/ZkDhHLx8HPO0RT9fABR0B/H+aDaEw==}
+ /@astrojs/markdown-remark/2.1.2_astro@packages+astro:
+ resolution: {integrity: sha512-rYkmFEv2w7oEk6ZPgxHkhWzwcxSUGc1vJU0cbCu5sHF8iFNnc1cmMsjXWa5DrU5sCEf8VVYE1iFlbbnFzvHQJw==}
peerDependencies:
astro: '*'
dependencies:
'@astrojs/prism': 2.1.1
astro: link:packages/astro
github-slugger: 1.5.0
- image-size: 1.0.2
import-meta-resolve: 2.2.1
rehype-raw: 6.1.1
rehype-stringify: 9.0.3
@@ -4288,7 +4289,7 @@ packages:
resolution: {integrity: sha512-mol57cw1jJMcQgKMRGn7p6cewajq6JTNtqj5aAZgROWam/phVDSOCbXj/WU3O9+3qFnyKtpczoufQKwJTQltAw==}
engines: {node: '>=16.12.0'}
dependencies:
- '@astrojs/markdown-remark': 2.1.0_astro@packages+astro
+ '@astrojs/markdown-remark': 2.1.2_astro@packages+astro
'@astrojs/prism': 2.1.1
'@mdx-js/mdx': 2.3.0
'@mdx-js/rollup': 2.3.0
@@ -6922,25 +6923,26 @@ packages:
resolution: {integrity: sha512-rr/UVhxbKWNUr+3qRyvZk+glC7v7ph8Gk/W0z96YG64COJKf9ilnWY6JGW77TRqhrRMmS2nsvAXOyQgcF+4jrA==}
dependencies:
'@lit/reactive-element': 1.6.1
- lit: 2.6.1
- lit-html: 2.6.1
+ lit: 2.7.0
+ lit-html: 2.7.0
dev: false
- /@lit-labs/ssr-dom-shim/1.0.0:
- resolution: {integrity: sha512-ic93MBXfApIFTrup4a70M/+ddD8xdt2zxxj9sRwHQzhS9ag/syqkD8JPdTXsc1gUy2K8TTirhlCqyTEM/sifNw==}
+ /@lit-labs/ssr-dom-shim/1.1.0:
+ resolution: {integrity: sha512-92uQ5ARf7UXYrzaFcAX3T2rTvaS9Z1//ukV+DqjACM4c8s0ZBQd7ayJU5Dh2AFLD/Ayuyz4uMmxQec8q3U4Ong==}
- /@lit-labs/ssr/2.3.0:
- resolution: {integrity: sha512-uPaJoNf5w3t8DOVDpuI4WR6wo552mZwiiE9n9TpIvinh75lDgvl1ki07wvfrFI6VEbDVPRj4jHiCduBr1dVJ7A==}
+ /@lit-labs/ssr/3.1.0:
+ resolution: {integrity: sha512-D4Ut27bmmj5AV9iQaEOxdjPHSZGp11ww0DI3zAniyFf2KBOH7y/X2U163oOdmKh6KQNFLQDOkVx+A6NlmxcY4Q==}
engines: {node: '>=13.9.0'}
dependencies:
'@lit-labs/ssr-client': 1.0.1
+ '@lit-labs/ssr-dom-shim': 1.1.0
'@lit/reactive-element': 1.6.1
'@parse5/tools': 0.1.0
'@types/node': 16.18.12
enhanced-resolve: 5.12.0
- lit: 2.6.1
- lit-element: 3.2.2
- lit-html: 2.6.1
+ lit: 2.7.0
+ lit-element: 3.3.0
+ lit-html: 2.7.0
node-fetch: 3.3.0
parse5: 7.1.2
dev: false
@@ -6948,7 +6950,7 @@ packages:
/@lit/reactive-element/1.6.1:
resolution: {integrity: sha512-va15kYZr7KZNNPZdxONGQzpUr+4sxVu7V/VG7a8mRfPPXUyhEYj5RzXCQmGrlP3tAh0L3HHm5AjBMFYRqlM9SA==}
dependencies:
- '@lit-labs/ssr-dom-shim': 1.0.0
+ '@lit-labs/ssr-dom-shim': 1.1.0
/@ljharb/has-package-exports-patterns/0.0.2:
resolution: {integrity: sha512-4/RWEeXDO6bocPONheFe6gX/oQdP/bEpv0oL4HqjPP5DCenBSt0mHgahppY49N0CpsaqffdwPq+TlX9CYOq2Dw==}
@@ -8518,8 +8520,8 @@ packages:
/@vue/shared/3.2.47:
resolution: {integrity: sha512-BHGyyGN3Q97EZx0taMQ+OLNuZcW3d37ZEVmEAyeoA9ERdGvm9Irc/0Fua8SNyOtV1w6BS4q25wbMzJujO9HIfQ==}
- /@webcomponents/template-shadowroot/0.1.0:
- resolution: {integrity: sha512-ry84Vft6xtRBbd4M/ptRodbOLodV5AD15TYhyRghCRgIcJJKmYmJ2v2BaaWxygENwh6Uq3zTfGPmlckKT/GXsQ==}
+ /@webcomponents/template-shadowroot/0.2.1:
+ resolution: {integrity: sha512-fXL/vIUakyZL62hyvUh+EMwbVoTc0hksublmRz6ai6et8znHkJa6gtqMUZo1oc7dIz46exHSIImml9QTdknMHg==}
dev: false
/abab/2.0.6:
@@ -12203,23 +12205,24 @@ packages:
uhyphen: 0.1.0
dev: true
- /lit-element/3.2.2:
- resolution: {integrity: sha512-6ZgxBR9KNroqKb6+htkyBwD90XGRiqKDHVrW/Eh0EZ+l+iC+u+v+w3/BA5NGi4nizAVHGYvQBHUDuSmLjPp7NQ==}
+ /lit-element/3.3.0:
+ resolution: {integrity: sha512-M3OIoblNS7LZdRxOIk8g0wyLEA/lRw/UGJ1TX+767OpkuDsRdSoxBIvewpWqCo7sMd9xt1XedUNZIr9jUO1X3g==}
dependencies:
+ '@lit-labs/ssr-dom-shim': 1.1.0
'@lit/reactive-element': 1.6.1
- lit-html: 2.6.1
+ lit-html: 2.7.0
- /lit-html/2.6.1:
- resolution: {integrity: sha512-Z3iw+E+3KKFn9t2YKNjsXNEu/LRLI98mtH/C6lnFg7kvaqPIzPn124Yd4eT/43lyqrejpc5Wb6BHq3fdv4S8Rw==}
+ /lit-html/2.7.0:
+ resolution: {integrity: sha512-/zPOl8EfeB3HHpTzINSpnWgvgQ8N07g/j272EOAIyB0Ys2RzBqTVT23i+JZuUlNbB2WHHeSsTCFi92NtWrtpqQ==}
dependencies:
'@types/trusted-types': 2.0.2
- /lit/2.6.1:
- resolution: {integrity: sha512-DT87LD64f8acR7uVp7kZfhLRrHkfC/N4BVzAtnw9Yg8087mbBJ//qedwdwX0kzDbxgPccWRW6mFwGbRQIxy0pw==}
+ /lit/2.7.0:
+ resolution: {integrity: sha512-qSy2BAVA+OiWtNptP404egcC/izDdNRw6iHGIbUmkZtbMJvPKfNsaoKrNs8Zmsbjmv5ZX2tur1l9TfzkSWWT4g==}
dependencies:
'@lit/reactive-element': 1.6.1
- lit-element: 3.2.2
- lit-html: 2.6.1
+ lit-element: 3.3.0
+ lit-html: 2.7.0
/lite-vimeo-embed/0.1.0:
resolution: {integrity: sha512-XFzPdv4NaWlyaM9WpBaS5CIUkf+laIRZEXQGsBb2ZdDWkuMmnzfAZ5nriYv3a3MVx5tEEetGN0sNaUhAVRXr1g==}
@@ -15325,6 +15328,7 @@ packages:
/source-map/0.6.1:
resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==}
engines: {node: '>=0.10.0'}
+ requiresBuild: true
/source-map/0.7.4:
resolution: {integrity: sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==}