1. import React from 'react'
  2. import PropTypes from 'prop-types'
  3. import en from 'locales/en'
  4. import { ErrorBoundary } from 'ErrorBoundary'
  5. // context is only created when <MultiMapManager> is implemented (see constructor)
  6. export let MultiMapContext = null
  7. /**
  8. * A higher order component that manages MultiMapContext for connectToContext wrapped children
  9. * @component
  10. * @category MultiMap
  11. * @since 1.7.0
  12. */
  13. class MultiMapManager extends React.Component {
  14. constructor (props) {
  15. super(props)
  16. // state becomes an object of persistedStateKeys (or component names) with their persisted states'
  17. this.state = {
  18. initialized: false,
  19. maps: [],
  20. syncedState: [],
  21. visibleState: [],
  22. visibleMapCount: 0
  23. }
  24. this.syncedView = null
  25. this.promisesResolvers = []
  26. // create context when <MultiMapManager> is included in component tree
  27. MultiMapContext = React.createContext()
  28. }
  29. componentDidMount () {
  30. window.addEventListener('resize', this.refreshMaps)
  31. }
  32. refreshMaps = () => {
  33. this.forceUpdate()
  34. // this refresh needs to fire after the component updates the map views
  35. setTimeout(() => this.state.maps.map(map => map.updateSize()), 0)
  36. }
  37. syncableMapListener = (map, e) => {
  38. const { synced, type } = e
  39. if (type === 'synced' && !synced) {
  40. // reset view of newly desynced map
  41. const view = this.state[map.getTargetElement().id].view
  42. map.setView(view)
  43. } else {
  44. this.setSyncedView(map)
  45. }
  46. this.refreshMaps()
  47. }
  48. setSyncedView = map => {
  49. if (!this.syncedView) {
  50. // this is the first map, set the view in state
  51. this.syncedView = map.getView()
  52. } else {
  53. map.setView(this.syncedView)
  54. }
  55. }
  56. addToContext = (config, addToContextProp = () => {}) => {
  57. const { map } = config
  58. const mapId = map.getTargetElement().id
  59. const synced = map.getSyncedState()
  60. const visible = map.getVisibleState()
  61. const mapConfig = {
  62. ...config,
  63. synced,
  64. visible,
  65. view: map.getView()
  66. }
  67. const newState = { ...this.state, [mapId]: mapConfig }
  68. // call original prop
  69. addToContextProp(config)
  70. this.setState({ ...newState })
  71. // attach listener
  72. const listener = e => this.syncableMapListener(map, e)
  73. map.on(['synced', 'visible'], listener)
  74. if (synced) this.setSyncedView(map)
  75. }
  76. onMapInitOverride = async (map, onMapInitProp = () => {}) => {
  77. const maps = [...this.state.maps, map]
  78. this.setState({ maps })
  79. const promise = new Promise((resolve) => {
  80. // call original prop
  81. onMapInitProp(map)
  82. this.promisesResolvers.push(resolve)
  83. })
  84. // check for that last time this is called & initialize
  85. if (maps.length === this.props.mapsConfig.length) this.initialize()
  86. return promise
  87. }
  88. initialize = async () => {
  89. const { maps } = this.state
  90. const { contextProps } = await this.props.onMapsInit(maps)
  91. // resolve all onMapInit promises now
  92. this.promisesResolvers.map(resolve => resolve())
  93. this.refreshMaps()
  94. this.setState({ initialized: true, ...contextProps })
  95. }
  96. getContextValue = () => {
  97. const { contextProps, translations } = this.props
  98. const { maps } = this.state
  99. const map = maps[0]
  100. return {
  101. ...this.state,
  102. map,
  103. onMapAdded: this.onMapAdded,
  104. onMapRemoved: this.onMapRemoved,
  105. syncedState: maps.map(m => m.getSyncedState()),
  106. translations,
  107. visibleState: maps.map(m => m.getVisibleState()),
  108. visibleMapCount: maps.map(m => m.getVisibleState()).filter(e => e).length,
  109. ...contextProps
  110. }
  111. }
  112. childModifier = rawChildren => {
  113. const { initialized } = this.state
  114. const children = !Array.isArray(rawChildren) ? [rawChildren] : rawChildren
  115. const adoptedChildren = children.map((child, i) => {
  116. // only render FlexMap & FullScreenFlex until initialized
  117. const allow = initialized || child.props.disableAsyncRender
  118. if (child.props.isMultiMap) {
  119. // we caught a map
  120. const addToContextOverride = config => this.addToContext(config, child.props.addMapToContext)
  121. const onMapInitOverride = map => this.onMapInitOverride(map, child.props.onMapInit)
  122. const propsOverride = {
  123. ...child.props,
  124. addMapToContext: addToContextOverride,
  125. onMapInit: onMapInitOverride,
  126. _ol_kit_context_id: child.props.id,
  127. isMultiMap: true,
  128. key: i,
  129. map: null // important so <Map> creates a SyncableMap
  130. }
  131. const adoptedChild = React.cloneElement(child, propsOverride)
  132. return adoptedChild
  133. } else if (Array.isArray(child)) {
  134. // child is an array of children
  135. return this.childModifier(child)
  136. } else if (child?.props?.children) {
  137. // loop through children of children
  138. return allow && React.cloneElement(child, { ...child.props }, [this.childModifier(child.props.children)])
  139. } else {
  140. // this allows the Maps to render and initialize first before all other comps
  141. return allow ? child : null
  142. }
  143. })
  144. console.log('children rendered by MultiMapManager: ', adoptedChildren)
  145. return adoptedChildren
  146. }
  147. render () {
  148. const adoptedChildren = this.childModifier(this.props.children)
  149. return (
  150. <ErrorBoundary floating={true}>
  151. <MultiMapContext.Provider value={this.getContextValue()}>
  152. {adoptedChildren}
  153. </MultiMapContext.Provider>
  154. </ErrorBoundary>
  155. )
  156. }
  157. }
  158. MultiMapManager.defaultProps = {
  159. contextProps: {},
  160. groups: [],
  161. onMapsInit: () => {},
  162. translations: en
  163. }
  164. MultiMapManager.propTypes = {
  165. /** Pass components as children of Provider component */
  166. children: PropTypes.oneOfType([
  167. PropTypes.arrayOf(PropTypes.node),
  168. PropTypes.node
  169. ]).isRequired,
  170. /** Add any custom props to context and they will be passed to all components wrapped by connectToContext */
  171. contextProps: PropTypes.object,
  172. /** Nested arrays of ids grouped together to syncronize events across multiple maps */
  173. groups: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.string)),
  174. /** Array of map config objects */
  175. mapsConfig: PropTypes.array.isRequired,
  176. /** callback called with array of map objects after all multimaps have been created */
  177. onMapsInit: PropTypes.func,
  178. /** Object with key/value pairs for component translation strings */
  179. translations: PropTypes.object
  180. }
  181. export default MultiMapManager