Jekyll Workflow With Gulp
Introduction
This weekend I decided to take this blog's design and create a theme. Of course it's not ready yet, because in the process of refactoring the SASS code I came across a problem. I 'm using flexbox but I didn't have prefixes! It's not the end of the world but this led to the fact that because of the github pages limitations I 'm limiting my workflow too. This is a problem.
Gulp to the rescue
So, we are going to include Gulp and some plugins to make our lives easier.
This process has two parts. The first one is the Jekyll's _config.yml configuration and the creation of a new folder we are going to have our development files, and our gulpfile.js configuration/setup.
Part I: Jekyll
Let's add some configuration to _config.yml.
exclude: ["_dev",
"gulpfile.js",
"node_modules",
"package.json"]
exclude
is a configuration setting variable in Jekyll's core to exlude paths or files from the site's conversion. (aka _site folder).
Exclude directories and/or files from the conversion. These exclusions are relative to the site's source directory and cannot be outside the source directory.
So, we don't want these files in our production site. I think it's pretty straightforward what we did here, but what you are goin to say is 'What the f*uck is the _dev folder?' and you 'll be right. The _dev folder is going to be the folder to keep our sass,js...(and whatever you want) source files which we don't want to be visible to the website.
Our project folder should be like this. The partials folder isn't mandatory, it's the way I structure my sass code. (if you feel comfortable tweaking the gulpfile.js you can work with any structure you want :) ):
|-- project |-- gulpfile.js |-- index.html |-- package.json |-- _config.yml |-- assets | |-- css | | |-- style.css | |-- img | |-- js |-- _dev | |-- src | |-- sass | |-- style.scss | |-- partials | |-- _base.scss | |-- _functions.scss | |-- _grid.scss | |-- _layout.scss | |-- _reset.scss | |-- _syntax.scss | |-- _variables.scss |-- _includes |-- _layouts |-- _posts |-- _site
And that's all we need to do for the Jekyll part.
Part II: Gulp
Now, let's install all the things!
You can copy paste this to a package.json file
{
"devDependencies": {
"gulp": "^3.9.0",
"gulp-autoprefixer": "^3.1.0",
"gulp-connect": "^2.3.1",
"gulp-clean-css": "^2.0.13",
"gulp-plumber": "^1.0.1",
"gulp-rename": "^1.2.2",
"gulp-sass": "^2.1.1",
"gulp-util": "^3.0.7"
}
}
and install by typing in your terminal
npm install
or create an empty package.json by typing
npm init
and follow the instructions to create a package.json file.
And then install by
npm install --save-dev gulp gulp-sass gulp-util gulp-plumber gulp-rename gulp-clean-css gulp-autoprefixer gulp-connect
Now we have our dependences let's build our gulpfile.js.
// Require all the things
const gulp = require('gulp'),
sass = require('gulp-sass'),
gutil = require('gulp-util'),
plumber = require('gulp-plumber'),
rename = require('gulp-rename'),
minifyCSS = require('gulp-clean-css'),
prefixer = require('gulp-autoprefixer'),
connect = require('gulp-connect');
cp = require('child_process');
// Set the path variables
const base_path = './',
src = base_path + '_dev/src',
dist = base_path + 'assets',
paths = {
js: src + '/js/*.js',
scss: [ src +'/sass/*.scss',
src +'/sass/**/* .scss',
src +'/sass/**/**/*.scss'],
jekyll: ['index.html', '_posts/*', '_layouts/*', '_includes/*' , 'assets/*', 'assets/**/*']
};
// Compile sass to css
gulp.task('compile-sass', () => {
return gulp.src(paths.scss)
.pipe(plumber((error) => {
gutil.log(gutil.colors.red(error.message));
gulp.task('compile-sass').emit('end');
}))
.pipe(sass())
.pipe(prefixer('last 3 versions', 'ie 9'))
.pipe(minifyCSS())
.pipe(rename({dirname: dist + '/css'}))
.pipe(gulp.dest('./'));
});
// Rebuild Jekyll
gulp.task('build-jekyll', (code) => {
return cp.spawn('jekyll', ['build', '--incremental'], { stdio: 'inherit' }) // Adding incremental reduces build time.
.on('error', (error) => gutil.log(gutil.colors.red(error.message)))
.on('close', code);
})
// Setup Server
gulp.task('server', () => {
connect.server({
root: ['_site'],
port: 4000
});
})
// Watch files
gulp.task('watch', () => {
gulp.watch(paths.scss, ['compile-sass']);
gulp.watch(paths.jekyll, ['build-jekyll']);
});
// Start Everything with the default task
gulp.task('default', [ 'compile-sass', 'build-jekyll', 'server', 'watch' ]);
Important For Windows users: You need to replace in the build-jekyll task the 'jekyll' string with the path of your jekyll.bat. In my case it was in the C:\Ruby22-x64\bin
so I changed this code to:
2016/11/20 Update: As Ron Dyar commented, in some Windows OS cases a jekyll.bat is enough, without the path. You can try both and see which one works for you.
// Rebuild Jekyll
gulp.task('build-jekyll', (code) => {
return cp.spawn('C:\\your_ruby_path\\bin\\jekyll.bat', ['build'], {stdio: 'inherit'})
.on('error', (error) => gutil.log(gutil.colors.red(error.message)))
.on('close', code)
})
2016/01/10 Update: Alex Lockwood was kind enough to create a gist for older npm versions on which ES6 features cause errors.
2017/02/11 Update: A user's comment I cannot find requested to update the ruby path with a newer version. So, in the your_ruby_path
you need to set your ruby path. For example, mine was Ruby22-x64
at the time.
gulp
and you can go to http://localhost:4000
and see your site. All changes will be watched and compiled.
What our gulpfile does:
- Builds Jekyll
- Minifies/compiles/autoprefixes SASS
- Sets up a local server
- Watches our source files(Jekyll's and static assets) and recompile when changes on update.
Let's see what each plugin does for our case:
gulp: You know what this does!
gulp-sass: Takes our sass files and compiles them to css.
gulp-util/gulp-plumber: Some gulp utilities to catch and report errors. For some reason the gulp-sass plugin stops gulp on error and we have to start it again, so with these two we can prevent that.
gulp-rename: Renames our CSS files because if we didn't we would get a file like this /assets/css/src/css/style.css
.
gulp-minifyclean-css: Simple!
gulp-autoprefixer: What started all this! This prefixes our css. There are some great options for this plugin, be sure to check them out! Github
gulp-connect: Setups a local web server to preview our site. It's inevitable because we can't use jekyll serve
like we did before.
child_process: This is not a plugin. This is a Node.js core module. We are using this because (very roughly) Gulp is running in one process and we need to run the jekyll build
process at the same time, we cannot stop Gulp. So we are 'attaching' the jekyll command to a child process that returns that everything went fine when it finishes. I want to see this in depth but I got carried away with the Windows problem I warned you before.
That's it! As I said before just run gulp
and work with your masterpiece!
Conclusion
Having this workflow might be strange at the beginning but it will help you, I promise. I didn't include the js files manipulation on purpose so that you can do some tests. But it will be available to the theme workflow I am working right now. Stay tuned!
If you have and questions or comments don't hesitate to contact me here or at my twitter account.
Cheers!