Awesome
react-ol-fiber
react-ol-fiber is a <a href="https://reactjs.org/docs/codebase-overview.html#renderers">React renderer</a> for <a href="https://openlayers.org/">OpenLayers</a>.
Build your maps declaratively with re-usable, self-contained components that react to state, are readily interactive and can participate in React's ecosystem.
npm install ol react-ol-fiber
Being a renderer and not a wrapper it's not tied to a specific version of OpenLayers, and allows easy extensibility.
Quick Start Code
<img src="https://i.imgur.com/k5AerZY.gif" />import React, { useEffect, useState } from 'react';
import ReactDOM from 'react-dom';
import { MapComponent } from 'react-ol-fiber';
import 'ol/ol.css';
function Shapes() {
const [active, setActive] = useState(false);
useEffect(() => {
const interval = setInterval(() => setActive(a => !a), 500);
return () => clearInterval(interval);
}, []);
return (
<vectorLayer>
<styleStyle>
<fillStyle arg={{ color: active ? 'blue' : 'yellow' }} />
<strokeStyle arg={{ color: 'red' }} />
</styleStyle>
<vectorSource>
{new Array(32 * 32).fill(0).map((_, i) => (
<feature>
<circleGeometry args={[[(i % 32) * 100000, Math.floor(i / 32) * 100000], 30000]} />
</feature>
))}
</vectorSource>
</vectorLayer>
);
}
function App() {
return (
<MapComponent>
<tileLayer>
<oSMSource />
</tileLayer>
<Shapes />
<dragPanInteraction />
<mouseWheelZoomInteraction />
</MapComponent>
);
}
ReactDOM.render(<App />, document.getElementById('root'));
Docs
The MapComponent component
The most important component in react-ol-fiber is <MapComponent />
it instantiate an OpenLayer Map
object and mounts it in a full-width and full-height div. As children you can provide OpenLayer elements that can be mounted inside the map, such as layers, controls and interactions.
function App() {
return (
<MapComponent
view={{
center: fromLonLat([37.41, 8.82]),
zoom: 4,
}}
>
<tileLayer>
<oSMSource />
</tileLayer>
</MapComponent>
);
}
Using OpenLayers classes
To create instances of OpenLayers classes in react-ol-fiber you can use JSX primitives. As component name, use the original class name with the first letter in lower case, followed by its category.
To provide arguments to the class constructor use the args
prop, or arg
if the constructor has a single parameter. To attach the children to the parent you can use the attach
and attachAdd
props (even though they are inferred automatically whenever possible by the reconciler).
Some examples:
function Component() {
return (
<>
<feature /> {/* ol/Feature */}
<tileLayer /> {/* ol/layers/Tile */}
<vectorLayer /> {/* ol/layers/Vector */}
<circleGeometry args={[[0, 0], 10]} /> {/* ol/geom/Circle */}
<pointGeometry arg={[0, 0]} /> {/* ol/geom/Point */}
<dragPanInteraction /> {/* ol/geom/Point */}
<styleStyle /> {/* ol/style/Style */}
<strokeStyle arg={{ color: 'red', width: 2 }} /> {/* ol/style/Stroke */}
</>
);
}
Props
The props are applied using the setters found in the target object. The reconciler is optimized to call only the setters of the modified values.
function Component() {
// This will call setOpacity in the VectorLayer
return <vectorLayer opacity={0.75} />;
}
Event handlers
All the events described in the OpenLayers documentation are capitalized and prefixed with "on".
function Component() {
// This will set the 'select' event
return <selectInteraction onSelect={e => console.log(e)} />;
// This will set the 'change' event
return <vectorSource onChange={e => console.log(e)} />;
// It also works on the map component!
return <MapComponent onPointermove={e => console.log(e.coordinate)} />;
}
Hooks
Whenever you need to access the underlying OpenLayers map instance, you can use the useOL()
hook. Remember that this can work only inside a component that is child of a MapComponent. :warning:
function Inner() {
const { map } = useOL();
function centerOnFeatures(extent: number[]) {
const view = map.getView();
view.fit(extent);
}
return (
<vectorLayer>
<vectorSource onChange={e => centerOnFeatures(e.target.getExtent())}>
<feature>
<circleGeometry args={[[0, 0], 30000]} />
</feature>
</vectorSource>
</vectorLayer>
);
}
function Parent() {
// WARNING: you can't use useOL() here
return (
<MapComponent>
<Inner />
</MapComponent>
);
}
Spring Animation
Provisional react-spring support is available! You can use the spring api to animate your maps, using the a.
components. See this example to see how.
Using primitives
If you want to use your own already instanced objects, you can use the olPrimitive
wrapper and set a custom attach:
function Component() {
const features = myLoadFeatures();
return (
<vectorSource>
{features.map((feature, i) => (
<olPrimitive object={feature} key={i} attachAdd='feature' />
))}
</vectorSource>
);
}
:warning: Using the <olPrimitive />
instrinsic the props will not be checked. To have a generic primitive component, based on the object
prop type, use the <OLPrimitive />
wrapper instead.
Using functions
Sometimes in OpenLayers it's convenient to use a function for some objects, such as style functions, to avoid creating too many objects.
function Component() {
return (
<vectorLayer>
<olFn fn={feature => new Style({ fill: new Fill({ color: feature.get('color') }) })} attach='style' />
{/* OR */}
<olFn
fn={feature => (
<styleStyle>
<fillStyle arg={{ color: feature.get('color') }} />
</styleStyle>
)}
attach='style'
/>
<vectorSource>
<feature color='red'>
<circleGeometry args={[[0, 0], 20000]} />
</feature>
</vectorSource>
</vectorLayer>
);
}
:warning: Please note that the second option, using JSX, does NOT use React to render the elements, it manually creates instances reading the JSX. Please use it with caution and DON'T use components in there.
Extending the catalogue
To extend the available components reachable by react-ol-fiber, you can use the extend()
command. You can even implement your own props application logic using setters!
import BaseLayer from 'ol/layer/Base';
class MyLayer extends BaseLayer {
constructor(args: { ctorArg: boolean }) {
super({});
}
setMyNumber(value: number) {
console.log(value);
}
}
import { extend, MapComponent, TypeOLCustomClass } from 'react-ol-fiber';
extend({ MyLayer: MyLayer as any });
declare global {
namespace JSX {
interface IntrinsicElements {
myLayer: TypeOLCustomClass<typeof MyLayer>;
}
}
}
function Test() {
return (
<MapComponent>
<myLayer arg={{ ctorArg: false }} myNumber={42} />
</MapComponent>
);
}
FAQ
I'm not seeing my map and the entire page is blank
You need to add make your parent DOM elements full-height:
html,
body,
#root {
width: 100%;
height: 100%;
margin: 0;
}
Credits
This library was strongly inspired by <a href="https://github.com/pmndrs/react-three-fiber">react-three-fiber</a> and the technical details given by this amazing <a href="https://codyb.co/articles/a-technical-breakdown-of-react-three-fiber">article</a> by Cody Bennet.