React Native TPL Maps
React Native TPL Maps Android, iOS pacakge for React Native. It will help you to add maps in your application. The API automatically handles access to our TPL Maps servers, data downloading, map display, and response to map gestures. You can do add markers, shapes, POIs show/hide point of interests, custom map styles and much more.
Maintainers
TPL Maps
Platform Compatibility
This project is compatible with Android , iOS
This project is compatible with Android Minimum SDK 21.
Getting Started
Please follow the below steps:
1- npm install react-native-tpl-maps-view
2- Use the latest version of Package in Package.json file
3- Add your TPL Maps Key in Android Manifest File like below
<meta-data android:name="com.tplmaps.android.sdk.API_KEY"
android:value="YOUR_API_KEY" />
4- Make Sure to Add in Gradle in AllProject under buildscript tag
buildscript {
ext.safeExtGet = {prop ->
rootProject.ext.has(prop) ? rootProject.ext.get(prop) : project.properties['ReactNativeWebView_' + prop]
}
repositories {
google()
gradlePluginPortal()
}
dependencies {
classpath("com.android.tools.build:gradle:7.0.4")
classpath("com.facebook.react:react-native-gradle-plugin")
}
allprojects {
repositories {
jcenter()
google()
maven { url 'https://jitpack.io'}
maven { url "http://api.tplmaps.com:8081/artifactory/example-repo-local/"
allowInsecureProtocol = true}
}
}
}
4- Add tplservices.config file in iOS project (Download from api.tplmaps.com follow iOS guide.)
5- Add MyPacakge() in your Android MainApplication Class ArrayList inside getPackages method
Usage
Import the MyViewManager
component from react-native-tpl-maps-view
and use it like so to load Base Map:
import React, { useEffect, useRef, useState } from 'react';
import {
PixelRatio,
UIManager,
View,
findNodeHandle,
TextInput,
StyleSheet,
StatusBar,
Image,
FlatList,
Text,
TouchableOpacity,
Platform,
Dimensions,
} from 'react-native';
import MyViewManager, { AddCustomMarkerWithBase64, RemoveAllMarkers, RemoveAllPolyLines, getOptimizedSearch } from 'react-native-tpl-maps-view';
import { useWindowDimensions } from 'react-native';
import { AddMarker, CameraCallback, SetCameraAnimation, SetPolyLine, AddCustomMarker, getSearchPointData } from 'react-native-tpl-maps-view'
import { NativeModules , NativeEventEmitter } from "react-native"
import { useFocusEffect, useNavigation } from '@react-navigation/native'
import { Keyboard } from 'react-native';
const { CustomEventEmitterModule } = NativeModules;
const androidEventEmitter = new NativeEventEmitter(CustomEventEmitterModule);
var windowWidth, windowHeight;
const createFragment = (viewId: any) =>
UIManager.dispatchViewManagerCommand(
viewId,
"1", [viewId],
);
function Home(): JSX.Element {
const apikey= '$2a$10$S0hDhKNNZgcC9JR6nPXDBDrUAqj8f5gDLP2a0lLGqCTfNKq5GpvY6'
const [data, setData] = useState([]);
const [isListVisible, setListVisible] = useState(true);
const [tit, setTit] = useState("");
const [desc, setDesc] = useState("");
const [keyboardStatus, setKeyboardStatus] = useState(false);
const [coordinates, setCoordinates] = useState<{latitude: number | null, longitude: number | null}>({latitude: null, longitude: null});
useEffect(() => {
const viewId = findNodeHandle(ref.current);
createFragment(viewId);
const keyboardDidShowListener = Keyboard.addListener(
Platform.OS === 'ios' ? 'keyboardWillShow' : 'keyboardDidShow',
() => setKeyboardStatus(true)
);
const keyboardDidHideListener = Keyboard.addListener(
Platform.OS === 'ios' ? 'keyboardWillHide' : 'keyboardDidHide',
() => setKeyboardStatus(false)
);
return () => {
keyboardDidShowListener.remove();
keyboardDidHideListener.remove();
};
}, []);
const ref = useRef(null);
if (Platform.OS === 'android') {
useEffect(() => {
const viewId = findNodeHandle(ref.current);
createFragment(viewId);
const eventListener = androidEventEmitter.addListener('onReady', (event) => {
const { type, data } = event;
if(type === "onMapReady"){
console.log('onMapReady from Android', data);
}else if(type === "onMapChange"){
console.log('onMapChange from Android:', data);
const [latitude, longitude] = data.split(';');
setCoordinates({latitude: data.split(';')[0], longitude: data.split(';')[1]});
fetchReverGeocoding(latitude , longitude);
_placeMarkerAtCenterPosition(latitude , longitude)
}else if(type === "onMarkerClick"){
const { lat, lng, title, desc } = event;
console.log(`Marker clicked: lat=${lat}, lng=${lng}, title=${title}, desc=${desc}`);
}
});
return () => {
if (eventListener) {
eventListener.remove();
}
};
}, []);
}else{
useEffect(() => {
const iOSModuleListner = new NativeEventEmitter(NativeModules.CustomEventEmitter);
const eventListener = iOSModuleListner.addListener('onReady', (event) => {
const { type, data } = event;
if (type === 'onMapReady') {
console.log('Received Lat Lng string from iOS on Map Ready:', data);
const [latitude, longitude] = data.split(';');
} else if (type === 'onMapChange') {
console.log('Received Lat Lng string from iOS on Map Event:', data);
const [latitude, longitude] = data.split(';');
fetchReverGeocoding(latitude , longitude);
}else if (type === 'onMarkerClick') {
const { lat, lng, title, desc } = data;
console.log('Lat', lat);
console.log('Lng', lng);
console.log('Title', title);
console.log('Desc', desc);
}
return () => {
if (eventListener) {
eventListener.remove();
}
};
});
}, []);
}
useFocusEffect(
React.useCallback(() => {
const viewId = findNodeHandle(ref.current);
createFragment(viewId);
console.log(750)
return () => {
};
}, [])
);
windowWidth = useWindowDimensions().width;
windowHeight = useWindowDimensions().height;
const renderItem = ({ item }: { item: any }) => (
<TouchableOpacity onPress={() => handleItemClick(item)}>
<View style={{ padding: 5, backgroundColor: 'white' , marginLeft: 20 , marginRight: 20}}>
<Text style={{ color: 'black' ,fontWeight: 'bold', }}>{item.compound_address_parents}</Text>
<Text style={{ color: 'grey' ,fontWeight: 'normal', }}>{item.address}</Text>
</View>
</TouchableOpacity>
);
const handleItemClick = (item: any) => {
SetCameraAnimation(` ${item.lng};${item.lat}`, "15")
setData([]);
setListVisible(false);
};
const fetchSearchApi = async (text: any) => {
if (text.length < 3) {
return;
}
try {
if (coordinates.latitude !== null && coordinates.longitude !== null) {
const latitudeStr = coordinates.latitude.toString();
const longitudeStr = coordinates.longitude.toString();
const jsonData = await getOptimizedSearch(text ,latitudeStr , longitudeStr , apikey)
const responseString = JSON.stringify(jsonData);
const jsonData2 = await JSON.parse(responseString);
console.log(jsonData2)
setData(jsonData2);
setListVisible(true)
}
} catch (error) {
console.error('Error fetching data:', error);
}
};
async function fetchReverGeocoding (latitude: string, longitude: string) {
const jsonData = await getSearchPointData("" , latitude , longitude , apikey)
const responseString = JSON.stringify(jsonData);
const jsonData2 = await JSON.parse(responseString);
const dataArray = jsonData2.data;
setTit(dataArray[0].name)
setDesc(dataArray[0].compound_address_parents)
}
return (
<View style={{flex: 1}}>
{Platform.OS === 'ios' && (
<MyViewManager style={{ flex: 1, marginBottom: 0, marginTop: 0 }} />
)}
{Platform.OS === 'android' && (
<MyViewManager
ref={ref}
customWidth={PixelRatio.getPixelSizeForLayoutSize(windowWidth)}
customHeight={PixelRatio.getPixelSizeForLayoutSize(windowHeight)}
leftMargin={10}
rightMargin={0}
topMargin={0}
bottomMargin={120}
/>
)}
{/* Rounded corner EditText at the bottom */}
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Search for a location..."
placeholderTextColor="gray"
// Handle any changes to the text
onChangeText={(text) => fetchSearchApi(text)}
// Handle the "Done" button press
onSubmitEditing={(event) => fetchSearchApi(event.nativeEvent.text)}
/>
</View>
<View style={{ marginTop: 150, position: 'absolute'}}>
{isListVisible && (
<FlatList
data={data}
renderItem={renderItem}
keyExtractor={item => item}
/>
)}
</View>
{!keyboardStatus && (
<View style={styles.container}>
<Text style={styles.title}>{tit}</Text>
<Text style={styles.description}>{desc}</Text>
</View>
)}
</View>
);
}
const _placeMarkerAtCenterPosition = (lat: String , lng: String) => {
RemoveAllMarkers()
var base64String: String = " ";
AddCustomMarkerWithBase64(` ${lng};${lat};0;"";""` , base64String)
}
const _SetCameraAnimation = () => {
console.log("animate")
RemoveAllMarkers()
RemoveAllPolyLines()
SetCameraAnimation(" 67.0314459840222;24.801601802801088", "15")
}
const _setZoomOnUserLocation = () => {
RemoveAllMarkers()
RemoveAllPolyLines()
}
const _SetPolyLine = () => {
RemoveAllMarkers();
RemoveAllPolyLines();
SetCameraAnimation("67.103118;24.820186", "12")
const destinationTasks = ['24.801976;67.031951' , '24.883838;67.177270' , '24.846263;67.055640' , '24.835181;67.101558' , '24.820303;67.067755' ];
SetPolyLine("24.820186;67.103118", destinationTasks, "#1D8548" , "2" , "$2a$10$bWCSBfSJFAHSsa8yD3g3Sej2QfUC2vGJp1VoJHsZFlw9gwl7XBST2")
destinationTasks.forEach((task, index) => {
const [lat, lng] = task.split(';');
var base64String: String = "";
AddCustomMarkerWithBase64(` ${lng};${lat};0;"This is Title";"This is description"` , base64String)
});
}
const styles = StyleSheet.create({
inputContainer: {
width: '85%',
marginTop: 80,
marginLeft: 30,
position: 'absolute',
borderRadius: 10,
paddingHorizontal: 10,
paddingVertical: 5,
backgroundColor: 'white',
shadowColor: 'grey',
shadowOffset: { width: 0, height: 2 },
shadowOpacity: 0.2,
shadowRadius: 3,
elevation: 10,
},
input: {
fontSize: 14,
color: '#000',
marginLeft: 40,
height: 40
},
iconsContainer: {
position: 'absolute',
bottom: 120,
right: 15,
flexDirection: 'column',
alignItems: 'flex-end',
justifyContent: 'flex-end',
paddingVertical: 10,
paddingHorizontal: 5,
},
icons: {
width: 25,
height: 25,
resizeMode: 'contain',
},
icon: {
width: 60,
height: 60,
resizeMode: 'contain',
},
iconMargin: {
marginBottom: 1,
},
iconMargins: {
marginBottom: 5,
right: 18
},
container: {
flexDirection: 'column',
alignItems: 'flex-start',
justifyContent: 'flex-end',
position: 'absolute',
backgroundColor: 'white',
padding: 10,
margin: 20,
borderRadius: 10,
bottom: 60,
left: 10,
right: 10,
},
title: {
fontWeight: 'bold',
marginBottom: 3,
color: 'black'
},
description: {
marginBottom: 3,
color: 'black'
},
});
export default Home;
### BRANCH LOCATOR
import React, { useEffect, useRef, useState } from 'react';
import {
PixelRatio,
UIManager,
View,
findNodeHandle,
TextInput,
StyleSheet,
StatusBar,
Image,
FlatList,
Text,
TouchableOpacity,
Platform,
Dimensions,
} from 'react-native';
import MyViewManager, { AddCustomMarkerWithBase64, RemoveAllMarkers, RemoveAllPolyLines, getOptimizedSearch, setZoomOnCurrentLocation , EnableLocationButton} from 'react-native-tpl-maps-view';
import { useWindowDimensions } from 'react-native';
import { AddMarker, CameraCallback, SetCameraAnimation, SetPolyLine, AddCustomMarker, getSearchPointData } from 'react-native-tpl-maps-view'
import { NativeModules , NativeEventEmitter } from "react-native"
import { useFocusEffect, useNavigation } from '@react-navigation/native'
import { Keyboard } from 'react-native';
const { CustomEventEmitterModule } = NativeModules;
const androidEventEmitter = new NativeEventEmitter(CustomEventEmitterModule);
var windowWidth, windowHeight;
const cheeziousBranchesDummy = [
{ latitude: 24.899987, longitude: 67.163350 },
{ latitude: 24.860360, longitude: 67.001331 },
{ latitude: 29.344599, longitude: 71.695158 },
{ latitude: 29.392770, longitude: 71.691381 },
{ latitude: 29.417595, longitude: 71.669409 },
{ latitude: 32.065048, longitude: 72.692200 },
{ latitude: 32.101991, longitude: 72.716576 },
{ latitude: 32.025762, longitude: 72.634521 },
{ latitude: 33.718173, longitude: 73.047147 },
{ latitude: 33.715890, longitude: 73.014989 },
{ latitude: 33.702651, longitude: 72.985301 },
{ latitude: 33.676569, longitude: 73.031101 },
{ latitude: 31.582981, longitude: 74.306383 },
{ latitude: 31.547681, longitude: 74.296339 },
{ latitude: 31.486319, longitude: 74.339025 },
{ latitude: 31.465284, longitude: 74.429934 }
];
const createFragment = (viewId: any) =>
UIManager.dispatchViewManagerCommand(
viewId,
"1", [viewId],
);
var userCurrentLat = '';
var userCurrentLng = '';
function Home(): JSX.Element {
const apikey= '$2a$10$S0hDhKNNZgcC9JR6nPXDBDrUAqj8f5gDLP2a0lLGqCTfNKq5GpvY6'
const [data, setData] = useState([]);
const [isListVisible, setListVisible] = useState(true);
const [tit, setTit] = useState("");
const [desc, setDesc] = useState("");
const [keyboardStatus, setKeyboardStatus] = useState(false);
useEffect(() => {
const viewId = findNodeHandle(ref.current);
createFragment(viewId);
const keyboardDidShowListener = Keyboard.addListener(
Platform.OS === 'ios' ? 'keyboardWillShow' : 'keyboardDidShow',
() => setKeyboardStatus(true)
);
const keyboardDidHideListener = Keyboard.addListener(
Platform.OS === 'ios' ? 'keyboardWillHide' : 'keyboardDidHide',
() => setKeyboardStatus(false)
);
return () => {
keyboardDidShowListener.remove();
keyboardDidHideListener.remove();
};
}, []);
const ref = useRef(null);
if (Platform.OS === 'android') {
useEffect(() => {
const viewId = findNodeHandle(ref.current);
createFragment(viewId);
const eventListener = androidEventEmitter.addListener('onReady', (event) => {
const { type, data } = event;
if(type === "onMapReady"){
console.log('onMapReady from Android', event);
const delayToShowIcon = setTimeout(() => {
RemoveAllMarkers();
RemoveAllPolyLines();
showAllBranches()
setCurrentUserIcon(data.split(';')[0], data.split(';')[1])
userCurrentLat = data.split(';')[0]
userCurrentLng = data.split(';')[1]
}, 3000);
return () => clearTimeout(delayToShowIcon);
}else if(type === "onMapChange"){
console.log('onMapChange from Android:', data);
EnableLocationButton(false)
const [latitude, longitude] = data.split(';');
}else if(type === "onMarkerClick"){
const { lat, lng, title, desc } = event;
console.log('Lat', lat);
console.log('Lng', lng);
console.log('Title', title);
console.log('Desc', desc);
console.log(userCurrentLat)
_SetPolyLine(userCurrentLat , userCurrentLng , lat, lng)
}
});
return () => {
if (eventListener) {
eventListener.remove();
}
};
}, []);
}else{
useEffect(() => {
const iOSModuleListner = new NativeEventEmitter(NativeModules.CustomEventEmitter);
const eventListener = iOSModuleListner.addListener('onReady', (event) => {
const { type, data } = event;
if (type === 'onMapReady') {
console.log('Received Lat Lng string from iOS on Map Ready:', data);
const [latitude, longitude] = data.split(';');
} else if (type === 'onMapChange') {
console.log('Received Lat Lng string from iOS on Map Event:', data);
const [latitude, longitude] = data.split(';');
}else if (type === 'onMarkerClick') {
const { markerLat, markerLng, title, desc } = data;
console.log('Lat', markerLat);
console.log('Lng', markerLng);
console.log('Title', title);
console.log('Desc', desc);
}
return () => {
if (eventListener) {
eventListener.remove();
}
};
});
}, []);
}
useFocusEffect(
React.useCallback(() => {
const viewId = findNodeHandle(ref.current);
createFragment(viewId);
console.log(750)
return () => {
};
}, [])
);
windowWidth = useWindowDimensions().width;
windowHeight = useWindowDimensions().height;
return (
<View style={{flex: 1}}>
{Platform.OS === 'ios' && (
<MyViewManager style={{ flex: 1, marginBottom: 0, marginTop: 0 }} />
)}
{Platform.OS === 'android' && (
<MyViewManager
ref={ref}
customWidth={PixelRatio.getPixelSizeForLayoutSize(windowWidth)}
customHeight={PixelRatio.getPixelSizeForLayoutSize(windowHeight)}
leftMargin={10}
rightMargin={0}
topMargin={0}
bottomMargin={300}
/>
)}
{!keyboardStatus && (
<View style={styles.container}>
<Text style={styles.title}>{"Bottom Navigation Space"}</Text>
<Text style={styles.description}>{"Bottom Navigation Space"}</Text>
</View>
)}
</View>
);
}
const _placeMarkerAtCenterPosition = (lat: String , lng: String) => {
RemoveAllMarkers()
var base64String: String = " ";
AddCustomMarkerWithBase64(` ${lng};${lat};0;"";""` , base64String)
}
const _SetCameraAnimation = () => {
console.log("animate")
RemoveAllMarkers()
RemoveAllPolyLines()
SetCameraAnimation(" 67.0314459840222;24.801601802801088", "15")
}
const _setZoomOnUserLocation = () => {
RemoveAllMarkers()
RemoveAllPolyLines()
}
const _SetPolyLine = (currentLat: String, currentLng: String , destinationLat: String, destinationLng: String) => {
RemoveAllPolyLines();
SetCameraAnimation(`${currentLng};${currentLat}`, "12")
const destinationTasks = [`${destinationLat};${destinationLng}`];
console.log(destinationTasks)
console.log(`${currentLat};${currentLng}`)
SetPolyLine(`${currentLat};${currentLng}`, destinationTasks, "#1D8548" , "2" , "$2a$10$bWCSBfSJFAHSsa8yD3g3Sej2QfUC2vGJp1VoJHsZFlw9gwl7XBST2")
}
const setCurrentUserIcon = (lat: String, lng: String) => {
var base64String: String = "";
AddCustomMarkerWithBase64(` ${lng};${lat};0;"This is Title";"This is description"` , base64String)
}
const showAllBranches = () => {
var base64String: String = "";
cheeziousBranchesDummy.forEach((coord, index) => {
AddCustomMarkerWithBase64(`${coord.longitude};${coord.latitude};0;"Cheezious Branch ${index+1}";"This is description"` , base64String)
});
}
const styles = StyleSheet.create({
inputContainer: {
width: '85%',
marginTop: 80,
marginLeft: 30,
position: 'absolute',
borderRadius: 10,
paddingHorizontal: 10,
paddingVertical: 5,
backgroundColor: 'white',
shadowColor: 'grey',
shadowOffset: { width: 0, height: 2 },
shadowOpacity: 0.2,
shadowRadius: 3,
elevation: 10,
},
input: {
fontSize: 14,
color: '#000',
marginLeft: 40,
height: 40
},
iconsContainer: {
position: 'absolute',
bottom: 120,
right: 15,
flexDirection: 'column',
alignItems: 'flex-end',
justifyContent: 'flex-end',
paddingVertical: 10,
paddingHorizontal: 5,
},
icons: {
width: 25,
height: 25,
resizeMode: 'contain',
},
icon: {
width: 60,
height: 60,
resizeMode: 'contain',
},
iconMargin: {
marginBottom: 1,
},
iconMargins: {
marginBottom: 5,
right: 18
},
container: {
flexDirection: 'column',
alignItems: 'center',
justifyContent: 'flex-end',
position: 'absolute',
backgroundColor: 'white',
padding: 10,
margin: 20,
borderRadius: 10,
bottom: 10,
left: 10,
right: 10,
},
title: {
fontWeight: 'bold',
marginBottom: 3,
color: 'black'
},
description: {
marginBottom: 3,
color: 'black'
},
});
export default Home;
Add Marker on Map with LatLng
import {AddMarker} from 'react-native-tpl-maps-view'
AddMarker("67.125973;24.820649")
Camera Zoom on Map with LatLng and Zoom Level
import {SetCameraAnimation} from 'react-native-tpl-maps-view'
SetCameraAnimation("67.125973;24.820649" , "15")
Add Custom Marker
var base64String: String = "";
AddCustomMarkerWithBase64("74.2756865690001;31.4616382070001;90;title;description" , base64String)
RemoveAllMarkers()
RemoveAllPolyLines()
Contributing
Please report your issues and bugs on daniyal.khan@tplmaps.com
License
MIT