Soar is a minimal opinionated static site generator built around JSX and filesystem routing.
Core ideas:
- Embrace JSX without serving client-side JS
- Leverage ES modules and Node's custom module hooks
- Filesystem routing
- Scoped styles through CSS-in-JS and CSS modules
- Scripts and styles must be included explicitly to keep sites lean (manual
<script>
and<link>
tags) - Extensible pipelines per file type
# install (global)
$ npm i -g soar
$ pnpm i -g soar
# build site
$ soar build src/
# start dev server
$ soar serve src/
Soar does not need a config file for most things, but can be provided with one.
The default export of Soar.ts
in the source directory will be used.
See config.ts for available options.
The file tree within the source directory is used for building the output site. This makes routing simple as creating a directory structure. There are two exceptions to this rule:
- Pages are built into
$PAGE/index.html
and not$PAGE.html
. ignore
d patterns in theSoar.ts
config file are not included in the output.
Pipelines transform source files before building and serving.
Pipelines are applied according to the file extension.
Files beginning with an underscore _
are not passed through any pipelines, and are copied as-is.
The JSX pipeline treats .tsx
and .jsx
files with a default
export as pages.
A custom JSX runtime is used to render JSX into HTML. Only functional elements are supported.
Component and layout files can be created without getting treated as pages by prefixing the file or directory name with an underscore (_component.tsx
).
Attributes in JSX are passed through to HTML, except for those starting with an underscore _
.
Props in JSX are never passed to HTML implicitly.
For example, classes must be directly applied to the inner HTML element for <Component class={class}>
.
Basic example:
import Layout from "./_layout.tsx"
export default () => (
<Layout>
<h1 id="hello-world">Hello world!</h1>
</Layout>
)
Styles created with css
and spread onto JSX nodes are automatically scoped and inlined into HTML.
This CSS-in-JS tranform is done at build time without client-side JS.
For example:
import { css } from "soar"
const styles = {
red: css`
color: red;
`,
green: css`
color: green;
`,
}
const Red = () => <span {...styles.red}>This is red</span>
const Component = () => (
<div
{...css`
color: blue;
`}
>
<span>This is blue</span>
<Red />
<span {...styles.green}>This is green</span>
</div>
)
Stylesheets can be created directly through CSS files. See the CSS pipeline.
Inline styling can be done through the style
attribute, which supports both HTML-like string styles and JS-like object styles.
.css
files are bundled and transformed with Lightning CSS.
These files must be included in pages explicitly with <link>
tags, like you would do for an HTML file.
The CSS pipeline supports CSS modules:
import styles from "./styles.css"
// <span class="xxxxxx_red">
const Red = ({ children }) => <span class={styles.red}>{children}</span>
.mdx
files are built into JSX and the JSX pipeline is applied.
Custom remark
and rehype
plugins can be specified in the config file.
.ts
and .js
files are bundled and built into .js
files.
These JS files must be included in pages explicitly with <link>
tags, like you would do for an HTML file.