-
-
Notifications
You must be signed in to change notification settings - Fork 498
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Official Asset Pipeline #272
Comments
This repository is now using lodash style issue management for enhancements. This means enhancement issues will now be closed instead of leaving them open. The enhancement backlog can be found here: https://github.com/11ty/eleventy/issues?utf8=%E2%9C%93&q=label%3Aneeds-votes+sort%3Areactions-%2B1-desc+ Don’t forget to upvote the top comment with 👍! |
In my opinion, Eleventy’s job is to expose the relevant pieces of content at appropriate stages of transformation so that available tooling can be used via their Node packages. There is already some interesting examples of how to realize this is in the docs (e.g. compressing CSS). It would be interesting to figure out how to use Eleventy with JavaScript bundlers like Parcel.js or Webpack. Can we somehow offer a simple way of using Eleventy and; for example, Parcel.js without resorting to starting both Eleventy and Parcel is separate terminal windows in parallel? I think Eleventy has great potential in solving this build tool pain points that a lot of static site generators bring with them. |
I wouldn't have considered picking up Eleventy were it not for the Eleventyone project. I'm not a developer, so though I'm sure it's possible to add yourself, it wasn't something I wanted to attempt. I've used grunt before, but again, didn't set it up to start with. If it's helpful, here's some things I wanted: My must haves:
For me, gulp has worked well. I can have different scripts for dev and production so different things get built - such as uncompressed / sass-maped sources on dev, and fully minified sources on prod. I don't concatenate all files together - assets that may not be needed, or aren't needed by most pages are kept and served separately. Gulp made this easy. It's possibly related to my background, but I prefer the idea of css concatenation happening separately from templates. A possible useful starter might be a tutorial / guidance on adding a build pipeline. I suspect you want to keep the base repo rather agnostic about this - this is where the tutorial section could really help I think. |
I'd certainly vote for this to be outside the core, as an explicit opt-in or just a sample/starter project. I've had luck with I've been working on setting up this kind of workflow with webpack v4 (mix is still at v3), but I've also been playing with parcel.js. I'll post a repo once I have something reasonable to share. It would probably bypass browserify completely, but I'd still want live reloading and watchers for all files. |
I'm looking into this as well. 11ty works well with assets in the site content (custom scripts and styles) but how to manage npm dependencies is proving difficult. For example, I added bootstrap to package.json now I need my pipeline to bundle all of bootstraps assets with mine. Currently I'm looking at calling browserify or webpack from gulp but looking for a simple answer if someone has solved it already. Thanks! |
Worth mentioning this here, please upvote if you like this idea of configuring eleventy to watch other globs: |
I think it would be great to include documentation about how to include some different asset pipelines easily (and maybe go so far as to provide those as plugins), but as others have said, it'd be nice to not include those by default in core. I really like it being "Bring your own ____" (markup/styles/asset pipeline/cms/etc.) |
I managed to get something working pretty well with webpack4 and eleventy.
I tried getting it working with webpack-dev-server, but I couldn't manage to get both the 11ty side and the webpack side to recognize each others' changed files — auto refresh wouldn't work. I'll probably put up a fresh repo soon and link to it here, maybe w/ a brief tutorial about this approach. I could make that repo as a starter project, too/instead. A plugin of sorts would be nice as a simple drop-in, yet I don't know how I'd approach it as a standalone plugin. (@zachleat any thoughts on how to approach this?) It currently uses Looking to gauge interest and get all your opinions/thoughts on this as starter project vs a drop-in plugin type thing. I think it makes more sense as a starter project or just a demo/tutorial. A drop-in plugin may end up hiding too much, getting us further away from "bring your own x". I'm not married to webpack (I'm certainly no webpack pro), so I may try another bundler like parcel.js and see if it's any easier. |
One thing that I think is common in the webpack space is to give files cache busting hashes and use the HTMLWebpackPlugin to make sure the proper hashes get included in the html files: For example:
@jevets did you do any of this with your eleventy integration? |
@robdodson No, I didn't get that far but am aware of the need. May ask for some input from you if you're down. Would love to have some others help on this. (Been away for a few weeks, will try to post some code this week.) |
@robdodson You could use something like webpack-manifest-plugin or webpack-asset-manifest to generate a json file that maps your entry points to the generated files. // Example manifest.json
{
"main.js": "assets/js/main.12345645423.js"
} In eleventy a custom shortcode or filter could than be used to lookup the generate file name from the json file. |
I'd prefer to be bundler agnostic (yes, I don't like WebPack, but prefer Rollup). |
I've been using a With this route I get to avoid pulling in any additional packages (webpack, gulp, parcel, etc.). It's been working very well for me lately, but I haven't yet used this route for sites heavy on client-side JS. Something like this: // stylesheet.11ty.js
module.exports = class {
data() {
return {
permalink: '/main.css'
}
}
render() {
return this.doTheCompileWorkAndGetTheResultingString()
}
doTheCompileWorkAndGetTheResultingString() {
return postcss([...])
}
} /* _includes/styles.css */
body {
font-family: sans-serif;
/* lives in `_includes` so eleventy automatically watches it for me */
} |
For anyone stumbling across here, this is my general asset setup for static site generators.
|
@jevets do you have a working example of |
@jevets I had the same problem with trying to get webpack and Eleventy to "see" each other's stuff. If I used Eleventy's built-in Browsersync instance, textual changes were auto-refreshed but SCSS changes were not. The reverse was true — text yes, SCSS no — if I used webpack-dev-server with Eleventy set only to watch and not to serve. So, after thinking about the issue for a few days, today I tried a separate instance of Browsersync via browser-sync-webpack-plugin with, again, Eleventy set only to watch and not to serve — and, lo and behold, that worked. Now I get auto-refresh on any change. I have no doubt there are more elegant solutions out there, but this one, finally, did the trick for me. I reproduced the relevant config files in my now-updated post here, in case they might help someone. |
What I'm doing which works pretty well is the following:
This is great because it's fast (webpack runs incremental) and I can utilize webpacks I have a (slightly outdated) version of this idea up here. I'll update this later with some small improvements I made in my actual (closed source) project. Only point of friction is that it's inherently a separate system. So adding another entry point is not just a matter of dropping a JS file in a directory, but also needs a change in the webpack config. Not a huge problem since the entrypoints usually don't change much, but still a slight annoyance. |
@pascalw That smart method raises another thought. I often see people wishing for a pagination function that would give each individual post the “next post” and ”previous post” links (e.g., #529); indeed, I wish for it, too; right now, I manually give each post an |
can be helpful too eleventyConfig.setBrowserSyncConfig({
files: [
'public/css',
'public/js'
]
}); |
Updated 1/11/2021 I wanted to share, this is how I'm using Gulp to:
So instead of tasking Eleventy with being a build pipeline that can do everything else, I'm including Eleventy in my external build pipeline, where everything else is handled. Gulp is invoking Eleventy via the API, instead of Eleventy being invoked via the CLI. This lets me grab the HTML built by Eleventy in a Gulp stream, where I can then post-process the HTML which was built by Eleventy. From the CLI:
|
@jevets I like your solution, would it be possible to do something like that but for bundling JS? |
I like the idea of keeping asset management independent of 11ty, but I think it would be nice to have some recipes in the docs for stuff like cache busting. It's great to allow people the freedom to configure everything the way they want, but it's also nice to not have to blaze your own trail for every task — especially common tasks like cache busting. |
I just wrote up a technique for processing styles/scripts using Eleventy that allows you to transform assets (i.e. compile SCSS into CSS, or bundle your JS) and use hashed file names for cache-busting. I think it's a bit abusive of things like Eleventy's global data files, but it has the advantage of not requiring additional dependencies like Webpack or Parcel. I can't necessarily recommend that you do this, but perhaps it can serve as a strawman for coming up with an official technique for managing assets in Eleventy. |
Very nice! I recently posted this article about doing it with a PostCSS plugin, but your solution is better and definitely more inclusive, I think. |
I know not everyone is using Netlify, but I was trying to find a solution for asset bundling in Eleventy and found that it's a built-in option with Netlify either through the admin uI or the |
Adding another vote for this. I realise that people want to use different pipelines, but since SASS and JS bundling is so common, it would be great to have a recommended starting point for newcomers where they can easily drop in to their project and get up and running fast. I've always found webpack pretty slow and unwieldy, so am hoping Vite.js overtakes it at some point (from the creators of Vue JS). Whether it's mature enough to use as the default, I'm not sure, but it is designed to be pre-configured for most typical tasks, which is the kind of thing that would be great to have in Eleventy. |
I found using esbuild it's pretty simple and straightforward: // /assets/js/main.11ty.js
module.exports = class {
data() {
return {
layout: '',
permalink: false,
eleventyExcludeFromCollections: true,
};
}
async render(data) {
/* https://esbuild.github.io/getting-started/#build-scripts */
require('esbuild')
.build({
entryPoints: ['_includes/js/main.js'],
bundle: true,
outfile: '_site/assets/js/main.js',
})
.catch(() => process.exit(1));
}
}; And for CSS: // /assets/css/style.11ty.js
const path = require('path')
const sass = require('node-sass-promise')
const CleanCSS = require('clean-css')
const inputFile = path.join(__dirname, '../../_includes/scss/main.scss')
const outputFile = path.join(__dirname, '../../assets/css/style.css')
module.exports = class {
data() {
return {
layout: '',
permalink: 'assets/css/style.css',
eleventyExcludeFromCollections: true
}
}
async render() {
const { css } = await sass.render({ file: inputFile })
const output = new CleanCSS({}).minify(css.toString()).styles
return output
}
} |
Just my two cents: I found nice and interesting examples as well, e.g. https://hipsterbrown.com/musings/musing/esbuild-with-11ty/ or https://github.com/jamshop/eleventy-plugin-esbuild (and so many others as mentioned here). Very exciting to see so many examples! While this is a very nice challenge for nerds to test on which packages they want to rely - in my personal opinion this could lead to "decision fatigue" and a robust starter would be good for many users of 11ty. A good and simple test case could be bootstrap5 I guess, because scss and js compiling is needed as well PostCSS autoprefixer is recommended (https://getbootstrap.com/docs/5.0/getting-started/download/#source-files). npm install bootstrap @popperjs/core --save-dev /* app.js */
import 'bootstrap'; /* app.scss,
see: https://getbootstrap.com/docs/5.0/customize/sass/ */
@import "~bootstrap/scss/bootstrap"; I recently found https://github.com/bergwerk/11ty-mix (Laravel Mix => Webpack) for this and got it working by modifying the package.json as suggested for my own projects. As far as I understand https://laravel-mix.com/ was built for the same challenge - keeping asset building simple for devs. (Here is a simple guide for bootstrap5 and laravel mix in general https://5balloons.info/setting-up-bootstrap-5-workflow-using-laravel-mix-webpack/, laravel mix also has autoprefixer automatically activated https://laravel-mix.com/docs/6.0/autoprefixer) But the most readable and accessible way for (new) 11ty users would probably be using the 'afterBuild' hook in eleventy config? :-) |
Yes, very much agreed. A robust quickstart guide to enabling things like Bootstrap, SCSS and JS would be much appreciated, otherwise it takes a fair bit of research to read through the options of varying quality to work out which is going to work best. Even Jekyll has better SCSS support out-of-the-box than Eleventy at the moment. |
@mandrasch @SimonEast - have you tried the Vite plugin and associated youtube video? |
No, haven't seen or tried. Thanks very much, exciting times ahead! 👍 (Although we'd have to wait for stable release of Eleventy 2.0+ to use this in production if I understand it correctly. So for Eleventy 1.0 a simple starter would be still a nice improvement) |
@mandrasch I've been building a site on eleventy 2.0 for a few weeks now (not production yet) and haven't come across any issues preventing me from continuing. |
Thanks for sharing your experience on this! Optional information for the interested readers regarding Eleventy v1: |
Just wanted to share that I recently wrote a plugin integrating Parcel in Eleventy's build pipeline. It also lets you (optionally) use the Parcel dev server as middleware. It's templated after Zach's new eleventy-plugin-vite, and similarly requires Eleventy 2.x. Vite's looking really promising, but I was curious about Parcel as well! |
I always install webc plugin, and in my base.webc, in the head. I do this:
Result:
By adding the at layer rule on top, you get the natural cascade you want and later, in your components, you add the @layer rule you need in order to have the css in it's proper place in the cascade. I'm still experimenting with this technique to see if it holds true. Voila, your css is bundled in one neat file! Ok. Maybe only a small part of the asset pipeline ... :) |
Surfaced in #270.
Maybe included as a separate starter project, plugin, and/or with the eleventy-base-blog project
The text was updated successfully, but these errors were encountered: