nx-logo

Introduction

Nx is a tool that manages monorepos and standardises tooling within that monorepo.

As part of these standards, Nx provides a set of tools that allow users to quickly generate a static vanilla HTML website, without any complicated tooling.

Prerequisites

To begin, make sure we have an Nx workspace setup.

npx create-nx-workspace@latest <workspace-name>
✔ Which stack do you want to use? · none
✔ Package-based monorepo, integrated monorepo, or standalone project? · integrated
✔ Do you want Nx Cloud to make your CI fast? · skip

This will generate an Nx workspace, that we can assign any name to. I choose an integrated repo because I generally use Nx to work with multiple projects that interact with each other at the monorepo level.

If we face an error stating that the npx command is not found, we can get it by installing the latest version of NodeJS. npx will come with that version of NodeJS.

For reference, the version that this article uses is NodeJS is v20.12.2.

Scaffolding our HTML Site

To generate our site, we will use the Nx provided plugin @nx/web. First we get this plugin by installing it via npm into the root of our monorepo, then use it to generate our application.

cd <workspace-name>
npm install -D @nx/web

npx nx generate @nx/web:application
✔ What name would you like to use for the application? · <app-name>
✔ Which stylesheet format would you like to use? · css
✔ Which bundler do you want to use? · webpack/vite
✔ Which E2E test runner would you like to use? · none

This will generate a folder with the following structure:

.
├── index.html
├── project.json
├── public
│   └── favicon.ico
├── src
│   ├── app
│   │   ├── app.element.css
│   │   ├── app.element.spec.ts
│   │   └── app.element.ts
│   ├── assets
│   ├── main.ts
│   └── styles.css
├── tsconfig.app.json
├── tsconfig.json
├── tsconfig.spec.json
└── vite.config.ts

Commands

To serve or build, we can use inferred commands from the @nx/web plugin. To find out more, read about inferred tasks.

npx nx run <app-name>:serve   # to serve
npx nx run <app-name>:build   # to build

Observations

Folder structure

Those familiar with Nx will find the project.json file familiar. Here we store all our project related configurations for Nx to parse, and which also helps to define the project-level behavior of our static site within the context of the Nx monorepo.

Like all other static sites, we can identify that the entrypoint of our project is index.html. Within it the line <script type="module" src="/src/main.ts"></script> loads the module located at src/main.ts, which sets the ground for exporting web components that we can then use in the index.html file. The usage of our very first web component is already provided by the line <static-site-root></static-site-root> in said file.

More on Web Components

Because web components seems to be the preferred way of handling componentisation in the @nx/web plugin, we can look at it a little more.

The contents of the app.element.ts file should be familiar to those familiar with web components.

import './app.element.css';

export class AppElement extends HTMLElement {
  public static observedAttributes = [];

  connectedCallback() { ... }
}

customElements.define('static-site-root', AppElement);

This basically defines and exports a web component, allowing us to build resuable components for our static site.

For those unfamiliar with web components, feel free to check out my overly detailed write up about it here.

Limitations

Some key limitations of this generator include

  1. No easily handled routing, unlike in more advanced static html generators
  2. Modularisation and template reuse with web components is not necessarily developer friendly, because it still mixes JS with HTML

Conclusion

In this short article, we have looked at using Nx to scaffold a static vanilla HTML site. Hope it was useful :)