Create a map component
You can extend Toolpad Studio with custom code components.
You can create a custom component to display any geographical map, like so:
data:image/s3,"s3://crabby-images/41077/41077fd355da71c140ce18029611993babc45f72" alt="Map component"
The map component
Creating the component
In the Toolpad Studio editor
- To get started creating this, hover over the component library and click on the Create button in the Custom Components section.
data:image/s3,"s3://crabby-images/a59a0/a59a0275b81fdfc984009a573784d9451c8475e2" alt="Create custom component"
Create a custom component
A dialog box appears asking you to name it. Name it "Map".
data:image/s3,"s3://crabby-images/11f56/11f56e8f19dcd32fd6c7d19a370ccda1006870fb" alt="Name custom component"
Naming a custom component
- A snackbar appears acknowledging the successful creation of the component. A starter file is created in
toolpad/components
. Use the Open button to open this file in your code editor:
data:image/s3,"s3://crabby-images/25515/2551537d1a17f6401b2208587275fc343fdb9f4e" alt="Open custom component"
Open the custom component
In the code editor
- A file with some sample code for a custom component is initialised for you. Replace its content with the following code:
import * as React from 'react';
import { createComponent } from '@toolpad/studio/browser';
import * as L from 'leaflet';
export interface LeafletProps {
lat: number;
long: number;
zoom: number;
}
async function createLeafletStyles(doc) {
let styles = doc.getElementById('leaflet-css');
if (styles) {
return;
}
const res = await fetch('https://esm.sh/leaflet/dist/leaflet.css');
if (!res.ok) {
throw new Error(`HTTP ${res.status}: ${res.statusText}`);
}
const css = await res.text();
styles = doc.createElement('style');
styles.id = 'leaflet-css';
styles.appendChild(doc.createTextNode(css));
doc.head.appendChild(styles);
}
function Leaflet({ lat, long, zoom }: LeafletProps) {
const root: any = React.useRef(null);
const mapRef = React.useRef<any>();
const [stylesInitialized, setStylesIitialized] = React.useState(false);
const [error, setError] = React.useState<Error>();
React.useEffect(() => {
const doc = root.current.ownerDocument;
createLeafletStyles(doc).then(
() => setStylesIitialized(true),
(err) => setError(err),
);
}, []);
React.useEffect(() => {
if (!mapRef.current && stylesInitialized) {
mapRef.current = L.map(root.current);
L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
maxZoom: 19,
attribution: '© OpenStreetMap',
}).addTo(mapRef.current);
}
if (mapRef.current) {
mapRef.current.setView([lat, long], zoom);
}
}, [stylesInitialized, lat, long, zoom]);
return (
<div style={{ height: 400, width: 600 }}>
{error ? (
error.message
) : (
<div style={{ width: '100%', height: '100%' }} ref={root} />
)}
</div>
);
}
export default createComponent(Leaflet, {
argTypes: {
lat: {
type: 'number',
defaultValue: 51.505,
},
long: {
type: 'number',
defaultValue: -0.09,
},
zoom: {
type: 'number',
defaultValue: 13,
},
},
});
label
and value
are the props that you will make available for binding in the Toolpad Studio editor.
- MapDisplay is now available as a custom component in the component library:
data:image/s3,"s3://crabby-images/02859/028591fbffb7656eedd4c190998955780edb5d9c" alt="map component in library"
The map component appears in the component library
Using the component
- Drag two of the map components on the canvas and select the first one. In the inspector, you'll see both the
label
andvalue
properties available as bindable properties.
data:image/s3,"s3://crabby-images/eeee7/eeee7cf636db8d65cf56e3edee4a8aad0af63876" alt="Use map components with numbers and labels"
Using the map component