- Use Knockout to display model data in views
- Configure WorldWind layers and layer categories
- Enable and disable layers
- Configure WMS/WMTS layers
In a hurry? View the completed code: Lesson 3
Knockout provides a mechanism for displaying model data (e.g., WorldWindow.layers) in views (e.g., the Layers panel). You put observable model data into in simple view-model objects which are then bound to the HTML views.
When the view-model data changes, your UI automatically changes.
We will use Knockout hosted on a CDN. Add this line of code to the list of JavaScript scripts at the bottom of your web page:
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.4.2/knockout-min.js"></script>
In our web app we need to know when change occurs within a layer category so the UI can be updated. We will use a simple timestamp associated with each category to capture changes. The timestamps will be contained within a Knockout observable. When the timestamp is updated, Knockout will signal the change to any subscribers to the observable.
In our Globe
's constructor, add a categoryTimestamps property that maps
categories to observable timestamps.
// Holds a map of category and observable timestamp pairs
this.categoryTimestamps = new Map();
Add the following two methods to our Globe
that operate on the
categoryTimestamps
property.
/**
* Returns an observable containing the last update timestamp for the category.
* @param {String} category
* @returns {Observable}
*/
getCategoryTimestamp(category) {
if (!this.categoryTimestamps.has(category)) {
this.categoryTimestamps.set(category, ko.observable());
}
return this.categoryTimestamps.get(category);
}
/**
* Updates the timestamp for the given category.
* @param {String} category
*/
updateCategoryTimestamp(category) {
let timestamp = this.getCategoryTimestamp(category);
timestamp(new Date());
}
Also add the following toggleLayer
method that will be used by the view models
to toggle the enabled
property of a layer. Note that toggling a layer
will invoke updateCategoryTimestamp
. Also note how our method enforces
an application rule that only allows one 'base' layer to be enabled at a time.
/**
* Toggles the enabled state of the given layer and updates the layer
* catetory timestamp. Applies a rule to the 'base' layers the ensures
* only one base layer is enabled.
* @param {WorldWind.Layer} layer
*/
toggleLayer(layer) {
// Multiplicity Rule: only [0..1] "base" layers can be enabled at a time
if (layer.category === 'base') {
this.wwd.layers.forEach(function (item) {
if (item.category === 'base' && item !== layer) {
item.enabled = false;
}
});
}
// Toggle the selected layer's visibility
layer.enabled = !layer.enabled;
// Trigger a redraw so the globe shows the new layer state ASAP
this.wwd.redraw();
// Signal a change in the category
this.updateCategoryTimestamp(layer.category);
}
And finally, we need to update the category timestamps when we add a layer.
Copy/paste this code to the end of our Globe.addLayers
method:
// Signal that this layer category has changed
this.getCategoryTimestamp(layer.category);
Now we will create a view model for the layers. Our view model will have two
observable arrays
whose contents are displayed in the Layers panel: one for the base layers and the
other for overlays. Our view model will subscribe to the globe's
categoryTimestamp
observables. When a change occurs in a layer category, the
view model with update the corresponding observable array, which will automatically
update the view.
Add the following JavaScript code to app.js below the Globe
class.
/**
* View model for the layers panel.
* @param {Globe} globe - Our globe object
*/
function LayersViewModel(globe) {
var self = this;
self.baseLayers = ko.observableArray(globe.getLayers('base').reverse());
self.overlayLayers = ko.observableArray(globe.getLayers('overlay').reverse());
// Update the view model whenever the model changes
globe.getCategoryTimestamp('base').subscribe(newValue =>
self.loadLayers(globe.getLayers('base'), self.baseLayers));
globe.getCategoryTimestamp('overlay').subscribe(newValue =>
self.loadLayers(globe.getLayers('overlay'), self.overlayLayers));
// Utility to load the layers in reverse order to show last rendered on top
self.loadLayers = function(layers, observableArray) {
observableArray.removeAll();
layers.reverse().forEach(layer => observableArray.push(layer));
};
// Click event handler for the layer panel's buttons
self.toggleLayer = function(layer) {
globe.toggleLayer(layer);
};
}
We will also create a view model for the settings. Layers within the setting category will be displayed and managed by the Settings panel. The same principles described for the layers view model apply here as well.
Add the following JavaScript code to app.js below the LayersViewModel
.
/**
* View model for the settings.
* @param {Globe} globe - Our globe object (not a WorldWind.Globe)
*/
function SettingsViewModel(globe) {
var self = this;
self.settingLayers = ko.observableArray(globe.getLayers('setting').reverse());
// Update the view model whenever the model changes
globe.getCategoryTimestamp('setting').subscribe(newValue =>
self.loadLayers(globe.getLayers('setting'), self.settingLayers));
// Utility to load layers in reverse order
self.loadLayers = function(layers, observableArray) {
observableArray.removeAll();
layers.reverse().forEach(layer => observableArray.push(layer));
};
// Click event handler for the setting panel's buttons
self.toggleLayer = function(layer) {
globe.toggleLayer(layer);
};
}
Now we will create buttons for all the layers in an observable array via a Knockout view template. The template
Add the following script to the web page. Place it close to the elements that will
use it, like between the .worldwind-overlay
<div/>
and the search #preview
<div/>
.
<!--Layer List Template-->
<script type="text/html" id="layer-list-template">
<button type="button" class="list-group-item list-group-item-action"
data-bind="click: $root.toggleLayer, css: { active: $data.enabled }">
<span data-bind="text: $data.displayName"></span>
</button>
</script>
Now we will reference the "layer-list-template" in the Layers and Settings panels.
In the Layers panel, replace the .card-body
<div/>
contents with this HTML:
<div class="list-group" data-bind="template: { name: 'layer-list-template', foreach: overlayLayers}"></div>
<hr/>
<div class="list-group" data-bind="template: { name: 'layer-list-template', foreach: baseLayers}"></div>
In the Settings panel, replace the .card-body
<div/>
contents with this HTML:
<div class="list-group" data-bind="template: { name: 'layer-list-template', foreach: settingLayers}"></div>
Add the following JavaScript code to app.js below the globe and layer initialization code:
// Activate the Knockout bindings between our view models and the html
let layersViewModel = new LayersViewModel(globe);
let settingsViewModel = new SettingsViewModel(globe);
ko.applyBindings(layersViewModel, document.getElementById('layers'));
ko.applyBindings(settingsViewModel, document.getElementById('settings'));
The web app is more fun to play with when there are more layers to use. We well add some more layers to the web app and we'll explore a few of the WorldWind.Layer properties that we can set via the options object
- enabled: controls the initial visibility of the layer.
- opacity: controls the opacity of a layer. 0 is transparent and 1 is opaque.
- detailControl: controls when a higher level-of-detail is requested for the zoom level.
- time: on the
AtmosphereLayer
it enables the day/night effect.
In app.js replace the globe and layer initialization code with this block:
// Create a globe
let globe = new Globe("globe-canvas");
// Add layers to the globe
// Add layers ordered by drawing order: first to last
globe.addLayer(new WorldWind.BMNGLayer(), {
category: "base"
});
globe.addLayer(new WorldWind.BMNGLandsatLayer(), {
category: "base",
enabled: false
});
globe.addLayer(new WorldWind.BingAerialLayer(), {
category: "base",
enabled: false
});
globe.addLayer(new WorldWind.BingAerialWithLabelsLayer(), {
category: "base",
enabled: false,
detailControl: 1.5
});
globe.addLayer(new WorldWind.BingRoadsLayer(), {
category: "overlay",
enabled: false,
detailControl: 1.5,
opacity: 0.75
});
globe.addLayer(new WorldWind.CoordinatesDisplayLayer(globe.wwd), {
category: "setting"
});
globe.addLayer(new WorldWind.ViewControlsLayer(globe.wwd), {
category: "setting"
});
globe.addLayer(new WorldWind.CompassLayer(), {
category: "setting",
enabled: false
});
globe.addLayer(new WorldWind.StarFieldLayer(), {
category: "setting",
enabled: false
displayName: "Stars"
});
globe.addLayer(new WorldWind.AtmosphereLayer(), {
category: "setting",
enabled: false,
time: null // new Date()
});
At this stage you have a web app with a functioning globe, navigation and now layer management.
Here's what we accomplished:
- Enabled change notification for the layer categories via observable timestamps.
- Created view models with observable arrays for the layers and settings data.
- Created a Knockout view template used in the Layers and Settings panels.
- Bound the view models to the views.
- Examined more WorldWind layers and their properties.
Here's the complete code for lesson 3: a web app with layer management.
<iframe width="100%" height="700" src="//jsfiddle.net/emxsys/sggs24bL/embedded/" allowpaymentrequest allowfullscreen="allowfullscreen" frameborder="0"></iframe>