This tutorial will show how you can use the webpack module bundler with Cesium. Some basic knowlege of webpack will be assumed but code samples will be provided for reference. These code samples are located here. There are two ways to use Cesium with webpack: the first is to use the pre-built bundle that is provided with every release; the other way is to require
individual Cesium files and let webpack make sure everything is properly included. This tutorial will cover both paths to using Cesium in your application.
Cesium is a complex library. It uses webworkers for performance (in separate files) and has images used for basic layers and calculations stored as json (also in separate files). In short, it has a lot of extra assets that don't fit the typical mould of a javascript library. So to integrate with module bundlers like webpack, require.js, and browserify, you can't just point to a single file, say require('cesium')
, and be done. But with some work, you can integrate Cesium with module bundlers.
Webpack is a very powerful bundler. It has the ability to parse both AMD and Common.js modules together without crazy shimming. It traces dependencies of your code, not just across javascript files you define as dependencies, but across file types. For example, if you say that you need a css file for a particular module, and if that css file references a font file that is served locally, webpack can make sure that the font file is copied over and placed where your css and browser will know where to find it.
This feature is great for your application, but it can get a little tricky when webpack tries to trace these dependencies for a complex library like Cesium that wasn't built with webpack.
Luckily webpack has a huge list of plugins and is very modular. There are two basic ways of getting it to work with Cesium: using the pre-compiled version, and using the source files directly.
If you are a webpack expert, or already have webpack up and running, just head on down to the bottom of this tutorial where I describe the steps without getting webpack set up.
This method essentially tells webpack to pull in the Cesium.js file as you would at the top of your <html>
tag.
Install node, webpack, and webpack-dev-server.
npm install -g webpack webpack-dev-server
You'll need to install all the dependencies located in the package.json (https://github.com/Aviture/cesium-webpack/blob/master/using-pre-built/package.json) for the source code example. npm install
-
Copy the built version of Cesium to your project; in this example it's in the 'lib' directory.
-
Next, copy over all the files from
lib/Cesium/*
to./public
, creating thepublic
directory if you haven't already done so. Note:public
should NOT be checked into your source control. -
Create an app.js file in your project, and make it look like this:
window.CESIUM_BASE_URL = './'; require('./lib/CesiumUnminified/Cesium.js'); require('./lib/Cesium/Widgets/widgets.css'); var Cesium = window.Cesium; var viewer = new Cesium.Viewer('cesiumContainer');
This is your web application's "entry" point, defined later in our webpack config. In it, we do 3 things. First, we set the
CESIUM_BASE_URL
property to be./
. This is the URL that Cesium uses for AJAX requests to the server. Since everything is served out ofpublic
in our example, we're telling Cesium to look there. Cesium will make requests to ourpublic/Assets/Textures/*
directory (as an example). Since we movedAssets
in step 2 above, we're making sure everything lines up.Next we
require
Cesium.js. I'm doing the unminified version but you can switch yours to do the minified version. This is where we're pulling in the main Cesium.js into our project.Next, we
require
in our widgets.css file. Huh? Yeah, webpack will let you 'require' css files, and it will take care of making sure all the css is copied over for us. It's a little strange, and an optional step (you can put the css link in your own index.html), but you get used to it.Finally, after Cesium is
require
d, it's sitting on the global namespace as 'Cesium'. So we grab it and instantiate a viewer on a div with idcesiumContainer
.So far so good, but how do we tell webpack to do its thing? That's in the webpack.config.js file coming up next.
-
Create a webpack.config.js file, and make it look like this:
var HtmlPlugin = require('html-webpack-plugin'); module.exports = { entry: "./app.js", output: { path: __dirname + '/public', filename: "bundle.js" }, plugins: [ new HtmlPlugin({ template: 'index.html', inject : true }) ], devServer: { contentBase: './public', }, module: { loaders: [ { test: /\.css$/, loader: "style!css" }, { test: /\.(png|gif|jpg|jpeg)$/, loader: 'file-loader' }, { test: /Cesium\.js$/, loader: 'script' } ] } };
This is the main config file for webpack, telling it how to load up any asset you
require
. First off, you'll see theentry
field, which specifies app.js from above as ourentry
. Next, is theoutput
config, which says to put everything in thepublic
directory and name the final bundle bundle.js. Next, we have theplugins
. We're using HtmlPlugin, which lets us have a little control over our html file (note you could skip this step too and just put an index.html file inpublic
). We'll cover index.html in a later step. Next up, devServer, which tells webpack-dev-server to serve everything up frompublic
.Finally, we get to the
loaders
section. Each entry here has a regextest
and a loader that matches it. The first one is our css loader, saying that if a file ends in.css
, use the css loader, which is what allows us torequire
css.Did you know the widgets.css file loads up other files, some of which have urls referencing images? Webpack is smart enough to trace all these down and make sure they're all available to you, but it has to know what to do when it finds a file that ends in .png. That's what the
file-loader
will do.Finally, we have a regex for Cesium.js, and we tell webpack to use the
script
loader. This is equivalent to having webpack put ascript
tag on your page and runs the script like that. -
Finally, create an index.html file, and make it look like:
<html> <head> <meta charset="utf-8"> </head> <body> <div id="cesiumContainer"></div> </body> </html>
Our app.js file is creating a new Viewer on a div with id
cesiumContainer
. So we need to make sure we have a div there. This HTML file lets us create our initial app structure right away. Webpack will insert the bundled app.js (with everything else) in our index.html for us (that's what thatinject: true
setting does). -
With all that setup, you should be able to run
webpack-dev-server
on your console and navigate to localhost:8080 to and see your Cesium viewer up and running.
If you are a webpack expert, you'll probably see a lot of steps in here that aren't strictly necessary, but this will get you started.
Using the source directly can be your best bet when you really only want to pull in the parts of Cesium that you're using, and leave out the stuff you're not. This can lead to faster loading times and keep your app more light-weight, always a good thing! Unfortunately things can get slightly more tricky with this route depending on what version of Cesium you're using, but don't worry, I'll identity what steps are necessary.
-
Getting Cesium ready for webpack. There are three options here that you may run into depending on the version of Cesium you're using:
pre 1.15: Before v1.15, Cesium included a set of package.json files in its
Source
directory. These really trip up webpack (and Browserify for that matter). So, delete all package.json files in Cesium'sSource
directory. You can do this on *nix systems with this command:find path/to/Cesium/Source -name "package.json" -type f -delete
. On Windows you should be able to run from the Source directory:del /s package.json
. Note this is untested and be careful as you type it. To be safe, usedel /s /p package.json
which should ask for confirmation for each file.Then you'll need to run the build: which you can do by running
ant build
.v1.15 : Run the Cesium build by running
gulp
.v1.16 and later: As noted here, Cesium is now available on npm! This makes the step as simple as
npm install cesium
. -
Next, copy over all the files from
Cesium/*
to./public
, creating thepublic
directory if you haven't already done so. Note:public
should NOT be checked into your source control. -
Create an app.js file and make it look like this:
require('cesium/Source/Widgets/widgets.css'); var BuildModuleUrl = require('cesium/Source/Core/ buildModuleUrl'); BuildModuleUrl.se tBaseUrl('./'); var Viewer = require('cesium/Source/Widgets/Viewer/ Viewer'); var viewer = new Viewer('cesiumContainer');
This is our
entry
point again. It's slightly different than before. We're still loading in thecss
file via arequire
call, but instead of doing awindow.CESIUM_BASE_URL
we're loading in theBuildModuleUrl
module and setting the base url there.Next we're
require
ing in the viewer from the source directory. This is how you'll use Cesium modules from now on.Finally we just use the viewer and instantiate it as before!
-
Next, let's look at our
webpack.config.js
file.var HtmlPlugin = require('html-webpack-plugin'); module.exports = { entry: "./app.js", output: { path: __dirname + '/public', filename: "bundle.js", sourcePrefix: '' }, plugins: [ new HtmlPlugin({ template: 'index.html', inject : true }) ], devServer: { contentBase: './public', }, module: { unknownContextCritical: false, loaders: [ { test: /\.css$/, loader: "style!css" }, { test: /\.(png|gif|jpg|jpeg)$/, loader: 'file-loader' } ] } };
A lot of this is the same as before but with some differences:
-
In the
output
object we havesourcePrefix: ''
. This is required because Cesium uses some multi-line strings in its code and webpack indents them improperly. -
There's also the
unknownContextCritical : false
which tells webpack to ignore some warnings due to the way Cesium dynamically builds module paths.
-
-
With all that setup, you should be able to run
webpack-dev-server
on your console and navigate to localhost:8080 to and see your Cesium viewer up and running.
You still have the two choices. Pre-built or using Source. In both cases you need to copy over the assets to your output directory.
-
Set up a script loader
{ test: /Cesium\.js$/, loader: 'script' }
and make sure you have a file loader set up as well. -
Set the
CESIUM_BASE_URL
property.window.CESIUM_BASE_URL = './';
before you load orrequire
Cesium../
is assuming you just copy the assets directly to your output directory. If you copy them tooutput/Cesium
then change the path accordingly.
-
You'll need to get Cesium ready for webpack. Check out step 1 from above. If you're using v1.16 or above, it's just
npm install cesium
, but it's slightly more complicated with earlier versions. -
Your webpack config will at a minimum need these options configured. Note, you'll need more, but these options are the minimum to get Cesium working from source:
{ output: { sourcePrefix: '' }, module: { unknownContextCritical: false, loaders: [ { test: /\.css$/, loader: "style!css" }, { test: /\.(png|gif|jpg|jpeg)$/, loader: 'file-loader' } ] } }
-
The
sourcePrefix
: '' is required because Cesium uses some multi-line strings in its code and webpack indents them improperly. -
The
unknownContextCritical: false
is not strictly required, but Cesium (and the version of knockout included) uses dynamic module loading with require.js and this confuses webpack. These warnings are safe to ignore and this flag ignores them. Be careful though as this ignores all of those warnings. -
The loaders are probably already included in your webpack config, but they are necessary to
require
the 'widgets.css' file and all the images they end up loading.
-
-
Finally, you'll want to set the
BuildModuleUrl
's base URL. Before you require Cesium, place these statements: var BuildModuleUrl = require('cesium/Source/Core/buildModuleUrl'); BuildModuleUrl.setBaseUrl('./');
You can see all of this in action here where there are two directories with sample projects set up. The using-pre-built
has a working example with using pre-built Cesium, and the using-source
directory has the same for using the source directly. For the using source example, I used v1.14 of Cesium, which is the most complicated. If you are using later versions, then getting Cesium ready is much easier.
Each example can be run by following these steps:
-
cd
to the directory and runnpm install
. Once it is done, this runs thepostinstall.sh
script, which takes care of some of the steps I mentioned above automatically. Note: best to run this in git bash in windows or translate yourself to windows commands. -
Run
npm start
, which will run any pre-processing tasks and then start up webpack-dev-server. -
Head on over to localhost:8080 to see Cesium running with webpack.