forked from google-map-react/google-map-react
-
Notifications
You must be signed in to change notification settings - Fork 0
/
GMapOptim.js
124 lines (117 loc) · 3.08 KB
/
GMapOptim.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
// Example to test the React Reconciler. This example
// is 100x faster in development mode,
// and 1.5-2x faster with NODE_ENV==='production'
// The idea was to not draw map children on hovers, but subscribe inside children on hover change
// see ./markers/ReactiveMarker source
import React from 'react';
import {
compose,
defaultProps,
withHandlers,
withState,
withProps,
withPropsOnChange,
} from 'recompose';
import { createSelector } from 'reselect';
import { susolvkaCoords, generateMarkers } from './data/fakeData';
import GoogleMapReact from '../src';
// import SimpleMarker from './markers/SimpleMarker';
import ReactiveMarker from './markers/ReactiveMarker';
import ptInBounds from './utils/ptInBounds';
import props2Stream from './utils/props2Stream';
import withStateSelector from './utils/withStateSelector';
import { GOOGLE_API_KEY } from './config/Google_API_key';
export const gMap = (
{
style,
hoverDistance,
options,
mapParams: { center, zoom },
onChange,
onChildMouseEnter,
onChildMouseLeave,
markers,
draggable,
}
) => (
<GoogleMapReact
bootstrapURLKeys={{
key: GOOGLE_API_KEY,
}}
style={style}
options={options}
draggable={draggable}
hoverDistance={hoverDistance}
zoom={zoom}
center={center}
onChange={onChange}
onChildMouseEnter={onChildMouseEnter}
onChildMouseLeave={onChildMouseLeave}
experimental
>
{markers}
</GoogleMapReact>
);
export const gMapHOC = compose(
defaultProps({
clusterRadius: 60,
hoverDistance: 30,
options: {
minZoom: 3,
maxZoom: 15,
},
style: {
position: 'relative',
margin: 0,
padding: 0,
flex: 1,
},
}),
// withState so you could change markers if you want
withStateSelector('markers', 'setMarkers', () =>
createSelector(
({ route: { markersCount = 20 } }) => markersCount,
markersCount => generateMarkers(markersCount)
)),
withState('hoveredMarkerId', 'setHoveredMarkerId', -1),
withState('mapParams', 'setMapParams', { center: susolvkaCoords, zoom: 6 }),
// describe events
withHandlers({
onChange: ({ setMapParams }) =>
({ center, zoom, bounds }) => {
setMapParams({ center, zoom, bounds });
},
onChildMouseEnter: ({ setHoveredMarkerId }) =>
(hoverKey, { id }) => {
setHoveredMarkerId(id);
},
onChildMouseLeave: ({ setHoveredMarkerId }) =>
() => {
setHoveredMarkerId(-1);
},
}),
withPropsOnChange(['markers', 'mapParams'], ({
markers,
mapParams: { bounds },
}) => ({
markers: bounds ? markers.filter(m => ptInBounds(bounds, m)) : [],
})),
withProps(({ hoveredMarkerId }) => ({
draggable: hoveredMarkerId === -1,
})),
props2Stream('hoveredMarkerId'),
withPropsOnChange(['markers', 'hoveredMarkerId$'], ({
markers,
hoveredMarkerId$,
}) => ({
markers: markers.map(({ ...markerProps, id }) => (
<ReactiveMarker
key={id}
id={id}
hoveredMarkerId$={hoveredMarkerId$}
{...markerProps}
/>
)),
}))
);
export default gMapHOC(gMap);