Introduction to Webpack tutorial
- Checkout this repository
- run
npm install
The following is a simple walk-through that will get you familiar with the basics of Webpack. The tutorial should take about fifteen minutes.
- Open index.html in the file in the browser.
- Open the browser developer tools, and observe that the browser returns a 404 when trying to get
bundle.js
. - Let's build
bundle.js
using Webpack. In a terminal, typewebpack ./entry.js bundle.js
. - Refresh the browser. You should see the words "Hello, world." in the browser.
- Open
entry.js
in your editor. Change "world" to "Workfront" and save the file. - In the terminal, run
webpack ./entry.js bundle.js
again and refresh the browser. You should see "Hello, Workfront.". - Manually running the webpack command after every change is tedious. Run
webpack -w ./entry.js bundle.js
, where-w
is the watch flag. When your entry file changes or any of its dependencies, it will rebuild the bundle. - Let's try out the watcher. Edit
entry.js
so that it ends the sentence with a "!" instead of a ".". Save the file. You should see in the terminal that it saw that your file changed and rebuilt by the bundle. Refresh your browser to see the change. - So far, we've been using Webpack as a glorified file copier. Let's add a dependency to our
entry.js
file and see what Webpack does. Changeentry.js
as follows:
'use strict';
document.write('Hello, Workfront! The current time is: ' + require('./displayTime'));
We are including displayTime.js
as a CommonJS dependency in our file. Open up displayTime.js
. You'll see the following:
define(['moment'], function(moment) {
return moment().format('dddd, MMMM D, YYYY, h:mm:ss a');
});
This file defines an AMD module. Webpack natively understands both CommonJS and AMD syntax. Additionally, it uses a third party module, moment, which is a date parsing, formatting, and localization library. Where does it get this dependency from? It actually looks in your node_modules directory for any dependencies. Moment was installed as a npm dependency.
Refresh your browser, and you'll see the changes that you made, along with the current date and time.
- Open
bundle.js
in your editor. While the file is fairly easy to read, it has a lot of extra Webpack noise that makes it difficult to debug in the browser. Let's add source maps. Stop the Webpack watch that you're currently running, and instead runwebpack -w --devtool source-map ./entry.js bundle.js
. The new switch,--devtool
, will generate a source map as part of the build. This will make debugger easier. - Refresh the browser, and take a look at the browser dev tools "source" section. You'll see a root named "webpack://". Expand the source tree there and you'll find the source for both "./entry.js" and "./displayTime.js".
- Let's try debugging the application using Webpack and the source maps. Add a
debugger;
statement indisplayTime.js
, like so:
define(['moment'], function(moment) {
debugger;
return moment().format('dddd, MMMM D, YYYY, h:mm:ss a');
});
- Refresh the browser, and notice that the app stops at the debugger breakpoint. Convenient, huh? Hit the continue button to let the page continue. Once you're done playing with the debugger, remove the debugger statement from the code.
- Our page looks kind of plain. Let's add some style. Create a file,
style.css
, in the root of your project and add the following content to it:
body {
font-size: 2em;
background-color: greenyellow;
}
- How do we use this new stylesheet? Let's require it in
entry.js
:
'use strict';
require('./style.css');
// ... rest of file
- Save
entry.js
. You'll see an error in your terminal similar to the following:
ERROR in ./style.css
Module parse failed: /Users/jeremylund/dev/training/webpack-start/style.css Line 1: Unexpected token {
You may need an appropriate loader to handle this file type.
| body {
| font-size: 2em;
| background-color: greenyellow;
@ ./entry.js 2:0-22
By default, Webpack treats every file as JavaScript. Since this isn't JavaScript, it doesn't know what to do with it. Thankfully, the error message gives us a hint as to what to do. We need to install a loader. Loaders are used to transform files into a form that Webpack can handle.
- Stop the Webpack watcher, and type the following in the terminal:
npm install style-loader css-loader -D
. This will install the two loaders that we need. Modify the where we are requiring the stylesheet inentry.js
to the following:
require('style!css!./style.css');
We are using two loaders, and they work like a pipeline. First, it applies the css-loader (loaders working right-to-left), which transforms CSS into a string, and the style-loader applies the style to the page. Start your Webpack watch command back up, refresh the browser, and you should see the style in the browser. Change some of the style rules, and watch how Webpack rebuilds the bundle with every change. You can refresh the browser to see the changes.
- While it was convenient to add the loaders right to the require function call, it kind of deviates from what a normal require call. Let's add a Webpack configuration file to handle the loaders. This will also give us an opportunity to remove some of the switches that we are using from the command line. Create a
webpack.config.js
file in the root of your project, and add the following content to it:
module.exports = {
entry: './entry.js',
output: {
filename: 'bundle.js'
},
devtool: 'source-map'
};
Run webpack -w
, and everything should still work.
- Of course, we added the config file so that we don't have to put the loaders on every CSS file. Add a module loaders section to
webpack.config.js
:
module.exports = {
entry: './entry.js',
output: {
filename: 'bundle.js'
},
devtool: 'source-map',
module: {
loaders: [
{test: /\.css$/, loader: 'style!css!'}
]
}
};
The loaders section defines an array of loaders to apply when the filename matches the test regular expression. The rule we just added will apply to the css-loader and style-loader to every CSS file it includes. After you make this change, remove the loaders from the require() call, then restart webpack -w
to apply the changes to the Webpack configuration.
- One thing that has been rather annoying to do is refresh the browser after any change. Using the Webpack Dev Server, your changes can not only rebuilt by Webpack, but also pushed to the browser. This is an easy switch.
- Stop any Webpack watcher that is currently running.
- In the terminal, run
webpack-dev-server --inline --hot
. This will start a web server on port 8080. The--inline
flag tells Webpack to inline the hot reload client in the bundle, and the--hot
flag enabled hot reloading. - Close the browser window you've been using, and instead navigate to
http://localhost:8080
. You should see the same content that we've already been working with. - Make some changes to the the stylesheet, to the text in
entry.js
, or any other changes. As soon as you save any of these files, notice how Webpack Dev Server re-builds the bundle and pushes the changes to the browser. Very nice!