2016-04-13 13:20:51 -04:00
include ../_util-fns
2017-01-22 06:51:02 -05:00
// The docs standard h4 style uppercases, making code terms unreadable. Override it.
style.
h4 {font-size: 17px !important; text-transform: none !important;}
.syntax { font-family: Consolas, 'Lucida Sans', Courier, sans-serif; color: black; font-size: 85%; }
2016-04-13 13:20:51 -04:00
:marked
[**Webpack**](https://webpack.github.io/) is a popular module bundler,
2017-03-23 18:30:12 -04:00
a tool for bundling application source code in convenient _chunks_
2016-04-13 13:20:51 -04:00
and for loading that code from a server into a browser.
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
It's an excellent alternative to the *SystemJS* approach used elsewhere in the documentation.
This guide offers a taste of Webpack and explains how to use it with Angular applications.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
<a id="top"></a>
2017-03-23 18:30:12 -04:00
# Contents
* [What is Webpack?](#what-is-webpack)
2017-03-31 18:25:27 -04:00
2016-04-13 13:20:51 -04:00
* [Entries and outputs](#entries-outputs)
2017-03-23 18:30:12 -04:00
* [Multiple bundles](#multiple-bundles)
2016-04-13 13:20:51 -04:00
* [Loaders](#loaders)
* [Plugins](#plugins)
2017-03-31 18:25:27 -04:00
2017-03-23 18:30:12 -04:00
* [Configuring Webpack](#configure-webpack)
2017-03-31 18:25:27 -04:00
2017-03-23 18:30:12 -04:00
* [Polyfills](#polyfills)
2016-04-13 13:20:51 -04:00
* [Common configuration](#common-configuration)
2017-03-23 18:30:12 -04:00
* [Inside `webpack.common.js`](#inside-webpack-commonjs)
2017-03-31 18:25:27 -04:00
* [entry](#common-entries)
* [resolve extension-less imports](#common-resolves)
* [`module.rules`](#common-rules)
* [Plugins](#plugins)
* [`CommonsChunkPlugin`](#commons-chunk-plugin)
* [`HtmlWebpackPlugin`](#html-webpack-plugin)
2017-03-23 18:30:12 -04:00
* [Environment specific configuration](#environment-configuration)
2016-04-13 13:20:51 -04:00
* [Development configuration](#development-configuration)
* [Production configuration](#production-configuration)
* [Test configuration](#test-configuration)
2017-03-31 18:25:27 -04:00
2017-03-23 18:30:12 -04:00
* [Trying it out](#try)
* [Highlights](#highlights)
* [Conclusion](#conclusion)
2017-02-28 17:31:24 -05:00
You can also <a href="/resources/zips/webpack/webpack.zip">download the final result.</a>
2016-04-13 13:20:51 -04:00
.l-main-section
<a id="what-is-webpack"></a>
:marked
## What is Webpack?
2017-03-23 18:30:12 -04:00
Webpack is a powerful module bundler.
A _bundle_ is a JavaScript file that incorporates _assets_ that *belong* together and
2016-04-13 13:20:51 -04:00
should be served to the client in a response to a single file request.
A bundle can include JavaScript, CSS styles, HTML, and almost any other kind of file.
2017-03-23 18:30:12 -04:00
Webpack roams over your application source code,
looking for `import` statements, building a dependency graph, and emitting one or more _bundles_.
With plugins and rules, Webpack can preprocess and minify different non-JavaScript files such as TypeScript, SASS, and LESS files.
2016-04-13 13:20:51 -04:00
2017-01-22 06:51:02 -05:00
You determine what Webpack does and how it does it with a JavaScript configuration file, `webpack.config.js`.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
a(id="entries-outputs")
.l-main-section
:marked
### Entries and outputs
2017-03-23 18:30:12 -04:00
You supply Webpack with one or more *entry* files and let it find and incorporate the dependencies that radiate from those entries.
2017-03-21 00:37:49 -04:00
The one entry point file in this example is the application's root file, `src/main.ts`:
2017-03-23 18:30:12 -04:00
2017-03-21 00:37:49 -04:00
+makeExample('webpack/ts/config/webpack.common.js', 'one-entry', 'webpack.config.js (single entry)')(format=".")
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
:marked
Webpack inspects that file and traverses its `import` dependencies recursively.
2017-03-23 18:30:12 -04:00
2017-03-21 00:37:49 -04:00
+makeExample('webpack/ts/src/app/app.component.ts', 'component', 'src/main.ts')(format=".")
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
It sees that you're importing `@angular/core` so it adds that to its dependency list for potential inclusion in the bundle.
It opens the `@angular/core` file and follows _its_ network of `import` statements until it has built the complete dependency graph from `main.ts` down.
2016-04-13 13:20:51 -04:00
Then it **outputs** these files to the `app.js` _bundle file_ designated in configuration:
2017-03-23 18:30:12 -04:00
2017-03-21 00:37:49 -04:00
.code-example
code-example(name="webpack.config.js (single output)" language="javascript").
output: {
filename: 'app.js'
}
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
This `app.js` output bundle is a single JavaScript file that contains the application source and its dependencies.
You'll load it later with a `<script>` tag in the `index.html`.
a#multiple-bundles
:marked
2016-04-13 13:20:51 -04:00
#### Multiple bundles
2017-01-22 06:51:02 -05:00
You probably don't want one giant bundle of everything.
It's preferable to separate the volatile application app code from comparatively stable vendor code modules.
2017-03-23 18:30:12 -04:00
2017-03-21 00:37:49 -04:00
Change the configuration so that it has two entry points, `main.ts` and `vendor.ts`:
.code-example
code-example(language="javascript").
entry: {
app: 'src/app.ts',
vendor: 'src/vendor.ts'
},
output: {
filename: '[name].js'
}
2016-04-13 13:20:51 -04:00
:marked
Webpack constructs two separate dependency graphs
2017-03-23 18:30:12 -04:00
and emits *two* bundle files, one called `app.js` containing only the application code and
2016-04-13 13:20:51 -04:00
another called `vendor.js` with all the vendor dependencies.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
.l-sub-section
:marked
2017-01-22 06:51:02 -05:00
The `[name]` in the output name is a *placeholder* that a Webpack plugin replaces with the entry names,
`app` and `vendor`. Plugins are [covered later](#commons-chunk-plugin) in the guide.
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
To tell Webpack what belongs in the vendor bundle,
2017-01-22 06:51:02 -05:00
add a `vendor.ts` file that only imports the application's third-party modules:
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/src/vendor.ts', null,'src/vendor.ts')(format=".")
a(id="loaders")
.l-main-section
:marked
### Loaders
2017-03-23 18:30:12 -04:00
Webpack can bundle any kind of file: JavaScript, TypeScript, CSS, SASS, LESS, images, HTML, fonts, whatever.
2017-01-22 06:51:02 -05:00
Webpack _itself_ only understands JavaScript files.
2017-03-23 18:30:12 -04:00
Teach it to transform non-JavaScript file into their JavaScript equivalents with *loaders*.
2017-01-22 06:51:02 -05:00
Configure loaders for TypeScript and CSS as follows.
2017-03-21 00:37:49 -04:00
.code-example
code-example(language="javascript").
rules: [
{
test: /\.ts$/,
loader: 'awesome-typescript-loader'
},
{
test: /\.css$/,
loaders: 'style-loader!css-loader'
}
]
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
When Webpack encounters `import` statements like the following,
it applies the `test` RegEx patterns.
2017-03-21 00:37:49 -04:00
.code-example
code-example(language="typescript").
import { AppComponent } from './app.component.ts';
2017-03-23 18:30:12 -04:00
2017-03-21 00:37:49 -04:00
import 'uiframework/dist/uiframework.css';
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
When a pattern matches the filename, Webpack processes the file with the associated loader.
2016-10-20 20:01:16 -04:00
The first `import` file matches the `.ts` pattern so Webpack processes it with the `awesome-typescript-loader`.
2017-03-23 18:30:12 -04:00
The imported file doesn't match the second pattern so its loader is ignored.
The second `import` matches the second `.css` pattern for which you have *two* loaders chained by the (!) character.
Webpack applies chained loaders *right to left*. So it applies
the `css` loader first to flatten CSS `@import` and `url(...)` statements.
Then it applies the `style` loader to append the css inside `<style>` elements on the page.
2016-04-13 13:20:51 -04:00
a(id="plugins")
.l-main-section
:marked
### Plugins
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
Webpack has a build pipeline with well-defined phases.
2017-01-22 06:51:02 -05:00
Tap into that pipeline with plugins such as the `uglify` minification plugin:
2017-03-23 18:30:12 -04:00
2017-03-21 00:37:49 -04:00
.code-example
code-example(language="javascript").
plugins: [
new webpack.optimize.UglifyJsPlugin()
]
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
a(id="configure-webpack")
.l-main-section
:marked
2017-03-23 18:30:12 -04:00
## Configuring Webpack
After that brief orientation, you are ready to build your own Webpack configuration for Angular apps.
2016-04-13 13:20:51 -04:00
Begin by setting up the development environment.
2017-03-23 18:30:12 -04:00
Create a new project folder.
2017-01-22 06:51:02 -05:00
code-example(language="sh" class="code-shell").
2016-09-19 23:24:40 -04:00
mkdir angular-webpack
cd angular-webpack
2016-04-13 13:20:51 -04:00
:marked
2017-02-23 07:26:27 -05:00
Add these files:
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeTabs(
`webpack/ts/package.webpack.json,
2017-02-23 07:26:27 -05:00
webpack/ts/src/tsconfig.1.json,
2016-04-13 13:20:51 -04:00
webpack/ts/webpack.config.js,
webpack/ts/karma.webpack.conf.js,
webpack/ts/config/helpers.js`,
null,
`package.json,
2017-02-23 07:26:27 -05:00
src/tsconfig.json,
2016-04-13 13:20:51 -04:00
webpack.config.js,
karma.conf.js,
config/helpers.js`
)
.l-sub-section
:marked
2017-01-22 06:51:02 -05:00
Many of these files should be familiar from other Angular documentation guides,
2017-03-23 18:30:12 -04:00
especially the [Typescript configuration](../guide/typescript-configuration.html) and
[npm packages](../guide/npm-packages.html) guides.
2017-01-22 06:51:02 -05:00
2017-03-23 18:30:12 -04:00
Webpack, the plugins, and the loaders are also installed as packages.
2017-01-22 06:51:02 -05:00
They are listed in the updated `packages.json`.
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
Open a terminal window and install the npm packages.
2017-01-22 06:51:02 -05:00
code-example(language="sh" class="code-shell").
npm install
a#polyfills
.l-main-section
:marked
### Polyfills
You'll need polyfills to run an Angular application in most browsers as explained
2017-03-23 18:30:12 -04:00
in the [Browser Support](browser-support.html) guide.
2017-01-22 06:51:02 -05:00
Polyfills should be bundled separately from the application and vendor bundles.
Add a `polyfills.ts` like this one to the `src/` folder.
+makeExample('webpack/ts/src/polyfills.ts', '', 'src/polyfills.ts')(format=".")
.callout.is-critical
header Loading polyfills
:marked
Load `zone.js` early within `polyfills.ts`, immediately after the other ES6 and metadata shims.
:marked
2017-03-23 18:30:12 -04:00
Because this bundle file will load first, `polyfills.ts` is also a good place to configure the browser environment
2017-01-22 06:51:02 -05:00
for production or development.
2016-04-13 13:20:51 -04:00
a(id="common-configuration")
.l-main-section
:marked
2017-03-23 18:30:12 -04:00
### Common configuration
2017-01-22 06:51:02 -05:00
Developers typically have separate configurations for development, production, and test environments.
All three have a lot of configuration in common.
Gather the common configuration in a file called `webpack.common.js`.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.common.js', null, 'config/webpack.common.js')(format=".")
2017-03-23 18:30:12 -04:00
a#inside-webpack-commonjs
2016-04-13 13:20:51 -04:00
:marked
2017-01-22 06:51:02 -05:00
### Inside _webpack.common.js_
2017-03-23 18:30:12 -04:00
Webpack is a NodeJS-based tool that reads configuration from a JavaScript commonjs module file.
2017-01-22 06:51:02 -05:00
The configuration imports dependencies with `require` statements
and exports several objects as properties of a `module.exports` object.
2017-03-23 18:30:12 -04:00
* [`entry`](#common-entries)—the entry-point files that define the bundles.
* [`resolve`](#common-resolve)—how to resolve file names when they lack extensions.
* [`module.rules`](#common-rules)— `module` is an object with `rules` for deciding how files are loaded.
* [`plugins`](#common-plugins)—creates instances of the plugins.
2017-01-22 06:51:02 -05:00
a#common-entries
:marked
2017-03-23 18:30:12 -04:00
#### _entry_
The first export is the `entry` object:
2017-01-22 06:51:02 -05:00
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.common.js', 'entries', 'config/webpack.common.js')(format=".")
:marked
2017-03-23 18:30:12 -04:00
This `entry` object defines the three bundles:
* `polyfills`—the polyfills needed to run Angular applications in most modern browsers.
* `vendor`—the third-party dependencies such as Angular, lodash, and bootstrap.css.
* `app`—the application code.
2017-01-22 06:51:02 -05:00
a#common-resolve
2016-04-13 13:20:51 -04:00
:marked
2017-01-22 06:51:02 -05:00
#### _resolve_ extension-less imports
2017-03-23 18:30:12 -04:00
The app will `import` dozens if not hundreds of JavaScript and TypeScript files.
2017-01-22 06:51:02 -05:00
You could write `import` statements with explicit extensions like this example:
2017-03-21 00:37:49 -04:00
.code-example
code-example(language="typescript").
import { AppComponent } from './app.component.ts';
2016-04-13 13:20:51 -04:00
:marked
2017-01-22 06:51:02 -05:00
But most `import` statements don't mention the extension at all.
Tell Webpack to resolve extension-less file requests by looking for matching files with
`.ts` extension or `.js` extension (for regular JavaScript files and pre-compiled TypeScript files).
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.common.js', 'resolve', 'config/webpack.common.js')(format=".")
.l-sub-section
:marked
2017-01-22 06:51:02 -05:00
If Webpack should resolve extension-less files for styles and HTML,
add `.css` and `.html` to the list.
a#common-rules
2016-04-13 13:20:51 -04:00
:marked
2017-01-22 06:51:02 -05:00
:marked
#### _module.rules_
2017-03-23 18:30:12 -04:00
Rules tell Webpack which loaders to use for each file, or module:
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.common.js', 'loaders', 'config/webpack.common.js')(format=".")
:marked
2017-03-23 18:30:12 -04:00
* `awesome-typescript-loader`—a loader to transpile the Typescript code to ES5, guided by the `tsconfig.json` file.
* `angular2-template-loader`—loads angular components' template and styles.
* `html-loader`—for component templates.
* images/fonts—Images and fonts are bundled as well.
* CSS—the first pattern matches application-wide styles; the second handles
component-scoped styles (the ones specified in a component's `styleUrls` metadata property).
2016-04-13 13:20:51 -04:00
.l-sub-section
:marked
2017-02-23 07:26:27 -05:00
The first pattern is for the application-wide styles. It excludes `.css` files within the `src/app` directory
where the component-scoped styles sit. The `ExtractTextPlugin` (described below) applies the `style` and `css`
loaders to these files.
2017-03-23 18:30:12 -04:00
The second pattern filters for component-scoped styles and loads them as strings via the `raw-loader`,
2016-04-13 13:20:51 -04:00
which is what Angular expects to do with styles specified in a `styleUrls` metadata property.
2016-07-04 18:46:57 -04:00
.l-sub-section
:marked
2017-01-22 06:51:02 -05:00
Multiple loaders can be chained using the array notation.
2016-07-04 18:46:57 -04:00
2017-01-22 06:51:02 -05:00
a#common-plugins
:marked
2016-04-13 13:20:51 -04:00
:marked
2017-01-22 06:51:02 -05:00
#### _plugins_
Finally, create instances of three plugins:
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.common.js', 'plugins', 'config/webpack.common.js')(format=".")
a(id="commons-chunk-plugin")
:marked
#### *CommonsChunkPlugin*
2017-03-23 18:30:12 -04:00
The `app.js` bundle should contain only application code. All vendor code belongs in the `vendor.js` bundle.
Of course the application code imports vendor code.
On its own, Webpack is not smart enough to keep the vendor code out of the `app.js` bundle.
The `CommonsChunkPlugin` does that job.
2016-04-13 13:20:51 -04:00
.l-sub-section
:marked
2017-03-23 18:30:12 -04:00
The `CommonsChunkPlugin` identifies the hierarchy among three _chunks_: `app` -> `vendor` -> `polyfills`.
2016-04-13 13:20:51 -04:00
Where Webpack finds that `app` has shared dependencies with `vendor`, it removes them from `app`.
2017-03-23 18:30:12 -04:00
It would remove `polyfills` from `vendor` if they shared dependencies, which they don't.
2016-04-13 13:20:51 -04:00
a(id="html-webpack-plugin")
:marked
2017-03-23 18:30:12 -04:00
#### _HtmlWebpackPlugin_
Webpack generates a number of js and CSS files.
2017-01-22 06:51:02 -05:00
You _could_ insert them into the `index.html` _manually_. That would be tedious and error-prone.
Webpack can inject those scripts and links for you with the `HtmlWebpackPlugin`.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
a(id="environment-configuration")
.l-main-section
:marked
### Environment-specific configuration
2017-03-23 18:30:12 -04:00
The `webpack.common.js` configuration file does most of the heavy lifting.
2017-01-22 06:51:02 -05:00
Create separate, environment-specific configuration files that build on `webpack.common`
by merging into it the peculiarities particular to the target environments.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
These files tend to be short and simple.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
a(id="development-configuration")
.l-main-section
:marked
2017-03-23 18:30:12 -04:00
### Development configuration
Here is the `webpack.dev.js` development configuration file.
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.dev.js', null, 'config/webpack.dev.js')(format=".")
:marked
2017-01-22 06:51:02 -05:00
The development build relies on the Webpack development server, configured near the bottom of the file.
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
Although you tell Webpack to put output bundles in the `dist` folder,
2016-04-13 13:20:51 -04:00
the dev server keeps all bundles in memory; it doesn't write them to disk.
2017-03-23 18:30:12 -04:00
You won't find any files in the `dist` folder, at least not any generated from *this development build*.
The `HtmlWebpackPlugin`, added in `webpack.common.js`, uses the `publicPath` and the `filename` settings to generate
appropriate `<script>` and `<link>` tags into the `index.html`.
2017-01-22 06:51:02 -05:00
The CSS styles are buried inside the Javascript bundles by default. The `ExtractTextPlugin` extracts them into
2017-03-23 18:30:12 -04:00
external `.css` files that the `HtmlWebpackPlugin` inscribes as `<link>` tags into the `index.html`.
Refer to the [Webpack documentation](https://webpack.github.io/docs/) for details on these and
other configuration options in this file.
2016-04-13 13:20:51 -04:00
Grab the app code at the end of this guide and try:
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
code-example(language="sh" class="code-shell").
2016-04-13 13:20:51 -04:00
npm start
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
a(id="production-configuration")
.l-main-section
:marked
2017-03-23 18:30:12 -04:00
### Production configuration
Configuration of a *production* build resembles *development* configuration with a few key changes.
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.prod.js', null, 'config/webpack.prod.js')(format=".")
:marked
2017-01-22 06:51:02 -05:00
You'll deploy the application and its dependencies to a real production server.
You won't deploy the artifacts needed only in development.
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
Put the production output bundle files in the `dist` folder.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
Webpack generates file names with cache-busting hash.
2017-03-23 18:30:12 -04:00
Thanks to the `HtmlWebpackPlugin`, you don't have to update the `index.html` file when the hash changes.
2016-04-13 13:20:51 -04:00
There are additional plugins:
2017-03-23 18:30:12 -04:00
* *`NoEmitOnErrorsPlugin`—stops the build if there is an error.
* *`UglifyJsPlugin`—minifies the bundles.
* *`ExtractTextPlugin`—extracts embedded css as external files, adding cache-busting hash to the filename.
* *`DefinePlugin`—use to define environment variables that you can reference within the application.
* *`LoaderOptionsPlugins`—to override options of certain loaders.
Thanks to the `DefinePlugin` and the `ENV` variable defined at top, you can enable Angular production mode like this:
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/src/main.ts', 'enable-prod')(format=".")
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
:marked
Grab the app code at the end of this guide and try:
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
code-example(language="sh" class="code-shell").
2016-04-13 13:20:51 -04:00
npm run build
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
a(id="test-configuration")
.l-main-section
:marked
2017-03-23 18:30:12 -04:00
### Test configuration
You don't need much configuration to run unit tests.
2017-01-22 06:51:02 -05:00
You don't need the loaders and plugins that you declared for your development and production builds.
You probably don't need to load and process the application-wide styles files for unit tests and doing so would slow you down;
you'll use the `null` loader for those CSS files.
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
You could merge the test configuration into the `webpack.common` configuration and override the parts you don't want or need.
2016-04-13 13:20:51 -04:00
But it might be simpler to start over with a completely fresh configuration.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/webpack.test.js', null, 'config/webpack.test.js')(format=".")
:marked
2017-03-23 18:30:12 -04:00
Reconfigure [Karma](https://karma-runner.github.io/1.0/index.html) to use Webpack to run the tests:
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/karma.conf.js', null, 'config/karma.conf.js')(format=".")
:marked
2017-01-22 06:51:02 -05:00
You don't precompile the TypeScript; Webpack transpiles the Typescript files on the fly, in memory, and feeds the emitted JS directly to Karma.
2016-04-13 13:20:51 -04:00
There are no temporary files on disk.
2017-03-23 18:30:12 -04:00
The `karma-test-shim` tells Karma what files to pre-load and
2016-04-13 13:20:51 -04:00
primes the Angular test framework with test versions of the providers that every app expects to be pre-loaded.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeExample('webpack/ts/config/karma-test-shim.js', null, 'config/karma-test-shim.js')(format=".")
:marked
2017-01-22 06:51:02 -05:00
Notice that you do _not_ load the application code explicitly.
You tell Webpack to find and load the test files (the files ending in `.spec.ts`).
2017-03-23 18:30:12 -04:00
Each spec file imports all—and only—the application source code that it tests.
2017-01-22 06:51:02 -05:00
Webpack loads just _those_ specific application files and ignores the other files that you aren't testing.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
:marked
Grab the app code at the end of this guide and try:
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
code-example(language="sh" class="code-shell").
2016-04-13 13:20:51 -04:00
npm test
<a id="try"></a>
:marked
## Trying it out
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
Here is the source code for a small application that bundles with the
Webpack techniques covered in this guide.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeTabs(
`webpack/ts/src/index.html,
webpack/ts/src/main.ts,
2017-02-23 07:26:27 -05:00
webpack/ts/src/assets/css/styles.css`,
2016-04-13 13:20:51 -04:00
null,
`src/index.html,
src/main.ts,
2017-02-23 07:26:27 -05:00
src/assets/css/styles.css`
2016-04-13 13:20:51 -04:00
)
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
+makeTabs(
`webpack/ts/src/app/app.component.ts,
webpack/ts/src/app/app.component.html,
webpack/ts/src/app/app.component.css,
2016-08-09 12:38:25 -04:00
webpack/ts/src/app/app.component.spec.ts,
webpack/ts/src/app/app.module.ts`,
2016-04-13 13:20:51 -04:00
null,
`src/app/app.component.ts,
src/app/app.component.html,
src/app/app.component.css,
2016-08-09 12:38:25 -04:00
src/app/app.component.spec.ts,
src/app/app.module.ts`
2016-04-13 13:20:51 -04:00
)
2016-05-15 07:22:29 -04:00
2017-02-23 07:26:27 -05:00
:marked
2016-05-15 07:22:29 -04:00
The <code>app.component.html</code> displays this downloadable Angular logo
2017-03-31 05:59:16 -04:00
<a href="https://raw.githubusercontent.com/angular/angular.io/master/public/resources/images/logos/angular/angular.png" target="_blank">
<img src="/resources/images/logos/angular/angular.png" height="40px" title="download Angular logo"></a>.
2017-03-23 18:30:12 -04:00
Create a folder called `images` under the project's `assets` folder, then right-click (Cmd+click on Mac)
on the image and download it to that folder.
2016-05-15 07:22:29 -04:00
2017-01-22 06:51:02 -05:00
a#bundle-ts
:marked
Here again are the TypeScript entry-point files that define the `polyfills` and `vendor` bundles.
2016-04-13 13:20:51 -04:00
+makeTabs(
2017-01-22 06:51:02 -05:00
`webpack/ts/src/polyfills.ts,
webpack/ts/src/vendor.ts`,
2016-04-13 13:20:51 -04:00
null,
2017-01-22 06:51:02 -05:00
`src/polyfills.ts,
src/vendor.ts`
2016-04-13 13:20:51 -04:00
)
2016-08-21 08:03:58 -04:00
<a id="highlights"></a>
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
### Highlights
* There are no `<script>` or `<link>` tags in the `index.html`.
2016-04-13 13:20:51 -04:00
The `HtmlWebpackPlugin` inserts them dynamically at runtime.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
* The `AppComponent` in `app.component.ts` imports the application-wide css with a simple `import` statement.
2016-07-04 18:46:57 -04:00
2017-03-23 18:30:12 -04:00
* The `AppComponent` itself has its own html template and css file. WebPack loads them with calls to `require()`.
2016-07-04 18:46:57 -04:00
Webpack stashes those component-scoped files in the `app.js` bundle too.
2017-03-23 18:30:12 -04:00
You don't see those calls in the source code;
they're added behind the scenes by the `angular2-template-loader` plug-in.
2016-04-13 13:20:51 -04:00
* The `vendor.ts` consists of vendor dependency `import` statements that drive the `vendor.js` bundle.
The application imports these modules too; they'd be duplicated in the `app.js` bundle
if the `CommonsChunkPlugin` hadn't detected the overlap and removed them from `app.js`.
2017-03-23 18:30:12 -04:00
<a id="conclusion"></a>
2016-04-13 13:20:51 -04:00
:marked
2017-03-23 18:30:12 -04:00
## Conclusion
You've learned just enough Webpack to configurate development, test and production builds
2016-04-13 13:20:51 -04:00
for a small Angular application.
2017-03-23 18:30:12 -04:00
2017-01-22 06:51:02 -05:00
_You could always do more_. Search the web for expert advice and expand your Webpack knowledge.
2017-03-23 18:30:12 -04:00
2016-04-13 13:20:51 -04:00
[Back to top](#top)