On This Page
Web Standards
Throughout our docs we make heavy use of, and reference to, some of the following Web APIs, either indirectly or as part of the core surface area of Greenwood itself. This section is a general introduction to them with relevant links and resources.
Import Attributes
Building upon ECMAScript Modules, Greenwood supports Import Attributes on the client and on the server seamlessly, supporting both CSS and JSON modules out of the box.
// returns a Constructable StyleSheet
// https://developer.mozilla.org/en-US/docs/Web/API/CSSStyleSheet/CSSStyleSheet
import sheet from "./styles.css" with { type: "css" };
console.log({ sheet });
// returns an object
import data from "./data.json" with { type: "json" };
console.log({ data });
⚠️ Although Import Attributes are not baseline yet, Greenwood supports polyfilling them with a configuration flag.
Web Components
Web Components are a collection of standard Web APIs that can be mixed and matched to create your own encapsulated styles and behaviors:
- Custom Elements - Define your own custom HTML tags
- Shadow DOM - Encapsulation mechanism for custom elements
- <template> tag - Commonly used for initializing the HTML contents of a Shadow Root
A simple example putting it all together might look like this:
import sheet from "./card.css" with { type: "css" };
// create a template element
// to be populated with dynamic HTML
const template = document.createElement("template");
export default class Card extends HTMLElement {
connectedCallback() {
// this block can be SSR'd and thus wont need to be re-run on the client
if (!this.shadowRoot) {
const thumbnail = this.getAttribute("thumbnail");
const title = this.getAttribute("title");
template.innerHTML = `
<div class="card">
<h3>${title}</h3>
<img src="${thumbnail}" alt="${title}" loading="lazy" width="100%">
</div>
`;
// attach our template to our Shadow root
this.attachShadow({ mode: "open" });
this.shadowRoot.appendChild(template.content.cloneNode(true));
}
// adopt our CSS Module Script
this.shadowRoot.adoptedStyleSheets = [sheet];
}
}
// defining the HTML tag that will invoke this definition
customElements.define("x-card", Card);
And would be used like this:
<x-card title="My Title" thumbnail="/path/to/image.png"></x-card>
Greenwood promotes Web Components not only as a great way to add sprinkles of JavaScript to an otherwise static site, but also for static templating through prerendering with all the power and expressiveness of JavaScript as well as completely full-stack web components.
Fetch (and Friends)
Fetch is a web standard for making HTTP requests and is supported both on the client and the server. It also brings along "companion" APIs like Request
, Response
, and Headers
.
This suite of APIs is featured prominently in our API Route examples:
// a standard request object is passed in
// and a standard response object should be returned
export async function handler(request) {
console.log("endpoint visited", request.url);
return new Response("...", {
headers: new Headers({
/* ... */
}),
});
}
Import Maps
During local development, Greenwood serves all resources to your browser unbundled right off disk using efficient E-Tag caching. Import Maps allow bare specifiers for ESM compatible packages installed from npm to work natively in the browser. When Greenwood sees a package in the dependency field of your package.json, Greenwood will walk all your dependencies and build up an import map to be injected into the <head>
of your HTML automatically, in conjunction with Greenwood's dev server.
Below is a sample of an import map that would be generated after having installed the lit package:
<html>
<head>
<script type="importmap">
{
"imports": {
"lit": "/node_modules/lit/index.js",
"lit-html": "/node_modules/lit-html/lit-html.js",
"lit-element": "/node_modules/lit-element/index.js",
"...": "..."
}
}
</script>
</head>
<body>
<!-- ... -->
</body>
</html>
To generate this map, Greenwood first checks each package's exports field, then looks for a module field, and finally a main field. If none of these fields are found, Greenwood will log some diagnostics information. For exports field, Greenwood supports the following conditions in this priority order:
- import
- module-sync
- default
Compatibility
However in the land of node_modules, not all packages are created equal. Greenwood depends on packages following the standard conventions of the NodeJS entry point specification when resolving the location of dependencies on disk using import.meta.resolve
. For server-side only packages, this is is usually not an issue. Greenwood will output some diagnostic information that can be used when reaching out for help in case something ends up not working as expected, but if it works, it works!
Some known issues / examples observed so far include:
ERR_MODULE_NOT_FOUND
- Observed with packages like @types/trusted-types which has an empty string for the main field, or font-awesome, which has no entry point at all, at least as ofv4.x
. This is also a fairly common issue with packages that primarily deal with shipping types since they will likely only define atypes
field in their package.json.ERR_PACKAGE_PATH_NOT_EXPORTED
- Encountered with packages like geist-font or @libsql/core, which has no default export in their exports map, which is assumed by the NodeJS resolver algorithm.
In almost all of our observed diagnostic cases, they would all go away if this feature was added to NodeJS, so please add an up-vote to that issue! 👍
If you have any issues or questions, please reach out in our discussion on the topic.
URL
The URL
constructor provides an elegant way for referencing static assets on the client and on the server, and it works great when combined with URLSearchParams
for easily interacting with search params in a request.
Below is an example used in an API Route handler:
export async function handler(request) {
const params = new URLSearchParams(request.url.slice(request.url.indexOf("?")));
const name = params.has("name") ? params.get("name") : "World";
const msg = `Hello, ${name}! `;
return new Response(JSON.stringify({ msg }), {
headers: new Headers({
"Content-Type": "application/json",
}),
});
}
FormData
FormData
is a very useful Web API that works great both on the client and the server, when dealing with forms.
In the browser, it can be used to easily gather the inputs of a <form>
tag for communicating with a backend API:
<html>
<head>
<script>
window.addEventListener("DOMContentLoaded", () => {
window.document.querySelector("form").addEventListener("submit", async (e) => {
e.preventDefault();
// with FormData we can pass the whole <form> to the constructor
// and send a URL Encoded request to the API backend
const formData = new FormData(e.currentTarget);
const term = formData.get("term");
const response = await fetch("/api/search", {
method: "POST",
body: new URLSearchParams({ term }).toString(),
headers: new Headers({
"content-type": "application/x-www-form-urlencoded",
}),
});
// ...
});
});
</script>
</head>
<body>
<h1>Search Page</h1>
<form>
<label for="term">
<input type="search" name="term" placeholder="a product..." required />
</label>
<button type="submit">Search</button>
</form>
</body>
</html>
On the server, we can use the same API to collect the inputs from that form request:
// src/pages/api/search.js
// we pull in WCC here to generate HTML fragments for us
import { getProductsBySearchTerm } from "../../db/client.js";
export async function handler(request) {
// use the web standard FormData to get the incoming form submission
const formData = await request.formData();
const term = formData.has("term") ? formData.get("term") : "";
const products = await getProductsBySearchTerm(term);
// ...
}