Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/riatelab/geotoolbox
geojson toolbox
https://github.com/riatelab/geotoolbox
Last synced: 2 months ago
JSON representation
geojson toolbox
- Host: GitHub
- URL: https://github.com/riatelab/geotoolbox
- Owner: riatelab
- License: mit
- Created: 2022-05-20T11:17:37.000Z (over 2 years ago)
- Default Branch: main
- Last Pushed: 2024-02-12T15:40:56.000Z (11 months ago)
- Last Synced: 2024-05-23T10:05:32.705Z (8 months ago)
- Language: JavaScript
- Size: 1.08 MB
- Stars: 57
- Watchers: 5
- Forks: 11
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-frontend-gis - geotoolbox - Provides several GIS operations for use with geojson properties. Useful for thematic cartography. (👨💻 JavaScript Libraries / Data Processing)
README
![logo](img/geotoolbox.svg)
![npm](https://img.shields.io/npm/v/geotoolbox) ![jsdeliver](https://img.shields.io/jsdelivr/npm/hw/geotoolbox) ![license](https://img.shields.io/badge/license-MIT-success) ![code size](https://img.shields.io/github/languages/code-size/neocarto/bertin)
**`geotoolbox`** is a javascript tool for geographers. It allows one to manage GeoJSON properties (attribute data) and provides several useful GIS operations for thematic cartography.
From the version 2.0, it is mailny based on [geos-wasm](https://github.com/chrispahm/geos-wasm).
## 1. Installation
#### 1.1. In browser
Latest version
``` html
```
Pinned version
``` html
```
#### 1.2. In [Observable](https://observablehq.com/)
Latest version
``` js
geo = require("geotoolbox");
```Pinned version
``` js
geo = require("[email protected]");
```### 2. Demo
Find a demo of all functions [here](https://observablehq.com/@neocartocnrs/hello-geotoolbox?collection=@neocartocnrs/geotoolbox).
### 3. Documentation
#### 3.1. Handle properties
\_Here we are talking about some basic functions which are useful for handling attribute data. [Example](https://observablehq.com/@neocartocnrs/handle-properties?collection=@neocartocnrs/geotoolbox).
**add** allows adding a new field in the attribute table. This function returns a new object and does not modify the initial object.
``` js
geo.properties.add({
x: world, // a GeoJSON object
field: "gdppc", // new colname (string)
expression: "gdp/pop*1000" // a string containing an expression
})
```**head** allows getting the n top values from a given field. This function returns a new object and does not modify the initial object.
``` js
geo.properties.head({
x: world, // a GeoJSON object
field: "gdp", // a colname (string)
nb: 5 // default:10. Number of features to get. Here, the 5 richest countries.
})
```**keep** allows selecting one or several columns to keep in the attribute table. All other columns are deleted. This function returns a new object and does not modify the initial object.
``` js
geo.properties.keep({
x: world, // a GeoJSON object
field: ["ISO3", "pop2020"] // colname(s) (string or array of strings)
})
```**remove** allows removing one or several columns in the attribute table. This function returns a new object and does not modify the initial object.
``` js
geo.properties.remove({
x: world, // a GeoJSON object
field: ["tmp", "FID"] // colname(s) (string or array of strings)
})
```**select** allows filtering a GeoJSON FeatureCollection by its attribute table. This function returns a new object and does not modify the initial object.
``` js
geo.properties.select({
x: world, // a GeoJSON object
expression: "pop2022 >= 100000" // an expression (string)
})
```**subset** allows creating a subset from an array of values. This function returns a new object and does not modify the initial object.
``` js
geo.properties.subset({
x: world, // a GeoJSON object
field: "ISO3", // colname (string)
selection: ["USA", "CAN", "MEX"], // values to be kept. Here, North American countries
inverse: false // default: false. If true, all countries except USA, CAN and MEX are kept
})
```**table** allows getting a GeoJSON FeatureCollection attribute table.
``` js
geo.properties.table(world // a GeoJSON object
```**tail** allows getting the n bottom values from a given field. This function returns a new object and does not modify the initial object.
``` js
geo.properties.tail({
x: world, // a GeoJSON object
field: "gdp", // a colname (string)
nb: 5 // default:10. Number of features to get. Here, the 5 least wealthy countries
})
```#### 3.2 Handle geometries
*Here we are talking about some basic functions which are useful for thematic maps, based on [topojson](https://github.com/topojson/topojson), [d3geo](https://github.com/d3/d3-geo) and [jsts](https://github.com/bjornharrtell/jsts).*
**aggregate** allows merging geometries based on their topology. To merge polygon geometries, see `union`. [Example](https://observablehq.com/@neocartocnrs/aggregate?collection=@neocartocnrs/geotoolbox)
![aggregate](img/aggregate.svg)
``` js
geo.aggregate(world) // a GeoJSON object
```With options, you can compute an aggregate by id.
``` js
continents = geo.aggregate(
world, // a GeoJSON object
{
id: "continent" // ids
})
```**bbox** allows returning a [geographic bounding box](https://www.jasondavies.com/maps/bounds/) as GeoJSON from a GeoJSON or an array defining a bounding box `[[left, bottom], [right, top]]`. This function is based on Jacob Rus [code](https://observablehq.com/@jrus/sphere-resample). [Example](https://observablehq.com/@neocartocnrs/bbox?collection=@neocartocnrs/geotoolbox)
![bbox](img/bbox.svg)
``` js
geo.bbox(world) // a GeoJSON object
```**border** allows extracting boundaries from a GeoJSON object (polygons). With options, you can get ids and calculate discontinuities. [Example](https://observablehq.com/@neocartocnrs/border?collection=@neocartocnrs/geotoolbox)
![border](img/border.svg)
``` js
geo.border(world) // a GeoJSON object
```With options:
``` js
geo.border(
world, // a GeoJSON object
{
id: "ISO3", // ids
values: "pop", // values
type: "abs", // type of discontinuities calculated: rel (relative), abs(absolute) (default:"rel")
share: 0.8 // share of kept borders (default: null for all)
})
```**buffer** allows building a buffer from points, lines or polygones. The distance is in kilometers.
![buffer](img/buffer.svg)
``` js
geo.buffer(geojson, { dist: 1000 }) // 1000 km
```The `distance` value can also be contained in a GeoJSON field (in the properties). In this case, you just have to indicate the name of this field.
``` js
geo.buffer(geojson, { dist: "a field" }) // a field in properties
```The merge option allows merging all the output buffers.
``` js
geo.buffer(geojson, { dist: 1000, merge:true })
```The `clip` option prevents the buffers from sticking out of the world outline. Not having coordinates that exceed \[-90, 90\] in latitude and \[-180, 180\] in longitude is necessary for the d3.js projection functions to work properly.
``` js
geo.buffer(geojson, { dist: 1000, clip:true })
```The `step` option allows defining the precision of the buffer (default:8)
``` js
geo.buffer(geojson, { dist: 1000, step:1 })
```You can use `wgs84=false` if your GeoJSON is not in wgs84. In this case, the distance will be given in the map coordinates.
``` js
geo.buffer(geojson, { dist: 1000, wgs84:false })
```**clip** allows clipping geometries. [Example](https://observablehq.com/@neocartocnrs/clip?collection=@neocartocnrs/geotoolbox)
![clip](img/clip.svg)
``` js
geo.clip(geojson1, {clip:geojson2})
```With the option reverse:true, you can do a difference operation.
![clip](img/clip_reverse.svg)
``` js
geo.clip(geojson1, {clip:geojson2, reverse: true})
```You can also define a buffer in km around the clip.
``` js
geo.clip(geojson1, {clip:geojson2, buffer: 100})
```**centroid** allows computing centroids from polygons. [Example](https://observablehq.com/@neocartocnrs/centroid?collection=@neocartocnrs/geotoolbox)
![centroid](img/centroid.svg)
``` js
geo.centroid(world) // a GeoJSON (polygons) object
```By default, the centroid is placed in the largest polygon. But you can avoid it.
``` js
geo.centroid(
world, // a GeoJSON object
{
largest: false // largest polygon. true/false (default: true)
})
```It may happen that the coordinates of your base map are not in latitudes and longitudes, but already projected. In this case you can use the option `planar = true`.
``` js
geo.centroid(
world, // a GeoJSON object
{
largest: false, // largest polygon. true/false (default: true)
planar: true // if geometries are already projected
})
```**coords2geo** allows building a GeoJSON object from a table with lat,lng coordinates. [Example](https://observablehq.com/@neocartocnrs/coords2geo?collection=@neocartocnrs/geotoolbox)
![coords2geo](img/coords2geo.svg)
``` js
geo.coords2geo(
data, // a json object
{
lat: "lat", // the field containing latitude coordinates (you can use also `latitude`)
lng: "lon" // the field containing longitude coordinates (you can use also `longitude`)
})
```This function works also if coordinates are stored in a single field.
``` js
geo.coords2geo(
data, // a json object
{
coords: "Coordinates" // the field containing coordinates separated by a comma or other separator. Coordinates in wkt format are also accepted.
})
```For an even simpler automatic use, you don't have to specify the name of the variables containing the coordinates. If your data table contains the following fields (`lat`, `latitude`, `lon`,`lng`, `longitude`, `coords`, `coordinates`, `coordinate`), they are automatically selected. It is convenient, but for a better control and a faster calculation, it is still better to define where the coordinates are.
``` js
geo.coords2geo(data) // a json object
```Sometimes the coordinates can be inverted. In this case you can set the reverse option to true.
``` js
geo.coords2geo(
data, // a json object
{
reverse: true // to reverse latitudes and longitude coordinates
})
```**densify** allows to densify a geojson [Example](https://observablehq.com/@neocartocnrs/densify?collection=@neocartocnrs/geotoolbox)
![densify](img/densify.svg)
``` js
geo.densify(geojson, { dist: 0.4 })
```The `dist` value determines the minimum distance between nodes.
**dissolve** allows dissolving geometries (multipart to single parts). [Example](https://observablehq.com/@neocartocnrs/dissolve?collection=@neocartocnrs/geotoolbox)
![dissolve](img/dissolve.svg)
``` js
geo.dissolve(world) // a GeoJSON object
```**nodes** allows to retrieve geometry nodes. [Example](https://observablehq.com/@neocartocnrs/nodes?collection=@neocartocnrs/geotoolbox)
![nodes](img/nodes.svg)
``` js
geo.nodes(world) // a GeoJSON object
```**union** allows merging polygon geometries. [Example](https://observablehq.com/@neocartocnrs/union?collection=@neocartocnrs/geotoolbox)
![union](img/union.svg)
``` js
geo.union(world) // a GeoJSON object
```With options, you can compute a union by id.
``` js
continents = geo.union(
world, // a GeoJSON object
{
id: "continent" // ids
})
```**simplify** allows simplifying geometries while preserving topology (`topojson.simplify` algorithm). [Example](https://observablehq.com/@neocartocnrs/simplify?collection=@neocartocnrs/geotoolbox)
![simplify](img/simplify.svg)
``` js
geo.simplify(
world, // a GeoJSON object
{
k, // factor of simplification (default: 0.5)
})
```**tissot** tissot allows getting the Tissot's indicatrix. [Example](https://observablehq.com/@neocartocnrs/tissot?collection=@neocartocnrs/geotoolbox)
![tissot](img/tissot.svg)
``` js
geo.tissot(20) // step (default; 10)
```**geolines** allows getting the natural geographic lines such as the equator, tropics & polar circles. [Example](https://observablehq.com/@neocartocnrs/geolines?collection=@neocartocnrs/geotoolbox)
![geolines](img/geolines.svg)
``` js
geo.geolines()
```#### 3.3 Utils
**filter** allows to create and returns a new GeoJSON containing all the elements of the original GeoJSON that meet a condition determined by the callback function applied on properties (or geometries). the function returns a new GeoJSON and does not modify the initial GeoJSON. Find examples [here](https://observablehq.com/@neocartocnrs/filter-geojson?collection=@neocartocnrs/geotoolbox).
![filter](img/filter.svg)
``` js
newworld = geo.filter(world, (d) => d.CNTR_ID == "BR")
```You can also do the same thing on geometries by specifying a third argument as "geometry". For example:
``` js
newworld = geo.filter(world, (d) => d3.geoArea(d) > area, "geometry")
```**map** allows to create a new GeoJSON with the results of a function call provided on each element of GeoJSON properties (or geometries). the function returns a new GeoJSON and does not modify the initial GeoJSON. Find examples [here](https://observablehq.com/@neocartocnrs/map-geojson?collection=@neocartocnrs/geotoolbox).
![map](img/map.svg)
``` js
renameproperties = geo.map(world, (d, i) => ({
index: i,
id: d.CNTR_ID,
name: d.CNTR_NAME
}))
```You can also do the same thing on geometries by specifying a third argument as "geometry". For example:
``` js
dot = geo.map(
world,
(d) => ({ type: "Point", coordinates: d3.geoCentroid(d) }),
"geometry"
)
```**rewind** allows to allows to rewind outer ring counterclockwise and inner rings clockwise. It is adapted from MapBox geojson-rewind [code](https://github.com/mapbox/geojson-rewind)) under ISC license. [Example](https://observablehq.com/@neocartocnrs/rewind?collection=@neocartocnrs/geotoolbox)
![rewind](img/rewind.svg)
``` js
geo.rewind(
data, // a featureCollection
{
outer: true, // Rewind Rings Outer (default: true)
mutate: true // to mutate the input geoJSON (default: true)
})
```**featurecollection** allows converting an array of features or an array of geometries to a well formatted GeoJSON. [Example](https://observablehq.com/@neocartocnrs/featurecollection?collection=@neocartocnrs/geotoolbox)
![featurecollection](img/featurecollection.svg)
``` js
geo.featurecollection(features)
```**makevalid** Returns a geometry which is valid according to the GEOS validity rules, and preserves as much as possible of the input geometry's extent, dimension, and structure. The returned geometry will be valid, or null if an exception occurs.
``` js
geo.makevalid(geojson)
```**type** allows get the geometry type of a GeoJSON. It return an object containg the number of dimensions of the geometries (1 for punctual, 2 for lineal, 3 for zonal and -1 for composite) and the types of the geometries ("Point", "LineString", "Polygon", "MultiPoint", "MultiLineString", "MultiPolygon").
``` js
geo.type(geojson)
```#### 3.4 GIS operators
**contains** tests if the first geometry contains the second geometry.
``` js
geo.op.contains(geojson, geojson)
```**covers** tests whether the first geometry covers the second geometry.
``` js
geo.op.covers(geojson, geojson)
```**crosses** tests if two geometries cross.
``` js
geo.op.crosses(geojson, geojson)
```**coveredby** tests whether the first geometry is covered by the second geometry.
``` js
geo.op.coverdby(geojson, geojson)
```**disjoint** tests if two geometries are disjoint.
``` js
geo.op.disjoint(geojson, geojson)
```**equals** tests whether two geometries are topologically equal.
``` js
geo.op.equals(geojson, geojson)
```**intersects** tests if two geometries intersect in interior or boundary points.
``` js
geo.op.intersects(geojson, geojson)
```**overlaps** tests if two geometries overlap in their interior points.
``` js
geo.op.overlaps(geojson, geojson)
```**touches** tests if two geometries touch at one or more points.
``` js
geo.op.touches(geojson, geojson)
```**within** tests if the first geometry is within the second geometry.
``` js
geo.op.within(geojson, geojson)
```