react-ol-fiber is a React renderer for OpenLayers.
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.
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'));
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>
);
}
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 */}
</>
);
}
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} />;
}
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)} />;
}
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.
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>
);
}
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.
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>
);
}
<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.
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>
);
}
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>
);
}
You need to add make your parent DOM elements full-height:
html,
body,
#root {
width: 100%;
height: 100%;
margin: 0;
}
This library was strongly inspired by react-three-fiber and the technical details given by this amazing article by Cody Bennet.