-
Notifications
You must be signed in to change notification settings - Fork 27
/
GoogleStaticMap.js
180 lines (138 loc) · 4.59 KB
/
GoogleStaticMap.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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
import {Image, PixelRatio} from 'react-native';
import React, {Component} from 'react';
import PropTypes from 'prop-types';
const defaultMapScale = ()=> {
const isRetina = PixelRatio.get() >= 2;
return isRetina ? 2 : 1;
};
const values = (obj)=> {
return Object.keys(obj).map(key => obj[key]);
};
const IMAGE_FORMATS = {
//png8 or png (default) specifies the 8-bit PNG format.
PNG: 'png',
//png32 specifies the 32-bit PNG format.
PNG32: 'png32',
//gif specifies the GIF format.
GIF: 'gif',
//jpg specifies the JPEG compression format.
JPG: 'jpg',
//jpg-baseline specifies a non-progressive JPEG compression format.
JPG_BASELINE: 'jpg-baseline'
};
const MAP_TYPES = {
//roadmap (default) specifies a standard roadmap image, as is normally shown on the Google Maps website.
ROADMAP: 'roadmap',
//satellite specifies a satellite image.
SATELLITE: 'satellite',
//terrain specifies a physical relief map image, showing terrain and vegetation.
TERRAIN: 'terrain',
//hybrid specifies a hybrid of the satellite and roadmap image,
// showing a transparent layer of major streets and place names on the satellite image.
HYBRID: 'hybrid'
};
const IMAGE_FORMATS_VALUES = values(IMAGE_FORMATS);
const MAP_TYPES_VALUES = values(MAP_TYPES);
// the Image's source should be ignored
const {source, ...imagePropTypes} = Image.propTypes;
/**
* A wrapper for Google's Static Maps
*
* @see https://developers.google.com/maps/documentation/staticmaps/intro#Overview
*
* @example: http://staticmapmaker.com/google/
*/
class GoogleStaticMap extends Component {
/**
* https://developers.google.com/maps/documentation/staticmaps/intro#api_key
*/
static ApiKey = null;
static RootUrl = 'https://maps.googleapis.com/maps/api/staticmap';
static ImageFormats = IMAGE_FORMATS;
static MapTypes = MAP_TYPES;
static propTypes = {
...imagePropTypes,
latitude: PropTypes.string.isRequired,
longitude: PropTypes.string.isRequired,
size: PropTypes.shape({
width: PropTypes.number.isRequired,
height: PropTypes.number.isRequired
}),
/**
* zoom (required if markers not present) defines the zoom level of the map,
* which determines the magnification level of the map.
*
* @see https://developers.google.com/maps/documentation/staticmaps/intro#Zoomlevels
*/
zoom: PropTypes.number.isRequired,
/**
* scale affects the number of pixels that are returned.
* scale=2 returns twice as many pixels as scale=1 while retaining the same coverage area and level of detail
* The default value is calculated from the screen PixelRatio.
*/
scale: PropTypes.number,
/**
* @see https://developers.google.com/maps/documentation/staticmaps/intro#ImageFormats
*/
format: PropTypes.oneOf(IMAGE_FORMATS_VALUES),
/**
* @see https://developers.google.com/maps/documentation/staticmaps/intro#MapTypes
*/
mapType: PropTypes.oneOf(MAP_TYPES_VALUES),
/**
* Add a marker on the center
*/
hasCenterMarker: PropTypes.bool,
apiKey: PropTypes.string.isRequired,
onError: PropTypes.func,
onLoad: PropTypes.func
};
static defaultProps = {
scale: defaultMapScale(),
format: IMAGE_FORMATS.PNG,
mapType: MAP_TYPES.ROADMAP,
hasCenterMarker: true,
onLoad: () => {},
onError: () => {}
};
render() {
return (
<Image
style={[this.props.style, this.props.size]}
source={{uri: this.staticMapUrl}}
onError={({nativeEvent: {error}}) => this.props.onError()}
onLoad={({nativeEvent: {error}}) => this.props.onLoad()}
>
{this.props.children}
</Image>
);
}
get staticMapUrl() {
const {
latitude,
longitude,
zoom,
size,
scale,
format,
mapType
} = this.props;
const {width, height} = size;
const rootUrl = this.constructor.RootUrl;
return `${rootUrl}?center=${latitude},${longitude}&zoom=${zoom}&scale=${scale}&size=${width}x${height}&maptype=${mapType}&format=${format}&${this.markerParams}&${this.apiKeyParam}`;
}
get markerParams() {
const {
latitude,
longitude,
hasCenterMarker
} = this.props;
const markerParams = `markers=${latitude},${longitude}`;
return hasCenterMarker ? markerParams : '';
}
get apiKeyParam() {
const apiKey = this.props.apiKey;
return apiKey ? `key=${apiKey}` : '';
}
}
export default GoogleStaticMap;