A minimal base HTML5 template and the esbuild setup to compile your SolidJS app within 11ty.
Includes:
- 11ty/is-land
- WebC,
- esbuild
- Minifiying and autoprefixing of styles using Lightning CSS
- Uses Purgecss to remove unused styles via a slightly modified version of esbuild-plugin-purgecss-2
- A shortcode to compile SolidJS inline
- Cashebusting via an esbuild generated hash.
- A manifest.json file and a buildmeta.json file
- You can also import HTTP URLs into JavaScript code using esbuild-plugin-http.
Add your_solid.jsx
file to the src/scripts/jsx
or the src/scripts/js
folders. Esbuild will output a minified js file. To configure esbuild modify config/build/esbuild.js
.
If you need to compile your js script inline, use this shortcode:
{% solid "filename", "bundled" %}
your.solid.jsx.code
{% endsolid %}
The shortcode will generate a module script tag.
<script type="module">your.es6.js.code</script>
There are two arguments:
filename
(required): The name of the file which is saved todist/app
. This name is automatically prefixed bysolid-
.bundled
(optional): The solid.jsx is bundled by default. To switch bundling off pass the value:"bundleOff"
.
To configure esbuild for js/jsx files modify config/build/esbuild.js
or to configure the shortcode, modify config/shortcode/solidify.js
. For further info check out the esbuild-plugin-solid github repo by amoutonbrady.
/** Configuration options for esbuild-plugin-solid */
export interface Options {
/** The options to use for @babel/preset-typescript @default {} */
typescript: object
/**
* Pass any additional babel transform options. They will be merged with
* the transformations required by Solid.
*
* @default {}
*/
babel:
| TransformOptions
| ((source: string, id: string, ssr: boolean) => TransformOptions)
| ((source: string, id: string, ssr: boolean) => Promise<TransformOptions>);
/**
* Pass any additional [babel-plugin-jsx-dom-expressions](https://github.com/ryansolid/dom-expressions/tree/main/packages/babel-plugin-jsx-dom-expressions#plugin-options).
* They will be merged with the defaults sets by [babel-preset-solid](https://github.com/solidjs/solid/blob/main/packages/babel-preset-solid/index.js#L8-L25).
*
* @default {}
*/
solid: {
/**
* The name of the runtime module to import the methods from.
*
* @default "solid-js/web"
*/
moduleName?: string;
/**
* The output mode of the compiler.
* Can be:
* - "dom" is standard output
* - "ssr" is for server side rendering of strings.
* - "universal" is for using custom renderers from solid-js/universal
*
* @default "dom"
*/
generate?: 'ssr' | 'dom' | 'universal';
/**
* Indicate whether the output should contain hydratable markers.
*
* @default false
*/
hydratable?: boolean;
/**
* Boolean to indicate whether to enable automatic event delegation on camelCase.
*
* @default true
*/
delegateEvents?: boolean;
/**
* Boolean indicates whether smart conditional detection should be used.
* This optimizes simple boolean expressions and ternaries in JSX.
*
* @default true
*/
wrapConditionals?: boolean;
/**
* Boolean indicates whether to set current render context on Custom Elements and slots.
* Useful for seemless Context API with Web Components.
*
* @default true
*/
contextToCustomElements?: boolean;
/**
* Array of Component exports from module, that aren't included by default with the library.
* This plugin will automatically import them if it comes across them in the JSX.
*
* @default ["For","Show","Switch","Match","Suspense","SuspenseList","Portal","Index","Dynamic","ErrorBoundary"]
*/
builtIns?: string[];
};
}
Esbuild is configured to add a hash to the CSS and JS files it processes in the src/scripts/jsx
, src/scripts/js
and the dist/app/css
folders (it purges the prefixed output of the scss in situ). It outputs a manifest.json
file to the src/_data
directory.
The manifest.json file is used in the hash filter to modify the URL src or href in the html:
<script src="{{ '/app/app.js' | hash }}"></script>
<link rel="stylesheet" href="{{ '/app/style.css' | hash }}" />
As a bonus if the file has been minified in production it will alter the file extension to [hash]-min.js
or [hash]-min.css
, for example:
<script src="/app/app-S5YUTCHU.min.js"></script>
Your css files will be automatically purged of unused css in production. To configure the purgecssPlugin modify the config/build/purgecss.js
file. You can use any configuration pattern from purgecss, but you don't need to set the css:
options as this is automatically included from the buildmeta.json.
plugins: [
...
purgecssPlugin({
content: ["dist/index.html"]
}),
]
The pathPrefix
npm script argument from 11ty (--pathprefix=form-2-pdf",
) is passed through as an environment variable (along with all the npm script arguments) using esbuild's define api. You can add other environment variables by adding them to the defineEnv const in the config/build/esbuild.js
script.
const defineEnv = {
'process.env.PATHPREFIX': JSON.stringify(pathPrefix),
// Add other environment variables as needed
};
If you decide to use a client-side router such as solid router you could do the following in your jsx:
const pathPrefix = process.env.PATHPREFIX;
const urlPrefix = pathPrefix ? `/${pathPrefix}` : "";
render(
() => (
<Router>
<Route path={urlPrefix}> {/* solid-js router uses urlPrefix here to set the url path */}
<Route path="/" component={YourComponent} />
</Route>
</Router>
),
document.getElementById('app')
);
In a production build the css and js files are automatically compressed and output as minified, gzipped and brotli files:
app-S5YUTCHU.min.js
app-S5YUTCHU.min.js.br
app-S5YUTCHU.min.js.gz
To alter this behaviour modifiy the following in the config/build/esbuild.js
script:
if (isProd) {
esbuildOpts.plugins.push(gzipPlugin({
uncompressed: isProd,
gzip: isProd,
brotli: isProd,
}));
}
npm start
Run 11ty with hot reload at localhost:8080, including reload based on Sass and JS changes.
npm run cloud
Development build for use on cloud IDE's such as Stackblitz without the pathprefix of /11ty-solid-base/. The CSS is autoprefixed but no minification or purging happens.
If your using Stackblitz. To start the cloud dev server run: npm run cloud
then npm run start
.
npm run build
Staging build with autoprefixed CSS but no minification or purging.
npm run minify
Production build includes minified, autoprefixed and purged CSS
Use this as the "Publish command" if needed by hosting such as Netlify.
Lightningcss can be configured to support css modules by adding --css-modules
to the prefix
npm script. Alternatively you could use esbuild css modules plugin. An example setup of scoped css modules with esbuild.
- Look at adding js import maps
- The web component (webC example)
- improve styling and make prettier hydration examples
- Example using Solid Element web components
- Maybe look as using eleventy-plugin-bundle and esbuild to create per page solidjs asset buckets/bundles.
- Alter the solid shortcode API so that the arguements are
bundle
and bundleOff instead of the current API ofbundled
andbundleOff
. - Change the esbuild/solid internal configuration so that its an eleventy-plugin instead.
- Should I remove the sass dependency?