2015-03-09 18:05:10 -07:00
|
|
|
|
/**
|
2015-03-23 13:35:08 -07:00
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
|
* All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
2015-03-09 18:05:10 -07:00
|
|
|
|
*
|
|
|
|
|
* @providesModule MapView
|
2015-03-25 12:55:10 -07:00
|
|
|
|
* @flow
|
2015-03-09 18:05:10 -07:00
|
|
|
|
*/
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
|
|
var EdgeInsetsPropType = require('EdgeInsetsPropType');
|
|
|
|
|
var NativeMethodsMixin = require('NativeMethodsMixin');
|
2015-04-21 21:07:17 -07:00
|
|
|
|
var Platform = require('Platform');
|
2015-03-09 18:05:10 -07:00
|
|
|
|
var React = require('React');
|
2015-05-08 09:45:43 -07:00
|
|
|
|
var ReactNativeViewAttributes = require('ReactNativeViewAttributes');
|
2015-03-09 18:05:10 -07:00
|
|
|
|
var View = require('View');
|
|
|
|
|
|
2015-05-08 09:45:43 -07:00
|
|
|
|
var createReactNativeComponentClass = require('createReactNativeComponentClass');
|
2015-03-09 18:05:10 -07:00
|
|
|
|
var deepDiffer = require('deepDiffer');
|
|
|
|
|
var insetsDiffer = require('insetsDiffer');
|
|
|
|
|
var merge = require('merge');
|
2015-04-21 21:07:17 -07:00
|
|
|
|
var requireNativeComponent = require('requireNativeComponent');
|
2015-03-09 18:05:10 -07:00
|
|
|
|
|
2015-03-25 12:55:10 -07:00
|
|
|
|
type Event = Object;
|
2015-04-21 09:01:09 -07:00
|
|
|
|
type MapRegion = {
|
|
|
|
|
latitude: number;
|
|
|
|
|
longitude: number;
|
|
|
|
|
latitudeDelta: number;
|
|
|
|
|
longitudeDelta: number;
|
|
|
|
|
};
|
2015-03-25 12:55:10 -07:00
|
|
|
|
|
2015-03-09 18:05:10 -07:00
|
|
|
|
var MapView = React.createClass({
|
|
|
|
|
mixins: [NativeMethodsMixin],
|
|
|
|
|
|
2015-06-25 09:07:19 -07:00
|
|
|
|
checkAnnotationIds: function (annotations: Array<Object>) {
|
|
|
|
|
|
|
|
|
|
var newAnnotations = annotations.map(function (annotation) {
|
|
|
|
|
if (!annotation.id) {
|
|
|
|
|
// TODO: add a base64 (or similar) encoder here
|
|
|
|
|
annotation.id = encodeURIComponent(JSON.stringify(annotation));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return annotation;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
|
annotations: newAnnotations
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
componentWillMount: function() {
|
|
|
|
|
if (this.props.annotations) {
|
|
|
|
|
this.checkAnnotationIds(this.props.annotations);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
componentWillReceiveProps: function(nextProps: Object) {
|
|
|
|
|
if (nextProps.annotations) {
|
|
|
|
|
this.checkAnnotationIds(nextProps.annotations);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2015-03-09 18:05:10 -07:00
|
|
|
|
propTypes: {
|
|
|
|
|
/**
|
|
|
|
|
* Used to style and layout the `MapView`. See `StyleSheet.js` and
|
|
|
|
|
* `ViewStylePropTypes.js` for more info.
|
|
|
|
|
*/
|
|
|
|
|
style: View.propTypes.style,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* If `true` the app will ask for the user's location and focus on it.
|
|
|
|
|
* Default value is `false`.
|
|
|
|
|
*
|
|
|
|
|
* **NOTE**: You need to add NSLocationWhenInUseUsageDescription key in
|
|
|
|
|
* Info.plist to enable geolocation, otherwise it is going
|
|
|
|
|
* to *fail silently*!
|
|
|
|
|
*/
|
|
|
|
|
showsUserLocation: React.PropTypes.bool,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* If `false` the user won't be able to pinch/zoom the map.
|
|
|
|
|
* Default `value` is true.
|
|
|
|
|
*/
|
|
|
|
|
zoomEnabled: React.PropTypes.bool,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* When this property is set to `true` and a valid camera is associated with
|
|
|
|
|
* the map, the camera’s heading angle is used to rotate the plane of the
|
|
|
|
|
* map around its center point. When this property is set to `false`, the
|
|
|
|
|
* camera’s heading angle is ignored and the map is always oriented so
|
|
|
|
|
* that true north is situated at the top of the map view
|
|
|
|
|
*/
|
|
|
|
|
rotateEnabled: React.PropTypes.bool,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* When this property is set to `true` and a valid camera is associated
|
|
|
|
|
* with the map, the camera’s pitch angle is used to tilt the plane
|
|
|
|
|
* of the map. When this property is set to `false`, the camera’s pitch
|
|
|
|
|
* angle is ignored and the map is always displayed as if the user
|
|
|
|
|
* is looking straight down onto it.
|
|
|
|
|
*/
|
|
|
|
|
pitchEnabled: React.PropTypes.bool,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* If `false` the user won't be able to change the map region being displayed.
|
|
|
|
|
* Default value is `true`.
|
|
|
|
|
*/
|
|
|
|
|
scrollEnabled: React.PropTypes.bool,
|
|
|
|
|
|
2015-06-11 10:46:28 -07:00
|
|
|
|
/**
|
|
|
|
|
* The map type to be displayed.
|
2015-06-25 09:07:19 -07:00
|
|
|
|
*
|
2015-06-11 10:46:28 -07:00
|
|
|
|
* - standard: standard road map (default)
|
|
|
|
|
* - satellite: satellite view
|
|
|
|
|
* - hybrid: satellite view with roads and points of interest overlayed
|
|
|
|
|
*/
|
|
|
|
|
mapType: React.PropTypes.oneOf([
|
2015-06-25 09:07:19 -07:00
|
|
|
|
'standard',
|
|
|
|
|
'satellite',
|
2015-06-11 10:46:28 -07:00
|
|
|
|
'hybrid',
|
|
|
|
|
]),
|
|
|
|
|
|
2015-03-09 18:05:10 -07:00
|
|
|
|
/**
|
|
|
|
|
* The region to be displayed by the map.
|
|
|
|
|
*
|
|
|
|
|
* The region is defined by the center coordinates and the span of
|
|
|
|
|
* coordinates to display.
|
|
|
|
|
*/
|
|
|
|
|
region: React.PropTypes.shape({
|
|
|
|
|
/**
|
|
|
|
|
* Coordinates for the center of the map.
|
|
|
|
|
*/
|
|
|
|
|
latitude: React.PropTypes.number.isRequired,
|
|
|
|
|
longitude: React.PropTypes.number.isRequired,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Distance between the minimun and the maximum latitude/longitude
|
|
|
|
|
* to be displayed.
|
|
|
|
|
*/
|
|
|
|
|
latitudeDelta: React.PropTypes.number.isRequired,
|
|
|
|
|
longitudeDelta: React.PropTypes.number.isRequired,
|
|
|
|
|
}),
|
|
|
|
|
|
2015-04-14 17:51:28 -07:00
|
|
|
|
/**
|
|
|
|
|
* Map annotations with title/subtitle.
|
|
|
|
|
*/
|
|
|
|
|
annotations: React.PropTypes.arrayOf(React.PropTypes.shape({
|
|
|
|
|
/**
|
|
|
|
|
* The location of the annotation.
|
|
|
|
|
*/
|
|
|
|
|
latitude: React.PropTypes.number.isRequired,
|
|
|
|
|
longitude: React.PropTypes.number.isRequired,
|
|
|
|
|
|
2015-06-25 09:07:19 -07:00
|
|
|
|
/**
|
|
|
|
|
* Whether the pin drop should be animated or not
|
|
|
|
|
*/
|
|
|
|
|
animateDrop: React.PropTypes.bool,
|
|
|
|
|
|
2015-04-14 17:51:28 -07:00
|
|
|
|
/**
|
|
|
|
|
* Annotation title/subtile.
|
|
|
|
|
*/
|
|
|
|
|
title: React.PropTypes.string,
|
|
|
|
|
subtitle: React.PropTypes.string,
|
2015-06-25 09:07:19 -07:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Whether the Annotation has callout buttons.
|
|
|
|
|
*/
|
|
|
|
|
hasLeftCallout: React.PropTypes.bool,
|
|
|
|
|
hasRightCallout: React.PropTypes.bool,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Event handlers for callout buttons.
|
|
|
|
|
*/
|
|
|
|
|
onLeftCalloutPress: React.PropTypes.func,
|
|
|
|
|
onRightCalloutPress: React.PropTypes.func,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* annotation id
|
|
|
|
|
*/
|
|
|
|
|
id: React.PropTypes.string
|
|
|
|
|
|
2015-04-14 17:51:28 -07:00
|
|
|
|
})),
|
|
|
|
|
|
2015-03-09 18:05:10 -07:00
|
|
|
|
/**
|
|
|
|
|
* Maximum size of area that can be displayed.
|
|
|
|
|
*/
|
|
|
|
|
maxDelta: React.PropTypes.number,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Minimum size of area that can be displayed.
|
|
|
|
|
*/
|
|
|
|
|
minDelta: React.PropTypes.number,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Insets for the map's legal label, originally at bottom left of the map.
|
|
|
|
|
* See `EdgeInsetsPropType.js` for more information.
|
|
|
|
|
*/
|
|
|
|
|
legalLabelInsets: EdgeInsetsPropType,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback that is called continuously when the user is dragging the map.
|
|
|
|
|
*/
|
|
|
|
|
onRegionChange: React.PropTypes.func,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback that is called once, when the user is done moving the map.
|
|
|
|
|
*/
|
|
|
|
|
onRegionChangeComplete: React.PropTypes.func,
|
2015-06-25 09:07:19 -07:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback that is called once, when the user is clicked on a annotation.
|
|
|
|
|
*/
|
|
|
|
|
onAnnotationPress: React.PropTypes.func,
|
2015-03-09 18:05:10 -07:00
|
|
|
|
},
|
|
|
|
|
|
2015-03-25 12:55:10 -07:00
|
|
|
|
_onChange: function(event: Event) {
|
2015-03-09 18:05:10 -07:00
|
|
|
|
if (event.nativeEvent.continuous) {
|
|
|
|
|
this.props.onRegionChange &&
|
|
|
|
|
this.props.onRegionChange(event.nativeEvent.region);
|
|
|
|
|
} else {
|
|
|
|
|
this.props.onRegionChangeComplete &&
|
|
|
|
|
this.props.onRegionChangeComplete(event.nativeEvent.region);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2015-06-25 09:07:19 -07:00
|
|
|
|
_onPress: function(event: Event) {
|
|
|
|
|
if (event.nativeEvent.action === 'annotation-click') {
|
|
|
|
|
this.props.onAnnotationPress && this.props.onAnnotationPress(event.nativeEvent.annotation);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (event.nativeEvent.action === 'callout-click') {
|
|
|
|
|
if (!this.props.annotations) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Find the annotation with the id of what has been pressed
|
|
|
|
|
for (var i = 0; i < this.props.annotations.length; i++) {
|
|
|
|
|
var annotation = this.props.annotations[i];
|
|
|
|
|
if (annotation.id === event.nativeEvent.annotationId) {
|
|
|
|
|
// Pass the right function
|
|
|
|
|
if (event.nativeEvent.side === 'left') {
|
|
|
|
|
annotation.onLeftCalloutPress && annotation.onLeftCalloutPress(event.nativeEvent);
|
|
|
|
|
} else if (event.nativeEvent.side === 'right') {
|
|
|
|
|
annotation.onRightCalloutPress && annotation.onRightCalloutPress(event.nativeEvent);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2015-03-09 18:05:10 -07:00
|
|
|
|
render: function() {
|
2015-06-25 09:07:19 -07:00
|
|
|
|
return <RCTMap {...this.props} onPress={this._onPress} onChange={this._onChange} />;
|
2015-03-09 18:05:10 -07:00
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
2015-04-21 21:07:17 -07:00
|
|
|
|
if (Platform.OS === 'android') {
|
2015-05-08 09:45:43 -07:00
|
|
|
|
var RCTMap = createReactNativeComponentClass({
|
2015-04-21 21:07:17 -07:00
|
|
|
|
validAttributes: merge(
|
2015-05-08 09:45:43 -07:00
|
|
|
|
ReactNativeViewAttributes.UIView, {
|
2015-06-25 21:02:53 -07:00
|
|
|
|
active: true,
|
2015-04-21 21:07:17 -07:00
|
|
|
|
showsUserLocation: true,
|
|
|
|
|
zoomEnabled: true,
|
|
|
|
|
rotateEnabled: true,
|
|
|
|
|
pitchEnabled: true,
|
|
|
|
|
scrollEnabled: true,
|
|
|
|
|
region: {diff: deepDiffer},
|
|
|
|
|
annotations: {diff: deepDiffer},
|
|
|
|
|
maxDelta: true,
|
|
|
|
|
minDelta: true,
|
|
|
|
|
legalLabelInsets: {diff: insetsDiffer},
|
|
|
|
|
}
|
|
|
|
|
),
|
|
|
|
|
uiViewClassName: 'RCTMap',
|
|
|
|
|
});
|
|
|
|
|
} else {
|
Added mechanism for directly mapping JS event handlers to blocks
Summary:
Currently, the system for mapping JS event handlers to blocks is quite clean on the JS side, but is clunky on the native side. The event property is passed as a boolean, which can then be checked by the native side, and if true, the native side is supposed to send an event via the event dispatcher.
This diff adds the facility to declare the property as a block instead. This means that the event side can simply call the block, and it will automatically send the event. Because the blocks for bubbling and direct events are named differently, we can also use this to generate the event registration data and get rid of the arrays of event names.
The name of the event is inferred from the property name, which means that the property for an event called "load" must be called `onLoad` or the mapping won't work. This can be optionally remapped to a different property name on the view itself if necessary, e.g.
RCT_REMAP_VIEW_PROPERTY(onLoad, loadEventBlock, RCTDirectEventBlock)
If you don't want to use this mechanism then for now it is still possible to declare the property as a BOOL instead and use the old mechanism (this approach is now deprecated however, and may eventually be removed altogether).
2015-09-02 05:58:10 -07:00
|
|
|
|
var RCTMap = requireNativeComponent('RCTMap', MapView, {
|
|
|
|
|
nativeOnly: {onChange: true, onPress: true}
|
|
|
|
|
});
|
2015-04-21 21:07:17 -07:00
|
|
|
|
}
|
2015-03-09 18:05:10 -07:00
|
|
|
|
|
|
|
|
|
module.exports = MapView;
|