Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/mourner/flatbush
A very fast static spatial index for 2D points and rectangles in JavaScript 🌱
https://github.com/mourner/flatbush
algorithm computational-geometry data-structures javascript r-tree spatial-index
Last synced: 4 days ago
JSON representation
A very fast static spatial index for 2D points and rectangles in JavaScript 🌱
- Host: GitHub
- URL: https://github.com/mourner/flatbush
- Owner: mourner
- License: isc
- Created: 2018-02-27T08:57:21.000Z (almost 7 years ago)
- Default Branch: main
- Last Pushed: 2024-10-03T11:49:22.000Z (3 months ago)
- Last Synced: 2025-01-07T08:05:41.153Z (6 days ago)
- Topics: algorithm, computational-geometry, data-structures, javascript, r-tree, spatial-index
- Language: JavaScript
- Homepage:
- Size: 133 KB
- Stars: 1,436
- Watchers: 25
- Forks: 57
- Open Issues: 6
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-frontend-gis - flatbush - A really fast static spatial index for 2D points and rectangles in JavaScript. ![GitHub stars](https://img.shields.io/github/stars/mourner/flatbush?style=social) (👨💻 JavaScript Libraries / Data Processing)
README
# Flatbush
A really fast **static spatial index** for 2D points and rectangles in JavaScript.
An efficient implementation of the [packed Hilbert R-tree](https://en.wikipedia.org/wiki/Hilbert_R-tree#Packed_Hilbert_R-trees) algorithm. Enables fast spatial queries on a very large number of objects (e.g. millions), which is very useful in maps, data visualizations and computational geometry algorithms. Similar to [RBush](https://github.com/mourner/rbush), with the following key differences:
- **Static**: you can't add/remove items after initial indexing.
- **Faster** indexing and search, with much lower **memory** footprint.
- Index is stored as a single **array buffer** (to [transfer](https://developer.mozilla.org/en-US/docs/Glossary/Transferable_objects) between threads or save as a compact binary file).Supports geographic locations with the [geoflatbush](https://github.com/mourner/geoflatbush) extension. See also: [KDBush](https://github.com/mourner/kdbush), a similar library for points.
[![Build Status](https://github.com/mourner/flatbush/workflows/Node/badge.svg?branch=main)](https://github.com/mourner/flatbush/actions) [![minzipped size](https://img.shields.io/bundlephobia/minzip/flatbush)](https://esm.run/flatbush) [![Simply Awesome](https://img.shields.io/badge/simply-awesome-brightgreen.svg)](https://github.com/mourner/projects)
## Usage
```js
// initialize Flatbush for 1000 items
const index = new Flatbush(1000);// fill it with 1000 rectangles
for (const p of items) {
index.add(p.minX, p.minY, p.maxX, p.maxY);
}// perform the indexing
index.finish();// make a bounding box query
const found = index.search(minX, minY, maxX, maxY).map((i) => items[i]);// make a k-nearest-neighbors query
const neighborIds = index.neighbors(x, y, 5);// instantly transfer the index from a worker to the main thread
postMessage(index.data, [index.data]);// reconstruct the index from a raw array buffer
const index = Flatbush.from(e.data);```
## Install
Install with NPM: `npm install flatbush`, then import as a module:
```js
import Flatbush from 'flatbush';
```Or use as a module directly in the browser with [jsDelivr](https://www.jsdelivr.com/esm):
```html
import Flatbush from 'https://cdn.jsdelivr.net/npm/flatbush/+esm';
```
Alternatively, there's a browser bundle with a `Flatbush` global variable:
```html
```
## API
#### `new Flatbush(numItems[, nodeSize, ArrayType, ArrayBufferType])`
Creates a Flatbush index that will hold a given number of items (`numItems`). Additionally accepts:
- `nodeSize`: size of the tree node (`16` by default); experiment with different values for best performance (increasing this value makes indexing faster and queries slower, and vise versa).
- `ArrayType`: the array type used for coordinates storage (`Float64Array` by default);
other types may be faster in certain cases (e.g. `Int32Array` when your data is integer).
- `ArrayBufferType`: the array buffer type used to store data (`ArrayBuffer` by default);
you may prefer `SharedArrayBuffer` if you want to share the index between threads (multiple `Worker`, `SharedWorker` or `ServiceWorker`).#### `index.add(minX, minY[, maxX, maxY])`
Adds a given rectangle to the index. Returns a zero-based, incremental number that represents the newly added rectangle.
If not provided, `maxX` and `maxY` default to `minX` and `minY` (essentially adding a point).#### `index.finish()`
Performs indexing of the added rectangles.
Their number must match the one provided when creating a `Flatbush` object.#### `index.search(minX, minY, maxX, maxY[, filterFn])`
Returns an array of indices of items intersecting or touching a given bounding box. Item indices refer to the value returned by [`index.add()`](#indexaddminx-miny-maxx-maxy).
```js
const ids = index.search(10, 10, 20, 20);
```If given a `filterFn`, calls it on every found item (passing an item index)
and only includes it if the function returned a truthy value.```js
const ids = index.search(10, 10, 20, 20, (i) => items[i].foo === 'bar');
```#### `index.neighbors(x, y[, maxResults, maxDistance, filterFn])`
Returns an array of item indices in order of distance from the given `x, y`
(known as K nearest neighbors, or KNN). Item indices refer to the value returned by [`index.add()`](#indexaddminx-miny-maxx-maxy).```js
const ids = index.neighbors(10, 10, 5); // returns 5 ids
````maxResults` and `maxDistance` are `Infinity` by default.
Also accepts a `filterFn` similar to `index.search`.#### `Flatbush.from(data[, byteOffset])`
Recreates a Flatbush index from raw `ArrayBuffer` or `SharedArrayBuffer` data
(that's exposed as `index.data` on a previously indexed Flatbush instance).
Very useful for transferring or sharing indices between threads or storing them in a file.### Properties
- `data`: array buffer that holds the index.
- `minX`, `minY`, `maxX`, `maxY`: bounding box of the data.
- `numItems`: number of stored items.
- `nodeSize`: number of items in a node tree.
- `ArrayType`: array type used for internal coordinates storage.
- `IndexArrayType`: array type used for internal item indices storage.## Performance
Running `node bench.js` with Node v14:
bench | flatbush | rbush
--- | --- | ---
index 1,000,000 rectangles | 273ms | 1143ms
1000 searches 10% | 575ms | 781ms
1000 searches 1% | 63ms | 155ms
1000 searches 0.01% | 6ms | 17ms
1000 searches of 100 neighbors | 24ms | 43ms
1 search of 1,000,000 neighbors | 133ms | 280ms
100,000 searches of 1 neighbor | 710ms | 1170ms## Ports
- [chusitoo/flatbush](https://github.com/chusitoo/flatbush) (C++ port)
- [jbuckmccready/static_aabb2d_index](https://github.com/jbuckmccready/static_aabb2d_index) (Rust port)
- [jbuckmccready/Flatbush](https://github.com/jbuckmccready/Flatbush) (C# port)
- [bmharper/flatbush-python](https://github.com/bmharper/flatbush-python) (Python port)
- [FlatGeobuf](https://github.com/flatgeobuf/flatgeobuf) (a geospatial format inspired by Flatbush)
- [IMQS/flatbush](https://github.com/IMQS/flatbush) (C++ port, no longer maintained)
- [msfstef/flatbush-dart](https://github.com/msfstef/flatbush-dart) (Dart port)