import React from 'react'; import deref from '@mapbox/mapbox-gl-style-spec/deref' import tokens from '../config/tokens.json' // Empty style is always used if no style could be restored or fetched const emptyStyle = ensureStyleValidity({ version: 8, sources: {}, layers: [], }) function generateId() { return Math.random().toString(36).substr(2, 9) } function ensureHasId(style) { if('id' in style) return style style.id = generateId() return style } function ensureHasNoInteractive(style) { const changedLayers = style.layers.map(layer => { const changedLayer = { ...layer } delete changedLayer.interactive return changedLayer }) const nonInteractiveStyle = { ...style, layers: changedLayers } return nonInteractiveStyle } function ensureHasNoRefs(style) { const derefedStyle = { ...style, layers: deref(style.layers) } return derefedStyle } function ensureStyleValidity(style) { return ensureHasNoInteractive(ensureHasNoRefs(ensureHasId(style))) } function indexOfLayer(layers, layerId) { for (let i = 0; i < layers.length; i++) { if(layers[i].id === layerId) { return i } } return null } function getAccessToken(sourceName, mapStyle, opts) { if(sourceName === "thunderforest_transport" || sourceName === "thunderforest_outdoors") { sourceName = "thunderforest" } const metadata = mapStyle.metadata || {} let accessToken = metadata[`maputnik:${sourceName}_access_token`] if(opts.allowFallback && !accessToken) { accessToken = tokens[sourceName] } return accessToken; } function replaceSourceAccessToken(mapStyle, sourceName, opts={}) { const source = mapStyle.sources[sourceName] if(!source) return mapStyle if(!source.hasOwnProperty("url")) return mapStyle const accessToken = getAccessToken(sourceName, mapStyle, opts) if(!accessToken) { // Early exit. return mapStyle; } const changedSources = { ...mapStyle.sources, [sourceName]: { ...source, url: source.url.replace('{key}', accessToken) } } const changedStyle = { ...mapStyle, sources: changedSources } if (mapStyle.glyphs) { changedStyle.glyphs = changedStyle.glyphs.replace('{key}', accessToken) } return changedStyle } function replaceAccessTokens(mapStyle, opts={}) { let changedStyle = mapStyle Object.keys(mapStyle.sources).forEach((sourceName) => { changedStyle = replaceSourceAccessToken(changedStyle, sourceName, opts); }) return changedStyle } export default { ensureStyleValidity, emptyStyle, indexOfLayer, generateId, replaceAccessTokens, }