Goal
In this tutorial, we create a basic HTML page containing a LuciadRIA map. It has quality background imagery from a public OpenStreetMap server, shows a coordinate grid, and uses an icon to mark a specific location.
Setup
We use webpack and webpack-dev-server to develop and serve the app.
Project structure setup
-
We create a new project directory with an
index.html
file and use webpack to serve this file.We start by creating a directory, initializing npm, and installing webpack and the dev-server packages from the command line:
mkdir hello-world cd hello-world npm init -y npm install webpack webpack-cli webpack-dev-server raw-loader --save-dev
-
We set up the standard webpack configuration, with a
src
directory for JavaScript sources and adist
directory for the output:Creating the standard configuration files from the command linemkdir dist mkdir src # Already create some empty files, which we'll populate later # On Mac and Linux, you use the touch command to create files. # The windows command line alternative for touch is type # type nul > dist/index.html # Alternatively, you can install touch as an npm package using `npm install --global touch-cli`. touch dist/index.html #create an empty index.html file in the dist directory touch src/index.js #create an empty index.js file in the src directory touch webpack.config.js #create an empty webpack configuration file
-
We make the web server serve the
dist
folder. We add this content to thewebpack.config.js
file:webpack.config.js
file additions to serve thedist
folderconst path = require('path'); module.exports = { devServer: { static: path.join(__dirname, 'dist') } };
-
To test our setup, we add basic content to the
index.html
file in thedist
folderInitial contents of thedist/index.html
file<!doctype html> <html lang="en"> <head> <title>Hello world</title> </head> <body> <h1>Hello world</h1> </body> </html>
The resulting folder and file structure looks like this:
hello-world |- dist |- index.html |- node_modules |- package.json |- package-lock.json | - src |- index.js |- webpack.config.js
-
To start the development server, we add a
start:dev
script inpackage.json
:Add an extra script to thepackage.json
file"scripts": { "start:dev": "webpack serve --mode development" }
Now we can run:
npm run start:dev
To see the web page, go to http://localhost:8080/ in your browser.
Installing the LuciadRIA npm packages
Your LuciadRIA distribution comes with a number of NPM packages under the packages
folder.
-
The
@luciad/ria
package has the core modules of the LuciadRIA library -
The
@luciad/geometry
and@luciad/symbology
packages are optional packages.
In this tutorial, we assume that these packages are already published in an npm
registry. We’re going to install them using npm install
.
# Install the LuciadRIA core package from your own private registry
npm install --registry http://my-private-registry:8073/ @luciad/ria
# Optional: if you purchased additional components, install them as well
# We won't be using any code from those packages in the remainder of this tutorial
# npm install --registry http://my-private-registry:8073/ @luciad/ria-geometry
# npm install --registry http://my-private-registry:8073/ @luciad/ria-milsym
We can now use the installed @luciad/ria
package in our index.js
file.
An alternative to using the
Consult the npm documentation for more information. |
Adding a LuciadRIA map to our application
First, we add a <div>
to our index.html
page to mark where we want our map.
The HTML file update immediately triggers the server to load our JavaScript code.
<body>
element, containing a <div>
for the map and the <script>
tag for the JS in the dist/index.html
file
<body>
<h1>Hello world</h1>
<!-- Create a div where our map should appear -->
<div id="map"/>
<!-- Load the javaScript file.
main.js is the file that webpack will generate for the src/index.js file
-->
<script src="main.js"></script>
</body>
We also add some CSS to give this <div>
a certain size.
To keep things simple in this tutorial, we add the CSS directly to the header of our file:
dist/index.html
file
<style>
#map {
position: absolute;
top: 1px;
right: 1px;
bottom: 1px;
left: 1px;
border: 1px solid grey;
}
</style>
The complete dist/index.html
file
<!doctype html>
<html lang="en">
<head>
<title>Hello world</title>
<style>
#map {
position: absolute;
top: 1px;
right: 1px;
bottom: 1px;
left: 1px;
border: 1px solid grey;
}
</style>
</head>
<body>
<h1>Hello world</h1>
<div id="map"/>
<script src="main.js"></script>
</body>
</html>
Now it’s time to write our first lines of LuciadRIA code.
In the src/index.js
file, we create a new @luciad/ria/view/Map
instance, and tell it to appear in that <div>
:
import { WebGLMap } from "@luciad/ria/view/WebGLMap.js";
import { getReference } from "@luciad/ria/reference/ReferenceProvider";
//Create a new 2D map instance, and display it in the div with the "map" id
const webMercatorReference = getReference("EPSG:3857");
const map = new WebGLMap("map", {
reference: webMercatorReference,
});
That’s it. We created our first LuciadRIA map.
If you visit the web page at this point, you bump into two problems, though:
-
The browser shows a license error because we didn’t install the LuciadRIA license in our application yet.
-
The page doesn’t show anything because the map doesn’t contain any data yet. We start adding data in Add OpenStreetMap data as a base layer.
Installing and activating the license
To activate the license, we must pass the contents of the license file to the License
class:
-
Copy the
luciadria_development.txt
license file to thesrc
folder. -
Create a new
src/license-loader.js
file -
Add the following contents to it:
Thesrc/license-loader.js
file:import {setLicenseText} from "@luciad/ria/util/License.js"; import license from 'raw-loader!./luciadria_development.txt'; setLicenseText(license);
We’re using the webpack raw-loader to load the contents of the license file as a string.
-
Import the
LicenseLoader
in thesrc/index.js
file.import "./license-loader";
Make sure that this import is the first import. You must load the license before you trigger any other LuciadRIA code.
Add OpenStreetMap data as a base layer
Now that the license is installed, it’s time to add some data to the map so that we can see the map.
OpenStreetMap offers worldwide free maps updated and maintained by volunteer contributors. It serves its maps as imagery data from tile servers. From LuciadRIA, you can access those tile servers and show OpenStreetMap imagery as a background layer on your map.
Access the OpenStreetMap data as a URL tileset model
OpenStreetMap offers its data through tile servers, so we create a UrlTileSetModel
that applies the OpenStreetMap raster tiles settings.
OpenStreetMap tile servers adhere to the tile URL pattern http(s)://baseUrl/${z}/${x}/${y}.png
. In that pattern, z, x, and y refer to zoom level, tile column, and tile row respectively. For an overview of available servers,
see How to visualize imagery from OpenStreetMap tile servers.
OpenStreetMap uses the "EPSG:3857" projection, a popular Mercator projection for web maps, so that is what use as the coordinate
reference for the bounds we are requesting.
Next, we use a RasterTileSetLayer
to visualize the model in a layer. We want the OpenStreetMap data as a background layer, so we add it to the bottom of the
map layer tree.
import { UrlTileSetModel } from "@luciad/ria/model/tileset/UrlTileSetModel";
import { RasterTileSetLayer } from "@luciad/ria/view/tileset/RasterTileSetLayer";
import { LayerType } from "@luciad/ria/view/LayerType";
import { createBounds } from "@luciad/ria/shape/ShapeFactory";
// Add OpenStreetMap background layer.
const backgroundLayer = createOpenStreetMapLayer();
map.layerTree.addChild(backgroundLayer, "bottom");
function createOpenStreetMapLayer() {
// Create a model that applies the OpenStreetMap raster tiles' settings, including:
// - the URL pattern consisting of baseUrl/{tileLevel}/{tileColumn}/{inverse tileRow}.png
// - a Web Mercator projection
// - a single top-level tile
const webMercatorReference = getReference("EPSG:3857");
const model = new UrlTileSetModel({
baseURL: "https://a.tile.openstreetmap.org/{z}/{x}/{-y}.png",
bounds: createBounds(webMercatorReference, [
-20037508.3427892,
2 * 20037508.3427892,
-20037508.3427892,
2 * 20037508.3427892,
]),
level0Columns: 1,
level0Rows: 1,
reference: webMercatorReference,
});
// Create a layer for the model.
return new RasterTileSetLayer(model, {
layerType: LayerType.BASE,
label: "OpenStreetMap",
});
}
For more information about adding background data to LuciadRIA maps, see How to get nice background data in your application.
Add the OpenStreetMap attribution
If you use OpenStreetMap data in your map application, you must attribute OpenStreetMap. You can add the attribution in a map overlay.
To create such a map overlay, add an extra <div>
to the map <div>
in the index.html
file.
<body>
<div id="map">
<div class="attribution" id="osm_attribution">©<a href="https://www.openstreetmap.org/copyright">
OpenStreetMap </a>contributors</div>
<script src="main.js"></script>
</div>
</body>
To position the attribution on top of the map, specify CSS associated with the attribution <div>
class`.
.attribution {
position: absolute;
bottom: 10px;
right: 10px;
pointer-events: none;
padding: 5px;
z-index: 5;
}
To learn more about overlaying components on your map, see Displaying items on top of your map.
Add a coordinate grid
A typical map displays a coordinate grid, so we add one here as well. We set up a longitude-latitude grid layer with our preferred styling and scale ranges.
With the scale range settings, we determine how many grid lines we want within each scale range.
In this example, we increase the grid line density with each scale level, so that more grid lines appear when map users zoom
in. This map shows grid lines at every 20 degrees longitude and 20 degrees latitude in the scale range between 5.0E-9
and 9.0E-9
.
For the styling of our grid, we set up default or fallback styling for grid lines and labels without specific styling. We also set up a specific style that shortens the coordinates in grid line labels when users zoom out into higher scale ranges.
Click here to expand the grid setup code
import { LonLatPointFormat } from "@luciad/ria/shape/format/LonLatPointFormat";
import { LonLatGrid } from "@luciad/ria/view/grid/LonLatGrid";
import { GridLayer } from "@luciad/ria/view/grid/GridLayer";
const gridLayer = createGridLayer();
map.layerTree.addChild(gridLayer);
function createGridLayer() {
//Define scale ranges and create a grid
const settings = [{
scale: 40000.0E-9,
deltaLon: 1 / 60,
deltaLat: 1 / 60
},
...
{
scale: 5000.0E-9,
deltaLon: 1 / 2,
deltaLat: 1 / 2
}, {
scale: 1000.0E-9,
deltaLon: 1,
deltaLat: 1
}, ...
{
scale: 20.0E-9,
deltaLon: 10,
deltaLat: 10
}, {
scale: 9.0E-9,
deltaLon: 20,
deltaLat: 20
}, {
scale: 5.0E-9,
deltaLon: 30,
deltaLat: 30
}, {
scale: 0,
deltaLon: 45,
deltaLat: 45
}];
const grid = new LonLatGrid(settings);
//Set the default styling for grid lines and labels
grid.fallbackStyle = {
labelFormat: new LonLatPointFormat({
pattern: "lat(+DM),lon(+DM)"
}),
originLabelFormat: new LonLatPointFormat({
pattern: "lat(+D),lon(+D)"
}),
originLineStyle: {
color: "rgba(230, 20, 20, 0.6)",
width: 2
},
lineStyle: {
color: "rgba(210,210,210,0.6)",
width: 1
},
originLabelStyle: {
fill: "rgba(210,210,210,0.8)",
halo: "rgba(230, 20, 20, 0.8)",
haloWidth: 3,
font: "12px sans-serif"
},
labelStyle: {
fill: "rgb(220,220,220)",
halo: "rgb(102,102,102)",
haloWidth: 3,
font: "12px sans-serif"
}
};
//Use simplified labels when zoomed out a lot.
const degreesOnlyFormat = new LonLatPointFormat({
pattern: "lat(+D),lon(+D)"
});
grid.setStyle(grid.scales.indexOf(0), {
labelFormat: degreesOnlyFormat
});
grid.setStyle(grid.scales.indexOf(5.0E-9), {
labelFormat: degreesOnlyFormat
});
grid.setStyle(grid.scales.indexOf(9.0E-9), {
labelFormat: degreesOnlyFormat
});
grid.setStyle(grid.scales.indexOf(20.0E-9), {
labelFormat: degreesOnlyFormat
});
grid.setStyle(grid.scales.indexOf(200.0E-9), {
labelFormat: degreesOnlyFormat
});
return new GridLayer(grid, {
label: "Grid"
});
}
To learn more about displaying coordinate grids in LuciadRIA, see Visualizing a grid.
Add a placemark feature
Now, we want to draw attention to a particular map location. We do so by making the map automatically zoom into the general area. Then, we mark the point of interest on the map with a large red dot.
Create the placemark
We are going to add a placemark to the location of the Luciad office. We create the office location as a point shape at the office coordinates.
To save that point along with more information about the office, we create a LuciadRIA feature of it. To be able to work with
the feature, we add it to a feature model and a MemoryStore
.
A feature model allows you to manage feature data.
A Store serves a go-between for the source of the data, the computer memory in this case, and the feature model. It handles
communication and converts the features between the source data format and the LuciadRIA format.
import { createPoint } from "@luciad/ria/shape/ShapeFactory";
import { MemoryStore } from "@luciad/ria/model/store/MemoryStore";
import { Feature } from "@luciad/ria/model/feature/Feature";
import { FeatureModel } from "@luciad/ria/model/feature/FeatureModel";
const reference = getReference("CRS:84");
// Add the location of the Luciad office:
// Create a point for the location.
const luciadLocation = createPoint(reference, [4.66935, 50.8648]);
// Create a feature containing the location and some properties.
const luciadFeature = new Feature(
luciadLocation,
{
name: "Luciad office",
company: "Hexagon",
},
"id1"
);
// Add the feature to a data store and model.
// The model is used to manage the data.
// The store is the link between the model and the data provider, which is one hardcoded feature in this example.
const store = new MemoryStore({
data: [luciadFeature],
});
const model = new FeatureModel(store, {
reference: reference,
});
Now that we have feature management set up, we can turn to the visualization of our feature. We visualize LuciadRIA map data as layers, so we add the model with the office feature to a new feature layer. We want it to have a style that is distinct from the style applied by the default LuciadRIA feature painter, so we assign our own painter to the layer.
In our custom feature painter definition, we are going to use an SVG icon to visualize the office feature.
First, we create an images
subfolder in our dist
folder, and add the preferred SVG icon to it.
Next, we specify that folder as the location of the SVG icon. We also specify how we want to style the point object when map
users select it.
import { FeatureLayer } from "@luciad/ria/view/feature/FeatureLayer";
import { FeaturePainter } from "@luciad/ria/view/feature/FeaturePainter";
// Add the model to a layer.
// The layer is used to visualize the data.
const layer = new FeatureLayer(model, {
label: "Luciad office",
painter: createLocationPainter(), // we use a painter that visualizes the data by means of an SVG icon
selectable: true,
});
function createLocationPainter() {
const featurePainter = new FeaturePainter();
// Use a paintBody implementation that paints an icon at the feature's location.
featurePainter.paintBody = function (
geoCanvas,
feature,
shape,
layer,
map,
paintState
) {
let destinationStyle = {
url: "images/location.svg",
width: paintState.selected ? "110%" : "100%",
height: paintState.selected ? "110%" : "100%",
};
geoCanvas.drawIcon(shape, destinationStyle);
};
return featurePainter;
}
Fit the map on the placemark area
Directing the map focus to a specific area by zooming and centering on that area, is called fitting. In this map application, the map automatically fits to a box-shaped area, also known as bounds, around the point of interest.
MapNavigator
is the class you can use for map fitting.
To specify the bounds location for this map fit, we supply the decimal longitude and latitude coordinates of the bottom left
corner of the bounds.
With the longitude, we specify the bounds width in degrees along the X axis starting from the bounds location.
With the latitude, we specify the bounds height in degrees along the Y axis starting from the bounds location.
map.mapNavigator.defaults.snapToScaleLevels = true;
// Fit on an area around the Luciad office and the city of Leuven.
map.mapNavigator.fit({
bounds: createBounds(reference, [4.5446, 0.23, 50.8386, 0.05]),
animate: true,
});
Finally, we set animate
to true so that the application uses an automatication to automatically fit the map at startup.
At this point, we’re ready to show our feature point on the map.
// Add the layer to the map.
map.layerTree.addChild(layer, "top");
Add a balloon with feature information
We want to show feature information with the feature.
We’re going to use a balloon for that. For the balloon contents, we’re getting the properties stored with the office feature.
We set up a BalloonContentProvider
function with the office feature as its parameter.
For the balloon content, we make it return a HTML table element with the table rows filled with the feature properties.
// Add a balloon content provider for the layer, capable of showing properties for a selected feature in a balloon.
layer.balloonContentProvider = (feature) => {
let content =
"<table style='width:100%'> <tr> <td style='vertical-align: top; padding:2px; padding-bottom: 6px; font-weight: bold;'> PROPERTIES </td> </tr>";
const properties = feature.properties;
for (const property in properties) {
if (properties.hasOwnProperty(property)) {
const propertyText = property.replace(/_/g, " ");
content += "<tr style=''>";
content += `<td style='vertical-align: top; padding:2px; font-weight: bold'>${propertyText}: </td><td style='vertical-align: top; padding:2px'> ${properties[property]}</td>`;
content += "</tr>";
}
}
content += "</table>";
console.log(content);
return content;
};
You style a LuciadRIA balloon with CSS. In this application, we added the CSS styling to index.html
.
Click here to expand the balloon CSS styling
.luciad .lcdBalloon {
z-index: 400;
}
.luciad .lcdBalloon .lcdFrame {
position: relative;
color: #ffffff;
background-color: #15202d;
-webkit-box-shadow: 5px 5px 2px rgba(13, 20, 28, 0.7);
box-shadow: 5px 5px 2px rgba(13, 20, 28, 0.7);
}
.luciad .lcdBalloon .lcdHeader {
width: 100%;
height: 20px;
margin: 0;
border: 0;
padding: 0;
}
.luciad .lcdBalloon .lcdContent {
width: 100%;
max-width: 280px;
padding: 0 10px 0 10px;
overflow: auto;
box-sizing: border-box;
word-wrap: anywhere;
}
.luciad .lcdBalloon .lcdClose {
position: absolute;
width: 20px;
height: 20px;
top: 3px;
right: 3px;
background-image: url("");
background-size: 100%;
-moz-background-size: 20px 20px;
}
.luciad .lcdBalloon .lcdClose:hover {
background-image: url("");
}
Finally, we show the balloon on the map:
// Show a balloon for the Luciad office feature.
map.showBalloon({
layer,
feature: luciadFeature
})
For more information about adding balloons, see How to add balloons to your map.
Instead of balloons, you can also use LuciadRIA labels to show feature information or other text. See Labeling feature data. |