$ yarn add altalogy-map-npm
import AltaMap from 'altalogy-map-npm'
class MyComponent extends Component {
constructor(props) {
super(props)
this.altaRef = React.createRef()
}
onAltaMapChange() {
let elements = this.altaRef.current.getElements()
}
render() {
return (
<div>
<AltaMap
ref={this.altaRef}
onChange={this.onAltaMapChange}
/>
</div>
)
}
}
The AltaMap
stores own internal state and contains functions to manipulate this state. To get its state or to use its functions (let's call it library's API), you need to use ref
. In above code it is shown how to initialize ref
in constructor and assign it to <AltaMap ref={} />
.
To sync application's state, Redux store etc., you can use onChange
to get AltaMap current state.
Check Options section to see advanced configuration.
<AltaMap
ref={this.altaRef}
onChange={this.updateState}
controlPanel={{
enabled: true,
title: 'AltaMap',
}}
searchAddress={{
enabled: true,
position: 'top',
}}
googleAPI='PUT_KEY_HERE'
/>
Property | Values | Description |
---|---|---|
controlPanel | [Map] | Display the control Panel on the Map |
onChange | [Function] | Callback triggered on AltaMap state change |
controlPanel | [Map] | Enables and customizes the control panel |
searchAddress | [Map] | Enables and customizes the Google autocomplete search address input |
googleAPI | [String] | Google API Key |
React reference. Create it in constructor and assign to component.
constructor(props) {
super(props)
this.altaRef = React.createRef()
}
<AltaMap
ref={this.altaRef} />
Then, you access the AltaMap
state and API:
this.altaRef.current.addMarker('50.270908', '19.039993','Some label')
Function triggered every time that AltaMap
state being changed.
updateState() {
let elements = this.altaRef.current.getElements()
this.setState({
elements: elements
})
}
<AltaMap
ref={this.altaRef}
onChange={this.updateState}
/>
<AltaMap
controlPanel={{
enabled: true,
title: 'AltaMap',
}}
/>
Input with address autocomplete.
<AltaMap
searchAddress={{
enabled: true, // skip or set to false if you don't want to display searchAddress
position: 'top', // set position.
}}
googleAPI='PUT_GOOGLE_API_KEY_HERE'
/>
- enabled - [Boolean]
- controlPanel - [String] displays address search input in the control panel
Add Google API key if you use Address Search or Google map layers.
Usage:
(...)
constructor(props) {
super(props)
this.altaRef = React.createRef()
}
(...)
doSomething() {
# Examples:
# 1. change viewport:
this.altaRef.current.setViewport()
# 2. add element:
this.altaRef.current.addElements([{...}])
}
(...)
<AltaMap
ref={this.altaRef}
/>
(...)
API | Params | Description |
---|---|---|
setViewport(lat,lng,zoom) | lat,lng,zoom | Set mapCenter and zoom |
addElements(elements) | elements - JSON | Add elements to the map |
deleteElements(tags) | array of strings | Delete elements with specific tags |
-
Clone repository with
git clone
-
Install:
$ yarn install
- Run:
$ yarn start
Open http://localhost:3000 to view it in the browser.
$ yarn docs
Script builds application to the build
folder and then moves entire folder to docs
folder. It is required to use docs
folder to publish page on GitHub pages.
$ yarn publish:npm
$ npm login
$ npm publish
Launches the test runner in the interactive watch mode.
See the section about running tests for more information.
Project contains:
dist # 1. NPM distribution
docs # 2. GitHub Pages documentation
public # 3. Public assets
src # 4. Application's source code to present the functionalities of the Altalogy Maps library
|- ...
|- components
|- AltaMap # 5. The implementation of Altalogy Maps library
|- ...
README.md
package.json
The folder: src/components/AltaMap
contains the library source code and its structure is:
AltaMap
|- components # 4.1. Contains components
|- models # 4.2. Models
|- ...
|- MapElements # 4.3. Groups objects of MapElement as a one set
|- MapElement # 4.4. Anything displayed on the map should extends this class
|- ...
|- AltaMap.jsx # 4.5. Entrypoint for the library. Displays map, MapElements and other components
Fields:
Field | Type | Description |
---|---|---|
id | UUIDv4 | ID. If not set, random UUIDv4 is generated |
type | string | Type of the MapElement, e.g. Marker, Heatmap |
tags | array of strings | Tags help to group MapElements so then we can delete group of elements with the same tags |
options | map | contains options specific for the type, for example: icon of the marker |
data | map | contains data specific for the type, e.g. Marker may contain: lat, lng |
Methods:
Method | Params | Description |
---|---|---|
- | - | - |
Fields:
Field | Type | Description |
---|---|---|
elements | array [] | List of mapElements |
Methods:
Method | Params | Description |
---|---|---|
getElements() | Returns all mapElements | |
getElementsById(id) | string - tag | Returns all mapElements with specific id |
getElementsByTag(tag) | string - id | Returns all mapElements with specific tags |
addElements(elements) | elements - JSON | Add elements to the map |
deleteElement(tag) | string - tag | Delete element with specific tags |
deleteElementById(id) | string - id | Delete element with specific id |
hideElements(tag) | string - tag | Toggle element with specific tags |
hideElementById(id) | string - id | Toggle element with specific id |
Project contains:
- AltaMap.jsx - initialize the Leaflet map and imports components.
- models - models should define the fields and methods to manipulate object.
- MapElements - it is a collection of MapElement objects and should contain methods to manipulate that set.
- MapElement - basic class / model for elements displayed on the map. It means that every new model of the element displayed on the map should extend this class.
- components - the recommended approach is to make one folder (component) per feature.
AltaMap stores own independent state which can be read and write via ref
. It's required to provide solution compatible with this approach.
The property onChange
can be used as callback to any event changing the AltaMap internal state.
Every element displayed on the map, like markers and shapes, should be an object of class extending MapElement model or just a MapElement. All of these elements (MapElement) are grouped in a collection MapElements.
The AltaMap.jsx
contains the section Handlers and then const handlers = { ... }
in the render.
constructor(props) {
super(props)
(...)
/*** HANDLERS BINDS ***/
// >>> HERE BIND HANDLERS
this.setViewport = this.setViewport.bind(this)
this.addElements = this.addElements.bind(this)
this.addMarker = this.addMarker.bind(this)
/*** END: HANDLERS BINDS ***/
}
/***************** HANDLERS *********************/
// >>> HERE PUT HANDLERS
setViewport(lat,lng,zoom) { ... }
addElements(elements) { ... }
addMarker(lat,lng,popup) { ... }
/* ================= END HANDLERS ===================== */
render() {
const handlers = {
// >>> HERE ADD HANDLERS TO VARIABLE PASSED TO CHILDREN
addElements: this.addElements,
(...)
}
(...)
}
Handlers are just a functions doing mostly two things:
- running model's function, e.g.
mapElements.addElements()
, - using
setState
to change AltaMap state and force re-render.
It's important to use handlers to get the re-render effect.
The onChange
property is a function that is triggered every time that the AltaMap internal state is changed.
Usage:
(...)
constructor(props) {
super(props)
this.state = {
elements: []
}
this.onAltaMapChange = this.onAltaMapChange.bind(this)
this.altaRef = React.createRef()
}
(...)
onAltaMapChange(mapState) {
# `mapState` gives the current state of the AltaMap
# ... but we can access state also via ref
let elements = this.altaRef.current.getElements()
this.setState({
elements: elements
})
}
(...)
<AltaMap
ref={this.altaRef}
onChange={this.onAltaMapChange}
/>
(...)
- Each element placed on the Map should extend
MapElement
model. MapElements
model is a set ofMapElement
objects.- Add to
MapElements
methods which are the operations on a set ofMapElement
- To
MapElement
model, add methods common for eachMapElement
object - Only one feature per a new branch.
- Try to make each commit as a new record in the Changelog. (see Git section)
- AltaMap contains state so it should be able to use all functionalities without syncing its state with the parent project's state.
- Download and fetch
develop
branch on local machine - Create a new branch from
develop
. You can create:
feature/<feature-name>
- for new featurefix/<what-is-fixed>
- to add fixorg/<what-is-changed
- to make some organizational works, like update README, updatepackage.json
, fix vulnerabilities in dependencies
- Implement feature (check New features sections)
- Lint & clean code
- Make tests / add tests (?)
- Update README.md:
- Options - add here a new props and API's functions
- Features - add here a newly implemented feature
- Changelog - add to the list a name of the feature, fix or anything what will be released in a new version.
- Create pull request to
develop
branch - Ask for review & merge
To release a new version:
- Create pull request:
develop
tomaster
- Test
master
branch - Merge pull request
- Generate
docs
- Increment version in
package.json
- Update Changelog and others
- Commit changes with
release
text - Run script to publish library to NPM
Use master
branch only to store production versions, to generate docs and publish on NPM. Tag each version. Only develop
and hotfix/
branches can be merged to the master
.
The develop
is used to sync works of developers and start working on new features, make fixes and organizational works. Each branch should have one out of the following prefixes:
feature/<feature-name>
- for new featurefix/<what-is-fixed>
- to add fixorg/<what-is-changed
- to make some organizational works, like update README, updatepackage.json
, fix vulnerabilities in dependencies
Try to make that each commit in any feature/
or fix/
branch can be expressed in the Changelog.
Example 1:
Feature: feature/markers
Commits:
- Marker model and component initialization
- Adding markers
- Removing and hiding markers
Changelog:
[Feature] Markers - adding, removing and hiding
Example 2:
Feature: fix/hiding-markers
Commits:
- Fix problem of not showing back hidden markers
Changelog:
[Fix] Markers - fix of not showing back hidden markers
The general approach:
-
create a model(s). It should extend
MapElement
class if it is an element displayed right on the map. -
add to model methods
-
if needed, add in
AltaMap.jsx
handlers to the model methods (only to setters). Using model methods throughAltaMap.jsx
handlers is important when we want force re-render, because in handler we can do two things: -
run model's method:
const myModelTemp = this.state.myModel.method1()
-
refresh state:
this.setState({ myModel: myModelTemp })
More about handlers - here you find alos instructions to bind handler and add tohandlers
variable in the render function. -
create new component in
src/components/AltaMap/components
and its sub-components -
import and use component in
AltaMap.jsx
or in any other proper component
Remember to add comments to the models and components which describes:
- what is the expected input with PropTypes
- what it does - comment at the beginning of the file
Remember about README.md. Update Options section with props and/or other sections if is it related.
Format:
* [Type] <component/feature name> - <description>
[Type] - e.g.: [Feature], [Fix]
<component/feature name> - name of the component or feature that was implemented or fixed
<description> - what was implemented or fixed
- [Feature] AddressSearchBar - integration with Google Services, searching an address and adding as MapElement
- [Admin] README - add instructions how to use and develop. Added Changelog.
- [Feature] ControlPanel - panel with controls to add fake data or user input, delete elements, hide elements
- Project base architecture setup, docs, NPM configuration etc.
- [Feature] Leaflet Map - using leaflet map
- [Feature] Heatmaps
- [Feature] Drawer
- [Feature] Markers