Merge branch 'master' into master

This commit is contained in:
abrensch 2019-08-05 17:20:54 +02:00 committed by GitHub
commit 42abbbcb75
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
104 changed files with 26871 additions and 1753 deletions

2
.gitignore vendored
View file

@ -1,4 +1,3 @@
bower_components/
node_modules/
nbproject/
.idea/
@ -7,3 +6,4 @@ nbproject/
/keys.js
/dist
brouter-web.*.zip
yarn-error.log

10
.prettierignore Normal file
View file

@ -0,0 +1,10 @@
LICENSE
dist/
*.zip
yarn.lock
.idea
.gitignore
.prettierignore
.tx/
layers/
locales/*.json

4
.prettierrc Normal file
View file

@ -0,0 +1,4 @@
{
"singleQuote": true,
"tabWidth": 4
}

4
.travis.yml Normal file
View file

@ -0,0 +1,4 @@
language: node_js
node_js:
- lts/*
cache: yarn

10
.tx/config Normal file
View file

@ -0,0 +1,10 @@
[main]
host = https://www.transifex.com
minimum_perc = 1
lang_map = fr_CA:fr-CA,pt_BR:pt-BR,zh_CN:zh-CN,zh_HK:zh-HK,zh_TW:zh-TW,da_DK:da-DK,sv_SE:sv-SE,kn_IN:kn-IN,nl_NL:nl-NL,en_NL:en-NL,gl_ES:gl-ES
[brouter-web.brouter-website]
file_filter = locales/<lang>.json
source_file = locales/en.json
source_lang = en
type = JSON

View file

@ -1,17 +1,137 @@
BRouter-Web Changelog
=====================
# BRouter-Web Changelog
## 0.10.3 (2019-06-27)
See also [milestone 0.10.3](https://github.com/nrenner/brouter-web/milestone/11?closed=1)
### Bugfixes
- Warn when special characters in export name will get removed ([#194](https://github.com/nrenner/brouter-web/issues/194), [#202](https://github.com/nrenner/brouter-web/issues/202))
- Fix %-encoded export file name in Microsoft Edge ([#201](https://github.com/nrenner/brouter-web/issues/201))
- Fix error when no elevation data above 60° north, causing empty stats and disabled export, by implementing own missing data handling for elevation diagram ([#203](https://github.com/nrenner/brouter-web/issues/203))
### Improvements
- Reduce tile.openstreetmap.org usage ([#205](https://github.com/nrenner/brouter-web/issues/205))
- use a worldwide monolingual layer (de, fr, ru) as default when matching the browser language
- remember the last selected layers (like map view), so it doesn't load the default layer next time
- default zoom level 5 instead of 6, which seems to be cached longer
- Upgrade Gulp (build tool) to version 4.0.2 - by [@Phyks](https://github.com/Phyks) ([#209](https://github.com/nrenner/brouter-web/pull/209))
- Upgrade leaflet geocoder to properly parse lat/lng - by [@bagage](https://github.com/bagage) ([#134](https://github.com/nrenner/brouter-web/issues/134))
- Upgrade to latest Bootstrap (front-end framework) - by [@bagage](https://github.com/bagage) ([#186](https://github.com/nrenner/brouter-web/pull/186))
## 0.10.2 (2019-06-02)
See also [milestone 0.10.2](https://github.com/nrenner/brouter-web/milestone/10?closed=1)
### New Features
- Polish formatting and behaviour of track statistics bar - by [@rkflx](https://github.com/rkflx) ([#200](https://github.com/nrenner/brouter-web/pull/200))
### Bugfixes
- Fix unintentional shortcut activations when typing text - by [@rkflx](https://github.com/rkflx) ([#198](https://github.com/nrenner/brouter-web/pull/198))
- Fix export button translation - by [@bagage](https://github.com/bagage) ([#195](https://github.com/nrenner/brouter-web/issues/195))
- Fix downloads in Microsoft Edge - by [@bagage](https://github.com/bagage) ([#193](https://github.com/nrenner/brouter-web/issues/193))
## 0.10.1 (2019-05-22)
### Bugfixes
- Really ignore missing elevation points in elevation chart - by [@bagage](https://github.com/bagage)/[@nrenner](https://github.com/nrenner) ([#147](https://github.com/nrenner/brouter-web/issues/147))
## 0.10.0 (2019-05-21)
See also [milestone 0.10.0](https://github.com/nrenner/brouter-web/milestone/9?closed=1)
### New Features
- Export dialog with input field for file name and track title (replaces Download dropdown) - by [@bagage](https://github.com/bagage) ([#96](https://github.com/nrenner/brouter-web/issues/96))
### Bugfixes
- Fix broken nogo's - by [@bagage](https://github.com/bagage)/[@nrenner](https://github.com/nrenner) ([#183](https://github.com/nrenner/brouter-web/issues/183))
## 0.9.0 (2019-05-18)
See also [milestone 0.9.0](https://github.com/nrenner/brouter-web/milestone/8?closed=1)
### New Features
- Add delete last point button - by [@bagage](https://github.com/bagage) ([#33](https://github.com/nrenner/brouter-web/issues/33))
- Add reverse route button - by [@bagage](https://github.com/bagage) ([#54](https://github.com/nrenner/brouter-web/issues/54))
### Improvements
- Improve about dialog texts - by [@bagage](https://github.com/bagage) ([#176](https://github.com/nrenner/brouter-web/pull/176))
- Replace | with ; in URL - by [@bagage](https://github.com/bagage) ([#109](https://github.com/nrenner/brouter-web/issues/109))
### Bugfixes
- Ignore missing elevation points in elevation chart - by [@bagage](https://github.com/bagage) ([#147](https://github.com/nrenner/brouter-web/issues/147))
- Fix loading nogos with weight - by [@Phyks](https://github.com/Phyks) ([#174](https://github.com/nrenner/brouter-web/issues/174))
- Fix wrong version under tag ([#140](https://github.com/nrenner/brouter-web/issues/140))
## 0.8.0 (2019-05-04)
See also [milestone 0.8.0](https://github.com/nrenner/brouter-web/milestone/6?closed=1)
### New Features
- Optional layers tree ([#146](https://github.com/nrenner/brouter-web/issues/146))
- Let user upload GeoJSON file of nogos - by [@Phyks](https://github.com/Phyks) ([#161](https://github.com/nrenner/brouter-web/pull/161))
- Translations: make website localizable (i18n) - by [@bagage](https://github.com/bagage) ([#63](https://github.com/nrenner/brouter-web/issues/63))
- Fix polygon edition - by [@Phyks](https://github.com/Phyks) ([#158](https://github.com/nrenner/brouter-web/pull/158))
- Render polygons from URL hash and pass it to BRouter server - by [@Phyks](https://github.com/Phyks) ([#157](https://github.com/nrenner/brouter-web/pull/157))
- Start support of nogos polylines/polygons - by [@Phyks](https://github.com/Phyks) ([#148](https://github.com/nrenner/brouter-web/pull/148))
### Improvements
- Show line numbers in profile editor to help locating error message line ([81f2c08](https://github.com/nrenner/brouter-web/commit/81f2c0863f2569fa9079e5c96f4c9b09ef4c26e2))
- Hide StravaSegments control when layer is not active ([eaba5a0](https://github.com/nrenner/brouter-web/commit/eaba5a08217fd026fb7f83ec7beb7c1f1fdc2d69))
- Show strava error + update translations - by [@bagage](https://github.com/bagage) ([#163](https://github.com/nrenner/brouter-web/pull/163))
- Replace Bower with Yarn/npm - by [@bagage](https://github.com/bagage) ([#116](https://github.com/nrenner/brouter-web/issues/116))
- Add strava layer in overlays - by [@bagage](https://github.com/bagage) ([#152](https://github.com/nrenner/brouter-web/pull/152))
- Fix release script - by [@bagage](https://github.com/bagage) ([#150](https://github.com/nrenner/brouter-web/pull/150))
### Bugfixes
- Overlays hidden under custom layer ([#143](https://github.com/nrenner/brouter-web/issues/143))
## 0.7.0 (2018-10-10)
See also [milestone 0.7.0](https://github.com/nrenner/brouter-web/milestone/4?closed=1)
### New Features
- Redesign of the user interface to also support mobile devices - by [@bagage](https://github.com/bagage) and [@RoPP](https://github.com/RoPP) ([#34](https://github.com/nrenner/brouter-web/issues/34), [#66](https://github.com/nrenner/brouter-web/issues/66))
- Permalink replaced with auto-updating URL address bar - by [@bagage](https://github.com/bagage) ([#62](https://github.com/nrenner/brouter-web/issues/62))
- Allow user to add custom layers - by [@bagage](https://github.com/bagage) ([#77](https://github.com/nrenner/brouter-web/pull/77))
- Profile and data table now in a collapsible, full-height sidebar ([#90](https://github.com/nrenner/brouter-web/issues/90), [#114](https://github.com/nrenner/brouter-web/issues/114))
- No-go areas individually editable and deletable ([#100](https://github.com/nrenner/brouter-web/issues/100))
### Improvements
- New gulp debug task and watch CSS folder - by [@bagage](https://github.com/bagage) ([#58](https://github.com/nrenner/brouter-web/pull/58))
- Locate button not shown when no https ([#60](https://github.com/nrenner/brouter-web/issues/60))
- Support Leaflet 1.0 ([#65](https://github.com/nrenner/brouter-web/issues/65), [#69](https://github.com/nrenner/brouter-web/issues/69))
- Add a gulp command for release - by [@RoPP](https://github.com/RoPP) ([#85](https://github.com/nrenner/brouter-web/pull/85))
- Use https scheme whenever possible, to avoid mixed content issues - by [@bagage](https://github.com/bagage) ([#87](https://github.com/nrenner/brouter-web/pull/87))
- Add car-eco/fast profiles + display energy/time - by [@abrensch](https://github.com/abrensch) ([#95](https://github.com/nrenner/brouter-web/pull/95))
- Improve error message if no route found - by [@bagage](https://github.com/bagage) ([#99](https://github.com/nrenner/brouter-web/issues/99))
- Support zoom 19 for German style - by [@giggls](https://github.com/giggls) ([#128](https://github.com/nrenner/brouter-web/pull/128))
## 0.6.3 (2017-03-16)
* Fix data tab showing only two rows (regression from v0.6.2) ([#72](https://github.com/nrenner/brouter-web/issues/72))
- Fix data tab showing only two rows (regression from v0.6.2) ([#72](https://github.com/nrenner/brouter-web/issues/72))
## 0.6.2 (2017-03-14)
* Fix "API Key Required" in OpenCycleMap & Outdoors by registering for Thunderforest "Hobby Project" plan ([#70](https://github.com/nrenner/brouter-web/issues/70))
- Fix "API Key Required" in OpenCycleMap & Outdoors by registering for Thunderforest "Hobby Project" plan ([#70](https://github.com/nrenner/brouter-web/issues/70))
## 0.6.1 (2016-12-12)
* Add Esri World Imagery layer (DigitalGlobe is now also blocked because monthly usage limit is exceeded)
- Add Esri World Imagery layer (DigitalGlobe is now also blocked because monthly usage limit is exceeded)
## 0.6.0 (2016-10-11)
@ -19,66 +139,65 @@ See also [milestone 0.6.0](https://github.com/nrenner/brouter-web/milestone/1?cl
### Features/Improvements
* Update OpenTopoMap zoom range to 0-17
* [local installation] Option to remove default base layers ([#27](https://github.com/nrenner/brouter-web/issues/27))
* Add tooltip to display length in meter precision (3 digits) ([#38](https://github.com/nrenner/brouter-web/issues/38))
* Add "mean cost" to route statistics ([#39](https://github.com/nrenner/brouter-web/issues/39))
* Set route transparency slider to partially transparent by default ([#36](https://github.com/nrenner/brouter-web/issues/36))
* Show position in elevation diagram when hovering path on map ([#29](https://github.com/nrenner/brouter-web/issues/29))
* [local installation] Added ability to specify custom overlays in configuration - by [@saesh](https://github.com/saesh) ([#46](https://github.com/nrenner/brouter-web/pull/46))
* Add button to get/follow the current location (leaflet.locatecontrol plugin) - by [@bagage](https://github.com/bagage) ([#49](https://github.com/nrenner/brouter-web/pull/49))
* Save and restore last map position (leaflet.restoreview.js plugin) - by [@bagage](https://github.com/bagage) ([#49](https://github.com/nrenner/brouter-web/pull/49))
* Toggle drawing mode via panel button - by [@bagage](https://github.com/bagage) ([#50](https://github.com/nrenner/brouter-web/pull/50))
* [local installation] add keys.js to configure API keys instead of bingkey request
* Switch to new icon set (Font Awesome) with more options
- Update OpenTopoMap zoom range to 0-17
- [local installation] Option to remove default base layers ([#27](https://github.com/nrenner/brouter-web/issues/27))
- Add tooltip to display length in meter precision (3 digits) ([#38](https://github.com/nrenner/brouter-web/issues/38))
- Add "mean cost" to route statistics ([#39](https://github.com/nrenner/brouter-web/issues/39))
- Set route transparency slider to partially transparent by default ([#36](https://github.com/nrenner/brouter-web/issues/36))
- Show position in elevation diagram when hovering path on map ([#29](https://github.com/nrenner/brouter-web/issues/29))
- [local installation] Added ability to specify custom overlays in configuration - by [@saesh](https://github.com/saesh) ([#46](https://github.com/nrenner/brouter-web/pull/46))
- Add button to get/follow the current location (leaflet.locatecontrol plugin) - by [@bagage](https://github.com/bagage) ([#49](https://github.com/nrenner/brouter-web/pull/49))
- Save and restore last map position (leaflet.restoreview.js plugin) - by [@bagage](https://github.com/bagage) ([#49](https://github.com/nrenner/brouter-web/pull/49))
- Toggle drawing mode via panel button - by [@bagage](https://github.com/bagage) ([#50](https://github.com/nrenner/brouter-web/pull/50))
- [local installation] add keys.js to configure API keys instead of bingkey request
- Switch to new icon set (Font Awesome) with more options
### Bugfixes
* Replace Bing (usage limit exceeded) with DigitalGlobe Recent Imagery layer (newer images, but sometimes cloudy)
* [local installation] Show error message for invalid server response with custom profiles on Windows (still needs to be fixed) ([#53](https://github.com/nrenner/brouter-web/issues/53))
* Restrictive Cookie settings caused app to stop responding ([#47](https://github.com/nrenner/brouter-web/issues/47))
- Replace Bing (usage limit exceeded) with DigitalGlobe Recent Imagery layer (newer images, but sometimes cloudy)
- [local installation] Show error message for invalid server response with custom profiles on Windows (still needs to be fixed) ([#53](https://github.com/nrenner/brouter-web/issues/53))
- Restrictive Cookie settings caused app to stop responding ([#47](https://github.com/nrenner/brouter-web/issues/47))
## 0.5.2 (2015-08-27)
* switch search from MapQuest to Nominatim (MapQuest licensing change)
- switch search from MapQuest to Nominatim (MapQuest licensing change)
## 0.5.1 (2015-07-24)
* config option ``baseLayers`` to add custom base layers locally (#24)
* reset slider on page load to minimum opacity (#22),
customizable locally with config setting ``minOpacity``
* set OpenTopoMap max zoom back to z15 while on fallback server (#21),
also fix max zoom of other services
* overscale tiles to common max zoom (avoids gray screen when switching)
- config option `baseLayers` to add custom base layers locally (#24)
- reset slider on page load to minimum opacity (#22),
customizable locally with config setting `minOpacity`
- set OpenTopoMap max zoom back to z15 while on fallback server (#21),
also fix max zoom of other services
- overscale tiles to common max zoom (avoids gray screen when switching)
## 0.5.0 (2015-07-01)
### Features
* Load profile content for selected profile (needs extra server locally)
* Bing maps aerial layer (not working locally)
* track color magenta instead of blue + white casing, for better contrast
with background map (esp. OpenCycleMap)
* transparency slider for route track and markers
* button to delete route (#10)
* map scale
* download all dependencies in a bundle, instead using CDNs and separate files (#18)
* switch search plugin for result-dependent zoom
* "about" popup with a bit more infos and links
* closable error/warning messages, profile messages in place
- Load profile content for selected profile (needs extra server locally)
- Bing maps aerial layer (not working locally)
- track color magenta instead of blue + white casing, for better contrast
with background map (esp. OpenCycleMap)
- transparency slider for route track and markers
- button to delete route (#10)
- map scale
- download all dependencies in a bundle, instead using CDNs and separate files (#18)
- switch search plugin for result-dependent zoom
- "about" popup with a bit more infos and links
- closable error/warning messages, profile messages in place
### Bugfixes
* keys to enable/disable drawing (d, q/esc) now always work, not only when map is focused
* fix adding new waypoint after deleting the last (#11)
* fix profile/data scrolling on Firefox
* hide trailer over controls and outside map
- keys to enable/disable drawing (d, q/esc) now always work, not only when map is focused
- fix adding new waypoint after deleting the last (#11)
- fix profile/data scrolling on Firefox
- hide trailer over controls and outside map
## BRouter 1.2
* data/CSV aggregated over segments with same tags (for better performance)
- data/CSV aggregated over segments with same tags (for better performance)
## 0.4.0 (2015-03-08)
* data tab (slow with long routes, exp. on Firefox)
- data tab (slow with long routes, exp. on Firefox)

View file

@ -1,6 +1,6 @@
The MIT License (MIT)
Copyright (c) 2016 Norbert Renner and contributors
Copyright (c) 2018 Norbert Renner and contributors
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in

213
README.md
View file

@ -1,22 +1,33 @@
brouter-web
===========
# brouter-web
Web client (by [@nrenner](https://github.com/nrenner)) for the BRouter routing engine (by [@abrensch](https://github.com/abrensch)). *Work in progress*.
Web client (by [@nrenner](https://github.com/nrenner) and [contributors](https://github.com/nrenner/brouter-web/graphs/contributors)) for the BRouter routing engine (by [@abrensch](https://github.com/abrensch)). _Work in progress_.
**New web client available mobile-ready to test on [brouter.damsy.net](http://brouter.damsy.net).
Feedbacks are appreciated, do not hesitate to create issues about it!**
Instances:
BRouter online service (provided by [@abrensch](https://github.com/abrensch)):
http://brouter.de/brouter-web/
- [brouter.de/brouter-web](http://brouter.de/brouter-web/) _(provided by [@abrensch](https://github.com/abrensch))_
- [brouter.damsy.net](http://brouter.damsy.net) _(provided by [@bagage](https://github.com/bagage))_
This repository is only about the frontend. For the server/backend, BRouter routing engine, Android app, profiles, brouter.de site, see:
**This repository is only about the frontend**.
For the server/backend, BRouter routing engine, Android app, profiles, brouter.de site, see:
https://github.com/abrensch/brouter
More information:
http://brouter.de
## Contact
General BRouter discussions/questions, support:
https://groups.google.com/group/osm-android-bikerouting
- [`#brouter` on IRC OFTC](https://webchat.oftc.net/?channels=#brouter). You can also use [Riot](https://riot.im/app/#/room/#_oftc_#brouter:matrix.org).
- [Google Group](https://groups.google.com/group/osm-android-bikerouting)
## Translating
Translations are managed using the
[Transifex](https://www.transifex.com/openstreetmap/brouter-web/) platform. After
signing up, you can go to [BRouter's project
page](https://www.transifex.com/openstreetmap/brouter-web/dashboard/), select a language and
click **Translate** to start translating.
## Installation
@ -24,116 +35,148 @@ As an alternative to the above online version, the standalone server of BRouter
### Install BRouter (server with routing engine)
1. download and unzip latest [BRouter revision](http://brouter.de/brouter/revisions.html)
e.g. for Linux (replace ``~/opt/`` with your preferred install dir and ``1_4_1`` with latest version):
1. download and unzip latest [BRouter revision](http://brouter.de/brouter/revisions.html)
e.g. for Linux (replace `~/opt/` with your preferred install directory and `1_4_11` with latest version):
mkdir ~/opt/brouter
cd ~/opt/brouter
wget http://brouter.de/brouter_bin/brouter_1_4_1.zip
unzip brouter_1_4_1.zip
chmod +x ./standalone/server.sh
mkdir ~/opt/brouter
cd ~/opt/brouter
wget http://brouter.de/brouter_bin/brouter_1_4_11.zip
unzip brouter_1_4_11.zip
chmod +x ./standalone/server.sh
2. download one or more [data file(s)](http://brouter.de/brouter/segments4/) (rd5) into ``segments4`` dir
2. download one or more [data file(s)](http://brouter.de/brouter/segments4/) (rd5) into `segments4` directory
### Install BRouter-Web (client)
1. download BRouter-Web as subdirectory ``brouter-web`` of the ``brouter`` directory
* using the latest stable release - adjust to current version number - from
https://github.com/nrenner/brouter-web/releases:
1. download BRouter-Web as subdirectory `brouter-web` of the `brouter` directory
wget https://github.com/nrenner/brouter-web/archive/0.6.3.zip
unzip 0.6.3.zip
mv brouter-web-0.6.3 brouter-web
- using the latest stable release - adjust to current version number - from
https://github.com/nrenner/brouter-web/releases:
* OR the current development state (potentially instable and without runtime distributables):
wget https://github.com/nrenner/brouter-web/releases/download/0.7.0/brouter-web-0.7.0.zip
unzip brouter-web-0.7.0.zip -d brouter-web
wget https://github.com/nrenner/brouter-web/archive/master.zip
unzip master.zip
mv brouter-web-master brouter-web
- OR the current development state (potentially instable and without runtime distributables):
* build the distributable files required for runtime (only for development state), see section [Build](#build)
wget https://github.com/nrenner/brouter-web/archive/master.zip
unzip master.zip
mv brouter-web-master brouter-web
2. copy ``config.template.js`` to ``config.js``
3. configure URL to ``profiles2`` directory
set ``BR.conf.profilesUrl`` in config.js, e.g. uncomment:
- build the distributable files required for runtime (only for development state), see section [Build](#build)
BR.conf.profilesUrl = 'http://localhost:8000/profiles2/';
2. copy `config.template.js` to `config.js`
3. configure URL to `profiles2` directory
set `BR.conf.profilesUrl` in config.js, e.g. uncomment:
4. add your API keys (optional)
copy ``keys.template.js`` to ``keys.js`` and edit to add your keys
BR.conf.profilesUrl = 'http://localhost:8000/profiles2/';
4. add your API keys (optional)
copy `keys.template.js` to `keys.js` and edit to add your keys
### Run
1. start BRouter server in the ``standalone`` directory with ``./server.sh`` or ``server.cmd`` (Windows)
2. serve the ``brouter`` directory for BRouter-Web
This is needed for pre-loading the selected profile (unless you allowed local file access in the Browser). Depending on your setup (see [How to run things locally](https://github.com/mrdoob/three.js/wiki/How-to-run-things-locally)), start a web server in the ``brouter`` directory, e.g.:
1. start BRouter server in the `standalone` directory with `./server.sh` or `server.cmd` (Windows)
2. serve the `brouter` directory for BRouter-Web
This is needed for pre-loading the selected profile (unless you allowed local file access in the Browser). Depending on your setup (see [How to run things locally](https://github.com/mrdoob/three.js/wiki/How-to-run-things-locally)), start a web server in the `brouter` directory, e.g.:
python -m SimpleHTTPServer
python -m SimpleHTTPServer
2. open http://localhost:8000/brouter-web/
3. open http://localhost:8000/brouter-web/
## Build
### Dependencies
Requires [Node and npm](https://nodejs.org/) (or [io.js](https://iojs.org)), [Bower](https://bower.io/) and [Gulp](http://gulpjs.com/):
npm install -g bower
npm install -g gulp
Requires [Node.js](https://nodejs.org/) and [Yarn](https://yarnpkg.com/en/).
### Install
npm install
bower install
yarn
### Build
gulp #for release
gulp debug #for development
yarn build #for release
yarn build debug #for development
### Develop
gulp watch
yarn build watch
## License
Copyright (c) 2016 Norbert Renner and [contributors](https://github.com/nrenner/brouter-web/graphs/contributors), licensed under the [MIT License (MIT)](LICENSE)
Copyright (c) 2018 Norbert Renner and [contributors](https://github.com/nrenner/brouter-web/graphs/contributors), licensed under the [MIT License (MIT)](LICENSE)
## Credits and Licenses
* [BRouter](https://github.com/abrensch/brouter) (not included)
by abrensch; [GNU General Public License, version 3.0 (GPLv3)](https://github.com/abrensch/brouter/blob/master/LICENSE)
* [Leaflet](http://leafletjs.com/)
Copyright (c) 2010-2014, Vladimir Agafonkin; Copyright (c) 2010-2011, CloudMade; [2-clause BSD License](https://github.com/Leaflet/Leaflet/blob/master/LICENSE)
* [leaflet-routing](https://github.com/Turistforeningen/leaflet-routing)
Copyright (c) 2013, Turistforeningen, Hans Kristian Flaatten. All rights reserved. [2-clause BSD License](https://github.com/Turistforeningen/leaflet-routing/blob/gh-pages/LICENSE)
* [Leaflet.Elevation](https://github.com/MrMufflon/Leaflet.Elevation)
Copyright (c) 2013 Felix Bache; [MIT License](https://github.com/MrMufflon/Leaflet.Elevation/blob/master/LICENSE)
* [D3.js](https://github.com/mbostock/d3)
Copyright (c) 2013, Michael Bostock. All rights reserved.; [3-clause BSD License](https://github.com/mbostock/d3/blob/master/LICENSE)
* [Leaflet.draw](https://github.com/Leaflet/Leaflet.draw)
Copyright 2012 Jacob Toye; [MIT License](https://github.com/Leaflet/Leaflet.draw/blob/master/MIT-LICENCE.txt)
* [Leaflet Control Geocoder](https://github.com/perliedman/leaflet-control-geocoder)
Copyright (c) 2012 [sa3m](https://github.com/sa3m), Copyright (c) 2013 Per Liedman; [2-clause BSD License](https://github.com/perliedman/leaflet-control-geocoder/blob/master/LICENSE)
* [leaflet-plugins](https://github.com/shramov/leaflet-plugins)
Copyright (c) 2011-2012, Pavel Shramov; [2-clause BSD License](https://github.com/shramov/leaflet-plugins/blob/master/LICENSE)
* [Async.js](https://github.com/caolan/async)
Copyright (c) 2010-2014 Caolan McMahon; [MIT License](https://github.com/caolan/async/blob/master/LICENSE)
* [Bootstrap](https://getbootstrap.com/)
Copyright (c) 2011-2014 Twitter, Inc; [MIT License](https://github.com/twbs/bootstrap/blob/master/LICENSE)
* [jQuery](https://github.com/jquery/jquery)
Copyright 2005, 2014 jQuery Foundation and other contributors; [MIT License](https://github.com/jquery/jquery/blob/master/LICENSE.txt)
* [DataTables](https://github.com/DataTables/DataTables)
Copyright (C) 2008-2014, SpryMedia Ltd.; [MIT License](https://www.datatables.net/license/MIT-LICENCE)
* [Leaflet.EasyButton](https://github.com/CliffCloud/Leaflet.EasyButton)
Copyright (C) 2014 Daniel Montague; [MIT License](https://github.com/CliffCloud/Leaflet.EasyButton/blob/master/LICENSE)
* [Bootbox](https://github.com/makeusabrew/bootbox)
Copyright (C) 2011-2014 by Nick Payne; [MIT License](https://github.com/makeusabrew/bootbox/blob/master/LICENSE.md)
* [bootstrap-slider](https://github.com/seiyria/bootstrap-slider)
Copyright (c) 2015 Kyle Kemp, Rohit Kalkur, and contributors; [MIT License](https://github.com/seiyria/bootstrap-slider/blob/master/LICENSE.md)
* [Leaflet.RestoreView](https://github.com/makinacorpus/Leaflet.RestoreView)
Copyright (c) 2012 Makina Corpus, [MIT License](https://github.com/makinacorpus/Leaflet.RestoreView/blob/master/LICENSE)
* [Leaflet.Locate](https://github.com/domoritz/leaflet-locatecontrol)
Copyright (c) 2014 Dominik Moritz, [MIT License](https://github.com/domoritz/leaflet-locatecontrol/blob/gh-pages/LICENSE)
* [Font Awesome](http://fontawesome.io/license/)
by Dave Gandy; [SIL OFL 1.1](https://scripts.sil.org/OFL) (Font), MIT License (Code), CC BY 3.0 (Documentation)
- [BRouter](https://github.com/abrensch/brouter) (not included)
by abrensch; [GNU General Public License, version 3.0 (GPLv3)](https://github.com/abrensch/brouter/blob/master/LICENSE)
- [Leaflet](http://leafletjs.com/)
Copyright (c) 2010-2014, Vladimir Agafonkin; Copyright (c) 2010-2011, CloudMade; [2-clause BSD License](https://github.com/Leaflet/Leaflet/blob/master/LICENSE)
- [leaflet-routing](https://github.com/Turistforeningen/leaflet-routing)
Copyright (c) 2013, Turistforeningen, Hans Kristian Flaatten. All rights reserved. [2-clause BSD License](https://github.com/Turistforeningen/leaflet-routing/blob/gh-pages/LICENSE)
- [Leaflet.Elevation](https://github.com/MrMufflon/Leaflet.Elevation)
Copyright (c) 2013 Felix Bache; [MIT License](https://github.com/MrMufflon/Leaflet.Elevation/blob/master/LICENSE)
- [D3.js](https://github.com/mbostock/d3)
Copyright (c) 2013, Michael Bostock. All rights reserved.; [3-clause BSD License](https://github.com/mbostock/d3/blob/master/LICENSE)
- [Leaflet.Editable](https://github.com/Leaflet/Leaflet.Editable)
Yohan Boniface; WTFPL licence
- [Leaflet Control Geocoder](https://github.com/perliedman/leaflet-control-geocoder)
Copyright (c) 2012 [sa3m](https://github.com/sa3m), Copyright (c) 2013 Per Liedman; [2-clause BSD License](https://github.com/perliedman/leaflet-control-geocoder/blob/master/LICENSE)
- [leaflet-plugins](https://github.com/shramov/leaflet-plugins)
Copyright (c) 2011-2012, Pavel Shramov; [2-clause BSD License](https://github.com/shramov/leaflet-plugins/blob/master/LICENSE)
- [Async.js](https://github.com/caolan/async)
Copyright (c) 2010-2014 Caolan McMahon; [MIT License](https://github.com/caolan/async/blob/master/LICENSE)
- [Bootstrap](https://getbootstrap.com/)
Copyright (c) 2011-2014 Twitter, Inc; [MIT License](https://github.com/twbs/bootstrap/blob/master/LICENSE)
- [jQuery](https://github.com/jquery/jquery)
Copyright 2005, 2014 jQuery Foundation and other contributors; [MIT License](https://github.com/jquery/jquery/blob/master/LICENSE.txt)
- [DataTables](https://github.com/DataTables/DataTables)
Copyright (C) 2008-2014, SpryMedia Ltd.; [MIT License](https://www.datatables.net/license/MIT-LICENCE)
- [Leaflet.EasyButton](https://github.com/CliffCloud/Leaflet.EasyButton)
Copyright (C) 2014 Daniel Montague; [MIT License](https://github.com/CliffCloud/Leaflet.EasyButton/blob/master/LICENSE)
- [Bootbox](https://github.com/makeusabrew/bootbox)
Copyright (C) 2011-2014 by Nick Payne; [MIT License](https://github.com/makeusabrew/bootbox/blob/master/LICENSE.md)
- [bootstrap-slider](https://github.com/seiyria/bootstrap-slider)
Copyright (c) 2015 Kyle Kemp, Rohit Kalkur, and contributors; [MIT License](https://github.com/seiyria/bootstrap-slider/blob/master/LICENSE.md)
- [Leaflet.RestoreView](https://github.com/makinacorpus/Leaflet.RestoreView)
Copyright (c) 2012 Makina Corpus, [MIT License](https://github.com/makinacorpus/Leaflet.RestoreView/blob/master/LICENSE)
- [Leaflet.Locate](https://github.com/domoritz/leaflet-locatecontrol)
Copyright (c) 2014 Dominik Moritz, [MIT License](https://github.com/domoritz/leaflet-locatecontrol/blob/gh-pages/LICENSE)
- [Font Awesome](http://fontawesome.io/license/)
by Dave Gandy; [SIL OFL 1.1](https://scripts.sil.org/OFL) (Font), MIT License (Code), CC BY 3.0 (Documentation)
- [url-search-params](https://github.com/WebReflection/url-search-params)
Copyright (C) 2015-2017 Andrea Giammarchi - @WebReflection; [MIT License](https://github.com/WebReflection/url-search-params/blob/master/LICENSE.txt)
- [bootstrap-select](https://github.com/snapappointments/bootstrap-select)
Copyright (c) 2012-2018 SnapAppointments, LLC; [MIT License](https://github.com/snapappointments/bootstrap-select/blob/v1.13.0-dev/LICENSE)
- [leaflet-sidebar-v2](https://github.com/nickpeihl/leaflet-sidebar-v2)
Copyright (c) 2013 Tobias Bieniek; [MIT License](https://github.com/nickpeihl/leaflet-sidebar-v2/blob/master/LICENSE)
- [CodeMirror](https://github.com/codemirror/CodeMirror)
Copyright (C) 2017 by Marijn Haverbeke <marijnh@gmail.com> and others; [MIT License](https://github.com/codemirror/CodeMirror/blob/master/LICENSE)
- [Map BBCode](https://github.com/MapBBCode/mapbbcode)
Ilya Zverev; [Do What The F\*ck You Want To Public License](https://github.com/MapBBCode/mapbbcode/blob/master/LICENSE)
- [Leafet.StravaSegments](https://gitlab.com/bagage/leaflet.stravasegments)
Copyright (c) 2018 Gautier Pelloux-Prayer; [MIT License](https://gitlab.com/bagage/leaflet.stravasegments/blob/master/LICENSE)
- [polyline](https://github.com/mapbox/polyline)
Copyright (c), Development Seed; [BSD 3-Clause License](https://github.com/mapbox/polyline/blob/master/LICENSE)
- [leaflet-fullHash](https://github.com/KoGor/leaflet-fullHash)
Copyright (c) 2014 KoGor; [MIT License](https://github.com/KoGor/leaflet-fullHash/blob/master/LICENSE)
- [Turf.js](https://github.com/Turfjs/turf)
Copyright (c) 2019 Morgan Herlocker; [MIT License](https://github.com/Turfjs/turf/blob/master/LICENSE)
- [i18next](https://github.com/i18next/i18next), [i18next-browser-languageDetector](https://github.com/i18next/i18next-browser-languageDetector), [i18next-xhr-backend](https://github.com/i18next/i18next-xhr-backend), [jquery-i18next](https://github.com/i18next/jquery-i18next/blob/master/LICENSE)
Copyright (c) 2017 i18next; [MIT License](https://github.com/i18next/i18next/blob/master/LICENSE)
- [Leaflet TriangleMarker](https://github.com/themeler/leaflet-triangle-marker)
Copyright (c) 2018 Przemysław Melnarowicz; [MIT License](https://github.com/themeler/leaflet-triangle-marker/blob/master/LICENSE)
- [jsTree ](https://github.com/vakata/jstree)
Copyright (c) 2014 Ivan Bozhanov; [MIT License](https://github.com/vakata/jstree/blob/master/LICENSE-MIT)
- [Leaflet.snogylop](https://github.com/ebrelsford/leaflet.snogylop)
Copyright (c) 2014 Eric Brelsford; [MIT License](https://github.com/ebrelsford/Leaflet.snogylop/blob/master/LICENSE)
- [JOSM maps](https://josm.openstreetmap.de/wiki/Maps)
imagery database is licensed under [CC-BY-SA](https://creativecommons.org/licenses/by-sa/3.0/)
- [LayersCollection](https://github.com/Edward17/LayersCollection/tree/gh-pages)
Copyright (c) 2016 Eduard &lt;edward17&gt;; [MIT License](https://github.com/Edward17/LayersCollection/blob/gh-pages/LICENSE.md)
- [Leaflet-providers](https://github.com/leaflet-extras/leaflet-providers)
Copyright (c) 2013 Leaflet Providers contributors All rights reserved.; [2-clause BSD License](https://github.com/leaflet-extras/leaflet-providers/blob/master/license.md)
- [Fetch polyfill](https://github.com/Github/fetch)
Copyright (c) 2014-2016 GitHub, Inc.; [MIT License](https://github.com/github/fetch/blob/master/LICENSE)
- [Promise Polyfill](https://github.com/taylorhakes/promise-polyfill)
Copyright (c) 2014 Taylor Hakes, Copyright (c) 2014 Forbes Lindesay; [MIT License](https://github.com/taylorhakes/promise-polyfill/blob/master/LICENSE)

View file

@ -1,117 +0,0 @@
{
"name": "brouter-web",
"version": "0.6.3",
"main": [
"dist/brouter-web.css",
"dist/brouter-web.js"
],
"ignore": [
"**/.*",
"bower_components"
],
"dependencies": {
"leaflet": "^1.0.3",
"leaflet-plugins": "~3.0.0",
"leaflet-routing": "nrenner/leaflet-routing#leaflet-1.0",
"async": "~0.9.2",
"d3": "~3.5.5",
"leaflet.draw": "~0.4.9",
"bootstrap": "4.0.0-alpha.5",
"DataTables": "~1.10.13",
"leaflet.elevation": "MrMufflon/Leaflet.Elevation#master",
"leaflet-control-geocoder": "1.5.3",
"Leaflet.EasyButton": "*",
"bootbox": "~4.4.0",
"seiyria-bootstrap-slider": "^9.8.1",
"url-search-params": "~0.5.0",
"Leaflet.RestoreView": "makinacorpus/Leaflet.RestoreView#master",
"leaflet.locatecontrol": "^0.60.0",
"font-awesome": "^4.7.0",
"bootstrap-select": "hugdx/bootstrap-select#patch-1",
"leaflet-sidebar": "^0.1.9",
"autosize": "^3.0.20"
},
"overrides": {
"leaflet": {
"main": [
"dist/leaflet-src.js",
"dist/leaflet.css",
"dist/images/*.png"
]
},
"leaflet-plugins": {
"main": [
"layer/tile/Bing.js"
]
},
"leaflet-routing": {
"main": [
"src/utils/LineUtil.Snapping.js",
"src/utils/Marker.Snapping.js",
"src/L.Routing.js",
"src/L.Routing.Draw.js",
"src/L.Routing.Edit.js"
]
},
"leaflet.draw": {
"main": [
"dist/leaflet.draw-src.js",
"dist/leaflet.draw.css",
"dist/images/*.png",
"dist/images/*.svg"
],
"dependencies": null
},
"bootstrap-select": {
"main": [
"js/bootstrap-select.js",
"dist/css/bootstrap-select.css"
]
},
"bootstrap": {
"main": [
"dist/js/bootstrap.js",
"dist/css/bootstrap.css"
]
},
"leaflet.elevation": {
"main": [
"dist/leaflet.elevation-0.0.4.src.js",
"dist/leaflet.elevation-0.0.4.css",
"dist/images/*.png"
],
"dependencies": null
},
"leaflet-control-geocoder": {
"main": [
"dist/Control.Geocoder.js",
"dist/Control.Geocoder.css",
"images/*.+(png|gif)"
]
},
"url-search-params": {
"main": "build/url-search-params.js"
},
"Leaflet.RestoreView": {
"main": "leaflet.restoreview.js"
},
"font-awesome": {
"main": [
"css/font-awesome.css",
"fonts/*"
]
},
"autosize": {
"main": "dist/autosize.js"
},
"seiyria-bootstrap-slider": {
"dependencies": {
"jquery": "*",
"bootstrap": "*"
}
}
},
"resolutions": {
"leaflet": "^1.0.3"
}
}

View file

@ -1,5 +1,4 @@
(function() {
var hostname = window.location.hostname;
var params = new URLSearchParams(window.location.search.slice(1));
@ -13,13 +12,13 @@
//BR.conf.transit = params.has('transit') && (params.get('transit') === 'true');
if (hostname === 'brouter.de' ) {
// online service (brouter.de) configuration
BR.conf.profiles = [
'trekking',
'fastbike',
'car-test',
'car-eco',
'car-fast',
'safety',
'shortest',
'trekking-ignore-cr',
@ -38,17 +37,20 @@
BR.conf.host = 'http://brouter.de:443';
BR.conf.profilesUrl = 'http://brouter.de/brouter/profiles2/';
} else {
// desktop configuration
BR.conf.profiles = [
'trekking',
'fastbike',
'car-eco',
'car-fast',
'shortest',
'moped',
'car-test'
'vm-forum-liegerad-schnell',
'vm-forum-velomobil-schnell',
'fastbike-lowtraffic',
'fastbike-asia-pacific'
];
BR.conf.host = 'http://0.0.0.0:17777';
@ -73,6 +75,9 @@
//'Mapsforge Tile Server': 'http://localhost:6090/{z}/{x}/{y}.png'
};
// Base layer to show on start, as position number in the layer switcher, starting from 0, default is first
BR.conf.defaultBaseLayerIndex = 0;
// Initial route line transparency (0-1, overridden by stored slider setting)
BR.conf.defaultOpacity = 0.67;
@ -105,7 +110,6 @@
// transit (intermodal routing) demo config
if (BR.conf.transit) {
BR.conf.profiles = [
'../im/bike',
'../im/foot',
@ -117,6 +121,8 @@
'moped',
'car-test'
];
}
// regex needs to be in sync with server, see ServerHandler.getTrackName()
BR.conf.tracknameAllowedChars = 'a-zA-Z0-9 \\._\\-';
})();

View file

@ -1,5 +1,5 @@
div.line-mouse-marker {
background-color: white;
border: 4px solid magenta;
border-radius: 8px;
background-color: white;
border: 4px solid magenta;
border-radius: 8px;
}

View file

@ -1,48 +1,83 @@
html, body, #map {
html,
body,
#map {
height: 100%;
}
/* This is important so that bootstrap-select list goes over leaflet buttons
Since the list is in navbar and leaflet buttons within map, we create a
stacking context on their common ancestor */
body, #content {
body,
#content {
position: relative;
z-index: 1;
}
.flexcolumn {
display: -webkit-box;
display: -moz-box;
display: -ms-flexbox;
display: -webkit-flex;
#content {
overflow: hidden;
}
.flexcolumn,
.leaflet-sidebar-pane.active,
.dataTables_wrapper,
.dataTables_scroll,
.dataTables_scrollBody {
display: flex;
flex-flow: column;
}
.flexrow {
display: -webkit-box;
display: -moz-box;
display: -ms-flexbox;
display: -webkit-flex;
display: flex;
flex-direction: row;
}
#content {
-webkit-box-flex: 1;
-moz-box-flex: 1;
-webkit-flex: 1;
-ms-flex: 1;
.flexgrow {
flex: 1;
}
/* 'auto' (1 1 auto) instead of '1' (1 1 0) needed for DataTables tab in Firefox */
.dataTables_wrapper,
.dataTables_scroll,
.dataTables_scrollBody,
table.dataTable {
flex: auto;
}
/* wrap toolbar controls */
.leaflet-top.leaflet-left {
bottom: 0;
margin-bottom: 10px;
display: flex;
flex-direction: column;
align-items: flex-start;
flex-wrap: wrap;
}
/* bottom underneath top controls when overlapping */
.leaflet-bottom {
z-index: 999;
}
.navbar {
/* align with leaflet-control */
padding: 0.5rem 10px;
}
.nav-link .fa {
margin-right: 0.2em;
}
.navbar-dark .navbar-toggler {
background-image: none;
}
/* disabled style for "Custom" option, but not for selected items */
.bootstrap-select.btn-group .dropdown-menu li.disabled:not(.selected) a {
color: #777;
}
footer {
-webkit-box-flex: none;
-moz-box-flex: none;
-webkit-flex: none;
-ms-flex: none;
flex: none;
background-color: #f7f7f9;
@ -52,39 +87,76 @@ footer {
flex-grow: 1;
margin: 0;
padding: 0;
text-align: center;
}
#stats li {
margin: 0 1rem;
display: inline-block;
}
@media (max-width: 767px) {
#stats li {
margin: 0 0.5rem;
}
#stats {
padding-top: 0.4em;
}
p.stats-label {
margin-bottom: 0.4em;
}
}
#elevation-chart {
height: 175px;
font-size: 80%;
.stats-label {
word-break: break-all;
font-weight: bold;
}
.stats-label abbr[title],
#distance {
text-decoration: none;
border-bottom: 1px dotted #818a91;
}
.form-group {
margin-bottom: 0
margin-bottom: 0;
}
.bootstrap-select.btn-group:not(.input-group-btn),
.bootstrap-select.btn-group[class*="col-"] {
margin-left: 10px;
input#trackname:invalid,
input#trackname:focus:invalid {
border-color: orange;
}
:not(output):-moz-ui-invalid:not(:focus),
:not(output):-moz-ui-invalid:-moz-focusring:not(:focus) {
box-shadow: none;
}
.validation-warning {
color: orange;
font-size: small;
}
.axis path,
.axis line {
stroke: #818a91;
fill: none;
/*
* elevation diagram
*/
.steelblue-theme.leaflet-control.elevation .background {
display: block;
font-size: 80%;
border-radius: 0;
}
/* diagram colors lighter, less dominant */
.steelblue-theme.leaflet-control.elevation .background {
background-color: rgba(105, 155, 196, 0.2);
}
.steelblue-theme.leaflet-control.elevation .axis path,
.steelblue-theme.leaflet-control.elevation .axis line {
stroke: #8398aa;
shape-rendering: crispEdges;
}
.axis text {
fill: #818a91;
.steelblue-theme.leaflet-control.elevation .axis text {
fill: #8398aa;
}
.area {
fill: rgba(129, 138, 145, 0.45);
.steelblue-theme.leaflet-control.elevation .area {
fill: #699bc4;
}
#elevation-btn {
@ -96,24 +168,48 @@ footer {
cursor: crosshair;
}
#map {
/* center error message horizontally */
display: flex;
justify-content: center;
/* map click/drag selects text in controls in Firefox because of display flex */
-moz-user-select: none;
}
.leaflet-control-container,
#message .alert {
-moz-user-select: text;
}
#message {
position: absolute;
left: 446px; /* 400 + 10 + 26 + 10 */
top: 0px;
margin-top: 10px;
margin: 10px 46px; /* 10 + 26 + 10 */
z-index: 3000;
font-size: 1rem;
cursor: auto;
}
#profile_buttons {
padding-top: 4px;
}
/* track messages (data tab) */
#tab_data, #profile_upload {
#tab_data,
.CodeMirror {
font-size: x-small;
}
/* transit demo */
#itinerary pre {
font-size: small;
/* turn off bootstrap 'break-word' */
word-wrap: normal;
margin: 0;
}
/* dashed line animation, derived from Chris Coyier and others
https://css-tricks.com/svg-line-animation-works/
*/
.loading-trailer {
-webkit-animation: dash 0.4s linear infinite;
animation: dash 0.4s linear infinite;
}
@-webkit-keyframes dash {
@ -140,12 +236,19 @@ https://css-tricks.com/svg-line-animation-works/
.control-slider {
background: #fff;
border-radius: 10px;
padding-top: 10px;
padding-bottom: 10px;
}
.slider#overlay {
display: block;
}
.slider.slider-vertical {
height: 80px;
margin: 10px 0;
}
.slider.slider-horizontal {
width: 180px;
margin: 0 10px;
}
/* invert track and selection styles to get partial gradient for "selection" */
@ -153,16 +256,22 @@ https://css-tricks.com/svg-line-animation-works/
width: 8px;
margin-left: 1px;
background-image: linear-gradient(to right, #f0f0f0 0%, #e9e9e9 100%);
box-shadow: inset -1px -0px 1px rgba(55, 55, 55, 0.3), inset 1px 0px 1px rgba(230, 230, 230, 1);
box-shadow: inset -1px -0px 1px rgba(55, 55, 55, 0.3),
inset 1px 0px 1px rgba(230, 230, 230, 1);
}
.slider.slider-horizontal .slider-track {
background-image: linear-gradient(to bottom, #f0f0f0 0%, #e9e9e9 100%);
box-shadow: inset -0px -1px 1px rgba(55, 55, 55, 0.3),
inset 0px 1px 1px rgba(230, 230, 230, 1);
}
.control-slider:hover .slider-track,
.control-slider:active .slider-track {
.control-slider:hover #route .slider-track,
.control-slider:active #route .slider-track {
background-image: linear-gradient(to bottom, magenta 0%, white 100%);
}
.slider-selection {
background-color: #C6C6C6;
background-color: #c6c6c6;
background-image: none;
box-shadow: inset 1px 1px 3px rgba(0, 0, 0, 0.6);
}
@ -170,27 +279,56 @@ https://css-tricks.com/svg-line-animation-works/
.slider.slider-vertical .slider-tick,
.slider.slider-vertical .slider-handle {
cursor: ns-resize;
}
.slider .slider-tick,
.slider .slider-handle {
cursor: ew-resize;
box-sizing: border-box;
background: none;
outline: none;
/* bootstrap .btn-default */
background-image: linear-gradient(to bottom,#fff 0,#e0e0e0 100%);
/* bootstrap .btn-secondary */
background-image: linear-gradient(to bottom, #fff 0, #e0e0e0 100%);
background-repeat: repeat-x;
box-shadow: inset 0 1px 0 rgba(255,255,255,.15),0 1px 1px rgba(0,0,0,.075);
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.15),
0 1px 1px rgba(0, 0, 0, 0.075);
border: 1px solid #adadad;
}
/* activated Font Awesome icon and Bootstrap button
(for EasyButton add ' active' to icon class property) */
.fa.active, .btn.active, .btn.active:hover, .btn.active:focus {
/* activated Font Awesome icon (for EasyButton add ' active' to icon class property) */
.fa.active {
/* use same color as leaflet-locatecontrol */
color: #2074B6;
color: #2074b6;
}
/* Bootstrap button */
.btn-secondary:not(:disabled):not(.disabled).active,
.btn-secondary.active,
.btn-secondary:hover,
.btn-outline-secondary:not(:disabled):not(.disabled).active,
.btn-outline-secondary:hover {
/* use same color as leaflet-locatecontrol */
color: #2074b6;
background-color: #e6e6e6;
}
.stats-label {
word-break: break-all;
font-weight: bold;
button.btn {
box-shadow: none !important;
}
.leaflet-bar button {
/* override button for horizontal fa center in Firefox */
padding: 0;
}
.leaflet-bar .fa {
font-size: 14px;
/* override fa with Leaflet button height for vertical center */
line-height: 26px;
}
.leaflet-touch .leaflet-bar .fa {
font-size: 16px;
line-height: 30px;
}
/* smaller tab height */
@ -198,10 +336,29 @@ https://css-tricks.com/svg-line-animation-works/
padding: 2px 15px;
}
#profile_message .alert {
margin-top: 3px;
}
.alert {
margin-bottom: 0px;
padding-left: 35px;
}
.alert span.fa {
position: relative;
left: -23px;
margin-right: -1em;
}
/*
* DataTables
*/
table.dataTable {
/* avoid getting centered and header misaligned with flex row (sidebar) */
margin: 0;
}
table.dataTable.mini thead th,
table.dataTable.mini thead td {
padding: 3px 13px 3px 2px;
@ -224,5 +381,147 @@ table.dataTable.hover tbody tr.even:hover,
table.dataTable.display tbody tr:hover,
table.dataTable.display tbody tr.odd:hover,
table.dataTable.display tbody tr.even:hover {
background-color: rgba(255,255,0,0.2);
background-color: rgba(255, 255, 0, 0.2);
}
/*
* No-go areas
*/
.nogo-delete-marker {
text-align: center;
line-height: 15px;
font-size: 11px;
border-radius: 8px;
}
.leaflet-touch .nogo-delete-marker {
border-radius: 12px;
line-height: 24px;
}
/* tooltip */
.editing-tooltip,
.editing-tooltip-create {
color: #fff;
background-color: rgba(0, 0, 0, 0.7);
/* for direction arrows that inherit */
border-color: rgba(0, 0, 0, 0.7);
/* no border but still set a color for direction arrows */
border-width: 0px;
}
.editing-tooltip-create {
/* no (invisible) direction arrow for cursor tooltip */
border-color: transparent;
}
.leaflet-tooltip-bottom:before {
border-bottom-color: inherit;
}
.leaflet-tooltip-right:before {
border-right-color: inherit;
}
/*
* leaflet-sidebar-v2
*/
.leaflet-sidebar-pane#tab_profile,
.leaflet-sidebar-pane#tab_data,
.leaflet-sidebar-pane#tab_itinerary {
/* Full height for content with inner scrolling,
overrides scroll fix for long content in Firefox */
height: 100%;
}
.leaflet-sidebar-content {
/* for optional-layers-tree */
overflow-x: auto;
}
/* layers control as sidebar tab */
#tab_layers_control {
font-size: 0.9rem;
line-height: normal;
}
#layers-control-wrapper label {
/* override Bootstrap/Reboot label */
display: block;
margin-bottom: 0px;
/* normalize label height
| Firefox | Chrome |
Leaflet only | 21 | 20 |
Bootstrap | 23 | 22 |*/
height: 23px;
}
#layers-button-group {
display: flex;
justify-content: space-between;
}
#tree-button-group {
margin-right: 5px;
}
#optional-layers-tree {
margin-top: 5px;
}
.tree-code {
font-family: Menlo, Consolas, Monaco, Liberation Mono, Lucida Console,
monospace;
margin-right: 7px;
color: #666; /* like root nodes, jstree-disabled */
}
/* hide currently unused bottom tabs container because of touch border artefacts */
.leaflet-sidebar-tabs > ul:last-child {
display: none;
}
/* layers svg icon not properly centered */
.leaflet-sidebar-tabs > ul > li > a[href='#tab_layers_control'] {
display: flex;
align-items: center;
justify-content: center;
}
/*
* CodeMirror
*/
.CodeMirror {
/* auto instead of 1 to avoid overflow to content height in Firefox */
flex: auto;
/* override default 300px, behaves like min-height with flex auto */
height: 0;
border: 1px solid #ddd;
font-family: Menlo, Consolas, Monaco, Liberation Mono, Lucida Console,
monospace;
line-height: 1.2em;
}
#loadNogos fieldset {
display: block;
margin-left: 2px;
margin-right: 2px;
padding-top: 0.35em;
padding-bottom: 0.625em;
padding-left: 0.75em;
padding-right: 0.75em;
border: 2px groove;
}
#loadNogos legend {
display: block;
padding-left: 2px;
padding-right: 2px;
border: none;
}
.nav-link.disabled {
/* by default, even if disabled, modals are opened by disabled nav-link
so we ignore pointer events in this situation to avoid that*/
pointer-events: none;
}

View file

@ -1,13 +1,12 @@
var gulp = require('gulp');
var concat = require('gulp-concat');
var concatCss = require('gulp-concat-css');
var minifyCss = require('gulp-minify-css');
var postcss = require('gulp-postcss');
var autoprefixer = require('autoprefixer');
var uglify = require('gulp-uglify');
var sourcemaps = require('gulp-sourcemaps');
var gulpDebug = require('gulp-debug');
var mainBowerFiles = require('main-bower-files');
var mainNpmFiles = require('npmfiles');
var del = require('del');
var tap = require('gulp-tap');
var path = require('path');
var cached = require('gulp-cached');
var remember = require('gulp-remember');
@ -20,219 +19,376 @@ var semver = require('semver');
var git = require('gulp-git');
var replace = require('gulp-replace');
var release = require('gulp-github-release');
var cleanCSS = require('gulp-clean-css');
var modifyCssUrls = require('gulp-modify-css-urls');
var sort = require('gulp-sort');
var scanner = require('i18next-scanner');
var jsonConcat = require('gulp-json-concat');
var rename = require('gulp-rename');
var debug = false;
var paths = {
// see overrides in bower.json
scriptsConfig: mainBowerFiles('**/url-search-params/**/*.js'),
scripts: mainBowerFiles([
'**/*.js',
'!**/*.min.js',
'!**/url-search-params/**/*.js'
]).concat([
'js/Browser.js',
'js/Util.js',
'js/Map.js',
'js/router/BRouter.js',
'js/plugin/*.js',
'js/control/*.js',
'js/index.js'
]),
styles: mainBowerFiles('**/*.css').concat('css/*.css'),
images: mainBowerFiles('**/*.+(png|gif|svg)'),
fonts: mainBowerFiles('**/font-awesome/fonts/*'),
dest: 'dist',
destName: 'brouter-web'
// see overrides in package.json
scriptsConfig: mainNpmFiles().filter(f =>
RegExp('url-search-params/.*\\.js', 'i').test(f)
),
scripts: [
'node_modules/jquery/dist/jquery.js',
'node_modules/tether/dist/js/tether.js',
'node_modules/async/lib/async.js',
'node_modules/leaflet/dist/leaflet-src.js'
]
.concat(
mainNpmFiles().filter(
f =>
RegExp('.*\\.js', 'i').test(f) &&
!RegExp('.*\\.min\\.js', 'i').test(f) &&
!RegExp('url-search-params/.*\\.js', 'i').test(f)
)
)
.concat([
'js/Browser.js',
'js/Util.js',
'js/Map.js',
'js/LayersConfig.js',
'js/router/BRouter.js',
'js/plugin/*.js',
'js/control/*.js',
'js/index.js'
]),
styles: mainNpmFiles()
.filter(
f =>
RegExp('.*\\.css', 'i').test(f) &&
!RegExp('.*\\.min\\.css', 'i').test(f)
)
.concat('css/*.css'),
images: mainNpmFiles().filter(f =>
RegExp('.*.+(png|gif|svg)', 'i').test(f)
),
fonts: mainNpmFiles().filter(f =>
RegExp('font-awesome/fonts/.*', 'i').test(f)
),
locales: 'locales/*.json',
layers: 'layers/**/*.geojson',
layersDestName: 'layers.js',
layersConfig: [
'layers/config/config.js',
'layers/config/tree.js',
'layers/config/overrides.js',
'layers/config/geometry.js'
],
layersConfigDestName: 'layersConf.js',
dest: 'dist',
destName: 'brouter-web'
};
gulp.task('clean', function(cb) {
del(paths.dest + '/**/*', cb);
});
// libs that require loading before config.js
gulp.task('scripts_config', ['clean'], function() {
// just copy for now
return gulp.src(paths.scriptsConfig)
.pipe(gulp.dest(paths.dest));
gulp.task('scripts_config', function() {
// just copy for now
return gulp.src(paths.scriptsConfig).pipe(gulp.dest(paths.dest));
});
gulp.task('scripts', function() {
if (debug)
gutil.log( gutil.colors.yellow('Running in Debug mode') );
else
gutil.log( gutil.colors.green('Running in Release mode') );
if (debug) gutil.log(gutil.colors.yellow('Running in Debug mode'));
else gutil.log(gutil.colors.green('Running in Release mode'));
return gulp.src(paths.scripts, { base: '.' })
.pipe(sourcemaps.init())
.pipe(cached('scripts'))
.pipe(gulpif(!debug, uglify()))
.pipe(remember('scripts'))
.pipe(concat(paths.destName + '.js'))
.pipe(sourcemaps.write('.'))
.pipe(gulp.dest(paths.dest));
return gulp
.src(paths.scripts, { base: '.' })
.pipe(sourcemaps.init())
.pipe(cached('scripts'))
.pipe(gulpif(!debug, uglify()))
.pipe(remember('scripts'))
.pipe(concat(paths.destName + '.js'))
.pipe(sourcemaps.write('.'))
.pipe(gulp.dest(paths.dest));
});
// separate, fallback task for debugging (switch manually in index.html)
gulp.task('concat', function() {
return gulp.src(paths.scripts)
.pipe(concat(paths.destName + '.src.js'))
.pipe(gulp.dest(paths.dest));
return gulp
.src(paths.scripts)
.pipe(concat(paths.destName + '.src.js'))
.pipe(gulp.dest(paths.dest));
});
gulp.task('styles', function() {
return gulp.src(paths.styles)
// hack for rewriting relative URLs to images/fonts in gulp-concat-css
// when src in css subfolder (remove '../')
// see also (?) https://github.com/mariocasciaro/gulp-concat-css/pull/10
.pipe(tap(function (file) {
if (path.basename(file.base) === 'css') {
file.path = 'css/' + file.relative;
file.base = './css';
} else {
file.path = file.relative;
file.base = '.';
}
}))
.pipe(concatCss(paths.destName + '.css'))
.pipe(minifyCss({
rebase: false
}))
.pipe(gulp.dest(paths.dest));
return gulp
.src(paths.styles)
.pipe(
modifyCssUrls({
modify(url, filePath) {
var distUrl = url;
var imageExt = ['.png', '.gif', '.svg'];
if (imageExt.indexOf(path.extname(url)) !== -1) {
distUrl = 'images/' + path.basename(url);
} else if (url.indexOf('font') !== -1) {
distUrl = 'fonts/' + path.basename(url);
}
return distUrl;
}
})
)
.pipe(concat(paths.destName + '.css'))
.pipe(
cleanCSS({
rebase: false
})
)
.pipe(postcss([autoprefixer({ remove: false })]))
.pipe(gulp.dest(paths.dest));
});
gulp.task('images', ['clean'], function() {
return gulp.src(paths.images)
.pipe(gulp.dest(paths.dest + '/images'));
gulp.task('images', function() {
return gulp.src(paths.images).pipe(gulp.dest(paths.dest + '/images'));
});
gulp.task('fonts', ['clean'], function() {
return gulp.src(paths.fonts)
.pipe(gulp.dest(paths.dest + '/fonts'));
gulp.task('fonts', function() {
return gulp.src(paths.fonts).pipe(gulp.dest(paths.dest + '/fonts'));
});
gulp.task('clean', function(cb) {
del(paths.dest + '/**/*' , cb);
gulp.task('locales', function() {
return gulp.src(paths.locales).pipe(gulp.dest(paths.dest + '/locales'));
});
gulp.task('watch', function() {
debug = true;
var watcher = gulp.watch(paths.scripts, ['scripts']);
watcher.on('change', function (event) {
if (event.type === 'deleted') {
delete cached.caches.scripts[event.path];
remember.forget('scripts', event.path);
}
});
gulp.watch(paths.styles, ['styles']);
debug = true;
var watcher = gulp.watch(paths.scripts, gulp.series('scripts'));
watcher.on('change', function(event) {
if (event.type === 'deleted') {
delete cached.caches.scripts[event.path];
remember.forget('scripts', event.path);
}
});
gulp.watch(paths.styles, gulp.series('styles'));
gulp.watch(paths.layersConfig, gulp.series('layers_config'));
});
// Print paths to console, for manually debugging the gulp build
// (comment out corresponding line of paths to print)
gulp.task('log', function() {
//return gulp.src(mainBowerFiles(['**/*.js', '!**/*.min.js']))
//return gulp.src(mainBowerFiles('**/*.css'))
return gulp.src(paths.scripts)
//return gulp.src(paths.styles)
//return gulp.src(paths.images)
.pipe(gulpDebug());
//return gulp.src(mainBowerFiles({debugging: true}));
//return gulp.src(paths.scripts)
//return gulp.src(paths.styles)
//return gulp.src(paths.images)
// return gulp.src(paths.locales)
return gulp
.src(
paths.scripts
.concat(paths.styles)
.concat(paths.images)
.concat(paths.locales)
)
.pipe(gulpDebug());
});
gulp.task('inject', function () {
var target = gulp.src('index.html');
var sources = gulp.src(paths.scripts, { base: '.', read: false });
gulp.task('inject', function() {
var target = gulp.src('index.html');
var sources = gulp.src(paths.scripts.concat(paths.styles), {
base: '.',
read: false
});
return target.pipe(inject(sources, { relative: true }))
.pipe(gulp.dest('.'));
});
gulp.task('default', ['clean', 'scripts_config', 'scripts', 'styles', 'images', 'fonts']);
gulp.task('debug', function() {
debug = true;
gulp.start('default');
return target
.pipe(inject(sources, { relative: true }))
.pipe(gulp.dest('.'));
});
var pkg = require('./package.json');
var tags = {patch: 'patch', minor: 'minor', major: 'major'};
var tags = { patch: 'patch', minor: 'minor', major: 'major' };
var nextVersion;
var ghToken;
gulp.task('release:init', function() {
var tag = gutil.env.tag;
if (!tag) {
gutil.log(gutil.colors.red('--tag is required'));
process.exit(1);
}
if (['major', 'minor', 'patch'].indexOf(tag) < 0) {
gutil.log(gutil.colors.red('--tag must be major, minor or patch'));
process.exit(2);
}
ghToken = gutil.env.token;
if (!ghToken) {
gutil.log(gutil.colors.red('--token is required (github personnal access token)'));
process.exit(3);
}
if (ghToken.length != 40) {
gutil.log(gutil.colors.red('--token length must be 40'));
process.exit(4);
}
git.status({args: '--porcelain', quiet: true}, function(err, stdout) {
if (err) throw err;
if (stdout.length > 0) {
gutil.log(gutil.colors.red('Repository is not clean. Please commit or stash your pending modification'));
process.exit(5);
gulp.task('release:init', function(cb) {
var tag = gutil.env.tag;
if (!tag) {
return cb(new Error('--tag is required'));
}
});
nextVersion = semver.inc(pkg.version, tag);
return;
if (['major', 'minor', 'patch'].indexOf(tag) < 0) {
return cb(new Error('--tag must be major, minor or patch'));
}
ghToken = gutil.env.token;
if (!ghToken) {
return cb(
new Error('--token is required (github personnal access token')
);
}
if (ghToken.length != 40) {
return cb(new Error('--token length must be 40'));
}
nextVersion = semver.inc(pkg.version, tag);
git.status({ args: '--porcelain', quiet: true }, function(err, stdout) {
if (err) return cb(err);
if (stdout.length > 0) {
return cb(
new Error(
'Repository is not clean. Please commit or stash your pending modification'
)
);
}
cb();
});
});
gulp.task('bump', ['bump:json', 'bump:html']);
gulp.task('bump:json', ['release:init'], function() {
gutil.log(gutil.colors.green('Bump to '+nextVersion));
return(gulp.src(['./package.json', './bower.json'])
.pipe(bump({version: nextVersion}))
.pipe(gulp.dest('./')));
gulp.task('bump:json', function() {
gutil.log(gutil.colors.green('Bump to ' + nextVersion));
return gulp
.src(['./package.json'])
.pipe(bump({ version: nextVersion }))
.pipe(gulp.dest('./'));
});
gulp.task('bump:html', ['release:init'], function() {
return(gulp.src('./index.html')
.pipe(replace(/<sup class="version">(.*)<\/sup>/, '<sup class="version">'+nextVersion+'</sup>'))
.pipe(gulp.dest('.')));
gulp.task('bump:html', function() {
return gulp
.src('./index.html')
.pipe(
replace(
/<sup class="version">(.*)<\/sup>/,
'<sup class="version">' + nextVersion + '</sup>'
)
)
.pipe(gulp.dest('.'));
});
gulp.task('release:commit', ['bump'], function() {
gulp.src(['./index.html', './package.json', './bower.json'])
.pipe(git.commit('release: '+nextVersion));
gulp.task('bump', gulp.series('bump:json', 'bump:html'));
gulp.task('release:commit', function() {
return gulp
.src(['./index.html', './package.json'])
.pipe(git.commit('release: ' + nextVersion));
});
gulp.task('release:tag', ['release:commit'], function() {
return(git.tag(nextVersion, '', function(err) {
if (err) throw err;
}));
gulp.task('release:tag', function(cb) {
return git.tag(nextVersion, '', cb);
});
gulp.task('release:push', ['release:tag'], function() {
git.push('origin', 'master', {args: '--tags'}, function(err) {
if (err) throw err;
});
gulp.task('release:push', function(cb) {
git.push('origin', 'master', { args: '--tags' }, cb);
});
gulp.task('release:zip', ['release:tag', 'default'], function() {
gutil.log(gutil.colors.green('Build brouter-web.'+nextVersion+'.zip'));
return(gulp.src(['dist/**', 'index.html', 'config.template.js', 'keys.template.js'], {'base': '.'})
.pipe(zip('brouter-web.'+nextVersion+'.zip'))
.pipe(gulp.dest('.')));
gulp.task('i18next', function() {
return gulp
.src([
'index.html',
'locales/keys.js',
'layers/config/overrides.js',
'js/**/*.js'
])
.pipe(sort())
.pipe(
scanner({
lngs: ['en'], // we only generate English version, other languages are handled by transifex via yarn transifex-pull/push
removeUnusedKeys: true,
sort: true,
resource: {
// the source path is relative to current working directory
loadPath: 'locales/{{lng}}.json',
// the destination path is relative to your `gulp.dest()` path
savePath: 'locales/{{lng}}.json'
}
})
)
.pipe(gulp.dest('.'));
});
gulp.task('release:publish', ['release:zip'], function() {
gulp.src('./brouter-web.'+nextVersion+'.zip')
.pipe(release({
tag: nextVersion,
token: ghToken,
manifeste: pkg,
}))
gulp.task('layers_config', function() {
return gulp
.src(paths.layersConfig)
.pipe(concat(paths.layersConfigDestName))
.pipe(gulp.dest(paths.dest));
});
gulp.task('release', ['release:init', 'bump', 'release:commit', 'release:tag',
'release:push', 'release:zip', 'release:publish']);
// Bundles layer files. To download and extract run "yarn layers"
gulp.task('layers', function() {
return (
gulp
.src(paths.layers)
// Workaround to get file extension removed from the dictionary key
.pipe(rename({ extname: '.json' }))
.pipe(
jsonConcat(paths.layersDestName, function(data) {
var header =
'// Licensed under the MIT License (https://github.com/nrenner/brouter-web#license + Credits and Licenses),\n' +
'// except JOSM imagery database (dataSource=JOSM) is licensed under Creative Commons (CC-BY-SA),\n' +
'// see https://josm.openstreetmap.de/wiki/Maps#Otherimportantinformation\n';
return Buffer.from(
header +
'BR.layerIndex = ' +
JSON.stringify(data, null, 2) +
';'
);
})
)
.pipe(gulp.dest(paths.dest))
);
});
gulp.task(
'default',
gulp.series(
'clean',
'scripts_config',
'layers_config',
'layers',
'scripts',
'styles',
'images',
'fonts',
'locales'
)
);
gulp.task(
'debug',
gulp.series(function(cb) {
debug = true;
cb();
}, 'default')
);
gulp.task('release:zip', function() {
gutil.log(gutil.colors.green('Build brouter-web.' + nextVersion + '.zip'));
return gulp
.src(
['dist/**', 'index.html', 'config.template.js', 'keys.template.js'],
{
base: '.'
}
)
.pipe(zip('brouter-web.' + nextVersion + '.zip'))
.pipe(gulp.dest('.'));
});
gulp.task('release:publish', function() {
return gulp.src('./brouter-web.' + nextVersion + '.zip').pipe(
release({
tag: nextVersion,
token: ghToken,
manifest: pkg
})
);
});
gulp.task(
'release',
gulp.series(
'release:init',
'bump',
'release:commit',
'release:tag',
'release:push',
'default',
'release:zip',
'release:publish'
)
);

1284
index.html

File diff suppressed because it is too large Load diff

View file

@ -1,24 +1,28 @@
(function () {
var touchScreen = (function () {
(function() {
var touchScreen = (function() {
var result = null;
if ('maxTouchPoints' in navigator) {
result = navigator.maxTouchPoints > 0;
} else if (window.matchMedia && window.matchMedia('(any-pointer:coarse),(any-pointer:fine),(any-pointer:none)').matches) {
result = window.matchMedia("(any-pointer:coarse)").matches;
} else if (
window.matchMedia &&
window.matchMedia(
'(any-pointer:coarse),(any-pointer:fine),(any-pointer:none)'
).matches
) {
result = window.matchMedia('(any-pointer:coarse)').matches;
} else if ('msMaxTouchPoints' in navigator) {
result = navigator.msMaxTouchPoints > 0;
};
}
return result;
}()),
touchScreenDetectable = touchScreen !== null;
})(),
touchScreenDetectable = touchScreen !== null,
touch = touchScreenDetectable ? touchScreen : L.Browser.touch;
BR.Browser = {
touchScreen: touchScreen,
touchScreenDetectable: touchScreenDetectable
touchScreenDetectable: touchScreenDetectable,
touch: touch
};
}());
})();

311
js/LayersConfig.js Normal file
View file

@ -0,0 +1,311 @@
BR.LayersConfig = L.Class.extend({
defaultBaseLayers: BR.confLayers.defaultBaseLayers,
defaultOverlays: BR.confLayers.defaultOverlays,
legacyNameToIdMap: BR.confLayers.legacyNameToIdMap,
initialize: function(map) {
this._map = map;
this._addLeafletProvidersLayers();
this._customizeLayers();
this.loadDefaultLayers();
this._addLanguageDefaultLayer();
},
loadDefaultLayers: function() {
if (BR.Util.localStorageAvailable()) {
var item = localStorage.getItem('map/defaultLayers');
if (item) {
var defaultLayers = JSON.parse(item);
this.defaultBaseLayers = defaultLayers.baseLayers;
this.defaultOverlays = defaultLayers.overlays;
}
}
},
storeDefaultLayers: function(baseLayers, overlays) {
if (BR.Util.localStorageAvailable()) {
var defaultLayers = {
baseLayers: baseLayers,
overlays: overlays
};
localStorage.setItem(
'map/defaultLayers',
JSON.stringify(defaultLayers)
);
}
},
_addLeafletProvidersLayers: function() {
var includeList = BR.confLayers.leafletProvidersIncludeList;
for (var i = 0; i < includeList.length; i++) {
var id = includeList[i];
var obj = {
geometry: null,
properties: {
id: id,
name: id.replace('.', ' '),
dataSource: 'leaflet-providers'
},
type: 'Feature'
};
BR.layerIndex[id] = obj;
}
},
_customizeLayers: function() {
var propertyOverrides = BR.confLayers.getPropertyOverrides();
for (id in propertyOverrides) {
var layer = BR.layerIndex[id];
if (layer) {
var properties = propertyOverrides[id];
for (key in properties) {
var value = properties[key];
layer.properties[key] = value;
}
} else {
console.error('Layer not found: ' + id);
}
}
BR.layerIndex['MtbMap'].geometry = BR.confLayers.europeGeofabrik;
BR.layerIndex['1069'].geometry = BR.confLayers.europeGeofabrik;
BR.layerIndex['OpenStreetMap.CH'].geometry =
BR.confLayers.switzerlandPadded;
BR.layerIndex['1017'].geometry = BR.confLayers.osmapaPl;
},
_addLanguageDefaultLayer: function() {
// language code -> layer id
var languageLayersMap = {};
var i;
for (i = 0; i < BR.confLayers.languageDefaultLayers.length; i++) {
var id = BR.confLayers.languageDefaultLayers[i];
var layer = BR.layerIndex[id];
if (layer) {
var layerLanguage = layer.properties['language_code'];
if (layerLanguage) {
languageLayersMap[layerLanguage] = id;
}
}
}
// iterate language code hierarchy, e.g ["de-DE", "de", "en"] (includes `i18next.options.fallbackLng`)
for (i = 0; i < i18next.languages.length; i++) {
var language = i18next.languages[i];
var layerId = languageLayersMap[language];
if (layerId) {
this.defaultBaseLayers.unshift(layerId);
break;
}
}
},
isDefaultLayer: function(id, overlay) {
var result = false;
if (overlay) {
result = this.defaultOverlays.indexOf(id) > -1;
} else {
result = this.defaultBaseLayers.indexOf(id) > -1;
}
return result;
},
getBaseLayers: function() {
return this._getLayers(this.defaultBaseLayers);
},
getOverlays: function() {
return this._getLayers(this.defaultOverlays);
},
_getLayers: function(ids) {
var layers = {};
for (var i = 0; i < ids.length; i++) {
var layerId = ids[i];
var layerData = BR.layerIndex[layerId];
if (layerData) {
// when key required only add if configured
var keyObj = this.getKeyName(layerData.properties.url);
if (!keyObj || (keyObj && BR.keys[keyObj.name])) {
layers[layerData.properties.name] = this.createLayer(
layerData
);
}
} else {
console.error('Layer not found: ' + layerId);
}
}
return layers;
},
// own convention: key placeholder with prefix
// e.g. ?api_key={keys_openrouteservice}
getKeyName: function(url) {
var result = null;
// L.Util.template only matches [\w_-]
var prefix = 'keys_';
var regex = new RegExp('{' + prefix + '([^}]*)}');
var found, name;
if (!url) return result;
found = url.match(regex);
if (found) {
name = found[1];
result = {
name: name,
urlVar: prefix + name
};
}
return result;
},
createLayer: function(layerData) {
var props = layerData.properties;
var url = props.url;
var layer;
// JOSM: https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png
// Leaflet: https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png
function convertUrlJosm(url) {
var rxSwitch = /{switch:[^}]*}/;
var rxZoom = /{zoom}/g;
var result = url.replace(rxSwitch, '{s}');
result = result.replace(rxZoom, '{z}');
return result;
}
// JOSM: https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png
// Leaflet: ['a','b','c']
function getSubdomains(url) {
var result = 'abc';
var regex = /{switch:([^}]*)}/;
var found = url.match(regex);
if (found) {
result = found[1].split(',');
}
return result;
}
function convertAttributionJosm(props) {
var result = '';
var attr = props.attribution;
if (attr) {
if (attr.html) {
result = attr.html;
} else if (attr.url && attr.text) {
result =
'<a href="' +
attr.url +
'" target="_blank" rel="noopener">' +
attr.text +
'</a>';
} else if (attr.text) {
result = attr.text;
}
}
if (!result) {
console.warn('No attribution: ' + props.id);
}
return result;
}
var options = {
maxZoom: this._map.getMaxZoom(),
bounds:
layerData.geometry && !props.worldTiles
? L.geoJson(layerData.geometry).getBounds()
: null
};
if (props.mapUrl) {
options.mapLink =
'<a target="_blank" href="' +
props.mapUrl +
'">' +
(props.nameShort || props.name) +
'</a>';
}
if (props.attribution) {
options.attribution = props.attribution;
}
var keyObj = this.getKeyName(url);
if (keyObj && BR.keys[keyObj.name]) {
options[keyObj.urlVar] = BR.keys[keyObj.name];
}
if (props.dataSource === 'leaflet-providers') {
layer = L.tileLayer.provider(props.id);
var layerOptions = L.Util.extend(options, {
maxNativeZoom: layer.options.maxZoom
});
L.setOptions(layer, layerOptions);
} else if (props.dataSource === 'LayersCollection') {
layer = L.tileLayer(
url,
L.Util.extend(options, {
minZoom: props.minZoom || 0,
maxNativeZoom: props.maxZoom
})
);
if (props.subdomains) {
layer.subdomains = props.subdomains;
}
} else {
// JOSM
var josmUrl = url;
var url = convertUrlJosm(josmUrl);
var josmOptions = L.Util.extend(options, {
minZoom: props.min_zoom || 0,
maxNativeZoom: props.max_zoom,
subdomains: getSubdomains(josmUrl),
attribution: convertAttributionJosm(props)
});
if (props.type && props.type === 'wms') {
layer = L.tileLayer.wms(
url,
L.Util.extend(josmOptions, {
layers: props.layers,
format: props.format
})
);
} else {
layer = L.tileLayer(url, josmOptions);
}
}
// Layer attribution here only as short link to original site,
// to keep current position use placeholders: {zoom}/{lat}/{lon}
// Copyright attribution in index.html #credits
var getAttribution = function() {
return this.options.mapLink;
};
layer.getAttribution = getAttribution;
layer.id = props.id;
return layer;
}
});
BR.layersConfig = function(map) {
return new BR.LayersConfig(map);
};

186
js/Map.js
View file

@ -1,94 +1,74 @@
BR.Map = {
initMap: function() {
var map,
layersControl;
var map, layersControl;
BR.keys = BR.keys || {};
var maxZoom = 19;
var osm = L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
maxZoom: maxZoom
});
var osmde = L.tileLayer('https://{s}.tile.openstreetmap.de/tiles/osmde/{z}/{x}/{y}.png', {
maxNativeZoom: 18,
maxZoom: maxZoom
});
var topo = L.tileLayer('https://{s}.tile.opentopomap.org/{z}/{x}/{y}.png', {
maxNativeZoom: 17,
maxZoom: maxZoom
});
var thunderforestAttribution = 'tiles &copy; <a target="_blank" href="https://www.thunderforest.com">Thunderforest</a> '
+ '(<a target="_blank" href="https://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA 2.0</a>)';
var thunderforestAuth = BR.keys.thunderforest ? '?apikey=' + BR.keys.thunderforest : '';
var cycle = L.tileLayer('https://{s}.tile.thunderforest.com/cycle/{z}/{x}/{y}.png' + thunderforestAuth, {
maxNativeZoom: 18,
maxZoom: maxZoom
});
var outdoors = L.tileLayer('https://{s}.tile.thunderforest.com/outdoors/{z}/{x}/{y}.png' + thunderforestAuth, {
maxNativeZoom: 18,
maxZoom: maxZoom
});
var esri = L.tileLayer('https://{s}.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}', {
maxNativeZoom: 19,
maxZoom: maxZoom,
subdomains: ['server', 'services'],
attribution: '<a target="_blank" href="http://goto.arcgisonline.com/maps/World_Imagery">World Imagery</a> '
+ '&copy; <a target="_blank" href="https://www.esri.com/">Esri</a>, sources: '
+ 'Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP, and the GIS User Community'
});
var cycling = L.tileLayer('https://tile.waymarkedtrails.org/cycling/{z}/{x}/{y}.png', {
maxNativeZoom: 18,
opacity: 0.7,
maxZoom: maxZoom
});
var hiking = L.tileLayer('https://tile.waymarkedtrails.org/hiking/{z}/{x}/{y}.png', {
maxNativeZoom: 18,
opacity: 0.7,
maxZoom: maxZoom
});
map = new L.Map('map', {
worldCopyJump: true
zoomControl: false, // add it manually so that we can translate it
worldCopyJump: true,
minZoom: 0,
maxZoom: maxZoom
});
L.control
.zoom({
zoomInTitle: i18next.t('map.zoomInTitle'),
zoomOutTitle: i18next.t('map.zoomOutTitle')
})
.addTo(map);
if (!map.restoreView()) {
map.setView([50.99, 9.86], 6);
map.setView([50.99, 9.86], 5);
}
// two attribution lines by adding two controls, prevents ugly wrapping on
// small screens, better separates static from layer-specific attribution
var osmAttribution =
$(map.getContainer()).outerWidth() >= 400
? i18next.t('map.attribution-osm-long')
: i18next.t('map.attribution-osm-short');
map.attributionControl.setPrefix(
'&copy; <a target="_blank" href="https://www.openstreetmap.org/copyright">' +
osmAttribution +
'</a>' +
' &middot; <a href="" data-toggle="modal" data-target="#credits">' +
i18next.t('map.copyright') +
'</a>' +
' &middot; <a target="_blank" href="http://brouter.de/privacypolicy.html">' +
i18next.t('map.privacy') +
'</a>'
);
$('#credits').on('show.bs.modal', function(event) {
BR.Map._renderLayerCredits(layersControl._layers);
});
new L.Control.PermalinkAttribution().addTo(map);
map.attributionControl.setPrefix(false);
map.attributionControl.addAttribution('<a href="" data-toggle="modal" data-target="#credits">Copyright & credits</a>')
var baseLayers = {
'OpenStreetMap': osm,
'OpenStreetMap.de': osmde,
'OpenTopoMap': topo,
'OpenCycleMap (Thunderf.)': cycle,
'Outdoors (Thunderforest)': outdoors,
'Esri World Imagery': esri
};
var overlays = {
'Cycling (Waymarked Trails)': cycling,
'Hiking (Waymarked Trails)': hiking
};
var layersConfig = BR.layersConfig(map);
var baseLayers = layersConfig.getBaseLayers();
var overlays = layersConfig.getOverlays();
if (BR.keys.bing) {
baseLayers['Bing Aerial'] = new BR.BingLayer(BR.keys.bing);
baseLayers[i18next.t('map.layer.bing')] = new BR.BingLayer(
BR.keys.bing
);
}
if (BR.keys.digitalGlobe) {
var recent = new L.tileLayer('https://{s}.tiles.mapbox.com/v4/digitalglobe.nal0g75k/{z}/{x}/{y}.png?access_token=' + BR.keys.digitalGlobe, {
minZoom: 1,
maxZoom: 19,
attribution: '&copy; <a href="https://www.digitalglobe.com/platforms/mapsapi">DigitalGlobe</a> ('
+ '<a href="https://bit.ly/mapsapiview">Terms of Use</a>)'
});
baseLayers['DigitalGlobe Recent Imagery'] = recent;
var recent = new L.tileLayer(
'https://{s}.tiles.mapbox.com/v4/digitalglobe.nal0g75k/{z}/{x}/{y}.png?access_token=' +
BR.keys.digitalGlobe,
{
minZoom: 1,
maxZoom: 19,
attribution:
'&copy; <a href="https://www.digitalglobe.com/platforms/mapsapi">DigitalGlobe</a> (<a href="https://bit.ly/mapsapiview">Terms of Use</a>)'
}
);
baseLayers[i18next.t('map.layer.digitalglobe')] = recent;
}
if (BR.conf.clearBaseLayers) {
@ -105,19 +85,27 @@ BR.Map = {
overlays[i] = L.tileLayer(BR.conf.overlays[i]);
}
}
// after applying custom base layer configurations, add first base layer to map
var firstLayer = baseLayers[Object.keys(baseLayers)[0]];
if (firstLayer) {
map.addLayer(firstLayer);
layersControl = BR.layersTab(layersConfig, baseLayers, overlays).addTo(
map
);
var secureContext =
'isSecureContext' in window
? isSecureContext
: location.protocol === 'https:';
if (secureContext) {
L.control
.locate({
strings: {
title: i18next.t('map.locate-me')
},
icon: 'fa fa-location-arrow',
iconLoading: 'fa fa-spinner fa-pulse'
})
.addTo(map);
}
layersControl = L.control.layers(baseLayers, overlays).addTo(map);
L.control.locate({
icon: 'fa fa-location-arrow',
iconLoading: 'fa fa-spinner fa-pulse',
}).addTo(map);
L.control.scale().addTo(map);
new BR.Layers().init(map, layersControl, baseLayers, overlays);
@ -126,15 +114,31 @@ BR.Map = {
BR.debug = BR.debug || {};
BR.debug.map = map;
var layersAndOverlays = baseLayers;
for (var o in overlays) {
layersAndOverlays[o] = overlays[o];
}
return {
map: map,
layersControl: layersControl,
layers: layersAndOverlays
layersControl: layersControl
};
}
},
_renderLayerCredits: function(layers) {
var dl = document.getElementById('credits-maps');
var i, obj, dt, dd, attribution;
L.DomUtil.empty(dl);
for (i = 0; i < layers.length; i++) {
obj = layers[i];
attribution = obj.layer.options.attribution;
if (attribution) {
dt = document.createElement('dt');
dt.innerHTML = obj.name;
dd = document.createElement('dd');
dd.innerHTML = obj.layer.options.attribution;
dl.appendChild(dt);
dl.appendChild(dd);
}
}
}
};

View file

@ -1,5 +1,4 @@
BR.Util = {
get: function(url, cb) {
var xhr = new XMLHttpRequest();
@ -16,17 +15,17 @@ BR.Util = {
};
try {
xhr.send();
} catch(e) {
} catch (e) {
cb(e);
}
},
getError: function(xhr) {
var msg = 'no response from server';
var msg = i18next.t('warning.no-response');
if (xhr.responseText) {
msg = xhr.responseText;
msg = xhr.responseText;
} else if (xhr.status || xhr.statusText) {
msg = xhr.status + ': ' + xhr.statusText;
msg = xhr.status + ': ' + xhr.statusText;
}
return new Error(msg);
},
@ -46,8 +45,7 @@ BR.Util = {
storage.setItem(x, x);
storage.removeItem(x);
return true;
}
catch(e) {
} catch (e) {
return false;
}
}

View file

@ -0,0 +1,99 @@
BR.ControlLayers = L.Control.Layers.extend({
getActiveLayers: function() {
var result = [];
for (var i = 0; i < this._layers.length; i++) {
var obj = this._layers[i];
if (this._map.hasLayer(obj.layer)) {
if (obj.overlay) {
result.push(obj);
} else {
result.unshift(obj);
}
}
}
return result;
},
getActiveBaseLayer: function() {
var activeLayers = this.getActiveLayers();
for (var i = 0; i < activeLayers.length; i++) {
var obj = activeLayers[i];
if (!obj.overlay) {
return obj;
}
}
return null;
},
removeActiveLayers: function() {
var removed = [];
for (var i = 0; i < this._layers.length; i++) {
var obj = this._layers[i];
if (this._map.hasLayer(obj.layer)) {
this._map.removeLayer(obj.layer);
removed.push(obj);
}
}
return removed;
},
getLayer: function(name) {
for (var i = 0; i < this._layers.length; i++) {
var obj = this._layers[i];
if (obj.name === name) {
return obj;
}
}
return null;
},
getBaseLayers: function() {
return this._layers.filter(function(obj) {
return !obj.overlay;
});
},
activateLayer: function(obj) {
if (!this._map.hasLayer(obj.layer)) {
this._map.addLayer(obj.layer);
}
},
activateFirstLayer: function() {
for (var i = 0; i < this._layers.length; i++) {
var obj = this._layers[i];
if (!obj.overlay) {
this.activateLayer(obj);
break;
}
}
},
activateBaseLayerIndex: function(index) {
var baseLayers = this.getBaseLayers();
var obj = baseLayers[index];
this.activateLayer(obj);
},
_addLayer: function(layer, name, overlay) {
L.Control.Layers.prototype._addLayer.call(this, layer, name, overlay);
// override z-index assignment to fix that base layers added later
// are on top of overlays; set all base layers to 0
if (this.options.autoZIndex && layer.setZIndex) {
if (!overlay) {
// undo increase in super method
this._lastZIndex--;
layer.setZIndex(0);
}
}
}
});

View file

@ -1,39 +0,0 @@
BR.Control = L.Control.extend({
options: {
position: 'bottomleft'
},
onAdd: function (map) {
var container = L.DomUtil.create('div', 'info'),
heading,
div;
if (this.options.heading) {
heading = L.DomUtil.create('div', 'heading', container);
heading.innerHTML = this.options.heading;
this._content = L.DomUtil.create('div', 'content', container);
} else {
this._content = container;
}
if (this.options.divId) {
div = L.DomUtil.get(this.options.divId);
L.DomUtil.removeClass(div, 'hidden');
this._content.appendChild(div);
}
var stop = L.DomEvent.stopPropagation;
L.DomEvent
.on(container, 'click', stop)
.on(container, 'mousedown', stop)
.on(container, 'dblclick', stop)
.on(container, 'mousewheel', stop)
.on(container, 'MozMousePixelScroll', stop);
// disabled because links not working, remove?
//L.DomEvent.on(container, 'click', L.DomEvent.preventDefault);
return container;
}
});

View file

@ -1,16 +0,0 @@
BR.Download = L.Class.extend({
update: function (urls) {
if (urls) {
['gpx', 'kml', 'geojson', 'csv'].forEach(function(e, i, a) {
var a = L.DomUtil.get('dl-'+e);
a.setAttribute('href', urls[e]);
a.setAttribute('download', 'brouter.'+e);
a.removeAttribute('disabled');
})
}
}
});
BR.download = function() {
return new BR.Download();
};

167
js/control/Export.js Normal file
View file

@ -0,0 +1,167 @@
BR.Export = L.Class.extend({
latLngs: [],
initialize: function(router) {
this.router = router;
this.exportButton = $('#exportButton');
var trackname = (this.trackname = document.getElementById('trackname'));
this.tracknameAllowedChars = BR.conf.tracknameAllowedChars;
if (this.tracknameAllowedChars) {
this.tracknameMessage = document.getElementById(
'trackname-message'
);
var patternRegex = new RegExp(
'[' + this.tracknameAllowedChars + ']+'
);
// warn about special characters getting removed by server quick fix (#194)
trackname.pattern = patternRegex.toString().slice(1, -1);
trackname.addEventListener(
'input',
L.bind(this._validationMessage, this)
);
}
this.exportButton.on('click', L.bind(this._generateTrackname, this));
L.DomUtil.get('submitExport').onclick = L.bind(this._export, this);
this.update([]);
},
update: function(latLngs) {
this.latLngs = latLngs;
if (latLngs.length < 2) {
this.exportButton.addClass('disabled');
} else {
this.exportButton.removeClass('disabled');
}
},
_export: function() {
var exportForm = document.forms['export'];
var format =
exportForm['format'].value ||
$('#export-format input:radio:checked').val();
var name = encodeURIComponent(exportForm['trackname'].value);
var includeWaypoints = exportForm['include-waypoints'].checked;
var uri = this.router.getUrl(
this.latLngs,
format,
name,
includeWaypoints
);
var evt = document.createEvent('MouseEvents');
evt.initMouseEvent(
'click',
true,
true,
window,
0,
0,
0,
0,
0,
false,
false,
false,
false,
0,
null
);
var link = document.createElement('a');
link.href = uri;
link.dispatchEvent(evt);
},
_validationMessage: function() {
var trackname = this.trackname;
var replaceRegex = new RegExp(
'[^' + this.tracknameAllowedChars + ']',
'g'
);
if (trackname.validity.patternMismatch) {
var replaced = trackname.value.replace(replaceRegex, '');
var patternStr = this.tracknameAllowedChars.replace(/\\/g, '');
this.tracknameMessage.textContent =
'[' + patternStr + '] --> ' + replaced;
} else {
this.tracknameMessage.textContent = '';
}
},
_generateTrackname: function() {
var trackname = this.trackname;
this._getCityAtPosition(
this.latLngs[0],
L.bind(function(from) {
this._getCityAtPosition(
this.latLngs[this.latLngs.length - 1],
L.bind(function(to) {
var distance = document.getElementById('distance')
.innerHTML;
if (this.tracknameAllowedChars) {
distance = distance.replace(',', '.'); // temp. fix (#202)
}
if (!from || !to) {
trackname.value = null;
} else if (from === to) {
trackname.value = i18next.t('export.route-loop', {
from: from,
distance: distance
});
} else {
trackname.value = i18next.t(
'export.route-from-to',
{ from: from, to: to, distance: distance }
);
}
if (this.tracknameAllowedChars) {
// temp. fix: replace and remove characters that will get removed by server quick fix (#194)
trackname.value = trackname.value
.replace(/[>)]/g, '')
.replace(/ \(/g, ' - ');
this._validationMessage();
}
}, this)
);
}, this)
);
},
_getCityAtPosition: function(lonlat, cb) {
var url = L.Util.template(
'https://nominatim.openstreetmap.org/reverse?lon={lng}&lat={lat}&format=json',
lonlat
);
BR.Util.get(
url,
L.bind(function(err, response) {
try {
var addr = JSON.parse(response).address;
cb(
addr.village ||
addr.town ||
addr.hamlet ||
addr.city_district ||
addr.city
);
} catch (err) {
BR.message.showError(
'Error getting position city "' + lonlat + '": ' + err
);
return cb(null);
}
})
);
}
});
BR.export = function() {
return new BR.Export();
};

View file

@ -1,24 +1,20 @@
BR.Itinerary = L.Class.extend({
options: {
heading: 'Itinerary'
},
onAdd: function (map) {
initialize: function() {
this._content = document.getElementById('itinerary');
L.DomUtil.removeClass(this._content.parentElement, 'hidden');
this.update();
},
update: function (polyline, segments) {
var i, j, iter, html = '';
update: function(polyline, segments) {
var i,
j,
iter,
html = '';
html += '<pre>';
for (i = 0; segments && i < segments.length; i++)
{
html += '<pre class="flexgrow">';
for (i = 0; segments && i < segments.length; i++) {
iter = segments[i].feature.iternity;
for (j = 0; iter && j < iter.length; j++)
{
html += iter[j] + '\n';
for (j = 0; iter && j < iter.length; j++) {
html += iter[j] + '\n';
}
}
html += '</pre>';

View file

@ -1,10 +1,9 @@
BR.Layers = L.Class.extend({
_loadLayers: function() {
this._customLayers = {};
if (BR.Util.localStorageAvailable()) {
var layers = JSON.parse(localStorage.getItem("map/customLayers"));
var layers = JSON.parse(localStorage.getItem('map/customLayers'));
for (a in layers) {
this._addLayer(a, layers[a].layer, layers[a].isOverlay);
}
@ -14,21 +13,28 @@ BR.Layers = L.Class.extend({
_loadTable: function() {
var layersData = [];
for (layer in this._customLayers) {
layersData.push([layer, this._customLayers[layer].layer._url, this._customLayers[layer].isOverlay ? "Overlay" : "Layer"]);
layersData.push([
layer,
this._customLayers[layer].layer._url,
this._customLayers[layer].isOverlay ? 'Overlay' : 'Layer'
]);
}
if (this._layersTable != null) {
this._layersTable.destroy();
}
this._layersTable = $('#custom_layers_table').DataTable({
this._layersTable = $('#custom_layers_table').DataTable({
data: layersData,
info: false,
searching: false,
paging: false,
language: {
emptyTable: i18next.t('sidebar.layers.table.empty')
},
columns: [
{ title: "Name" },
{ title: "URL" },
{ title: "Type" }
{ title: i18next.t('sidebar.layers.table.name') },
{ title: i18next.t('sidebar.layers.table.URL') },
{ title: i18next.t('sidebar.layers.table.type') }
]
});
},
@ -36,22 +42,29 @@ BR.Layers = L.Class.extend({
init: function(map, layersControl, baseLayers, overlays) {
this._layersControl = layersControl;
this._map = map;
this._layers = {}
for (var l in overlays)
this._layers[l] = [overlays[l], true];
for (var l in baseLayers)
this._layers[l] = [baseLayers[l], false];
this._layers = {};
for (var l in overlays) this._layers[l] = [overlays[l], true];
for (var l in baseLayers) this._layers[l] = [baseLayers[l], false];
L.DomUtil.get('custom_layers_add_base').onclick = L.bind(this._addBaseLayer, this);
L.DomUtil.get('custom_layers_add_overlay').onclick = L.bind(this._addOverlay, this);
L.DomUtil.get('custom_layers_remove').onclick = L.bind(this._remove, this);
L.DomUtil.get('custom_layers_add_base').onclick = L.bind(
this._addBaseLayer,
this
);
L.DomUtil.get('custom_layers_add_overlay').onclick = L.bind(
this._addOverlay,
this
);
L.DomUtil.get('custom_layers_remove').onclick = L.bind(
this._remove,
this
);
this._loadLayers();
this._loadTable();
var table = this._layersTable;
$('#custom_layers_table tbody').on( 'click', 'tr', function () {
if ( $(this).hasClass('selected') ) {
$('#custom_layers_table tbody').on('click', 'tr', function() {
if ($(this).hasClass('selected')) {
$(this).removeClass('selected');
} else {
table.$('tr.selected').removeClass('selected');
@ -59,16 +72,9 @@ BR.Layers = L.Class.extend({
}
});
addLayer = L.easyButton(
'fa-plus-square',
function () {
$('#custom_layers').modal();
},
'Add or remove custom layers',
{
position: 'topright'
}
).addTo(map);
L.DomUtil.get('custom_layers_button').onclick = function() {
$('#custom_layers').modal();
};
},
_remove: function(evt) {
@ -78,7 +84,10 @@ BR.Layers = L.Class.extend({
this._layersControl.removeLayer(this._customLayers[name].layer);
this._map.removeLayer(this._customLayers[name].layer);
delete this._customLayers[name];
this._layersTable.row('.selected').remove().draw( false );
this._layersTable
.row('.selected')
.remove()
.draw(false);
this._sync();
}
},
@ -98,16 +107,17 @@ BR.Layers = L.Class.extend({
},
_addLayer: function(layerName, layerUrl, isOverlay) {
if (layerName in this._layers)
return
if (layerName in this._layers) return;
if (layerName in this._customLayers)
return
if (layerName in this._customLayers) return;
try {
var layer = L.tileLayer(layerUrl);
this._customLayers[layerName] = {layer: layer, isOverlay: isOverlay};
this._customLayers[layerName] = {
layer: layer,
isOverlay: isOverlay
};
if (isOverlay) {
this._layersControl.addOverlay(layer, layerName);
@ -118,17 +128,20 @@ BR.Layers = L.Class.extend({
this._sync();
return layer;
} catch (e) {
console.warn("Oops:", e);
return
console.warn('Oops:', e);
return;
}
},
_sync: function() {
if (BR.Util.localStorageAvailable()) {
localStorage.setItem("map/customLayers", JSON.stringify(this._customLayers, function(k, v) {
// dont write Leaflet.Layer in localStorage; simply keep the URL
return v._url || v;
}));
localStorage.setItem(
'map/customLayers',
JSON.stringify(this._customLayers, function(k, v) {
// dont write Leaflet.Layer in localStorage; simply keep the URL
return v._url || v;
})
);
}
}
});

484
js/control/LayersTab.js Normal file
View file

@ -0,0 +1,484 @@
BR.LayersTab = BR.ControlLayers.extend({
previewLayer: null,
previewBounds: null,
saveLayers: [],
initialize: function(layersConfig, baseLayers, overlays, options) {
L.Control.Layers.prototype.initialize.call(
this,
baseLayers,
overlays,
options
);
this.layersConfig = layersConfig;
},
addTo: function(map) {
this._map = map;
this.onAdd(map);
L.DomUtil.get('layers-control-wrapper').appendChild(this._section);
this.initOpacitySlider(map);
this.initButtons();
this.initJsTree();
return this;
},
onAdd: function(map) {
BR.ControlLayers.prototype.onAdd.call(this, map);
map.on(
'baselayerchange overlayadd overlayremove',
this.storeActiveLayers,
this
);
},
onRemove: function(map) {
BR.ControlLayers.prototype.onRemove.call(this, map);
map.off(
'baselayerchange overlayadd overlayremove',
this.storeActiveLayers,
this
);
},
initOpacitySlider: function(map) {
var self = this;
var overlayOpacitySlider = new BR.OpacitySlider({
id: 'overlay',
reversed: false,
orientation: 'horizontal',
defaultValue: 1,
title: i18next.t('layers.opacity-slider'),
callback: function(opacity) {
for (var i = 0; i < self._layers.length; i++) {
if (
!self._layers[i].overlay ||
!map.hasLayer(self._layers[i].layer)
) {
continue;
}
self._layers[i].layer.setOpacity(opacity);
}
}
});
L.DomUtil.get(
'leaflet-control-layers-overlays-opacity-slider'
).appendChild(overlayOpacitySlider.getElement());
},
initButtons: function() {
var expandTree = function(e) {
this.jstree.open_all();
};
var collapseTree = function(e) {
this.jstree.close_all();
};
var toggleOptionalLayers = function(e) {
var button = L.DomUtil.get('optional_layers_button');
var treeButtons = L.DomUtil.get('tree-button-group');
var div = L.DomUtil.get('optional-layers-tree');
div.hidden = !div.hidden;
treeButtons.hidden = !treeButtons.hidden;
button.classList.toggle('active');
if (div.hidden) {
this.deselectNode();
}
};
L.DomUtil.get('expand_tree_button').onclick = L.bind(expandTree, this);
L.DomUtil.get('collapse_tree_button').onclick = L.bind(
collapseTree,
this
);
L.DomUtil.get('optional_layers_button').onclick = L.bind(
toggleOptionalLayers,
this
);
},
initJsTree: function() {
var layerIndex = BR.layerIndex;
var treeData = this.toJsTree(BR.confLayers.tree);
var oldSelected = null;
var onSelectNode = function(e, data) {
var layerData = layerIndex[data.node.id];
var selected = data.selected[0];
if (selected !== oldSelected) {
this.showPreview(layerData);
oldSelected = selected;
} else {
data.instance.deselect_node(data.node);
}
};
var onDeselectNode = function(e, data) {
this.hidePreview();
oldSelected = null;
};
var onCheckNode = function(e, data) {
var layerData = layerIndex[data.node.id];
var layer = this.createLayer(layerData);
var name = layerData.properties.name;
var overlay = layerData.properties.overlay;
if (overlay) {
this.addOverlay(layer, name);
} else {
this.addBaseLayer(layer, name);
}
this.storeDefaultLayers();
};
var onUncheckNode = function(e, data) {
var obj = this.getLayerById(data.node.id);
if (!obj) return;
this.removeLayer(obj.layer);
if (this._map.hasLayer(obj.layer)) {
this._map.removeLayer(obj.layer);
if (!obj.overlay) {
this.activateFirstLayer();
}
}
this.storeDefaultLayers();
};
$('#optional-layers-tree')
.on('select_node.jstree', L.bind(onSelectNode, this))
.on('deselect_node.jstree', L.bind(onDeselectNode, this))
.on('check_node.jstree', L.bind(onCheckNode, this))
.on('uncheck_node.jstree', L.bind(onUncheckNode, this))
.on('ready.jstree', function(e, data) {
data.instance.open_all();
})
.jstree({
plugins: ['checkbox'],
checkbox: {
whole_node: false,
tie_selection: false
},
core: {
multiple: false,
themes: {
icons: false,
dots: false
},
data: treeData
}
});
this.jstree = $('#optional-layers-tree').jstree(true);
},
toJsTree: function(layerTree) {
var data = {
children: []
};
var self = this;
function createRootNode(name) {
var rootNode = {
text: i18next.t('sidebar.layers.category.' + name, name),
state: {
disabled: true
},
children: []
};
return rootNode;
}
function getText(props, parent) {
var text = '';
var code = props.country_code || props.language_code;
if (code && parent.text !== code) {
text += '<span class="tree-code">' + code + '</span>';
}
text += props.name;
return text;
}
function createNode(id, layerData, parent) {
var props = layerData.properties;
var url = props.url;
var keyObj = self.layersConfig.getKeyName(url);
var childNode = null;
// when key required only add if configured
if (!keyObj || (keyObj && BR.keys[keyObj.name])) {
childNode = {
id: id,
text: getText(props, parent),
state: {
checked: self.layersConfig.isDefaultLayer(
id,
props.overlay
)
}
};
}
return childNode;
}
function walkTree(inTree, outTree) {
function walkObject(obj) {
for (name in obj) {
var value = obj[name];
var rootNode = createRootNode(name);
outTree.children.push(rootNode);
walkTree(value, rootNode);
}
}
if (Array.isArray(inTree)) {
for (var i = 0; i < inTree.length; i++) {
var entry = inTree[i];
if (typeof entry === 'object') {
walkObject(entry);
} else {
var layer = BR.layerIndex[entry];
if (layer) {
var childNode = createNode(entry, layer, outTree);
if (childNode) {
outTree.children.push(childNode);
}
} else {
console.error('Layer "' + entry + '" not found');
}
}
}
} else {
walkObject(inTree);
}
}
walkTree(layerTree, data);
return data.children;
},
storeDefaultLayers: function() {
var baseLayers = [];
var overlays = [];
for (var i = 0; i < this._layers.length; i++) {
var obj = this._layers[i];
// id set in LayersConfig.createLayer
var id = obj.layer.id;
if (id) {
if (obj.overlay) {
overlays.push(id);
} else {
baseLayers.push(id);
}
}
}
this.layersConfig.storeDefaultLayers(baseLayers, overlays);
},
createLayer: function(layerData) {
var layer = this.layersConfig.createLayer(layerData);
var overlay = layerData.properties.overlay;
// preview z-index, like in BR.ControlLayers._addLayer
layer.options.zIndex = overlay ? this._lastZIndex + 1 : 0;
return layer;
},
getLayerById: function(id) {
for (var i = 0; i < this._layers.length; i++) {
var obj = this._layers[i];
if (obj.layer.id === id) {
return obj;
}
}
return null;
},
getLayerByLegacyName: function(legacyName) {
var obj = null;
var id = this.layersConfig.legacyNameToIdMap[legacyName];
if (id) {
obj = this.getLayerById(id);
}
return obj;
},
activateDefaultBaseLayer: function() {
var index = BR.conf.defaultBaseLayerIndex || 0;
var activeBaseLayer = this.getActiveBaseLayer();
if (!activeBaseLayer) {
this.activateBaseLayerIndex(index);
}
},
saveRemoveActiveLayers: function() {
this.saveLayers = this.removeActiveLayers();
},
restoreActiveLayers: function(overlaysOnly) {
for (var i = 0; i < this.saveLayers.length; i++) {
var obj = this.saveLayers[i];
if (!overlaysOnly || (overlaysOnly && obj.overlay)) {
var hasLayer = !!this._getLayer(L.Util.stamp(obj.layer));
if (hasLayer) {
this.activateLayer(obj);
} else if (!obj.overlay) {
// saved base layer has been removed during preview, select first
this.activateFirstLayer();
}
}
}
this.saveLayers = [];
},
removePreviewLayer: function() {
if (this.previewLayer && this._map.hasLayer(this.previewLayer)) {
this._map.removeLayer(this.previewLayer);
this.previewLayer = null;
return true;
}
return false;
},
showPreviewBounds: function(layerData) {
if (layerData.geometry) {
this.previewBounds = L.geoJson(layerData.geometry, {
// fill/mask outside of bounds polygon with Leaflet.snogylop
invert: true,
// reduce unmasked areas appearing due to clipping while panning and zooming out
renderer: L.svg({ padding: 1 }),
color: '#333',
fillOpacity: 0.4,
weight: 2
}).addTo(this._map);
}
},
removePreviewBounds: function() {
if (this.previewBounds && this._map.hasLayer(this.previewBounds)) {
this._map.removeLayer(this.previewBounds);
this.previewBounds = null;
}
},
deselectNode: function() {
var selected = this.jstree.get_selected();
if (selected.length > 0) {
this.jstree.deselect_node(selected[0]);
}
},
onBaselayerchange: function() {
// execute after current input click handler,
// otherwise added overlay checkbox state doesn't update
setTimeout(
L.Util.bind(function() {
this.removePreviewBounds();
this.removePreviewLayer();
this.restoreActiveLayers(true);
this.deselectNode();
}, this),
0
);
},
showPreview: function(layerData) {
var layer = this.createLayer(layerData);
this._map.addLayer(layer);
this.removePreviewBounds();
if (!this.removePreviewLayer()) {
this.saveRemoveActiveLayers();
this._map.once('baselayerchange', this.onBaselayerchange, this);
}
this.previewLayer = layer;
this.showPreviewBounds(layerData);
},
hidePreview: function(layer) {
this._map.off('baselayerchange', this.onBaselayerchange, this);
this.removePreviewBounds();
this.removePreviewLayer();
this.restoreActiveLayers();
},
toLayerString: function(obj) {
return obj.layer.id ? obj.layer.id : obj.name;
},
getLayerFromString: function(layerString) {
var obj = this.getLayerById(layerString);
if (!obj) {
// fallback to name for custom and config layers
obj = this.getLayer(layerString);
if (!obj) {
// legacy layer name support
obj = this.getLayerByLegacyName(layerString);
}
}
return obj;
},
storeActiveLayers: function() {
if (BR.Util.localStorageAvailable()) {
var objList = this.getActiveLayers();
var idList = objList.map(
L.bind(function(obj) {
return this.toLayerString(obj);
}, this)
);
var str = JSON.stringify(idList);
localStorage.setItem('map/activeLayers', str);
}
},
loadActiveLayers: function() {
if (BR.Util.localStorageAvailable()) {
var item = localStorage.getItem('map/activeLayers');
if (item) {
var idList = JSON.parse(item);
for (var i = 0; i < idList.length; i++) {
var id = idList[i];
var obj = this.getLayerFromString(id);
if (obj) {
this.activateLayer(obj);
}
}
}
}
}
});
BR.layersTab = function(baseLayers, overlays, options) {
return new BR.LayersTab(baseLayers, overlays, options);
};

View file

@ -5,39 +5,51 @@ BR.Message = L.Class.extend({
alert: false
},
initialize: function (id, options) {
initialize: function(id, options) {
L.setOptions(this, options);
this.id = id;
},
_show: function (msg, type) {
_show: function(msg, type) {
var ele = L.DomUtil.get(this.id),
iconClass = (type === 'warning') ? 'fa-exclamation-triangle' : 'fa-times-circle',
alertClass = (type === 'warning') ? 'alert-warning' : 'alert-danger';
iconClass =
type === 'warning'
? 'fa-exclamation-triangle'
: 'fa-times-circle',
alertClass = type === 'warning' ? 'alert-warning' : 'alert-danger';
L.DomEvent.disableClickPropagation(ele);
ele.innerHTML =
'<div class="alert ' + alertClass + ' alert-dismissible fade in" role="alert">'
+ ' <button type="button" class="close" data-dismiss="alert" aria-label="Close">'
+ ' <span aria-hidden="true">&times;</span>'
+ ' </button>'
+ ' <span class="fa ' + iconClass + '" aria-hidden="true"/></span>'
+ msg
+ '</div>';
'<div class="alert ' +
alertClass +
' alert-dismissible fade show" role="alert">' +
' <button type="button" class="close" data-dismiss="alert" aria-label="Close">' +
' <span aria-hidden="true">&times;</span>' +
' </button>' +
' <span class="fa ' +
iconClass +
'" aria-hidden="true"/></span>' +
msg +
'</div>';
if (this.options.alert) {
$('#' + this.id + ' .alert').alert();
}
},
hide: function () {
hide: function() {
$('#' + this.id + ' .alert').alert('close');
},
showError: function (err) {
showError: function(err) {
if (err == 'Error: target island detected for section 0\n') {
err = i18next.t('warning.no-route-found');
}
this._show(err, 'error');
},
showWarning: function (msg) {
showWarning: function(msg) {
this._show(msg, 'warning');
}
});

View file

@ -1,67 +1,58 @@
BR.OpacitySlider = L.Control.extend({
options: {
position: 'topleft',
BR.OpacitySlider = L.Class.extend({
options: {
id: '',
reversed: true,
orientation: 'vertical',
defaultValue: BR.conf.defaultOpacity,
title: '',
callback: function(opacity) {}
},
onAdd: function (map) {
var container = L.DomUtil.create('div', 'leaflet-bar control-slider'),
input = $('<input id="slider" type="text"/>'),
item = BR.Util.localStorageAvailable() ? localStorage.opacitySliderValue : null,
value = item ? parseInt(item) : BR.conf.defaultOpacity * 100,
initialize: function(options) {
L.setOptions(this, options);
var input = (this.input = $(
'<input id="slider-' + this.options.id + '" type="text"/>'
)),
item = BR.Util.localStorageAvailable()
? localStorage['opacitySliderValue' + this.options.id]
: null,
value = item ? parseInt(item) : this.options.defaultValue * 100,
minOpacity = (BR.conf.minOpacity || 0) * 100;
if (value < minOpacity) {
value = minOpacity;
}
// prevent also dragging map in Chrome
L.DomEvent.disableClickPropagation(container);
input
.slider({
id: this.options.id,
min: 0,
max: 100,
step: 1,
value: value,
orientation: this.options.orientation,
reversed: this.options.reversed,
selection: this.options.reversed ? 'before' : 'after', // inverted, serves as track style, see css
tooltip: 'hide'
})
.on('slide slideStop', { self: this }, function(evt) {
evt.data.self.options.callback(evt.value / 100);
})
.on('slideStop', { self: this }, function(evt) {
if (BR.Util.localStorageAvailable()) {
localStorage[
'opacitySliderValue' + evt.data.self.options.id
] = evt.value;
}
});
var stopClickAfterSlide = function(evt) {
L.DomEvent.stop(evt);
removeStopClickListeners();
};
var removeStopClickListeners = function() {
document.removeEventListener('click', stopClickAfterSlide, true);
document.removeEventListener('mousedown', removeStopClickListeners, true);
};
$(container).html(input);
$(container).attr('title', 'Set transparency of route track and markers');
input.slider({
min: 0,
max: 100,
step: 1,
value: value,
orientation: 'vertical',
reversed : true,
selection: 'before', // inverted, serves as track style, see css
tooltip: 'hide'
}).on('slideStart', function (evt) {
// dragging beyond slider control selects zoom control +/- text in Firefox
L.DomUtil.disableTextSelection();
}).on('slide slideStop', { self: this }, function (evt) {
evt.data.self.options.callback(evt.value / 100);
}).on('slideStop', function (evt) {
if (BR.Util.localStorageAvailable()) {
localStorage.opacitySliderValue = evt.value;
}
L.DomUtil.enableTextSelection();
// When dragging outside slider and over map, click event after mouseup
// adds marker when active on Chromium. So disable click (not needed)
// once after sliding.
document.addEventListener('click', stopClickAfterSlide, true);
// Firefox does not fire click event in this case, so make sure stop listener
// is always removed on next mousedown.
document.addEventListener('mousedown', removeStopClickListeners, true);
});
this.getElement().title = this.options.title;
this.options.callback(value / 100);
},
return container;
getElement: function() {
return this.input.slider('getElement');
}
});

View file

@ -0,0 +1,64 @@
BR.OpacitySliderControl = L.Control.extend({
options: {
position: 'topleft'
},
onAdd: function(map) {
var container = L.DomUtil.create('div', 'leaflet-bar control-slider');
// prevent also dragging map in Chrome
L.DomEvent.disableClickPropagation(container);
// migrate legacy value
if (BR.Util.localStorageAvailable()) {
var value = localStorage.getItem('opacitySliderValue');
if (value !== null) {
localStorage.setItem(
'opacitySliderValue' + this.options.id,
value
);
localStorage.removeItem('opacitySliderValue');
}
}
var slider = new BR.OpacitySlider(this.options);
container.appendChild(slider.getElement());
var stopClickAfterSlide = function(evt) {
L.DomEvent.stop(evt);
removeStopClickListeners();
};
var removeStopClickListeners = function() {
document.removeEventListener('click', stopClickAfterSlide, true);
document.removeEventListener(
'mousedown',
removeStopClickListeners,
true
);
};
slider.input
.on('slideStart', function(evt) {
// dragging beyond slider control selects zoom control +/- text in Firefox
L.DomUtil.disableTextSelection();
})
.on('slideStop', { self: this }, function(evt) {
L.DomUtil.enableTextSelection();
// When dragging outside slider and over map, click event after mouseup
// adds marker when active on Chromium. So disable click (not needed)
// once after sliding.
document.addEventListener('click', stopClickAfterSlide, true);
// Firefox does not fire click event in this case, so make sure stop listener
// is always removed on next mousedown.
document.addEventListener(
'mousedown',
removeStopClickListeners,
true
);
});
return container;
}
});

View file

@ -1,11 +1,15 @@
BR.Profile = L.Class.extend({
BR.Profile = L.Evented.extend({
cache: {},
initialize: function () {
initialize: function() {
var textArea = L.DomUtil.get('profile_upload');
this.editor = CodeMirror.fromTextArea(textArea, {
lineNumbers: true
});
L.DomUtil.get('upload').onclick = L.bind(this._upload, this);
L.DomUtil.get('clear').onclick = L.bind(this.clear, this);
this.ele = L.DomUtil.get('profile_upload');
autosize(this.ele);
this.message = new BR.Message('profile_message', {
alert: true
});
@ -15,9 +19,7 @@ BR.Profile = L.Class.extend({
var button = evt.target || evt.srcElement;
evt.preventDefault();
this.ele.value = null;
this.ele.defaultValue = null;
autosize.update(this.ele);
this._setValue('');
this.fire('clear');
button.blur();
@ -26,39 +28,54 @@ BR.Profile = L.Class.extend({
update: function(options) {
var profileName = options.profile,
profileUrl,
ele = this.ele,
dirty = ele.defaultValue !== ele.value;
empty = !this.editor.getValue(),
clean = this.editor.isClean();
this.profileName = profileName;
if (profileName && BR.conf.profilesUrl && (!ele.value || !dirty)) {
if (profileName && BR.conf.profilesUrl && (empty || clean)) {
if (!(profileName in this.cache)) {
profileUrl = BR.conf.profilesUrl + profileName + '.brf';
BR.Util.get(profileUrl, L.bind(function(err, profileText) {
if (err) {
console.warn('Error getting profile from "' + profileUrl + '": ' + err);
return;
}
BR.Util.get(
profileUrl,
L.bind(function(err, profileText) {
if (err) {
console.warn(
'Error getting profile from "' +
profileUrl +
'": ' +
err
);
return;
}
this.cache[profileName] = profileText;
this.cache[profileName] = profileText;
// don't set when option has changed while loading
if (!this.profileName || this.profileName === profileName) {
ele.value = profileText;
ele.defaultValue = ele.value;
autosize.update(this.ele);
}
}, this));
// don't set when option has changed while loading
if (
!this.profileName ||
this.profileName === profileName
) {
this._setValue(profileText);
}
}, this)
);
} else {
ele.value = this.cache[profileName];
ele.defaultValue = ele.value;
autosize.update(this.ele);
this._setValue(this.cache[profileName]);
}
}
},
show: function() {
this.editor.refresh();
},
onResize: function() {
this.editor.refresh();
},
_upload: function(evt) {
var button = evt.target || evt.srcElement,
profile = this.ele.value;
profile = this.editor.getValue();
this.message.hide();
$(button).button('uploading');
@ -66,12 +83,15 @@ BR.Profile = L.Class.extend({
this.fire('update', {
profileText: profile,
callback: function () {
callback: function() {
$(button).button('reset');
$(button).blur();
}
});
},
_setValue: function(profileText) {
this.editor.setValue(profileText);
this.editor.markClean();
}
});
BR.Profile.include(L.Mixin.Events);

View file

@ -1,43 +1,45 @@
BR.RoutingOptions = BR.Control.extend({
onAdd: function (map) {
$('#profile-alternative').on('changed.bs.select', this._getChangeHandler());
BR.RoutingOptions = L.Evented.extend({
initialize: function() {
$('#profile-alternative').on(
'changed.bs.select',
this._getChangeHandler()
);
// build option list from config
var profiles = BR.conf.profiles;
var profiles_list = L.DomUtil.get('profile');
for (var i = 0; i < profiles.length; i++) {
var option = document.createElement("option");
var option = document.createElement('option');
option.value = profiles[i];
option.text = profiles[i];
option.text = i18next.t('navbar.profile.' + profiles[i]);
profiles_list.appendChild(option);
}
// set default value, used as indicator for empty custom profile
profiles_list.children[0].value = 'Custom';
// <custom> profile is empty at start, select next one
profiles_list.children[1].selected = true;
return BR.Control.prototype.onAdd.call(this, map);
},
refreshUI: function() {
var profile = $('#profile option:selected'),
alternative = $('#alternative option:selected');
$('#stat-profile').html(profile.text() + ' (' + alternative.text() +')');
// we do not allow to select more than one profile and/or alternative at a time
// so we disable the current selected items
$('#profile-alternative').find('option:disabled').each(function(index) {
$(this).prop('disabled', false);
});
$('#profile-alternative').find('option:selected').each(function(index) {
$(this).prop('disabled', true);
});
$('#profile-alternative')
.find('option:disabled')
.each(function(index) {
$(this).prop('disabled', false);
});
$('#profile-alternative')
.find('option:selected')
.each(function(index) {
$(this).prop('disabled', true);
});
// disable custom option if it has no value yet (default value is "Custom")
var custom = L.DomUtil.get('profile').children[0];
if (custom.value === "Custom") {
if (custom.value === 'Custom') {
custom.disabled = true;
}
$('.selectpicker').selectpicker('refresh')
$('.selectpicker').selectpicker('refresh');
},
getOptions: function() {
@ -53,8 +55,12 @@ BR.RoutingOptions = BR.Control.extend({
setOptions: function(options) {
var values = [
options.profile ? options.profile : $('#profile option:selected').val(),
options.alternative ? options.alternative : $('#alternative option:selected').val()
options.profile
? options.profile
: $('#profile option:selected').val(),
options.alternative
? options.alternative
: $('#alternative option:selected').val()
];
$('.selectpicker').selectpicker('val', values);
this.refreshUI();
@ -68,22 +74,28 @@ BR.RoutingOptions = BR.Control.extend({
},
setCustomProfile: function(profile, noUpdate) {
var profiles_grp,
option;
var profiles_grp, option;
profiles_grp = L.DomUtil.get('profile');
option = profiles_grp.children[0]
option.value = profile;
option = profiles_grp.children[0];
option.value = profile || 'Custom';
option.disabled = !profile;
$('#profile').find('option:selected').each(function(index) {
$(this).prop('selected', false);
});
if (profile) {
$('#profile')
.find('option:selected')
.each(function(index) {
$(this).prop('selected', false);
});
} else if (option.selected) {
// clear, select next in group when custom deselected
profiles_grp.children[1].selected = true;
}
option.selected = !!profile;
if (!noUpdate) {
this.fire('update', {options: this.getOptions()});
this.fire('update', { options: this.getOptions() });
}
},
@ -92,7 +104,7 @@ BR.RoutingOptions = BR.Control.extend({
option = profiles_grp.children[0],
profile = null;
if (!option.disabled) {
if (option.value !== 'Custom') {
profile = option.value;
}
return profile;
@ -100,9 +112,7 @@ BR.RoutingOptions = BR.Control.extend({
_getChangeHandler: function() {
return L.bind(function(evt) {
this.fire('update', {options: this.getOptions()});
this.fire('update', { options: this.getOptions() });
}, this);
}
});
BR.RoutingOptions.include(L.Mixin.Events);

View file

@ -1,47 +0,0 @@
BR.Tabs = BR.Control.extend({
options: {
divId: 'sidebar',
// tab a.hash > instance
tabs: {}
},
initialize: function (options) {
L.setOptions(this, options);
},
onAdd: function (map) {
var tabs = this.options.tabs;
for (var key in tabs) {
$('<li><a href="' + key + '" role="tab">' + tabs[key].options.heading + '</a></li>').appendTo('#tab');
if (tabs[key].onAdd) {
tabs[key].onAdd(map);
}
}
$('#tab a').click(function (e) {
e.preventDefault();
$(this).tab('show');
});
// e.target = activated tab
// e.relatedTarget = previous tab
$('#tab a').on('shown.bs.tab', L.bind(function (e) {
var tab = this.options.tabs[e.target.hash],
prevTab = e.relatedTarget ? this.options.tabs[e.relatedTarget.hash] : null;
if (tab && tab.show) {
tab.show();
}
if (prevTab && prevTab.hide) {
prevTab.hide();
}
}, this));
// activate first tab (instead of setting 'active' class in html)
$('#tab li:not(.hidden) a:first').tab('show');
return BR.Control.prototype.onAdd.call(this, map);
}
});

View file

@ -1,7 +1,5 @@
BR.TrackMessages = L.Class.extend({
options: {
heading: 'Segment data',
edgeStyle: {
color: 'yellow',
opacity: 0.8,
@ -13,31 +11,31 @@ BR.TrackMessages = L.Class.extend({
active: false,
columnOptions: {
'Longitude': { visible: false },
'Latitude': { visible: false },
'Elevation': { title: 'elev.', className: 'dt-body-right' },
'Distance': { title: 'dist.', className: 'dt-body-right' },
'CostPerKm': { title: '$/km', className: 'dt-body-right' },
'ElevCost': { title: 'elev$', className: 'dt-body-right' },
'TurnCost': { title: 'turn$', className: 'dt-body-right' },
'NodeCost': { title: 'node$', className: 'dt-body-right' },
'InitialCost': { title: 'initial$', className: 'dt-body-right' }
Longitude: { visible: false },
Latitude: { visible: false },
Elevation: { title: 'elev.', className: 'dt-body-right' },
Distance: { title: 'dist.', className: 'dt-body-right' },
CostPerKm: { title: '$/km', className: 'dt-body-right' },
ElevCost: { title: 'elev$', className: 'dt-body-right' },
TurnCost: { title: 'turn$', className: 'dt-body-right' },
NodeCost: { title: 'node$', className: 'dt-body-right' },
InitialCost: { title: 'initial$', className: 'dt-body-right' }
},
initialize: function (options) {
initialize: function(map, options) {
L.setOptions(this, options);
this._map = map;
var table = document.getElementById('datatable');
this.tableClassName = table.className;
this.tableParent = table.parentElement;
},
onAdd: function (map) {
this._map = map;
},
update: function (polyline, segments) {
var i, messages, columns, headings,
update: function(polyline, segments) {
var i,
messages,
columns,
headings,
data = [];
if (!this.active) {
@ -54,7 +52,7 @@ BR.TrackMessages = L.Class.extend({
this._destroyTable();
if (data.length === 0) {
return;
return;
}
headings = messages[0];
@ -67,13 +65,19 @@ BR.TrackMessages = L.Class.extend({
paging: false,
searching: false,
info: false,
// flexbox workaround: without scrollY height Firefox extends to content height
// (^= minimum height with flexbox?)
scrollY: 50,
scrollX: true,
order: []
});
// highlight track segment (graph edge) on row hover
this._setEdges(polyline, segments);
$('#datatable tbody tr').hover(L.bind(this._handleHover, this), L.bind(this._handleHoverOut, this));
$('#datatable tbody tr').hover(
L.bind(this._handleHover, this),
L.bind(this._handleHoverOut, this)
);
},
show: function() {
@ -88,9 +92,11 @@ BR.TrackMessages = L.Class.extend({
_destroyTable: function() {
var ele;
if ($.fn.DataTable.isDataTable('#datatable') ) {
if ($.fn.DataTable.isDataTable('#datatable')) {
// destroy option too slow on update, really remove elements with destroy method
$('#datatable').DataTable().destroy(true);
$('#datatable')
.DataTable()
.destroy(true);
// recreate original table element, destroy removes all
ele = document.createElement('table');
@ -144,7 +150,14 @@ BR.TrackMessages = L.Class.extend({
},
_setEdges: function(polyline, segments) {
var messages, segLatLngs, length, si, mi, latLng, i, segIndex,
var messages,
segLatLngs,
length,
si,
mi,
latLng,
i,
segIndex,
baseIndex = 0;
// track latLngs index for end node of edge
@ -185,7 +198,10 @@ BR.TrackMessages = L.Class.extend({
endIndex = this._edges[row.index()],
edgeLatLngs = trackLatLngs.slice(startIndex, endIndex + 1);
this._selectedEdge = L.polyline(edgeLatLngs, this.options.edgeStyle).addTo(this._map);
this._selectedEdge = L.polyline(
edgeLatLngs,
this.options.edgeStyle
).addTo(this._map);
},
_handleHoverOut: function(evt) {
@ -193,5 +209,3 @@ BR.TrackMessages = L.Class.extend({
this._selectedEdge = null;
}
});
BR.TrackMessages.include(L.Mixin.Events);

View file

@ -1,13 +1,61 @@
BR.TrackStats = L.Class.extend({
update: function (polyline, segments) {
var stats = this.calcStats(polyline, segments),
length1 = L.Util.formatNum(stats.trackLength/1000,1),
length3 = L.Util.formatNum(stats.trackLength/1000,3),
meanCostFactor = stats.trackLength ? L.Util.formatNum(stats.cost / stats.trackLength, 2) : ''
update: function(polyline, segments) {
if (segments.length == 0) {
$('#distance').html('-');
$('#distance').attr('title', '');
$('#ascend').html('-');
$('#plainascend').html('-');
$('#cost').html('-');
$('#meancostfactor').html('-');
$('#totaltime').html('-');
$('#totalenergy').html('-');
$('#meanenergy').html('-');
return;
}
$('#distance').html(length1 + ' <abbr title="kilometer">km</abbr>');
$('#ascend').html(stats.filteredAscend + ' (' + stats.plainAscend +')' + ' <abbr title="meter">m</abbr>');
$('#cost').html(stats.cost + ' (' + meanCostFactor + ')');
var stats = this.calcStats(polyline, segments),
length1 = L.Util.formatNum(
stats.trackLength / 1000,
1
).toLocaleString(),
length3 = L.Util.formatNum(
stats.trackLength / 1000,
3
).toLocaleString(),
formattedAscend = stats.filteredAscend.toLocaleString(),
formattedPlainAscend = stats.plainAscend.toLocaleString(),
formattedCost = stats.cost.toLocaleString(),
meanCostFactor = stats.trackLength
? L.Util.formatNum(
stats.cost / stats.trackLength,
2
).toLocaleString()
: '0',
formattedTime =
Math.trunc(stats.totalTime / 3600) +
':' +
('0' + Math.trunc((stats.totalTime % 3600) / 60)).slice(-2),
formattedEnergy = L.Util.formatNum(
stats.totalEnergy / 3600000,
2
).toLocaleString(),
meanEnergy = stats.trackLength
? L.Util.formatNum(
stats.totalEnergy / 36 / stats.trackLength,
2
).toLocaleString()
: '0';
$('#distance').html(length1);
// alternative 3-digit format down to meters as tooltip
$('#distance').attr('title', length3 + ' km');
$('#ascend').html(formattedAscend);
$('#plainascend').html(formattedPlainAscend);
$('#cost').html(formattedCost);
$('#meancostfactor').html(meanCostFactor);
$('#totaltime').html(formattedTime);
$('#totalenergy').html(formattedEnergy);
$('#meanenergy').html(meanEnergy);
},
calcStats: function(polyline, segments) {
@ -15,6 +63,8 @@ BR.TrackStats = L.Class.extend({
trackLength: 0,
filteredAscend: 0,
plainAscend: 0,
totalTime: 0,
totalEnergy: 0,
cost: 0
};
var i, props;
@ -24,6 +74,8 @@ BR.TrackStats = L.Class.extend({
stats.trackLength += +props['track-length'];
stats.filteredAscend += +props['filtered ascend'];
stats.plainAscend += +props['plain-ascend'];
stats.totalTime += +props['total-time'];
stats.totalEnergy += +props['total-energy'];
stats.cost += +props['cost'];
}

View file

@ -5,21 +5,26 @@
*/
(function() {
var mapContext;
function verifyTouchStyle(mapContext) {
// revert touch style (large icons) when touch screen detection is available and negative
// see https://github.com/nrenner/brouter-web/issues/69
if (L.Browser.touch && BR.Browser.touchScreenDetectable && !BR.Browser.touchScreen) {
L.DomUtil.removeClass(mapContext.map.getContainer(), 'leaflet-touch');
if (
L.Browser.touch &&
BR.Browser.touchScreenDetectable &&
!BR.Browser.touchScreen
) {
L.DomUtil.removeClass(
mapContext.map.getContainer(),
'leaflet-touch'
);
}
}
function initApp(mapContext) {
var map = mapContext.map,
layersControl = mapContext.layersControl,
mapLayers = mapContext.layers,
search,
router,
routing,
@ -28,20 +33,23 @@
stats,
itinerary,
elevation,
download,
tabs,
exportRoute,
profile,
trackMessages,
sidebar,
drawButton,
deleteButton,
deleteRouteButton,
drawToolbar,
urlHash,
reverseRoute,
saveWarningShown = false;
// By default bootstrap-select use glyphicons
$('.selectpicker').selectpicker({
iconBase: 'fa',
tickIcon: 'fa-check'
tickIcon: 'fa-check',
// don't overlap with footer
windowPadding: [0, 0, 40, 0]
});
search = new BR.Search();
@ -50,45 +58,75 @@
router = L.bRouter(); //brouterCgi dummyRouter
drawButton = L.easyButton({
states: [{
stateName: 'deactivate-draw',
icon: 'fa-pencil active',
onClick: function (control) {
routing.draw(false);
control.state('activate-draw');
states: [
{
stateName: 'deactivate-draw',
icon: 'fa-pencil active',
onClick: function(control) {
routing.draw(false);
control.state('activate-draw');
},
title: i18next.t('map.draw-route-stop')
},
title: 'Stop drawing route (ESC key)'
}, {
stateName: 'activate-draw',
icon: 'fa-pencil',
onClick: function (control) {
routing.draw(true);
control.state('deactivate-draw');
},
title: 'Draw route (D key)'
}]
{
stateName: 'activate-draw',
icon: 'fa-pencil',
onClick: function(control) {
routing.draw(true);
control.state('deactivate-draw');
},
title: i18next.t('map.draw-route-start')
}
]
});
deleteButton = L.easyButton(
reverseRouteButton = L.easyButton(
'fa-random',
function() {
routing.reverse();
},
i18next.t('map.reverse-route')
);
deletePointButton = L.easyButton(
'<span><i class="fa fa-caret-left"></i><i class="fa fa-map-marker" style="margin-left: 1px; color: gray;"></i></span>',
function() {
routing.removeWaypoint(routing.getLast(), function(
err,
data
) {});
},
i18next.t('map.delete-last-point')
);
deleteRouteButton = L.easyButton(
'fa-trash-o',
function () {
bootbox.confirm({
function() {
bootbox.prompt({
size: 'small',
message: "Delete route?",
title: i18next.t('map.delete-route'),
inputType: 'checkbox',
inputOptions: [
{
text: i18next.t('map.delete-nogo-areas'),
value: 'nogo'
}
],
callback: function(result) {
if (result) {
if (result !== null) {
routing.clear();
if (result.length > 0 && result[0] === 'nogo') {
nogos.clear();
}
onUpdate();
urlHash.onMapMove();
}
}
});
},
'Clear route'
i18next.t('map.clear-route')
);
drawToolbar = L.easyBar([drawButton, deleteButton]).addTo(map);
function updateRoute(evt) {
router.setOptions(evt.options);
@ -112,6 +150,11 @@
profile.update(evt.options);
});
BR.NogoAreas.MSG_BUTTON = i18next.t('map.nogo.draw');
BR.NogoAreas.MSG_BUTTON_CANCEL = i18next.t('map.nogo.cancel');
BR.NogoAreas.MSG_CREATE = i18next.t('map.nogo.click-drag');
BR.NogoAreas.MSG_DISABLED = i18next.t('map.nogo.edit');
BR.NogoAreas.MSG_ENABLED = i18next.t('map.nogo.help');
nogos = new BR.NogoAreas();
nogos.on('update', updateRoute);
@ -121,22 +164,26 @@
} else {
stats = new BR.TrackStats();
}
download = new BR.Download();
exportRoute = new BR.Export(router);
elevation = new BR.Elevation();
profile = new BR.Profile();
profile.on('update', function(evt) {
BR.message.hide();
var profileId = routingOptions.getCustomProfile();
router.uploadProfile(profileId, evt.profileText, function(err, profileId) {
router.uploadProfile(profileId, evt.profileText, function(
err,
profileId
) {
if (!err) {
routingOptions.setCustomProfile(profileId, true);
updateRoute({
options: routingOptions.getOptions()
});
if (!saveWarningShown) {
profile.message.showWarning('<strong>Note:</strong> Uploaded custom profiles are only cached temporarily on the server.'
+ '<br/>Please save your edits to your local PC.');
profile.message.showWarning(
i18next.t('warning.temporary-profile')
);
saveWarningShown = true;
}
} else {
@ -156,7 +203,7 @@
profile.message.hide();
routingOptions.setCustomProfile(null);
});
trackMessages = new BR.TrackMessages({
trackMessages = new BR.TrackMessages(map, {
requestUpdate: requestUpdate
});
@ -167,7 +214,9 @@
styles: BR.conf.routingStyles
});
routing.on('routing:routeWaypointEnd routing:setWaypointsEnd', function(evt) {
routing.on('routing:routeWaypointEnd routing:setWaypointsEnd', function(
evt
) {
search.clear();
onUpdate(evt && evt.err);
});
@ -191,8 +240,7 @@
var track = routing.toPolyline(),
segments = routing.getSegments(),
latLngs = routing.getWaypoints(),
segmentsLayer = routing._segments,
urls = {};
segmentsLayer = routing._segments;
elevation.update(track, segmentsLayer);
if (BR.conf.transit) {
@ -202,54 +250,62 @@
}
trackMessages.update(track, segments);
if (latLngs.length > 1) {
urls.gpx = router.getUrl(latLngs, 'gpx');
urls.kml = router.getUrl(latLngs, 'kml');
urls.geojson = router.getUrl(latLngs, 'geojson');
urls.csv = router.getUrl(latLngs, 'csv');
}
download.update(urls);
};
routingOptions.addTo(map);
exportRoute.update(latLngs);
}
routing.addTo(map);
elevation.addBelow(map);
tabs = new BR.Tabs({
tabs: {
'#tab_itinerary': itinerary,
'#tab_data': trackMessages
sidebar = BR.sidebar({
defaultTabId: BR.conf.transit ? 'tab_itinerary' : 'tab_profile',
listeningTabs: {
tab_profile: profile,
tab_data: trackMessages
}
});
if (!BR.conf.transit) {
delete tabs.options.tabs['#tab_itinerary'];
}).addTo(map);
if (BR.conf.transit) {
sidebar.showPanel('tab_itinerary');
}
map.addControl(tabs);
var sidebar = L.control.sidebar('sidebar', {
position: 'left'
});
sidebar.id = 'sidebar-control'; //required for persistence in local storage
map.addControl(sidebar);
nogos.addTo(map);
map.addControl(new BR.OpacitySlider({
callback: L.bind(routing.setOpacity, routing)
}));
drawToolbar = L.easyBar([
drawButton,
reverseRouteButton,
nogos.getButton(),
deletePointButton,
deleteRouteButton
]).addTo(map);
nogos.preventRoutePointOnCreate(routing);
if (BR.keys.strava) {
BR.stravaSegments(map, layersControl);
}
map.addControl(
new BR.OpacitySliderControl({
id: 'route',
title: i18next.t('map.opacity-slider'),
callback: L.bind(routing.setOpacity, routing)
})
);
// initial option settings (after controls are added and initialized with onAdd)
router.setOptions(nogos.getOptions());
router.setOptions(routingOptions.getOptions());
profile.update(routingOptions.getOptions());
// restore active layers from local storage when called without hash
// (check before hash plugin init)
if (!location.hash) {
layersControl.loadActiveLayers();
}
var onHashChangeCb = function(url) {
var url2params = function (s) {
var url2params = function(s) {
s = s.replace(/;/g, '|');
var p = {};
var sep = '&';
if (s.search('&amp;') !== -1)
sep = '&amp;';
if (s.search('&amp;') !== -1) sep = '&amp;';
var params = s.split(sep);
for (var i = 0; i < params.length; i++) {
var tmp = params[i].split('=');
@ -257,7 +313,7 @@
p[tmp[0]] = decodeURIComponent(tmp[1]);
}
return p;
}
};
if (url == null) return;
var opts = router.parseUrlParams(url2params(url));
router.setOptions(opts);
@ -283,35 +339,48 @@
// do not initialize immediately
urlHash = new L.Hash(null, null);
urlHash.additionalCb = function() {
var url = router.getUrl(routing.getWaypoints(), null).substr('brouter?'.length+1);
return url.length > 0 ? '&' + url : null;
};
var url = router
.getUrl(routing.getWaypoints(), null)
.substr('brouter?'.length + 1);
url = url.replace(/\|/g, ';');
return url.length > 0 ? '&' + url : null;
};
urlHash.onHashChangeCb = onHashChangeCb;
urlHash.onInvalidHashChangeCb = onInvalidHashChangeCb;
urlHash.layers = mapLayers;
urlHash.map = map;
urlHash.init(map, mapLayers);
urlHash.init(map, {
layersControl: layersControl
});
// activate configured default base layer or first if no hash,
// only after hash init, by using the same delay
setTimeout(function() {
layersControl.activateDefaultBaseLayer();
}, urlHash.changeDefer);
routingOptions.on('update', urlHash.onMapMove, urlHash);
nogos.on('update', urlHash.onMapMove, urlHash);
// waypoint add, move, delete (but last)
routing.on('routing:routeWaypointEnd', urlHash.onMapMove, urlHash);
// delete last waypoint
routing.on('waypoint:click', function (evt) {
var r = evt.marker._routing;
if (!r.prevMarker && !r.nextMarker) {
urlHash.onMapMove();
}
}, urlHash);
routing.on(
'waypoint:click',
function(evt) {
var r = evt.marker._routing;
if (!r.prevMarker && !r.nextMarker) {
urlHash.onMapMove();
}
},
urlHash
);
$(window).resize(function () {
$(window).resize(function() {
elevation.addBelow(map);
});
$('#elevation-chart').on('show.bs.collapse', function () {
$('#elevation-chart').on('show.bs.collapse', function() {
$('#elevation-btn').addClass('active');
});
$('#elevation-chart').on('hidden.bs.collapse', function () {
$('#elevation-chart').on('hidden.bs.collapse', function() {
$('#elevation-btn').removeClass('active');
// we must fetch tiles that are located behind elevation-chart
map._onResize();
@ -319,36 +388,149 @@
var onHide = function() {
if (this.id && BR.Util.localStorageAvailable()) {
localStorage[this.id] = 'true';
localStorage.removeItem(this.id);
}
};
var onShow = function() {
if (this.id && BR.Util.localStorageAvailable()) {
localStorage.removeItem(this.id);
localStorage[this.id] = 'true';
}
};
var toggleSidebar = function (event) {
sidebar.toggle();
$('#sidebar-btn').toggleClass('active');
};
$('#sidebar-btn').on('click', toggleSidebar);
sidebar.on('shown', onShow);
sidebar.on('hidden', onHide);
// on page load, we want to restore collapsible elements from previous usage
$('.collapse').on('hidden.bs.collapse', onHide)
.on('shown.bs.collapse', onShow)
.each(function() {
if (!(this.id && BR.Util.localStorageAvailable() && localStorage[this.id] === 'true' )) {
$(this).collapse('hide');
}
});
if (BR.Util.localStorageAvailable() && localStorage[sidebar.id] !== 'true') {
toggleSidebar();
}
$('.collapse')
.on('hidden.bs.collapse', onHide)
.on('shown.bs.collapse', onShow)
.each(function() {
if (
this.id &&
BR.Util.localStorageAvailable() &&
localStorage[this.id] === 'true'
) {
$(this).collapse('show');
}
});
$('#submitNogos').on('click', function() {
var geoJSONPromise;
var nogoURL = $('#nogoURL').val();
var nogoFile = $('#nogoFile')[0].files[0];
if (nogoURL) {
// TODO: Handle {{bbox}}
geoJSONPromise = fetch(nogoURL).then(function(response) {
response.json();
});
} else if (nogoFile) {
geoJSONPromise = new Promise(function(resolve, reject) {
var reader = new FileReader();
reader.onload = function() {
resolve(reader.result);
};
reader.readAsText(nogoFile);
}).then(function(response) {
return JSON.parse(response);
});
} else {
$('#nogoError').text('Error: Missing file or URL.');
$('#nogoError').css('display', 'block');
return false;
}
var nogoWeight = parseFloat($('#nogoWeight').val());
if (isNaN(nogoWeight)) {
$('#nogoError').text('Error: Missing default nogo weight.');
$('#nogoError').css('display', 'block');
return false;
}
var nogoRadius = parseFloat($('#nogoRadius').val());
if (isNaN(nogoRadius) || nogoRadius < 0) {
$('#nogoError').text('Error: Invalid default nogo radius.');
$('#nogoError').css('display', 'block');
return false;
}
var nogoBuffer = parseFloat($('#nogoBuffer').val());
if (isNaN(nogoBuffer)) {
$('#nogoError').text('Error: Invalid nogo buffering radius.');
$('#nogoError').css('display', 'block');
return false;
}
geoJSONPromise.then(function(response) {
// Iterate on features in order to discard features without geometry
var cleanedGeoJSONFeatures = [];
turf.featureEach(response, function(feature) {
if (turf.getGeom(feature)) {
var maybeBufferedFeature = feature;
// Eventually buffer GeoJSON
if (nogoBuffer != 0) {
maybeBufferedFeature = turf.buffer(
maybeBufferedFeature,
nogoBuffer,
{ units: 'meters' }
);
}
cleanedGeoJSONFeatures.push(maybeBufferedFeature);
}
});
var geoJSON = L.geoJson(
turf.featureCollection(cleanedGeoJSONFeatures),
{
onEachFeature: function(feature, layer) {
layer.options.nogoWeight =
feature.properties.nogoWeight || nogoWeight;
}
}
);
var nogosPoints = geoJSON.getLayers().filter(function(e) {
return e.feature.geometry.type === 'Point';
});
nogosPoints = nogosPoints.map(function(item) {
var radius = item.feature.properties.radius || nogoRadius;
if (radius > 0) {
return L.circle(item.getLatLng(), { radius: radius });
}
return null;
});
nogosPoints = nogosPoints.filter(function(e) {
return e;
});
nogos.setOptions({
nogos: nogosPoints,
polygons: geoJSON.getLayers().filter(function(e) {
return e.feature.geometry.type === 'Polygon';
}),
polylines: geoJSON.getLayers().filter(function(e) {
return e.feature.geometry.type === 'LineString';
})
});
updateRoute({
options: nogos.getOptions()
});
urlHash.onMapMove();
$('#nogoError').text('');
$('#nogoError').css('display', 'none');
$('#loadNogos').modal('hide');
});
return false;
});
}
mapContext = BR.Map.initMap();
verifyTouchStyle(mapContext);
initApp(mapContext);
i18next
.use(window.i18nextXHRBackend)
.use(window.i18nextBrowserLanguageDetector)
.init(
{
fallbackLng: 'en',
backend: {
loadPath: 'dist/locales/{{lng}}.json'
}
},
function(err, t) {
jqueryI18next.init(i18next, $);
$('html').localize();
mapContext = BR.Map.initMap();
verifyTouchStyle(mapContext);
initApp(mapContext);
}
);
})();

View file

@ -1,17 +1,19 @@
BR.BingLayer = L.BingLayer.extend({
options: {
maxZoom: 19,
attribution: '<a target="_blank" href="https://www.bing.com/maps/">Bing Maps</a>'
+ ' (<a target="_blank" href="https://go.microsoft.com/?linkid=9710837">TOU</a>)'
attribution:
'<a target="_blank" href="https://www.bing.com/maps/">Bing Maps</a>' +
' (<a target="_blank" href="https://go.microsoft.com/?linkid=9710837">TOU</a>)'
},
initialize: function(key, options) {
L.BingLayer.prototype.initialize.call(this, key, options);
this._logo = L.control({position: 'bottomleft'});
this._logo.onAdd = function (map) {
this._logo = L.control({ position: 'bottomleft' });
this._logo.onAdd = function(map) {
this._div = L.DomUtil.create('div', 'bing-logo');
this._div.innerHTML = '<img src="https://www.microsoft.com/maps/images/branding/Bing%20logo%20white_50px-19px.png">';
this._div.innerHTML =
'<img src="https://www.microsoft.com/maps/images/branding/Bing%20logo%20white_50px-19px.png">';
return this._div;
};
},

View file

@ -1,33 +1,47 @@
BR.Elevation = L.Control.Elevation.extend({
options: {
width:$('#map').outerWidth(),
width: $('#map').outerWidth(),
margins: {
top: 20,
right: 30,
bottom: 30,
left: 60
},
theme: "steelblue-theme"
theme: 'steelblue-theme'
},
onAdd: function (map) {
onAdd: function(map) {
var container = L.Control.Elevation.prototype.onAdd.call(this, map);
// revert registering touch events when touch screen detection is available and negative
// see https://github.com/MrMufflon/Leaflet.Elevation/issues/67
if (L.Browser.touch && BR.Browser.touchScreenDetectable && !BR.Browser.touchScreen) {
this._background.on("touchmove.drag", null).
on("touchstart.drag", null).
on("touchstart.focus", null);
L.DomEvent.off(this._container, 'touchend', this._dragEndHandler, this);
this._background.on("mousemove.focus", this._mousemoveHandler.bind(this)).
on("mouseout.focus", this._mouseoutHandler.bind(this)).
on("mousedown.drag", this._dragStartHandler.bind(this)).
on("mousemove.drag", this._dragHandler.bind(this));
L.DomEvent.on(this._container, 'mouseup', this._dragEndHandler, this);
if (
L.Browser.touch &&
BR.Browser.touchScreenDetectable &&
!BR.Browser.touchScreen
) {
this._background
.on('touchmove.drag', null)
.on('touchstart.drag', null)
.on('touchstart.focus', null);
L.DomEvent.off(
this._container,
'touchend',
this._dragEndHandler,
this
);
this._background
.on('mousemove.focus', this._mousemoveHandler.bind(this))
.on('mouseout.focus', this._mouseoutHandler.bind(this))
.on('mousedown.drag', this._dragStartHandler.bind(this))
.on('mousemove.drag', this._dragHandler.bind(this));
L.DomEvent.on(
this._container,
'mouseup',
this._dragEndHandler,
this
);
}
return container;
@ -36,7 +50,7 @@ BR.Elevation = L.Control.Elevation.extend({
addBelow: function(map) {
// waiting for https://github.com/MrMufflon/Leaflet.Elevation/pull/66
// this.width($('#map').outerWidth());
this.options.width = $('#map').outerWidth();
this.options.width = $('#content').outerWidth();
if (this.getContainer() != null) {
this.remove(map);
@ -44,10 +58,13 @@ BR.Elevation = L.Control.Elevation.extend({
function setParent(el, newParent) {
newParent.appendChild(el);
}
this.addTo(map);
}
this.addTo(map);
// move elevation graph outside of the map
setParent(this.getContainer(), document.getElementById('elevation-chart'));
setParent(
this.getContainer(),
document.getElementById('elevation-chart')
);
},
update: function(track, layer) {
@ -62,7 +79,7 @@ BR.Elevation = L.Control.Elevation.extend({
if (track && track.getLatLngs().length > 0) {
this.addData(track.toGeoJSON(), layer);
layer.on("mouseout", this._hidePositionMarker.bind(this));
layer.on('mouseout', this._hidePositionMarker.bind(this));
}
}
});

View file

@ -1,76 +1,504 @@
L.drawLocal.draw.toolbar.buttons.circle = 'Draw no-go area (circle)';
L.drawLocal.edit.toolbar.buttons.edit = 'Edit no-go areas';
L.drawLocal.edit.toolbar.buttons.remove = 'Delete no-go areas';
BR.NogoAreas = L.Control.Draw.extend({
initialize: function () {
this.drawnItems = new L.FeatureGroup();
L.Control.Draw.prototype.initialize.call(this, {
draw: {
position: 'topleft',
polyline: false,
polygon: false,
circle: true,
rectangle: false,
marker: false
},
edit: {
featureGroup: this.drawnItems,
//edit: false,
edit: {
selectedPathOptions: {
//opacity: 0.8
}
},
remove: true
}
});
BR.NogoAreas = L.Control.extend({
statics: {
MSG_BUTTON: 'Draw no-go area (circle)',
MSG_BUTTON_CANCEL: 'Cancel drawing no-go area',
MSG_CREATE: 'Click and drag to draw circle',
MSG_DISABLED: 'Click to edit',
MSG_ENABLED:
'&square; = move / resize, <span class="fa fa-trash-o"></span> = delete,<br>click nogo to quit editing',
STATE_CREATE: 'no-go-create',
STATE_CANCEL: 'cancel-no-go-create'
},
onAdd: function (map) {
map.addLayer(this.drawnItems);
style: {
color: '#f06eaa',
weight: 4,
opacity: 0.5,
fillColor: null, //same as color by default
fillOpacity: 0.2,
dashArray: null
},
map.on('draw:created', function (e) {
var layer = e.layer;
this.drawnItems.addLayer(layer);
this._fireUpdate();
}, this);
editStyle: {
color: '#fe57a1',
opacity: 0.6,
dashArray: '10, 10',
fillOpacity: 0.1
},
map.on('draw:editstart', function (e) {
this.drawnItems.eachLayer(function (layer) {
layer.on('edit', function(e) {
this._fireUpdate();
}, this);
}, this);
}, this);
initialize: function() {
this._wasRouteDrawing = false;
},
map.on('draw:deleted', function (e) {
this._fireUpdate();
}, this);
onAdd: function(map) {
var self = this;
return L.Control.Draw.prototype.onAdd.call(this, map);
this.drawnItems = new L.FeatureGroup().addTo(map);
this.drawnItems.on('click', function(e) {
L.DomEvent.stop(e);
e.layer.toggleEdit();
});
var editTools = (this.editTools = map.editTools = new L.Editable(map, {
circleEditorClass: BR.DeletableCircleEditor,
// FeatureGroup instead of LayerGroup to propagate events to members
editLayer: new L.FeatureGroup().addTo(map),
featuresLayer: this.drawnItems
}));
this.button = L.easyButton({
states: [
{
stateName: BR.NogoAreas.STATE_CREATE,
icon: 'fa-ban',
title: BR.NogoAreas.MSG_BUTTON,
onClick: function(control) {
// initial radius of 0 to detect click, see DeletableCircleEditor.onDrawingMouseUp
var opts = L.extend({ radius: 0 }, self.style);
editTools.startCircle(null, opts);
control.state('cancel-no-go-create');
}
},
{
stateName: BR.NogoAreas.STATE_CANCEL,
icon: 'fa-ban active',
title: BR.NogoAreas.MSG_BUTTON_CANCEL,
onClick: function(control) {
editTools.stopDrawing();
control.state('no-go-create');
}
}
]
});
this.editTools.on(
'editable:drawing:end',
function(e) {
self.button.state(BR.NogoAreas.STATE_CREATE);
setTimeout(
L.bind(function() {
// turn editing off after create; async to still fire 'editable:vertex:dragend'
e.layer.disableEdit();
}, this),
0
);
},
this
);
this.editTools.on(
'editable:vertex:dragend editable:deleted',
function(e) {
this._fireUpdate();
},
this
);
this.editTools.on(
'editable:enable',
function(e) {
e.layer.setStyle(this.editStyle);
},
this
);
this.editTools.on(
'editable:disable',
function(e) {
e.layer.setStyle(this.style);
},
this
);
this.tooltip = new BR.EditingTooltip(map, editTools, this.button);
this.tooltip.enable();
// dummy, no own representation, delegating to EasyButton
return L.DomUtil.create('div');
},
// prevent route waypoint added after circle create (map click after up)
preventRoutePointOnCreate: function(routing) {
this.editTools.on(
'editable:drawing:start',
function(e) {
this._wasRouteDrawing = routing.isDrawing();
routing.draw(false);
},
this
);
// after create
this.editTools.on(
'editable:drawing:end',
function(e) {
if (this._wasRouteDrawing) {
setTimeout(function() {
routing.draw(true);
}, 0);
}
},
this
);
},
getOptions: function() {
return {
nogos: this.drawnItems.getLayers()
nogos: this.drawnItems.getLayers().filter(function(e) {
return e instanceof L.Circle;
}),
polygons: this.drawnItems.getLayers().filter(function(e) {
return e instanceof L.Polygon;
}),
polylines: this.drawnItems.getLayers().filter(function(e) {
return e instanceof L.Polyline && !(e instanceof L.Polygon);
})
};
},
setOptions: function(options) {
var nogos = options.nogos;
this.drawnItems.clearLayers();
var polylines = options.polylines;
var polygons = options.polygons;
this._clear();
if (nogos) {
for (var i = 0; i < nogos.length; i++) {
nogos[i].setStyle(this.style);
this.drawnItems.addLayer(nogos[i]);
}
}
if (polylines) {
for (var i = 0; i < polylines.length; i++) {
polylines[i].setStyle(this.style);
this.drawnItems.addLayer(polylines[i]);
}
}
if (polygons) {
for (var i = 0; i < polygons.length; i++) {
polygons[i].setStyle(this.style);
this.drawnItems.addLayer(polygons[i]);
}
}
},
_fireUpdate: function () {
this.fire('update', {options: this.getOptions()});
_clear: function() {
this.drawnItems.clearLayers();
},
clear: function() {
this._clear();
this._fireUpdate();
},
_fireUpdate: function() {
this.fire('update', { options: this.getOptions() });
},
getFeatureGroup: function() {
return this.drawnItems;
},
getEditGroup: function() {
return this.editTools.editLayer;
},
getButton: function() {
return this.button;
}
});
BR.NogoAreas.include(L.Mixin.Events);
BR.NogoAreas.include(L.Evented.prototype);
L.Editable.prototype.createVertexIcon = function(options) {
return BR.Browser.touch
? new L.Editable.TouchVertexIcon(options)
: new L.Editable.VertexIcon(options);
};
BR.EditingTooltip = L.Handler.extend({
options: {
closeTimeout: 2000
},
initialize: function(map, editTools, button) {
this.map = map;
this.editTools = editTools;
this.button = button;
},
addHooks: function() {
// hack: listen to EasyButton click (instead of editable:drawing:start),
// to get mouse position from event for initial tooltip location
L.DomEvent.addListener(
this.button.button,
'click',
this._addCreate,
this
);
this.editTools.featuresLayer.on('layeradd', this._bind, this);
this.editTools.on('editable:drawing:end', this._postCreate, this);
this.editTools.on('editable:enable', this._enable, this);
this.editTools.on('editable:disable', this._disable, this);
},
removeHooks: function() {
L.DomEvent.removeListener(
this.button.button,
'click',
this._addCreate,
this
);
this.editTools.featuresLayer.off('layeradd', this._bind, this);
this.editTools.off('editable:drawing:end', this._postCreate, this);
this.editTools.off('editable:enable', this._enable, this);
this.editTools.off('editable:disable', this._disable, this);
},
_bind: function(e) {
// Position tooltip at bottom of circle, less distracting than
// sticky with cursor or at center.
var layer = e.layer;
layer.bindTooltip(BR.NogoAreas.MSG_DISABLED, {
direction: 'bottom',
className: 'editing-tooltip'
});
// Override to set position to south instead of center (circle latlng);
// works better with zooming than updating offset to match radius
layer.openTooltip = function(layer, latlng) {
if (!latlng && layer instanceof L.Layer) {
latlng = L.latLng(
layer.getBounds().getSouth(),
0.5 *
(layer.getBounds().getWest() +
layer.getBounds().getEast())
);
}
L.Layer.prototype.openTooltip.call(this, layer, latlng);
};
},
_addCreate: function(e) {
// button cancel
if (!this.editTools.drawing()) return;
var initialLatLng = this.map.mouseEventToLatLng(e);
var tooltip = L.tooltip({
// no effect with map tooltip
sticky: true,
// offset wrong with 'auto' when switching direction
direction: 'right',
offset: L.point(5, 28),
className: 'editing-tooltip-create'
});
// self-reference hack for _moveTooltip, as tooltip is not bound to layer
tooltip._tooltip = tooltip;
// simulate sticky feature (follow mouse) for map tooltip without layer
var onOffMove = function(e) {
var onOff = e.type === 'tooltipclose' ? 'off' : 'on';
this._map[onOff]('mousemove', this._moveTooltip, this);
};
this.map.on('tooltipopen', onOffMove, tooltip);
this.map.on('tooltipclose', onOffMove, tooltip);
var onTooltipRemove = function(e) {
this.map.off('tooltipopen', onOffMove, e.tooltip);
this.map.off('tooltipclose', onOffMove, e.tooltip);
this.map.off('tooltipclose', onTooltipRemove, this);
e.tooltip._tooltip = null;
};
this.map.on('tooltipclose', onTooltipRemove, this);
tooltip.setTooltipContent(BR.NogoAreas.MSG_CREATE);
this.map.openTooltip(tooltip, initialLatLng);
var closeTooltip = function() {
this.map.closeTooltip(tooltip);
};
this.editTools.once(
'editable:editing editable:drawing:cancel',
closeTooltip,
this
);
if (BR.Browser.touch) {
// can't move with cursor on touch devices, so show at start pos for a few seconds
setTimeout(L.bind(closeTooltip, this), this.options.closeTimeout);
}
},
_setCloseTimeout: function(layer) {
var timeoutId = setTimeout(function() {
layer.closeTooltip();
}, this.options.closeTimeout);
// prevent timer to close tooltip that changed in the meantime
layer.once('tooltipopen', function(e) {
clearTimeout(timeoutId);
});
},
_postCreate: function() {
// editing is disabled by another handler, tooltip won't stay open before
this.editTools.once(
'editable:disable',
function(e) {
// show for a few seconds, as mouse often not hovering circle after create
e.layer.openTooltip(e.layer);
this._setCloseTimeout(e.layer);
},
this
);
},
_enable: function(e) {
e.layer.setTooltipContent(BR.NogoAreas.MSG_ENABLED);
this.editTools.once(
'editable:editing',
function(e) {
e.layer.closeTooltip();
},
this
);
},
_disable: function(e) {
e.layer.setTooltipContent(BR.NogoAreas.MSG_DISABLED);
this._setCloseTimeout(e.layer);
}
});
BR.DeletableCircleEditor = L.Editable.CircleEditor.extend({
_computeDeleteLatLng: function() {
// While circle is not added to the map, _radius is not set.
var delta =
(this.feature._radius || this.feature._mRadius) *
Math.cos(Math.PI / 4),
point = this.map.project(this.feature._latlng);
return this.map.unproject([point.x - delta, point.y - delta]);
},
_updateDeleteLatLng: function() {
this._deleteLatLng.update(this._computeDeleteLatLng());
this._deleteLatLng.__vertex.update();
},
_addDeleteMarker: function() {
if (!this.enabled()) return;
this._deleteLatLng = this._computeDeleteLatLng();
return new BR.DeleteMarker(this._deleteLatLng, this);
},
_delete: function() {
this.disable();
this.tools.featuresLayer.removeLayer(this.feature);
},
delete: function() {
this._delete();
this.fireAndForward('editable:deleted');
},
initialize: function(map, feature, options) {
L.Editable.CircleEditor.prototype.initialize.call(
this,
map,
feature,
options
);
this._deleteLatLng = this._computeDeleteLatLng();
// FeatureGroup instead of LayerGroup to propagate events to members
this.editLayer = new L.FeatureGroup();
},
addHooks: function() {
L.Editable.CircleEditor.prototype.addHooks.call(this);
if (this.feature) {
this._addDeleteMarker();
}
return this;
},
reset: function() {
L.Editable.CircleEditor.prototype.reset.call(this);
this._addDeleteMarker();
},
onDrawingMouseDown: function(e) {
this._deleteLatLng.update(e.latlng);
L.Editable.CircleEditor.prototype.onDrawingMouseDown.call(this, e);
},
// override to cancel/remove created circle when added by click instead of drag, because:
// - without resize, edit handles stacked on top of each other
// - makes event handling more complicated (editable:vertex:dragend not called)
onDrawingMouseUp: function(e) {
if (this.feature.getRadius() > 0) {
this.commitDrawing(e);
} else {
this.cancelDrawing(e);
this._delete();
}
e.originalEvent._simulated = false;
L.Editable.PathEditor.prototype.onDrawingMouseUp.call(this, e);
},
onVertexMarkerDrag: function(e) {
this._updateDeleteLatLng();
L.Editable.CircleEditor.prototype.onVertexMarkerDrag.call(this, e);
}
});
BR.DeleteMarker = L.Marker.extend({
options: {
draggable: false,
icon: L.divIcon({
iconSize: BR.Browser.touch
? new L.Point(24, 24)
: new L.Point(16, 16),
className: 'leaflet-div-icon fa fa-trash-o nogo-delete-marker'
})
},
initialize: function(latlng, editor, options) {
// derived from L.Editable.VertexMarker.initialize
// We don't use this._latlng, because on drag Leaflet replace it while
// we want to keep reference.
this.latlng = latlng;
this.editor = editor;
L.Marker.prototype.initialize.call(this, latlng, options);
this.latlng.__vertex = this;
this.editor.editLayer.addLayer(this);
// to keep small circles editable, make sure delete button is below drag handle
// (not using "+ 1" to place at bottom of other vertex markers)
this.setZIndexOffset(editor.tools._lastZIndex);
},
onAdd: function(map) {
L.Marker.prototype.onAdd.call(this, map);
this.on('click', this.onClick);
},
onRemove: function(map) {
delete this.latlng.__vertex;
this.off('click', this.onClick);
L.Marker.prototype.onRemove.call(this, map);
},
onClick: function(e) {
this.editor.delete();
}
});

View file

@ -1,7 +1,7 @@
L.Routing.Draw.prototype._hideTrailer = function() {
if (this._trailer.options.opacity !== 0.0) {
this._trailer.setStyle({opacity: 0.0});
}
if (this._trailer.options.opacity !== 0.0) {
this._trailer.setStyle({ opacity: 0.0 });
}
};
BR.Routing = L.Routing.extend({
@ -20,7 +20,7 @@ BR.Routing = L.Routing.extend({
zIndexOffset: -2000
},
onAdd: function (map) {
onAdd: function(map) {
this._segmentsCasing = new L.FeatureGroup().addTo(map);
var container = L.Routing.prototype.onAdd.call(this, map);
@ -31,35 +31,50 @@ BR.Routing = L.Routing.extend({
this._waypoints.on('layeradd', this._setMarkerOpacity, this);
// turn line mouse marker off while over waypoint marker
this.on('waypoint:mouseover', function(e) {
// L.Routing.Edit._segmentOnMouseout without firing 'segment:mouseout' (enables draw)
if (this._dragging) { return; }
this.on(
'waypoint:mouseover',
function(e) {
// L.Routing.Edit._segmentOnMouseout without firing 'segment:mouseout' (enables draw)
if (this._dragging) {
return;
}
this._mouseMarker.setOpacity(0.0);
this._map.off('mousemove', this._segmentOnMousemove, this);
this._suspended = true;
}, this._edit);
this._mouseMarker.setOpacity(0.0);
this._map.off('mousemove', this._segmentOnMousemove, this);
this._suspended = true;
},
this._edit
);
this.on('waypoint:mouseout', function(e) {
this._segmentOnMouseover(e);
this._suspended = false;
}, this._edit);
this.on(
'waypoint:mouseout',
function(e) {
this._segmentOnMouseover(e);
this._suspended = false;
},
this._edit
);
this._edit._mouseMarker.setIcon(L.divIcon({
className: 'line-mouse-marker'
,iconAnchor: [8, 8] // size/2 + border/2
,iconSize: [16, 16]
}));
this._edit._mouseMarker.setIcon(
L.divIcon({
className: 'line-mouse-marker',
iconAnchor: [8, 8], // size/2 + border/2
iconSize: [16, 16]
})
);
// Forward mousemove event to snapped feature (for Leaflet.Elevation to
// update indicator), see also L.Routing.Edit._segmentOnMousemove
this._edit._mouseMarker.on('move', L.bind(function(e) {
var latLng = e.latlng;
if (latLng._feature) {
this._mouseMarker._feature = latLng._feature;
latLng._feature.fire('mousemove', e, true);
}
}, this._edit));
this._edit._mouseMarker.on(
'move',
L.bind(function(e) {
var latLng = e.latlng;
if (latLng._feature) {
this._mouseMarker._feature = latLng._feature;
latLng._feature.fire('mousemove', e, true);
}
}, this._edit)
);
var mouseoutHandler = function(e) {
if (this._mouseMarker._feature) {
this._mouseMarker._feature.fire('mouseout', e, true);
@ -79,12 +94,20 @@ BR.Routing = L.Routing.extend({
// intercept listener: only re-show draw trailer after marker hover
// when edit is not active (i.e. wasn't also supended)
this._parent.off('waypoint:mouseout' , this._catchWaypointEvent, this);
this.on('waypoint:mouseout' , function(e) {
if (!this._parent._edit._suspended) {
this._catchWaypointEvent(e);
}
}, this);
this._parent.off(
'waypoint:mouseout',
this._catchWaypointEvent,
this
);
this.on(
'waypoint:mouseout',
function(e) {
if (!this._parent._edit._suspended) {
this._catchWaypointEvent(e);
}
},
this
);
});
this._draw.on('disabled', function() {
L.DomUtil.removeClass(map.getContainer(), 'routing-draw-enabled');
@ -111,7 +134,12 @@ BR.Routing = L.Routing.extend({
this._map.off('mouseout', hide, this);
this._map.off('mouseover', show, this);
L.DomEvent.off(this._map._controlContainer, 'mouseout', show, this);
L.DomEvent.off(this._map._controlContainer, 'mouseover', hide, this);
L.DomEvent.off(
this._map._controlContainer,
'mouseover',
hide,
this
);
});
// Call show after deleting last waypoint, but hide trailer.
@ -119,180 +147,216 @@ BR.Routing = L.Routing.extend({
// mouseout to show again never fires when deleted. Click handler
// _onMouseClick aborts when hidden, so no waypoint can be added
// although enabled.
this.on('waypoint:click', function() {
if (this._hidden && !this._parent._waypoints._first) {
this._show();
this._hideTrailer();
}
}, this._draw);
this.on(
'waypoint:click',
function() {
if (this._hidden && !this._parent._waypoints._first) {
this._show();
this._hideTrailer();
}
},
this._draw
);
// keys not working when map container does not have focus, use document instead
L.DomEvent.removeListener(this._container, 'keyup', this._keyupListener);
L.DomEvent.removeListener(
this._container,
'keyup',
this._keyupListener
);
L.DomEvent.addListener(document, 'keyup', this._keyupListener, this);
// enable drawing mode
this.draw(true);
return container;
}
},
,_addSegmentCasing: function(e) {
var casing = L.polyline(e.layer.getLatLngs(), this.options.styles.trackCasing);
this._segmentsCasing.addLayer(casing);
e.layer._casing = casing;
this._segments.bringToFront();
}
_addSegmentCasing: function(e) {
var casing = L.polyline(
e.layer.getLatLngs(),
this.options.styles.trackCasing
);
this._segmentsCasing.addLayer(casing);
e.layer._casing = casing;
this._segments.bringToFront();
},
,_removeSegmentCasing: function(e) {
this._segmentsCasing.removeLayer(e.layer._casing);
}
_removeSegmentCasing: function(e) {
this._segmentsCasing.removeLayer(e.layer._casing);
},
,setOpacity: function(opacity) {
// Due to the second Polyline layer for casing, the combined opacity is less
// transparent than with a single layer and the slider is non-linear. The
// inverted formula is used to get the same result as with a single layer.
// SVG simple alpha compositing: Ca' = 1 - (1 - Ea) * (1 - Ca)
// https://www.w3.org/TR/SVG11/masking.html#SimpleAlphaBlending
var sourceOpacity = 1 - Math.sqrt(1 - opacity);
setOpacity: function(opacity) {
// Due to the second Polyline layer for casing, the combined opacity is less
// transparent than with a single layer and the slider is non-linear. The
// inverted formula is used to get the same result as with a single layer.
// SVG simple alpha compositing: Ca' = 1 - (1 - Ea) * (1 - Ca)
// https://www.w3.org/TR/SVG11/masking.html#SimpleAlphaBlending
var sourceOpacity = 1 - Math.sqrt(1 - opacity);
this.options.styles.track.opacity = sourceOpacity;
this.options.styles.trackCasing.opacity = sourceOpacity;
this.options.icons.opacity = opacity;
this.options.styles.track.opacity = sourceOpacity;
this.options.styles.trackCasing.opacity = sourceOpacity;
this.options.icons.opacity = opacity;
this._segments.setStyle({
opacity: sourceOpacity
});
this._segmentsCasing.setStyle({
opacity: sourceOpacity
});
this._waypoints.eachLayer(function(marker) {
marker.setOpacity(opacity);
});
}
,_setMarkerOpacity: function(e) {
e.layer.setOpacity(this.options.icons.opacity);
}
,_removeMarkerEvents: function(marker) {
marker.off('mouseover', this._fireWaypointEvent, this);
marker.off('mouseout' , this._fireWaypointEvent, this);
marker.off('dragstart', this._fireWaypointEvent, this);
marker.off('dragend' , this._fireWaypointEvent, this);
marker.off('drag' , this._fireWaypointEvent, this);
marker.off('click' , this._fireWaypointEvent, this);
}
,clear: function() {
var drawEnabled = this._draw._enabled;
var current = this._waypoints._first;
this.draw(false);
if (current === null) { return; }
this._removeMarkerEvents(current);
while (current._routing.nextMarker) {
var marker = current._routing.nextMarker;
this._removeMarkerEvents(marker);
current = marker;
};
this._waypoints._first = null;
this._waypoints._last = null;
this._waypoints.clearLayers();
this._segments.clearLayers();
if (drawEnabled) {
this.draw(true);
}
}
,setWaypoints: function(latLngs, cb) {
var i;
var callbackCount = 0;
var firstErr;
var $this = this;
var callback = function(err, data) {
callbackCount++;
firstErr = firstErr || err;
if (callbackCount >= latLngs.length) {
$this.fire('routing:setWaypointsEnd', { err: firstErr });
if (cb) {
cb(firstErr);
}
}
};
this.fire('routing:setWaypointsStart');
for (i = 0; latLngs && i < latLngs.length; i++) {
this.addWaypoint(latLngs[i], this._waypoints._last, null, callback);
}
}
// patch to fix error when line is null or error line
// (when called while still segments to calculate, e.g. permalink or fast drawing)
,toPolyline: function() {
var latLngs = [];
this._eachSegment(function(m1, m2, line) {
// omit if null (still calculating) or error
// NOTE: feature check specific to BRouter GeoJSON response, workaround to detect error line
if (line && line.feature) {
latLngs = latLngs.concat(line.getLatLngs());
}
});
return L.polyline(latLngs);
}
,_routeSegment: function(m1, m2, cb) {
var loadingTrailer;
// change segment color before request to indicate recalculation (mark old)
if (m1 && m1._routing.nextLine !== null) {
m1._routing.nextLine.setStyle({color: 'dimgray' });
}
// animate dashed trailer as loading indicator
if (m1 && m2) {
loadingTrailer = new L.Polyline([m1.getLatLng(), m2.getLatLng()], {
color: this.options.styles.track.color,
opacity: this.options.styles.trailer.opacity,
dashArray: [10, 10],
className: 'loading-trailer'
this._segments.setStyle({
opacity: sourceOpacity
});
loadingTrailer.addTo(this._map);
}
this._segmentsCasing.setStyle({
opacity: sourceOpacity
});
this._waypoints.eachLayer(function(marker) {
marker.setOpacity(opacity);
});
},
L.Routing.prototype._routeSegment.call(this, m1, m2, L.bind(function(err, data) {
if (loadingTrailer) {
this._map.removeLayer(loadingTrailer);
_setMarkerOpacity: function(e) {
e.layer.setOpacity(this.options.icons.opacity);
},
_removeMarkerEvents: function(marker) {
marker.off('mouseover', this._fireWaypointEvent, this);
marker.off('mouseout', this._fireWaypointEvent, this);
marker.off('dragstart', this._fireWaypointEvent, this);
marker.off('dragend', this._fireWaypointEvent, this);
marker.off('drag', this._fireWaypointEvent, this);
marker.off('click', this._fireWaypointEvent, this);
},
clear: function() {
var drawEnabled = this._draw._enabled;
var current = this._waypoints._first;
this.draw(false);
if (current === null) {
return;
}
this._removeMarkerEvents(current);
while (current._routing.nextMarker) {
var marker = current._routing.nextMarker;
this._removeMarkerEvents(marker);
current = marker;
}
cb(err, data);
}, this));
}
,getSegments: function() {
var segments = [];
this._waypoints._first = null;
this._waypoints._last = null;
this._waypoints.clearLayers();
this._segments.clearLayers();
this._eachSegment(function(m1, m2, line) {
// omit if null (still calculating) or error
// NOTE: feature check specific to BRouter GeoJSON response, workaround to detect error line
if (line && line.feature) {
segments.push(line);
}
});
if (drawEnabled) {
this.draw(true);
}
},
return segments;
}
setWaypoints: function(latLngs, cb) {
var i;
var callbackCount = 0;
var firstErr;
var $this = this;
// add 'esc' to disable drawing
,_keyupListener: function (e) {
if (e.keyCode === 27) {
this._draw.disable();
} else {
L.Routing.prototype._keyupListener.call(this, e);
var callback = function(err, data) {
callbackCount++;
firstErr = firstErr || err;
if (callbackCount >= latLngs.length) {
$this.fire('routing:setWaypointsEnd', { err: firstErr });
if (cb) {
cb(firstErr);
}
}
};
this.fire('routing:setWaypointsStart');
for (i = 0; latLngs && i < latLngs.length; i++) {
this.addWaypoint(latLngs[i], this._waypoints._last, null, callback);
}
},
// patch to fix error when line is null or error line
// (when called while still segments to calculate, e.g. permalink or fast drawing)
toPolyline: function() {
var latLngs = [];
this._eachSegment(function(m1, m2, line) {
// omit if null (still calculating) or error
// NOTE: feature check specific to BRouter GeoJSON response, workaround to detect error line
if (line && line.feature) {
latLngs = latLngs.concat(line.getLatLngs());
}
});
return L.polyline(latLngs);
},
_routeSegment: function(m1, m2, cb) {
var loadingTrailer;
// change segment color before request to indicate recalculation (mark old)
if (m1 && m1._routing.nextLine !== null) {
m1._routing.nextLine.setStyle({ color: 'dimgray' });
}
// animate dashed trailer as loading indicator
if (m1 && m2) {
loadingTrailer = new L.Polyline([m1.getLatLng(), m2.getLatLng()], {
color: this.options.styles.track.color,
opacity: this.options.styles.trailer.opacity,
dashArray: [10, 10],
className: 'loading-trailer'
});
loadingTrailer.addTo(this._map);
}
L.Routing.prototype._routeSegment.call(
this,
m1,
m2,
L.bind(function(err, data) {
if (loadingTrailer) {
this._map.removeLayer(loadingTrailer);
}
cb(err, data);
}, this)
);
},
getSegments: function() {
var segments = [];
this._eachSegment(function(m1, m2, line) {
// omit if null (still calculating) or error
// NOTE: feature check specific to BRouter GeoJSON response, workaround to detect error line
if (line && line.feature) {
segments.push(line);
}
});
return segments;
},
_keyupListener: function(e) {
// Suppress shortcut handling when a text input field is focussed
if (
document.activeElement.type == 'text' ||
document.activeElement.type == 'textarea'
) {
return;
}
// add 'esc' to disable drawing
if (e.keyCode === 27) {
this._draw.disable();
} else {
L.Routing.prototype._keyupListener.call(this, e);
}
},
isDrawing: function() {
return this._draw._enabled;
},
reverse: function() {
var waypoints = this.getWaypoints();
waypoints.reverse();
this.clear();
this.setWaypoints(waypoints);
}
}
});

View file

@ -1,18 +1,14 @@
BR.Search = L.Control.Geocoder.extend({
options: {
geocoder: new L.Control.Geocoder.Nominatim({
serviceUrl: 'https://nominatim.openstreetmap.org/'
geocoder: new L.Control.Geocoder.LatLng({
next: new L.Control.Geocoder.Nominatim({
serviceUrl: 'https://nominatim.openstreetmap.org/'
}),
sizeInMeters: 800
}),
position: 'topleft'
},
onAdd: function (map) {
map.attributionControl.addAttribution(
'search by <a href="https://wiki.openstreetmap.org/wiki/Nominatim" target="_blank">Nominatim</a>');
return L.Control.Geocoder.prototype.onAdd.call(this, map);
},
markGeocode: function(result) {
this._map.fitBounds(result.geocode.bbox, {
maxZoom: 17

115
js/plugin/Sidebar.js Normal file
View file

@ -0,0 +1,115 @@
BR.Sidebar = L.Control.Sidebar.extend({
storageId: 'sidebar-control',
options: {
position: 'right',
container: 'sidebar',
tabContainer: 'sidebarTabs',
autopan: false,
defaultTabId: '',
// Tabs to be notified when shown or hidden
// (tab div id -> object implementing show/hide methods)
listeningTabs: {}
},
initialize: function(id, options) {
L.Control.Sidebar.prototype.initialize.call(this, id, options);
this.oldTab = null;
},
addTo: function(map) {
L.Control.Sidebar.prototype.addTo.call(this, map);
this.on('content', this._notifyOnContent, this);
this.on('closing', this._notifyOnClose, this);
this.on('toggleExpand', this._notifyOnResize, this);
this.on(
'closing',
function() {
this._map.getContainer().focus();
},
this
);
this._rememberTabState();
if (
L.Browser.touch &&
BR.Browser.touchScreenDetectable &&
!BR.Browser.touchScreen
) {
L.DomUtil.removeClass(this._container, 'leaflet-touch');
L.DomUtil.removeClass(this._tabContainer, 'leaflet-touch');
}
return this;
},
showPanel: function(id) {
var tab = this._getTab(id);
tab.hidden = false;
return this;
},
_rememberTabState: function() {
if (BR.Util.localStorageAvailable()) {
this.on('content closing', this._storeActiveTab, this);
var tabId = localStorage.getItem(this.storageId);
// 'true': legacy value for toggling old sidebar
if (tabId === 'true') {
tabId = this.options.defaultTabId;
} else if (tabId === null) {
// not set: closed by default for new users
tabId = '';
}
if (tabId !== '') {
this.open(tabId);
}
}
},
_notifyShow: function(tab) {
if (tab && tab.show) {
tab.show();
}
},
_notifyHide: function(tab) {
if (tab && tab.hide) {
tab.hide();
}
},
_notifyOnContent: function(e) {
var tab = this.options.listeningTabs[e.id];
this._notifyHide(this.oldTab);
this._notifyShow(tab);
this.oldTab = tab;
},
_notifyOnClose: function(e) {
this._notifyHide(this.oldTab);
this.oldTab = null;
},
_notifyOnResize: function(e) {
var tab = this.oldTab;
if (tab && tab.onResize) {
tab.onResize();
}
},
_storeActiveTab: function(e) {
localStorage.setItem(this.storageId, e.id || '');
}
});
BR.sidebar = function(divId, options) {
return new BR.Sidebar(divId, options);
};

View file

@ -1,8 +1,9 @@
(function(window) {
var HAS_HASHCHANGE = (function() {
var doc_mode = window.documentMode;
return ('onhashchange' in window) &&
(doc_mode === undefined || doc_mode > 7);
return (
'onhashchange' in window && (doc_mode === undefined || doc_mode > 7)
);
})();
L.Hash = function(map, options) {
@ -14,17 +15,17 @@
};
L.Hash.parseHash = function(hash) {
if(hash.indexOf('#map=') === 0) {
if (hash.indexOf('#map=') === 0) {
hash = hash.substr(5);
}
var args = hash.split(/\&(.+)/);
var mapsArgs = args[0].split("/");
var mapsArgs = args[0].split('/');
if (mapsArgs.length == 4) {
var zoom = parseInt(mapsArgs[0], 10),
lat = parseFloat(mapsArgs[1]),
lon = parseFloat(mapsArgs[2]),
layers = decodeURIComponent(mapsArgs[3]).split('-'),
additional = args[1];
lat = parseFloat(mapsArgs[1]),
lon = parseFloat(mapsArgs[2]),
layers = this.parseLayers(mapsArgs[3]),
additional = args[1];
if (isNaN(zoom) || isNaN(lat) || isNaN(lon)) {
return false;
} else {
@ -40,32 +41,19 @@
}
};
L.Hash.formatHash = function(map) {
(L.Hash.formatHash = function(map) {
var center = map.getCenter(),
zoom = map.getZoom(),
precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2)),
layers = [];
layers = this.formatLayers();
//console.log(this.options);
var options = this.options;
//Check active layers
for(var key in options) {
if (options.hasOwnProperty(key)) {
if (map.hasLayer(options[key])) {
layers.push(key);
};
};
};
if (layers.length == 0) {
layers.push(Object.keys(options)[0]);
}
var params = [
zoom,
center.lat.toFixed(precision),
center.lng.toFixed(precision),
encodeURIComponent(layers.join("-"))
layers
];
url = "#map=" + params.join("/");
url = '#map=' + params.join('/');
if (this.additionalCb != null) {
var additional = this.additionalCb();
if (additional != null) {
@ -73,160 +61,244 @@
}
}
return url;
},
}),
(L.Hash.prototype = {
options: {
layerSeparator: ','
},
map: null,
lastHash: null,
L.Hash.prototype = {
map: null,
lastHash: null,
parseHash: L.Hash.parseHash,
formatHash: L.Hash.formatHash,
parseHash: L.Hash.parseHash,
formatHash: L.Hash.formatHash,
init: function(map, options) {
this.map = map;
L.Util.setOptions(this, options);
init: function(map, options) {
this.map = map;
L.Util.setOptions(this, options);
// reset the hash
this.lastHash = null;
this.onHashChange();
// reset the hash
this.lastHash = null;
this.onHashChange();
if (!this.isListening) {
this.startListening();
}
},
if (!this.isListening) {
this.startListening();
}
},
_parseLayers: function(layersParam, layerSeparator) {
var layers = layersParam.split(layerSeparator).map(
L.bind(function(layerEncoded) {
var obj = null;
var layerString = decodeURIComponent(layerEncoded);
removeFrom: function(map) {
if (this.changeTimeout) {
clearTimeout(this.changeTimeout);
}
if (layerString) {
obj = this.options.layersControl.getLayerFromString(
layerString
);
}
if (this.isListening) {
this.stopListening();
}
return obj;
}, this)
);
this.map = null;
},
return layers;
},
onMapMove: function() {
// bail if we're moving the map (updating from a hash),
// or if the map is not yet loaded
parseLayers: function(layersParam) {
var countFoundLayers = function(count, obj) {
if (obj) {
count++;
}
return count;
};
if (this.movingMap || !this.map._loaded) {
return false;
}
var layers = this._parseLayers(
layersParam,
this.options.layerSeparator
);
var found = layers.reduce(countFoundLayers, 0);
var hash = this.formatHash(this.map);
if (this.lastHash != hash) {
location.replace(hash);
this.lastHash = hash;
}
},
if (found < layers.length) {
// legacy support for name instead of id and '-' layer separator
var layersLegacy = this._parseLayers(layersParam, '-');
var foundLegacy = layersLegacy.reduce(countFoundLayers, 0);
movingMap: false,
update: function() {
var hash = location.hash;
if (hash === this.lastHash) {
return;
}
var parsed = this.parseHash(hash);
if (!parsed) {
// migration from old hash style to new one
if (this.onInvalidHashChangeCb != null) {
var newHash = this.onInvalidHashChangeCb(hash);
if (newHash != null && newHash != hash) {
parsed = this.parseHash(newHash);
if (foundLegacy > found) {
layers = layersLegacy;
}
}
}
if (parsed) {
this.movingMap = true;
return layers;
},
this.map.setView(parsed.center, parsed.zoom);
var layers = parsed.layers,
options = this.options,
that = this;
//Add/remove layer
this.map.eachLayer(function(layer) {
for (alayer in that.layers) {
if (that.layers[alayer] == layer) {
that.map.removeLayer(layer);
break;
activateLayers: function(layers) {
var layersControl = this.options.layersControl;
var added = false;
layersControl.removeActiveLayers();
layers.forEach(
L.bind(function(obj, index, array) {
if (obj) {
layersControl.activateLayer(obj);
if (obj && !obj.overlay) {
added = true;
}
}
}, this)
);
if (!added) {
// if we couldn't add layers (removed or invalid name), add the default one
layersControl.activateDefaultBaseLayer();
}
},
formatLayers: function() {
var objList = this.options.layersControl.getActiveLayers();
var layerList = objList.map(
L.bind(function(obj) {
return encodeURIComponent(
this.options.layersControl.toLayerString(obj)
);
}, this)
);
return layerList.join(this.options.layerSeparator);
},
removeFrom: function(map) {
if (this.changeTimeout) {
clearTimeout(this.changeTimeout);
}
if (this.isListening) {
this.stopListening();
}
this.map = null;
},
onMapMove: function() {
// bail if we're moving the map (updating from a hash),
// or if the map is not yet loaded
if (this.movingMap || !this.map._loaded) {
return false;
}
var hash = this.formatHash(this.map);
if (this.lastHash != hash) {
location.replace(hash);
this.lastHash = hash;
}
},
movingMap: false,
update: function() {
var hash = location.hash;
if (hash === this.lastHash) {
return;
}
var parsed = this.parseHash(hash);
if (!parsed) {
// migration from old hash style to new one
if (this.onInvalidHashChangeCb != null) {
var newHash = this.onInvalidHashChangeCb(hash);
if (newHash != null && newHash != hash) {
parsed = this.parseHash(newHash);
}
}
});
var added = false;
layers.forEach(function(element, index, array) {
if (element in options) {
added = true;
that.map.addLayer(options[element]);
}
if (parsed) {
this.movingMap = true;
this.map.setView(parsed.center, parsed.zoom);
this.activateLayers(parsed.layers);
if (this.onHashChangeCb != null) {
this.onHashChangeCb(parsed.additional);
}
});
if (!added) {
// if we couldn't add layers (custom ones or invalid name), add the default one
this.map.addLayer(options[Object.keys(options)[0]]);
this.movingMap = false;
} else {
this.onMapMove(this.map);
}
},
if (this.onHashChangeCb != null) {
this.onHashChangeCb(parsed.additional);
// defer hash change updates every 100ms
changeDefer: 100,
changeTimeout: null,
onHashChange: function() {
// throttle calls to update() so that they only happen every
// `changeDefer` ms
if (!this.changeTimeout) {
var that = this;
this.changeTimeout = setTimeout(function() {
that.update();
that.changeTimeout = null;
}, this.changeDefer);
}
},
this.movingMap = false;
} else {
this.onMapMove(this.map);
isListening: false,
hashChangeInterval: null,
startListening: function() {
this.map.on(
'moveend layeradd layerremove',
this.onMapMove,
this
);
if (HAS_HASHCHANGE) {
L.DomEvent.addListener(
window,
'hashchange',
this.onHashChange
);
} else {
clearInterval(this.hashChangeInterval);
this.hashChangeInterval = setInterval(
this.onHashChange,
50
);
}
this.isListening = true;
},
stopListening: function() {
this.map.off(
'moveend layeradd layerremove',
this.onMapMove,
this
);
if (HAS_HASHCHANGE) {
L.DomEvent.removeListener(
window,
'hashchange',
this.onHashChange
);
} else {
clearInterval(this.hashChangeInterval);
}
this.isListening = false;
},
_keyByValue: function(obj, value) {
for (var key in obj) {
if (obj.hasOwnProperty(key)) {
if (obj[key] === value) {
return key;
} else {
return null;
}
}
}
}
},
// defer hash change updates every 100ms
changeDefer: 100,
changeTimeout: null,
onHashChange: function() {
// throttle calls to update() so that they only happen every
// `changeDefer` ms
if (!this.changeTimeout) {
var that = this;
this.changeTimeout = setTimeout(function() {
that.update();
that.changeTimeout = null;
}, this.changeDefer);
}
},
isListening: false,
hashChangeInterval: null,
startListening: function() {
this.map.on("moveend layeradd layerremove", this.onMapMove, this);
if (HAS_HASHCHANGE) {
L.DomEvent.addListener(window, "hashchange", this.onHashChange);
} else {
clearInterval(this.hashChangeInterval);
this.hashChangeInterval = setInterval(this.onHashChange, 50);
}
this.isListening = true;
},
stopListening: function() {
this.map.off("moveend layeradd layerremove", this.onMapMove, this);
if (HAS_HASHCHANGE) {
L.DomEvent.removeListener(window, "hashchange", this.onHashChange);
} else {
clearInterval(this.hashChangeInterval);
}
this.isListening = false;
},
_keyByValue: function(obj, value) {
for(var key in obj) {
if (obj.hasOwnProperty(key)) {
if (obj[key] === value) {
return key;
} else { return null; };
};
};
}
};
});
L.hash = function(map, options) {
return new L.Hash(map, options);
};

View file

@ -0,0 +1,31 @@
BR.stravaSegments = function(map, layersControl) {
const stravaControl = L.control
.stravaSegments({
runningTitle: i18next.t('map.strava-running'),
bikingTitle: i18next.t('map.strava-biking'),
loadingTitle: i18next.t('map.loading'),
stravaToken: BR.keys.strava
})
.addTo(map);
layersControl.addOverlay(
stravaControl.stravaLayer,
i18next.t('map.layer.strava-segments')
);
stravaControl.onError = function(err) {
BR.message.showError(
i18next.t('warning.strava-error', {
error: err && err.message ? err.message : err
})
);
};
// hide strava buttons when layer is inactive
var toggleStravaControl = function() {
var stravaBar = stravaControl.runningButton.button.parentElement;
stravaBar.hidden = !stravaBar.hidden;
};
toggleStravaControl();
stravaControl.stravaLayer.on('add remove', toggleStravaControl);
return stravaControl;
};

View file

@ -2,7 +2,8 @@ L.BRouter = L.Class.extend({
statics: {
// NOTE: the routing API used here is not public!
// /brouter?lonlats=1.1,1.2|2.1,2.2|3.1,3.2|4.1,4.2&nogos=-1.1,-1.2,1|-2.1,-2.2,2&profile=shortest&alternativeidx=1&format=kml
URL_TEMPLATE: '/brouter?lonlats={lonlats}&nogos={nogos}&profile={profile}&alternativeidx={alternativeidx}&format={format}',
URL_TEMPLATE:
'/brouter?lonlats={lonlats}&nogos={nogos}&polylines={polylines}&polygons={polygons}&profile={profile}&alternativeidx={alternativeidx}&format={format}',
URL_PROFILE_UPLOAD: BR.conf.host + '/brouter/profile',
PRECISION: 6,
NUMBER_SEPARATOR: ',',
@ -10,18 +11,20 @@ L.BRouter = L.Class.extend({
ABORTED_ERROR: 'aborted'
},
options: {
},
options: {},
initialize: function (options) {
initialize: function(options) {
L.setOptions(this, options);
this.queue = async.queue(L.bind(function (task, callback) {
this.getRoute(task.segment, callback);
}, this), 1);
this.queue = async.queue(
L.bind(function(task, callback) {
this.getRoute(task.segment, callback);
}, this),
1
);
// patch to call callbacks on kill for cleanup (loadingTrailer)
this.queue.kill = function () {
this.queue.kill = function() {
var aborted = this.tasks;
this.drain = null;
this.tasks = [];
@ -37,15 +40,32 @@ L.BRouter = L.Class.extend({
getUrlParams: function(latLngs, format) {
params = {};
if (this._getLonLatsString(latLngs) != null)
params.lonlats = this._getLonLatsString(latLngs);
if (this._getNogosString(this.options.nogos).length > 0)
if (
this.options.nogos &&
this._getNogosString(this.options.nogos).length > 0
)
params.nogos = this._getNogosString(this.options.nogos);
if (this.options.profile != null)
params.profile = this.options.profile;
if (
this.options.polylines &&
this._getNogosPolylinesString(this.options.polylines).length > 0
)
params.polylines = this._getNogosPolylinesString(
this.options.polylines
);
if (
this.options.polygons &&
this._getNogosPolygonsString(this.options.polygons).length > 0
)
params.polygons = this._getNogosPolygonsString(
this.options.polygons
);
if (this.options.profile != null) params.profile = this.options.profile;
params.alternativeidx = this.options.alternative;
@ -53,10 +73,16 @@ L.BRouter = L.Class.extend({
params.format = format;
} else {
// do not put values in URL if this is the default value (format===null)
if (params.profile === BR.conf.profiles[0])
if (params.profile === BR.conf.profiles[0]) delete params.profile;
if (params.alternativeidx == 0) delete params.alternativeidx;
// don't add custom profile, as these are only stored temporarily
if (
params.profile &&
params.profile.substring(0, 7) === 'custom_'
) {
delete params.profile;
if (params.alternativeidx == 0)
delete params.alternativeidx;
}
}
return params;
@ -70,6 +96,12 @@ L.BRouter = L.Class.extend({
if (params.nogos) {
opts.nogos = this._parseNogos(params.nogos);
}
if (params.polylines) {
opts.polylines = this._parseNogosPolylines(params.polylines);
}
if (params.polygons) {
opts.polygons = this._parseNogosPolygons(params.polygons);
}
if (params.alternativeidx) {
opts.alternative = params.alternativeidx;
}
@ -79,24 +111,39 @@ L.BRouter = L.Class.extend({
return opts;
},
getUrl: function(latLngs, format) {
getUrl: function(latLngs, format, trackname, exportWaypoints) {
var urlParams = this.getUrlParams(latLngs, format);
var args = []
var args = [];
if (urlParams.lonlats != null && urlParams.lonlats.length > 0)
args.push(L.Util.template('lonlats={lonlats}', urlParams));
if (urlParams.nogos != null)
args.push(L.Util.template('nogos={nogos}', urlParams));
if (urlParams.polylines != null)
args.push(L.Util.template('polylines={polylines}', urlParams));
if (urlParams.polygons != null)
args.push(L.Util.template('polygons={polygons}', urlParams));
if (urlParams.profile != null)
args.push(L.Util.template('profile={profile}', urlParams));
if (urlParams.alternativeidx != null)
args.push(L.Util.template('alternativeidx={alternativeidx}', urlParams));
args.push(
L.Util.template('alternativeidx={alternativeidx}', urlParams)
);
if (urlParams.format != null)
args.push(L.Util.template('format={format}', urlParams));
if (trackname)
args.push(
L.Util.template('trackname={trackname}', {
trackname: trackname
})
);
if (exportWaypoints) args.push('exportWaypoints=1');
var prepend_host = (format != null);
var prepend_host = format != null;
return (prepend_host ? BR.conf.host : '') + '/brouter?' + args.join('&');
return (
(prepend_host ? BR.conf.host : '') + '/brouter?' + args.join('&')
);
},
getRoute: function(latLngs, cb) {
@ -104,26 +151,32 @@ L.BRouter = L.Class.extend({
xhr = new XMLHttpRequest();
if (!url) {
return cb(new Error('Error getting route URL'));
return cb(new Error(i18next.t('warning.cannot-get-route')));
}
xhr.open('GET', url, true);
xhr.onload = L.bind(this._handleRouteResponse, this, xhr, cb);
xhr.onerror = L.bind(function(xhr, cb) {
cb(BR.Util.getError(xhr));
}, this, xhr, cb);
xhr.onerror = L.bind(
function(xhr, cb) {
cb(BR.Util.getError(xhr));
},
this,
xhr,
cb
);
xhr.send();
},
_handleRouteResponse: function(xhr, cb) {
var layer,
geojson;
if (xhr.status === 200
&& xhr.responseText
// application error when not GeoJSON format (text/plain for errors)
&& xhr.getResponseHeader('Content-Type').split(';')[0] === 'application/vnd.geo+json') {
var layer, geojson;
if (
xhr.status === 200 &&
xhr.responseText &&
// application error when not GeoJSON format (text/plain for errors)
xhr.getResponseHeader('Content-Type').split(';')[0] ===
'application/vnd.geo+json'
) {
// leaflet.spin
//gpxLayer.fire('data:loaded');
@ -132,7 +185,7 @@ L.BRouter = L.Class.extend({
layer = L.geoJSON(geojson).getLayers()[0];
return cb(null, layer);
} catch(e) {
} catch (e) {
console.error(e, xhr.responseText);
return cb(e);
}
@ -147,7 +200,7 @@ L.BRouter = L.Class.extend({
uploadProfile: function(profileId, profileText, cb) {
var url = L.BRouter.URL_PROFILE_UPLOAD;
xhr = new XMLHttpRequest();
xhr = new XMLHttpRequest();
// reuse existing profile file
if (profileId) {
@ -158,7 +211,7 @@ L.BRouter = L.Class.extend({
xhr.onload = L.bind(this._handleProfileResponse, this, xhr, cb);
xhr.onerror = function(evt) {
var xhr = this;
cb('Upload error: ' + xhr.statusText);
cb(i18next.t('warning.upload-error', { error: xhr.statusText }));
};
// send profile text only, as text/plain;charset=UTF-8
@ -168,11 +221,15 @@ L.BRouter = L.Class.extend({
_handleProfileResponse: function(xhr, cb) {
var response;
if (xhr.status === 200 && xhr.responseText && xhr.responseText.length > 0) {
if (
xhr.status === 200 &&
xhr.responseText &&
xhr.responseText.length > 0
) {
response = JSON.parse(xhr.responseText);
cb(response.error, response.profileid);
} else {
cb('Profile error: no or empty response from server');
cb(i18next.t('warning.profile-error'));
}
},
@ -180,7 +237,7 @@ L.BRouter = L.Class.extend({
var s = '';
for (var i = 0; i < latLngs.length; i++) {
s += this._formatLatLng(latLngs[i]);
if (i < (latLngs.length - 1)) {
if (i < latLngs.length - 1) {
s += L.BRouter.GROUP_SEPARATOR;
}
}
@ -213,7 +270,16 @@ L.BRouter = L.Class.extend({
s += this._formatLatLng(circle.getLatLng());
s += L.BRouter.NUMBER_SEPARATOR;
s += Math.round(circle.getRadius());
if (i < (nogos.length - 1)) {
// -1 is default nogo exclusion, it should not be passed as a URL parameter.
if (
circle.options.nogoWeight !== undefined &&
circle.options.nogoWeight !== null &&
circle.options.nogoWeight !== -1
) {
s += L.BRouter.NUMBER_SEPARATOR;
s += circle.options.nogoWeight;
}
if (i < nogos.length - 1) {
s += L.BRouter.GROUP_SEPARATOR;
}
}
@ -231,26 +297,137 @@ L.BRouter = L.Class.extend({
groups = s.split(L.BRouter.GROUP_SEPARATOR);
for (var i = 0; i < groups.length; i++) {
// lng,lat,radius
// lng,lat,radius(,weight)
numbers = groups[i].split(L.BRouter.NUMBER_SEPARATOR);
// TODO refactor: pass simple obj, create circle in NogoAreas; use shapeOptions of instance
// [lat,lng],radius
nogos.push(L.circle([numbers[1], numbers[0]], {radius: numbers[2]}));
// Parse as a nogo circle
var nogoOptions = { radius: numbers[2] };
if (numbers.length > 3) {
nogoOptions.nogoWeight = numbers[3];
}
nogos.push(L.circle([numbers[1], numbers[0]], nogoOptions));
}
return nogos;
},
_getNogosPolylinesString: function(nogos) {
var s = '';
for (var i = 0, polyline, vertices; i < nogos.length; i++) {
polyline = nogos[i];
vertices = polyline.getLatLngs();
for (var j = 0; j < vertices.length; j++) {
if (j > 0) {
s += L.BRouter.NUMBER_SEPARATOR;
}
s += this._formatLatLng(vertices[j]);
}
// -1 is default nogo exclusion, it should not be passed as a URL parameter.
if (
polyline.options.nogoWeight !== undefined &&
polyline.options.nogoWeight !== null &&
polyline.options.nogoWeight !== -1
) {
s += L.BRouter.NUMBER_SEPARATOR;
s += polyline.options.nogoWeight;
}
if (i < nogos.length - 1) {
s += L.BRouter.GROUP_SEPARATOR;
}
}
return s;
},
_parseNogosPolylines: function(s) {
var groups,
numbers,
latlngs,
nogos = [];
groups = s.split(L.BRouter.GROUP_SEPARATOR);
for (var i = 0; i < groups.length; i++) {
numbers = groups[i].split(L.BRouter.NUMBER_SEPARATOR);
if (numbers.length > 1) {
latlngs = [];
for (var j = 0; j < numbers.length - 1; ) {
var lng = Number.parseFloat(numbers[j++]);
var lat = Number.parseFloat(numbers[j++]);
latlngs.push([lat, lng]);
}
var nogoWeight;
if (j < numbers.length) {
nogoWeight = Number.parseFloat(numbers[j++]);
}
nogos.push(L.polyline(latlngs, { nogoWeight: nogoWeight }));
}
}
return nogos;
},
_getNogosPolygonsString: function(nogos) {
var s = '';
for (var i = 0, polygon, vertices; i < nogos.length; i++) {
polygon = nogos[i];
vertices = polygon.getLatLngs()[0];
for (var j = 0; j < vertices.length; j++) {
if (j > 0) {
s += L.BRouter.NUMBER_SEPARATOR;
}
s += this._formatLatLng(vertices[j]);
}
// -1 is default nogo exclusion, it should not be passed as a URL parameter.
if (
polygon.options.nogoWeight !== undefined &&
polygon.options.nogoWeight !== null &&
polygon.options.nogoWeight !== -1
) {
s += L.BRouter.NUMBER_SEPARATOR;
s += polygon.options.nogoWeight;
}
if (i < nogos.length - 1) {
s += L.BRouter.GROUP_SEPARATOR;
}
}
return s;
},
_parseNogosPolygons: function(s) {
var groups,
numbers,
latlngs,
nogos = [];
groups = s.split(L.BRouter.GROUP_SEPARATOR);
for (var i = 0; i < groups.length; i++) {
numbers = groups[i].split(L.BRouter.NUMBER_SEPARATOR);
if (numbers.length > 1) {
latlngs = [];
for (var j = 0; j < numbers.length - 1; ) {
var lng = Number.parseFloat(numbers[j++]);
var lat = Number.parseFloat(numbers[j++]);
latlngs.push([lat, lng]);
}
var nogoWeight;
if (j < numbers.length) {
nogoWeight = Number.parseFloat(numbers[j++]);
}
nogos.push(L.polygon(latlngs, { nogoWeight: nogoWeight }));
}
}
return nogos;
},
// formats L.LatLng object as lng,lat string
_formatLatLng: function(latLng) {
var s = '';
s += L.Util.formatNum(latLng.lng, L.BRouter.PRECISION);
s += L.Util.formatNum(latLng.lng || latLng[1], L.BRouter.PRECISION);
s += L.BRouter.NUMBER_SEPARATOR;
s += L.Util.formatNum(latLng.lat, L.BRouter.PRECISION);
s += L.Util.formatNum(latLng.lat || latLng[0], L.BRouter.PRECISION);
return s;
}
});
L.bRouter = function (options) {
L.bRouter = function(options) {
return new L.BRouter(options);
};

View file

@ -24,6 +24,5 @@ var brouterCgi = (function() {
return {
getUrl: getUrl
}
};
})();

View file

@ -1,5 +1,4 @@
(function() {
// COPYING: Please get your own API keys from the sites listed below
BR.keys = {
@ -10,7 +9,12 @@
digitalGlobe: '',
// Thunderforest, https://thunderforest.com/pricing/
thunderforest: ''
};
thunderforest: '',
// Strava API token in case you want to display Strava segments
strava: '',
// OpenMapSurfer (OpenRouteService API), https://openrouteservice.org/plans/
openrouteservice: ''
};
})();

View file

@ -0,0 +1,20 @@
{
"geometry": null,
"properties": {
"name": "OpenStreetMap.se",
"maxZoom": 18,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://openstreetmap.se/\" target=\"_blank\">OpenStreetMap Sweden</a>",
"id": "1010",
"subdomains": [
"a",
"b",
"c",
"d",
"e",
"f"
],
"url": "http://{s}.tile.openstreetmap.se/hydda/full/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,12 @@
{
"geometry": null,
"properties": {
"name": "4UMaps",
"maxZoom": 15,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://www.4umaps.eu/\">4UMaps</a>",
"id": "1016",
"url": "http://4umaps.eu/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,14 @@
{
"geometry": null,
"properties": {
"name": "Osmapa.pl",
"maxZoom": 20,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://osmapa.pl/\" target=\"_blank\">Osmapa.pl</a>",
"id": "1017",
"threed": "true",
"language": "pl",
"url": "http://{s}.tile.openstreetmap.pl/osmapa.pl/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,14 @@
{
"geometry": null,
"properties": {
"name": "Спутник",
"maxZoom": 19,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://maps.sputnik.ru/\" target=\"_blank\">Спутник</a>",
"id": "1021",
"threed": "true",
"language": "ru",
"url": "http://{s}.tiles.maps.sputnik.ru/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,19 @@
{
"geometry": null,
"properties": {
"name": "Космоснимки",
"maxZoom": 18,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://osm.kosmosnimki.ru/\" target=\"_blank\">ScanEx</a>",
"id": "1023",
"subdomains": [
"a",
"b",
"c",
"d"
],
"language": "ru",
"url": "http://{s}.tile.osm.kosmosnimki.ru/kosmo/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,12 @@
{
"geometry": null,
"properties": {
"name": "ÖPNV Karte",
"maxZoom": 18,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://öpnvkarte.de/\" target=\"_blank\">ÖPNV Karte</a>",
"id": "1059",
"url": "http://tileserver.memomaps.de/tilegen/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,12 @@
{
"geometry": null,
"properties": {
"name": "Thunderforest Outdoors",
"maxZoom": 22,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://www.thunderforest.com/\" target=\"_blank\">Andy Allan</a>",
"id": "1061",
"url": "http://{s}.tile.thunderforest.com/outdoors/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,12 @@
{
"geometry": null,
"properties": {
"name": "Hike & Bike",
"maxZoom": 19,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://hikebikemap.de/\" target=\"_blank\">Colin Marquardt</a>",
"id": "1065",
"url": "http://toolserver.org/tiles/hikebike/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,13 @@
{
"geometry": null,
"properties": {
"name": "Refuges.info hiking",
"maxZoom": 18,
"attribution": "Map data: &copy; <a href=\"http://www.openstreetmap.org/copyright\" target=\"_blank\">OpenStreetMap</a> contributors, under ODbL | Tiles: &copy; <a href=\"http://maps.refuges.info/\" target=\"_blank\">sly</a>",
"id": "1069",
"old": "true",
"url": "http://maps.refuges.info/hiking/{z}/{x}/{y}.png",
"dataSource": "LayersCollection"
},
"type": "Feature"
}

View file

@ -0,0 +1,53 @@
const fs = require('fs');
const path = require('path');
const fetch = require('node-fetch');
const outDir = __dirname;
var includeList = [
"1016", // 4UMaps
"1065", // Hike & Bike Map
"1061", // Thunderforest Outdoors
"1021", // kosmosnimki.ru
"1017", // sputnik.ru
"1023", // Osmapa.pl - Mapa OpenStreetMap Polska
"1010", // OpenStreetMap.se (Hydda.Full)
"1069", // MRI (maps.refuges.info),
"1059" // ÖPNV Karte
];
function extract(constantsJs) {
eval(constantsJs);
for (let i = 0; i < includeList.length; i++) {
let id = includeList[i];
let layer = getLayerDataByID(id);
if (!layer) {
console.warn('Layer not found: ' + id);
continue;
}
//console.log(`${layer.id}, ${layer.name}, ${layer.address}`);
layer.url = layer.address;
delete layer.address;
let geoJson = {
geometry: null,
properties: layer,
type: "Feature"
};
geoJson.properties.dataSource = 'LayersCollection';
const outFileName = path.join(outDir, layer.id + '.geojson');
const data = JSON.stringify(geoJson, null, 2);
fs.writeFileSync(outFileName, data);
}
}
// https://github.com/Edward17/LayersCollection/blob/gh-pages/constants.js
fetch('http://edward17.github.io/LayersCollection/constants.js')
.then(res => res.text())
.then(text => extract(text))
.catch(err => console.error(err));

38
layers/config/config.js Normal file
View file

@ -0,0 +1,38 @@
BR.confLayers = {};
BR.confLayers.defaultBaseLayers = [
'standard',
'OpenTopoMap',
'Stamen.Terrain',
'Esri.WorldImagery'
];
// worldwide monolingual layers to add as default when browser language matches
BR.confLayers.languageDefaultLayers = [
'osm-mapnik-german_style',
'osmfr',
'1021' // sputnik.ru
];
BR.confLayers.defaultOverlays = [
'HikeBike.HillShading',
'Waymarked_Trails-Cycling',
'Waymarked_Trails-Hiking'
];
BR.confLayers.legacyNameToIdMap = {
'OpenStreetMap': 'standard',
'OpenStreetMap.de': 'osm-mapnik-german_style',
'OpenTopoMap': 'OpenTopoMap',
'Esri World Imagery': 'Esri.WorldImagery',
'Cycling (Waymarked Trails)': 'Waymarked_Trails-Cycling',
'Hiking (Waymarked Trails)': 'Waymarked_Trails-Hiking'
};
BR.confLayers.leafletProvidersIncludeList = [
'Stamen.Terrain',
'MtbMap',
'OpenStreetMap.CH',
'HikeBike.HillShading',
'Esri.WorldImagery'
];

1475
layers/config/geometry.js Normal file

File diff suppressed because it is too large Load diff

232
layers/config/overrides.js Normal file
View file

@ -0,0 +1,232 @@
BR.confLayers.getPropertyOverrides = function() {
return {
'standard': {
'name': i18next.t('map.layer.osm'),
'attribution': {
'html': '&copy; <a target="_blank" href="https://www.openstreetmap.org/copyright">openstreetmap.org</a>, <a target="_blank" href="https://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA 2.0</a>'
},
'mapUrl': 'https://www.openstreetmap.org/#map={zoom}/{lat}/{lon}'
},
'OpenTopoMap': {
'name': i18next.t('map.layer.topo'),
'attribution': {
'html': '&copy; <a target="_blank" href="https://opentopomap.org/about#verwendung">OpenTopoMap</a>, <a target="_blank" href="https://creativecommons.org/licenses/by-sa/3.0/">CC-BY-SA 3.0</a>; <a target="_blank" href="http://viewfinderpanoramas.org">SRTM</a>'
},
'mapUrl': 'https://opentopomap.org/#map={zoom}/{lat}/{lon}'
},
'Stamen.Terrain': {
'name': i18next.t('map.layer.stamen-terrain'),
'attribution': '&copy; <a href="http://stamen.com">Stamen Design</a>, <a href="http://creativecommons.org/licenses/by/3.0">CC BY 3.0</a>',
'mapUrl': 'http://maps.stamen.com/#terrain/{zoom}/{lat}/{lon}'
},
'Esri.WorldImagery': {
'name': i18next.t('map.layer.esri'),
'nameShort': i18next.t('credits.esri-tiles'),
'attribution': i18next.t('credits.esri-license'),
'mapUrl': 'http://www.arcgis.com/home/item.html?id=10df2279f9684e4a9f6a7f08febac2a9'
},
'wikimedia-map': {
'mapUrl': 'https://maps.wikimedia.org/#{zoom}/{lat}/{lon}'
},
'HDM_HOT': {
'nameShort': 'HOT',
'mapUrl': 'http://map.hotosm.org/#{zoom}/{lat}/{lon}'
},
// OpenStreetMap.se (Hydda.Full)
'1010': {
'mapUrl': 'https://maps.openstreetmap.se/#{zoom}/{lat}/{lon}'
},
'opencylemap': {
'name': i18next.t('map.layer.cycle'),
'nameShort': 'OpenCycleMap',
// add Thunderforest API key variable
'url': 'https://{switch:a,b,c}.tile.thunderforest.com/cycle/{zoom}/{x}/{y}.png?apikey={keys_thunderforest}',
'mapUrl': 'https://www.opencyclemap.org/?zoom={zoom}&lat={lat}&lon={lon}&layers=B0000'
},
'1061': {
'name': i18next.t('map.layer.outdoors'),
'nameShort': 'Outdoors',
'url': 'http://{s}.tile.thunderforest.com/outdoors/{z}/{x}/{y}.png?apikey={keys_thunderforest}',
'mapUrl': 'https://www.opencyclemap.org/?zoom={zoom}&lat={lat}&lon={lon}&layers=000B0'
},
// Hike & Bike Map
'1065': {
'mapUrl': 'http://hikebikemap.org/?zoom={zoom}&lat={lat}&lon={lon}&layer=HikeBikeMap'
},
// 4UMaps
'1016': {
'mapUrl': 'https://www.4umaps.com/map.htm?zoom={zoom}&lat={lat}&lon={lon}&layers=B00'
},
'openmapsurfer': {
'mapUrl': 'https://maps.openrouteservice.org/directions?n1={lat}&n2={lon}&n3={zoom}&b=0&c=0&k1=en-US&k2=km'
},
// ÖPNV Karte
'1059': {
'name': 'Public transport (ÖPNVKarte)',
'nameShort': 'ÖPNVKarte',
'mapUrl': 'http://öpnvkarte.de/#{lon};{lat};{zoom}'
},
'osm-mapnik-german_style': {
'name': i18next.t('map.layer.osmde'),
'language_code': 'de',
'attribution': {
'html': '&copy; <a target="_blank" href="https://openstreetmap.de/karte.html">openstreetmap.de</a>'
},
'mapUrl': 'https://www.openstreetmap.de/karte.html?zoom={zoom}&lat={lat}&lon={lon}&layers=B000TF'
},
'osmfr': {
'language_code': 'fr',
'nameShort': 'OSM French',
'mapUrl': 'http://tile.openstreetmap.fr/?layers=B00000000FFFFFF&zoom={zoom}&lat={lat}&lon={lon}'
},
// kosmosnimki.ru
'1023': {
'language_code': 'ru',
'mapUrl': 'http://kosmosnimki.ru/'
},
// sputnik.ru
'1021': {
'language_code': 'ru',
'mapUrl': 'http://maps.sputnik.ru/?lat={lat}&lng={lon}&zoom={zoom}'
},
'MtbMap': {
'mapUrl': 'http://mtbmap.cz/#zoom={zoom}&lat={lat}&lon={lon}',
'worldTiles': true // -z12
},
// MRI (maps.refuges.info)
'1069': {
'nameShort': 'Refuges.info',
'mapUrl': 'http://maps.refuges.info/?zoom={zoom}&lat={lat}&lon={lon}&layers=B',
'worldTiles': true
},
'osmfr-basque': {
'language_code': 'eu',
'nameShort': 'OSM Basque',
'mapUrl': 'http://tile.openstreetmap.fr/?layers=00000000BFFFFFF&zoom={zoom}&lat={lat}&lon={lon}',
'worldTiles': true
},
'osmfr-breton': {
'language_code': 'br',
'nameShort': 'OSM Breton',
'mapUrl': 'https://kartenn.openstreetmap.bzh/#map={zoom}/{lat}/{lon}',
'worldTiles': true
},
'osmfr-occitan': {
'language_code': 'oc',
'nameShort': 'OSM Occitan',
'mapUrl': 'http://tile.openstreetmap.fr/?layers=0000000B0FFFFFF&zoom={zoom}&lat={lat}&lon={lon}',
'worldTiles': true
},
'osmbe': {
'nameShort': 'OSM Belgium',
'mapUrl': 'https://tile.osm.be/#map={zoom}/{lat}/{lon}',
'worldTiles': true // -z7
},
'osmbe-fr': {
'nameShort': 'OSM Belgium (fr)',
'mapUrl': 'https://tile.osm.be/#map={zoom}/{lat}/{lon}',
'worldTiles': true // -z7
},
'osmbe-nl': {
'nameShort': 'OSM Belgium (nl)',
'mapUrl': 'https://tile.osm.be/#map={zoom}/{lat}/{lon}',
'worldTiles': true // -z7
},
'OpenStreetMap.CH': {
'country_code': 'CH',
'mapUrl': 'https://osm.ch/#{zoom}/{lat}/{lon}',
'worldTiles': true
},
'topplus-open': {
'country_code': 'DE',
'mapUrl': 'http://www.geodatenzentrum.de/geodaten/gdz_rahmen.gdz_div?gdz_spr=deu&gdz_user_id=0&gdz_akt_zeile=5&gdz_anz_zeile=1&gdz_unt_zeile=41',
'worldTiles': true // World -z9, Europe -z14
},
'OpenStreetMap-turistautak': {
'nameShort': 'OSM Turistautak',
'mapUrl': 'https://turistautak.openstreetmap.hu/?zoom={zoom}&lat={lat}&lon={lon}&layers=0B00F'
},
'Israel_Hiking': {
'mapUrl': 'https://israelhiking.osm.org.il/map/{zoom}/{lat}/{lon}'
},
'Israel_MTB': {
'mapUrl': 'https://israelhiking.osm.org.il/map/{zoom}/{lat}/{lon}'
},
'mtbmap-no': {
'mapUrl': 'https://mtbmap.no/#{zoom}/{lat}/{lon}'
},
// Osmapa.pl - Mapa OpenStreetMap Polska
'1017': {
"country_code": "PL",
'language_code': 'pl',
'mapUrl': 'http://osmapa.pl/#lat={lat}&lon={lon}&z={zoom}&m=os',
'worldTiles': true // -z13
},
'Freemap.sk-Car': {
'mapUrl': 'https://www.freemap.sk/?map={zoom}/{lat}/{lon}&layers=A'
},
'Freemap.sk-Hiking': {
'mapUrl': 'https://www.freemap.sk/?map={zoom}/{lat}/{lon}&layers=T'
},
'Freemap.sk-Cyclo': {
'mapUrl': 'https://www.freemap.sk/?map={zoom}/{lat}/{lon}&layers=C'
},
'osm-cambodia_laos_thailand_vietnam-bilingual': {
'country_code': 'TH+',
'nameShort': 'Thaimap',
'mapUrl': 'http://thaimap.osm-tools.org/?zoom={zoom}&lat={lat}&lon={lon}&layers=BT',
'worldTiles': true
},
'HikeBike.HillShading': {
'name': i18next.t('map.layer.hikebike-hillshading'),
'nameShort': i18next.t('map.hikebike-hillshading'),
'attribution': '&copy; <a target="_blank" href="http://hikebikemap.org">hikebikemap.org</a>; SRTM3 v2 (<a target="_blank" href="http://www2.jpl.nasa.gov/srtm/">NASA</a>)',
'mapUrl': 'http://hikebikemap.org/?zoom={zoom}&lat={lat}&lon={lon}&layer=HikeBikeMap',
'overlay': true
},
'Waymarked_Trails-Cycling': {
'name': i18next.t('map.layer.cycling'),
'nameShort': i18next.t('map.cycling'),
'attribution': {
'html': '&copy; <a target="_blank" href="https://cycling.waymarkedtrails.org/en/help/legal">waymarkedtrails.org</a>, <a target="_blank" href="https://creativecommons.org/licenses/by-sa/3.0/de/deed.en">CC-BY-SA 3.0 DE</a>'
},
'mapUrl': 'http://cycling.waymarkedtrails.org/#?map={zoom}!{lat}!{lon}'
},
'Waymarked_Trails-Hiking': {
'name': i18next.t('map.layer.hiking'),
'nameShort': i18next.t('map.hiking'),
'attribution': {
'html': '&copy; <a target="_blank" href="https://hiking.waymarkedtrails.org/en/help/legal">waymarkedtrails.org</a>, <a target="_blank" href="https://creativecommons.org/licenses/by-sa/3.0/de/deed.en">CC-BY-SA 3.0 DE</a>'
},
'mapUrl': 'http://hiking.waymarkedtrails.org/#?map={zoom}!{lat}!{lon}'
},
'Waymarked_Trails-MTB': {
'nameShort': 'MTB',
'mapUrl': 'http://mtb.waymarkedtrails.org/#?map={zoom}!{lat}!{lon}'
},
'mapillary-coverage-raster': {
'nameShort': 'Mapillary',
'mapUrl': 'https://www.mapillary.com/app/?lat={lat}&lng={lon}&z={zoom}&menu=false'
},
'openpt_map': {
'nameShort': 'openptmap',
'mapUrl': 'http://openptmap.org/?zoom={zoom}&lat={lat}&lon={lon}&layers=B0000TFT'
},
'historic-place-contours': {
'mapUrl': 'http://gk.historic.place/historische_objekte/?zoom={zoom}&lat={lat}&lon={lon}&pid=GhHaSaHe'
},
'hu-hillshade': {
'nameShort': 'Hillshade HU',
'mapUrl': 'http://map.turistautak.hu/?zoom={zoom}&lat={lat}&lon={lon}&layers=0B000FTF'
},
'mapaszlakow-cycle': {
'nameShort': 'Cycleways PL',
'mapUrl': 'http://mapaszlakow.eu/#{zoom}/{lat}/{lon}'
},
'mapaszlakow-routes': {
'nameShort': 'Routes PL',
'mapUrl': 'http://mapaszlakow.eu/#{zoom}/{lat}/{lon}'
}
};
};

82
layers/config/tree.js Normal file
View file

@ -0,0 +1,82 @@
BR.confLayers.tree = {
'base-layers': {
'worldwide-international': [
'standard',
'OpenTopoMap',
'Stamen.Terrain',
'Esri.WorldImagery',
'wikimedia-map',
'HDM_HOT',
'1010', // OpenStreetMap.se (Hydda.Full)
'opencylemap',
'1061', // Thunderforest Outdoors
'1065', // Hike & Bike Map
'1016', // 4UMaps,
'openmapsurfer',
'1059' // ÖPNV Karte
],
'worldwide-monolingual': [
'osm-mapnik-german_style',
'osmfr',
'1023', // kosmosnimki.ru
'1021' // sputnik.ru
],
'europe': [
'MtbMap',
'1069' // MRI (maps.refuges.info)
],
'europe-monolingual': [
'osmfr-basque',
'osmfr-breton',
'osmfr-occitan'
],
'country': [
{
'BE': [
'osmbe',
'osmbe-fr',
'osmbe-nl',
]
},
'OpenStreetMap.CH',
'topplus-open',
'OpenStreetMap-turistautak',
{
'IL': [
'Israel_Hiking',
'Israel_MTB',
]
},
'mtbmap-no',
'1017', // Osmapa.pl - Mapa OpenStreetMap Polska
{
'SK': [
'Freemap.sk-Car',
'Freemap.sk-Hiking',
'Freemap.sk-Cyclo',
]
},
'osm-cambodia_laos_thailand_vietnam-bilingual'
]
},
'overlays': {
'worldwide': [
'HikeBike.HillShading',
'Waymarked_Trails-Cycling',
'Waymarked_Trails-Hiking',
'Waymarked_Trails-MTB',
'mapillary-coverage-raster',
'openpt_map'
],
'country': [
'historic-place-contours',
'hu-hillshade',
{
'PL': [
'mapaszlakow-cycle',
'mapaszlakow-routes'
]
}
]
}
};

View file

@ -0,0 +1,45 @@
{
"geometry": {
"coordinates": [
[
[
5.45,
45
],
[
17,
45
],
[
17,
56.27
],
[
5.45,
56.27
],
[
5.45,
45
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"html": "<a href=\"https://www.eea.europa.eu/data-and-maps/data/eu-dem\" target=\"_blank\" rel=\"noopener\">Produced using Copernicus data and information funded by the European Union - EU-DEM layers</a>, download from <a href=\"http://opendem.info/index.html\" target=\"_blank\">OpenDEM</a>, Data sources <a href=\"https://www2.jpl.nasa.gov/srtm/\" target=\"_blank\">SRTM</a>, <a href=\"https://asterweb.jpl.nasa.gov/gdem.asp\" target=\"_blank\">ASTER GDEM</a>, Tiles from <a href=\"http://www.historic.place/\" target=\"_blank\" rel=\"noopener\">Historic.Place</a>"
},
"country_code": "DACH",
"id": "historic-place-contours",
"min_zoom": 12,
"max_zoom": 15,
"name": "Contours DACH",
"overlay": true,
"type": "tms",
"url": "http://tiles.historic.place/ele/de/{zoom}/{x}/{y}.png",
"valid-georeference": true
},
"type": "Feature"
}

View file

@ -0,0 +1,412 @@
{
"geometry": {
"coordinates": [
[
[
15.9751041,
54.3709213
],
[
16.311164,
54.5561775
],
[
17.1391878,
54.7845723
],
[
18.3448458,
54.9022727
],
[
19.6613689,
54.4737213
],
[
20.2815206,
54.4213456
],
[
21.4663914,
54.3406369
],
[
22.7759855,
54.3769755
],
[
22.8625989,
54.4233613
],
[
23.2956657,
54.2678633
],
[
23.5347186,
54.0955258
],
[
23.5208604,
53.9775182
],
[
23.7183389,
53.4629603
],
[
23.9296755,
53.1856735
],
[
23.9296755,
52.6887269
],
[
23.732197,
52.6067497
],
[
23.5658994,
52.5878101
],
[
23.2090523,
52.3302642
],
[
23.1951942,
52.2370089
],
[
23.5035377,
52.1860596
],
[
23.6906226,
52.0030113
],
[
23.5970802,
51.739903
],
[
23.6629063,
51.3888562
],
[
23.9366046,
50.9827781
],
[
24.1687284,
50.8604752
],
[
24.0197534,
50.8035823
],
[
24.1098313,
50.6610467
],
[
24.0578633,
50.4188439
],
[
23.6178674,
50.3083403
],
[
22.6824431,
49.5163532
],
[
22.7378756,
49.2094935
],
[
22.9041733,
49.0780441
],
[
22.8625989,
48.9940062
],
[
22.6096878,
49.0371785
],
[
22.0761495,
49.2004392
],
[
21.8474902,
49.3721872
],
[
21.3763135,
49.4488281
],
[
21.1026153,
49.3721872
],
[
20.9120659,
49.3022043
],
[
20.6452967,
49.3902311
],
[
20.1845136,
49.3315641
],
[
20.1186875,
49.2004392
],
[
19.9419962,
49.1302123
],
[
19.765305,
49.2117568
],
[
19.7479823,
49.3992506
],
[
19.6024718,
49.4150307
],
[
19.5089294,
49.5815389
],
[
19.4292451,
49.5905232
],
[
19.2317666,
49.4150307
],
[
18.9961783,
49.387976
],
[
18.9338167,
49.4916048
],
[
18.8368097,
49.4938552
],
[
18.8021643,
49.6623381
],
[
18.6427958,
49.7094091
],
[
18.521537,
49.8994693
],
[
18.0815412,
50.0109209
],
[
17.8875272,
49.9886512
],
[
17.7385522,
50.0687739
],
[
17.6068999,
50.1709584
],
[
17.7454813,
50.2153184
],
[
17.710836,
50.3017019
],
[
17.4163505,
50.2640668
],
[
16.9486384,
50.4453265
],
[
16.8932058,
50.4033889
],
[
17.0006064,
50.3105529
],
[
17.017929,
50.2241854
],
[
16.8135215,
50.186489
],
[
16.6402948,
50.0976742
],
[
16.4324227,
50.2862087
],
[
16.1968344,
50.4276731
],
[
16.4220291,
50.5885165
],
[
16.3388803,
50.6632429
],
[
16.2280152,
50.6368824
],
[
16.0547884,
50.6127057
],
[
15.5732181,
50.7641544
],
[
15.2683391,
50.8976368
],
[
15.2440873,
50.980597
],
[
15.0292862,
51.0133036
],
[
15.0015699,
50.8582883
],
[
14.8110205,
50.8735944
],
[
14.956531,
51.0721176
],
[
15.0188926,
51.2914636
],
[
14.9392083,
51.4601459
],
[
14.7209426,
51.5571799
],
[
14.7521234,
51.6260562
],
[
14.5996839,
51.8427626
],
[
14.70362,
52.0733396
],
[
14.5581095,
52.2497371
],
[
14.5165351,
52.425436
],
[
14.6031485,
52.5878101
],
[
14.1146491,
52.8208272
],
[
14.152759,
52.9733951
],
[
14.3502374,
53.0734212
],
[
14.4229927,
53.2665624
],
[
14.1977979,
53.8734759
],
[
14.2220497,
53.9958517
],
[
15.9751041,
54.3709213
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"text": "Tiles © OSM Mapa Szlaków",
"url": "http://mapaszlakow.eu/"
},
"country_code": "PL",
"id": "mapaszlakow-cycle",
"max_zoom": 16,
"min_zoom": 6,
"name": "Cycleways (Mapa Szlaków)",
"overlay": true,
"type": "tms",
"url": "http://mapaszlakow.eu/c{zoom}/{zoom}/{x}/{y}.png"
},
"type": "Feature"
}

View file

@ -0,0 +1,412 @@
{
"geometry": {
"coordinates": [
[
[
15.9751041,
54.3709213
],
[
16.311164,
54.5561775
],
[
17.1391878,
54.7845723
],
[
18.3448458,
54.9022727
],
[
19.6613689,
54.4737213
],
[
20.2815206,
54.4213456
],
[
21.4663914,
54.3406369
],
[
22.7759855,
54.3769755
],
[
22.8625989,
54.4233613
],
[
23.2956657,
54.2678633
],
[
23.5347186,
54.0955258
],
[
23.5208604,
53.9775182
],
[
23.7183389,
53.4629603
],
[
23.9296755,
53.1856735
],
[
23.9296755,
52.6887269
],
[
23.732197,
52.6067497
],
[
23.5658994,
52.5878101
],
[
23.2090523,
52.3302642
],
[
23.1951942,
52.2370089
],
[
23.5035377,
52.1860596
],
[
23.6906226,
52.0030113
],
[
23.5970802,
51.739903
],
[
23.6629063,
51.3888562
],
[
23.9366046,
50.9827781
],
[
24.1687284,
50.8604752
],
[
24.0197534,
50.8035823
],
[
24.1098313,
50.6610467
],
[
24.0578633,
50.4188439
],
[
23.6178674,
50.3083403
],
[
22.6824431,
49.5163532
],
[
22.7378756,
49.2094935
],
[
22.9041733,
49.0780441
],
[
22.8625989,
48.9940062
],
[
22.6096878,
49.0371785
],
[
22.0761495,
49.2004392
],
[
21.8474902,
49.3721872
],
[
21.3763135,
49.4488281
],
[
21.1026153,
49.3721872
],
[
20.9120659,
49.3022043
],
[
20.6452967,
49.3902311
],
[
20.1845136,
49.3315641
],
[
20.1186875,
49.2004392
],
[
19.9419962,
49.1302123
],
[
19.765305,
49.2117568
],
[
19.7479823,
49.3992506
],
[
19.6024718,
49.4150307
],
[
19.5089294,
49.5815389
],
[
19.4292451,
49.5905232
],
[
19.2317666,
49.4150307
],
[
18.9961783,
49.387976
],
[
18.9338167,
49.4916048
],
[
18.8368097,
49.4938552
],
[
18.8021643,
49.6623381
],
[
18.6427958,
49.7094091
],
[
18.521537,
49.8994693
],
[
18.0815412,
50.0109209
],
[
17.8875272,
49.9886512
],
[
17.7385522,
50.0687739
],
[
17.6068999,
50.1709584
],
[
17.7454813,
50.2153184
],
[
17.710836,
50.3017019
],
[
17.4163505,
50.2640668
],
[
16.9486384,
50.4453265
],
[
16.8932058,
50.4033889
],
[
17.0006064,
50.3105529
],
[
17.017929,
50.2241854
],
[
16.8135215,
50.186489
],
[
16.6402948,
50.0976742
],
[
16.4324227,
50.2862087
],
[
16.1968344,
50.4276731
],
[
16.4220291,
50.5885165
],
[
16.3388803,
50.6632429
],
[
16.2280152,
50.6368824
],
[
16.0547884,
50.6127057
],
[
15.5732181,
50.7641544
],
[
15.2683391,
50.8976368
],
[
15.2440873,
50.980597
],
[
15.0292862,
51.0133036
],
[
15.0015699,
50.8582883
],
[
14.8110205,
50.8735944
],
[
14.956531,
51.0721176
],
[
15.0188926,
51.2914636
],
[
14.9392083,
51.4601459
],
[
14.7209426,
51.5571799
],
[
14.7521234,
51.6260562
],
[
14.5996839,
51.8427626
],
[
14.70362,
52.0733396
],
[
14.5581095,
52.2497371
],
[
14.5165351,
52.425436
],
[
14.6031485,
52.5878101
],
[
14.1146491,
52.8208272
],
[
14.152759,
52.9733951
],
[
14.3502374,
53.0734212
],
[
14.4229927,
53.2665624
],
[
14.1977979,
53.8734759
],
[
14.2220497,
53.9958517
],
[
15.9751041,
54.3709213
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"text": "Tiles © OSM Mapa Szlaków",
"url": "http://mapaszlakow.eu/"
},
"country_code": "PL",
"id": "mapaszlakow-routes",
"max_zoom": 16,
"min_zoom": 6,
"name": "Routes (Mapa Szlaków)",
"overlay": true,
"type": "tms",
"url": "http://mapaszlakow.eu/b{zoom}/{zoom}/{x}/{y}.png"
},
"type": "Feature"
}

View file

@ -0,0 +1,17 @@
{
"geometry": null,
"properties": {
"attribution": {
"text": "Mapillary, CC BY",
"url": "https://www.mapillary.com"
},
"id": "mapillary-coverage-raster",
"max_zoom": 17,
"name": "Mapillary Coverage",
"overlay": true,
"type": "tms",
"url": "https://d6a1v2w10ny40.cloudfront.net/v0.1/{z}/{x}/{y}.png"
},
"type": "Feature"
}

View file

@ -0,0 +1,15 @@
{
"geometry": null,
"properties": {
"attribution": {
"html": "<a href=\"https://giscience.uni-hd.de\" target=\"_blank\" rel=\"noopener\">GIScience Research Group</a> @ University of Heidelberg (<a href=\"https://openrouteservice.org/\" target=\"_blank\" rel=\"noopener\">info</a>)"
},
"id": "openmapsurfer",
"max_zoom": 19,
"name": "OpenMapSurfer",
"type": "tms",
"url": "https://api.openrouteservice.org/mapsurfer/{zoom}/{x}/{y}.png?api_key={keys_openrouteservice}"
},
"type": "Feature"
}

View file

@ -0,0 +1,42 @@
{
"geometry": {
"coordinates": [
[
[
4.219,
46.317
],
[
16.875,
46.317
],
[
16.875,
55.776
],
[
4.219,
55.776
],
[
4.219,
46.317
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"html": "© <a href=\"http://www.bkg.bund.de\" target=\"_blank\" rel=\"noopener\">Bundesamt für Kartographie und Geodäsie</a> 2018, <a href=\"http://sg.geodatenzentrum.de/web_public/Datenquellen_TopPlus_Open.pdf\" target=\"_blank\" rel=\"noopener\">Datenquellen</a>"
},
"id": "topplus-open",
"name": "TopPlusOpen",
"type": "wms",
"url": "http://sgx.geodatenzentrum.de/wms_topplus_web_open",
"layers": "web",
"format": "image/png"
},
"type": "Feature"
}

View file

@ -0,0 +1,210 @@
{
"geometry": {
"coordinates": [
[
[
19.83682,
49.25529
],
[
19.80075,
49.42385
],
[
19.60437,
49.48058
],
[
19.49179,
49.63961
],
[
19.21831,
49.52604
],
[
19.16778,
49.42521
],
[
19.00308,
49.42236
],
[
18.97611,
49.5308
],
[
18.54685,
49.51425
],
[
18.31432,
49.33818
],
[
18.15913,
49.2961
],
[
18.05564,
49.11134
],
[
17.56396,
48.84938
],
[
17.17929,
48.88816
],
[
17.058,
48.81105
],
[
16.90426,
48.61947
],
[
16.79685,
48.38561
],
[
17.06762,
48.01116
],
[
17.32787,
47.97749
],
[
17.51699,
47.82535
],
[
17.74776,
47.73093
],
[
18.29515,
47.72075
],
[
18.67959,
47.75541
],
[
18.89755,
47.81203
],
[
18.79463,
47.88245
],
[
18.84318,
48.04046
],
[
19.46212,
48.05333
],
[
19.62064,
48.22938
],
[
19.89585,
48.09387
],
[
20.33766,
48.2643
],
[
20.55395,
48.52358
],
[
20.82335,
48.55714
],
[
21.10271,
48.47096
],
[
21.45863,
48.55513
],
[
21.74536,
48.31435
],
[
22.15293,
48.37179
],
[
22.61255,
49.08914
],
[
22.09997,
49.23814
],
[
21.9686,
49.36363
],
[
21.6244,
49.46989
],
[
21.06873,
49.46402
],
[
20.94336,
49.31088
],
[
20.73052,
49.44006
],
[
20.22804,
49.41714
],
[
20.05234,
49.23052
],
[
19.83682,
49.25529
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"logo-image": "https://raw.githubusercontent.com/FreemapSlovakia/freemap-v3-react/master/src/images/freemap-logo-small.png",
"required": true,
"text": "Copyright ©2007-2012 Freemap Slovakia (www.freemap.sk). Some rights reserved."
},
"country_code": "SK",
"icon": "https://raw.githubusercontent.com/FreemapSlovakia/freemap-v3-react/master/src/images/freemap-logo-small.png",
"id": "Freemap.sk-Car",
"max_zoom": 16,
"min_zoom": 8,
"name": "Freemap.sk Car",
"type": "tms",
"url": "https://{switch:a,b,c,d}.freemap.sk/A/{zoom}/{x}/{y}.jpeg",
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,210 @@
{
"geometry": {
"coordinates": [
[
[
19.83682,
49.25529
],
[
19.80075,
49.42385
],
[
19.60437,
49.48058
],
[
19.49179,
49.63961
],
[
19.21831,
49.52604
],
[
19.16778,
49.42521
],
[
19.00308,
49.42236
],
[
18.97611,
49.5308
],
[
18.54685,
49.51425
],
[
18.31432,
49.33818
],
[
18.15913,
49.2961
],
[
18.05564,
49.11134
],
[
17.56396,
48.84938
],
[
17.17929,
48.88816
],
[
17.058,
48.81105
],
[
16.90426,
48.61947
],
[
16.79685,
48.38561
],
[
17.06762,
48.01116
],
[
17.32787,
47.97749
],
[
17.51699,
47.82535
],
[
17.74776,
47.73093
],
[
18.29515,
47.72075
],
[
18.67959,
47.75541
],
[
18.89755,
47.81203
],
[
18.79463,
47.88245
],
[
18.84318,
48.04046
],
[
19.46212,
48.05333
],
[
19.62064,
48.22938
],
[
19.89585,
48.09387
],
[
20.33766,
48.2643
],
[
20.55395,
48.52358
],
[
20.82335,
48.55714
],
[
21.10271,
48.47096
],
[
21.45863,
48.55513
],
[
21.74536,
48.31435
],
[
22.15293,
48.37179
],
[
22.61255,
49.08914
],
[
22.09997,
49.23814
],
[
21.9686,
49.36363
],
[
21.6244,
49.46989
],
[
21.06873,
49.46402
],
[
20.94336,
49.31088
],
[
20.73052,
49.44006
],
[
20.22804,
49.41714
],
[
20.05234,
49.23052
],
[
19.83682,
49.25529
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"logo-image": "https://raw.githubusercontent.com/FreemapSlovakia/freemap-v3-react/master/src/images/freemap-logo-small.png",
"required": true,
"text": "Copyright ©2007-2012 Freemap Slovakia (www.freemap.sk). Some rights reserved."
},
"country_code": "SK",
"icon": "https://raw.githubusercontent.com/FreemapSlovakia/freemap-v3-react/master/src/images/freemap-logo-small.png",
"id": "Freemap.sk-Cyclo",
"max_zoom": 16,
"min_zoom": 8,
"name": "Freemap.sk Cyclo",
"type": "tms",
"url": "https://{switch:a,b,c,d}.freemap.sk/C/{zoom}/{x}/{y}.jpeg",
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,210 @@
{
"geometry": {
"coordinates": [
[
[
19.83682,
49.25529
],
[
19.80075,
49.42385
],
[
19.60437,
49.48058
],
[
19.49179,
49.63961
],
[
19.21831,
49.52604
],
[
19.16778,
49.42521
],
[
19.00308,
49.42236
],
[
18.97611,
49.5308
],
[
18.54685,
49.51425
],
[
18.31432,
49.33818
],
[
18.15913,
49.2961
],
[
18.05564,
49.11134
],
[
17.56396,
48.84938
],
[
17.17929,
48.88816
],
[
17.058,
48.81105
],
[
16.90426,
48.61947
],
[
16.79685,
48.38561
],
[
17.06762,
48.01116
],
[
17.32787,
47.97749
],
[
17.51699,
47.82535
],
[
17.74776,
47.73093
],
[
18.29515,
47.72075
],
[
18.67959,
47.75541
],
[
18.89755,
47.81203
],
[
18.79463,
47.88245
],
[
18.84318,
48.04046
],
[
19.46212,
48.05333
],
[
19.62064,
48.22938
],
[
19.89585,
48.09387
],
[
20.33766,
48.2643
],
[
20.55395,
48.52358
],
[
20.82335,
48.55714
],
[
21.10271,
48.47096
],
[
21.45863,
48.55513
],
[
21.74536,
48.31435
],
[
22.15293,
48.37179
],
[
22.61255,
49.08914
],
[
22.09997,
49.23814
],
[
21.9686,
49.36363
],
[
21.6244,
49.46989
],
[
21.06873,
49.46402
],
[
20.94336,
49.31088
],
[
20.73052,
49.44006
],
[
20.22804,
49.41714
],
[
20.05234,
49.23052
],
[
19.83682,
49.25529
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"logo-image": "https://raw.githubusercontent.com/FreemapSlovakia/freemap-v3-react/master/src/images/freemap-logo-small.png",
"required": true,
"text": "Copyright ©2007-2012 Freemap Slovakia (www.freemap.sk). Some rights reserved."
},
"country_code": "SK",
"icon": "https://raw.githubusercontent.com/FreemapSlovakia/freemap-v3-react/master/src/images/freemap-logo-small.png",
"id": "Freemap.sk-Hiking",
"max_zoom": 16,
"min_zoom": 8,
"name": "Freemap.sk Hiking",
"type": "tms",
"url": "https://{switch:a,b,c,d}.freemap.sk/T/{zoom}/{x}/{y}.jpeg",
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,19 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© OpenStreetMap contributors, tiles courtesy of Humanitarian OpenStreetMap Team",
"url": "https://www.hotosm.org/"
},
"icon": "https://wiki.openstreetmap.org/w/images/thumb/c/c9/Hot_logo.svg/300px-Hot_logo.svg.png",
"id": "HDM_HOT",
"max_zoom": 20,
"name": "HDM (Humanitarian OpenStreetMap Team)",
"type": "tms",
"url": "https://{switch:a,b,c}.tile.openstreetmap.fr/hot/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,187 @@
{
"geometry": {
"coordinates": [
[
[
34.64563,
32.92073
],
[
34.98374,
33.13352
],
[
35.15662,
33.09994
],
[
35.31781,
33.11463
],
[
35.36541,
33.06285
],
[
35.46229,
33.09994
],
[
35.51741,
33.12652
],
[
35.5266,
33.21531
],
[
35.53893,
33.25442
],
[
35.56446,
33.2969
],
[
35.61264,
33.27918
],
[
35.67429,
33.30627
],
[
35.70785,
33.34269
],
[
35.75363,
33.35091
],
[
35.81509,
33.3392
],
[
35.91531,
32.9406
],
[
35.80834,
32.772
],
[
35.77835,
32.72446
],
[
35.59491,
32.62828
],
[
35.5729,
32.36541
],
[
35.59461,
32.21856
],
[
35.55452,
32.02901
],
[
35.57225,
31.75415
],
[
35.48771,
31.41951
],
[
35.4209,
31.25116
],
[
35.47936,
31.1783
],
[
35.42771,
30.95172
],
[
35.3321,
30.77107
],
[
35.20709,
30.53307
],
[
35.17202,
30.11204
],
[
35.07514,
29.83713
],
[
35.02336,
29.64569
],
[
34.93992,
29.39946
],
[
34.89517,
29.37711
],
[
34.84785,
29.59084
],
[
34.69667,
30.10714
],
[
34.52423,
30.40912
],
[
34.48879,
30.64515
],
[
34.07929,
31.52265
],
[
34.64563,
32.92073
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "Tiles © IsraelHiking, CC BY-SA-NC 3.0. Data by OpenStreetMap under ODbL.",
"url": "https://israelhiking.osm.org.il/"
},
"country_code": "IL",
"description": "Israel Hiking map",
"icon": "https://israelhiking.osm.org.il/content/favicons/favicon.ico",
"id": "Israel_Hiking",
"max_zoom": 16,
"min_zoom": 7,
"name": "Israel Hiking",
"type": "tms",
"url": "https://israelhiking.osm.org.il/Tiles/{zoom}/{x}/{y}.png",
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,187 @@
{
"geometry": {
"coordinates": [
[
[
34.64563,
32.92073
],
[
34.98374,
33.13352
],
[
35.15662,
33.09994
],
[
35.31781,
33.11463
],
[
35.36541,
33.06285
],
[
35.46229,
33.09994
],
[
35.51741,
33.12652
],
[
35.5266,
33.21531
],
[
35.53893,
33.25442
],
[
35.56446,
33.2969
],
[
35.61264,
33.27918
],
[
35.67429,
33.30627
],
[
35.70785,
33.34269
],
[
35.75363,
33.35091
],
[
35.81509,
33.3392
],
[
35.91531,
32.9406
],
[
35.80834,
32.772
],
[
35.77835,
32.72446
],
[
35.59491,
32.62828
],
[
35.5729,
32.36541
],
[
35.59461,
32.21856
],
[
35.55452,
32.02901
],
[
35.57225,
31.75415
],
[
35.48771,
31.41951
],
[
35.4209,
31.25116
],
[
35.47936,
31.1783
],
[
35.42771,
30.95172
],
[
35.3321,
30.77107
],
[
35.20709,
30.53307
],
[
35.17202,
30.11204
],
[
35.07514,
29.83713
],
[
35.02336,
29.64569
],
[
34.93992,
29.39946
],
[
34.89517,
29.37711
],
[
34.84785,
29.59084
],
[
34.69667,
30.10714
],
[
34.52423,
30.40912
],
[
34.48879,
30.64515
],
[
34.07929,
31.52265
],
[
34.64563,
32.92073
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "Tiles © IsraelHiking, CC BY-SA-NC 3.0. Data by OpenStreetMap under ODbL.",
"url": "https://israelhiking.osm.org.il/"
},
"country_code": "IL",
"description": "Israel MTB map",
"icon": "https://israelhiking.osm.org.il/content/favicons/favicon.ico",
"id": "Israel_MTB",
"max_zoom": 16,
"min_zoom": 7,
"name": "Israel MTB",
"type": "tms",
"url": "https://israelhiking.osm.org.il/MTBTiles/{zoom}/{x}/{y}.png",
"dataSource": "JOSM"
},
"type": "Feature"
}

7
layers/josm/LICENSE Normal file
View file

@ -0,0 +1,7 @@
JOSM imagery database is licensed under Creative Commons (CC-BY-SA).
Content since April 2014 is dual licensed with LGPL (except contents copied from ELI, which is only CC-BY-SA).
https://creativecommons.org/licenses/by-sa/3.0/
Source: https://josm.openstreetmap.de/wiki/Maps#Otherimportantinformation

File diff suppressed because it is too large Load diff

File diff suppressed because one or more lines are too long

View file

@ -0,0 +1,20 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© waymarkedtrails.org, OpenStreetMap contributors, CC by-SA 3.0",
"url": "https://cycling.waymarkedtrails.org/en/help/legal"
},
"icon": "https://static.waymarkedtrails.org/img/map_cycling.png",
"id": "Waymarked_Trails-Cycling",
"max_zoom": 17,
"name": "Waymarked Trails: Cycling",
"overlay": true,
"type": "tms",
"url": "https://tile.waymarkedtrails.org/cycling/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,20 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© waymarkedtrails.org, OpenStreetMap contributors, CC by-SA 3.0",
"url": "https://hiking.waymarkedtrails.org/en/help/legal"
},
"icon": "https://static.waymarkedtrails.org/img/map_hiking.png",
"id": "Waymarked_Trails-Hiking",
"max_zoom": 17,
"name": "Waymarked Trails: Hiking",
"overlay": true,
"type": "tms",
"url": "https://tile.waymarkedtrails.org/hiking/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,20 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© waymarkedtrails.org, OpenStreetMap contributors, CC by-SA 3.0",
"url": "https://mtb.waymarkedtrails.org/en/help/legal"
},
"icon": "https://static.waymarkedtrails.org/img/map_mtb.png",
"id": "Waymarked_Trails-MTB",
"max_zoom": 17,
"name": "Waymarked Trails: MTB",
"overlay": true,
"type": "tms",
"url": "https://tile.waymarkedtrails.org/mtb/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

63
layers/josm/extract.js Normal file
View file

@ -0,0 +1,63 @@
const fs = require('fs');
const path = require('path');
const fetch = require('node-fetch');
const outDir = __dirname;
const includeList = [
"osmbe",
"osmbe-fr",
"osmbe-nl",
"osmfr-basque",
"osmfr-breton",
"osmfr-occitan",
"OpenStreetMap-turistautak",
"hu-hillshade",
"Israel_Hiking",
"Israel_MTB",
"mtbmap-no",
"Freemap.sk-Car",
"Freemap.sk-Hiking",
"Freemap.sk-Cyclo",
"opencylemap",
"standard",
"HDM_HOT",
"osmfr",
"osm-mapnik-german_style",
"OpenTopoMap",
"osm-cambodia_laos_thailand_vietnam-bilingual",
"Waymarked_Trails-Hiking",
"Waymarked_Trails-Cycling",
"Waymarked_Trails-MTB",
"wikimedia-map",
"openpt_map"
];
function extract(layersJosm) {
for (let i = 0; i < layersJosm.features.length; i++) {
let layer = layersJosm.features[i];
let props = layer.properties;
let id = props.id;
if (includeList.includes(id)) {
//console.log(`${id}, ${props.name}, ${props.url}`);
props.dataSource = 'JOSM';
const outFileName = path.join(outDir, id + '.geojson');
const data = JSON.stringify(layer, null, 2);
fs.writeFileSync(outFileName, data);
includeList.splice(includeList.indexOf(id), 1);
}
}
if (includeList.length > 0) {
console.warn('Layers not found: ', includeList);
}
}
fetch('https://josm.openstreetmap.de/maps?format=geojson')
.then(res => res.json())
.then(json => extract(json))
.catch(err => console.error(err));

View file

@ -0,0 +1,46 @@
{
"geometry": {
"coordinates": [
[
[
15,
45
],
[
24,
45
],
[
24,
49
],
[
15,
49
],
[
15,
45
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "SRTM"
},
"country_code": "HU",
"id": "hu-hillshade",
"max_zoom": 18,
"min_zoom": 0,
"name": "Hillshade Hungary",
"overlay": true,
"type": "tms",
"url": "https://{switch:a,b,c}.map.turistautak.hu/tiles/shading/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,134 @@
{
"geometry": {
"coordinates": [
[
[
31.904253,
70.4368136
],
[
28.4765186,
71.3289643
],
[
23.6865015,
71.2514263
],
[
16.8090601,
70.0730823
],
[
11.1620655,
67.5253903
],
[
9.975542,
64.811576
],
[
4.2187061,
62.1449966
],
[
4.3725367,
59.1871966
],
[
6.1743055,
57.8915032
],
[
7.932118,
57.7393554
],
[
10.777577,
58.8649103
],
[
11.7224012,
58.762509
],
[
12.722157,
60.1141506
],
[
13.0517469,
61.3493518
],
[
12.5243921,
63.6169922
],
[
14.2382593,
63.9856094
],
[
15.1171656,
65.9016624
],
[
18.6987085,
68.3749083
],
[
20.0610132,
68.2612583
],
[
21.0058375,
68.7841518
],
[
25.2465601,
68.3506025
],
[
26.9384546,
69.8472011
],
[
28.7621851,
69.6112133
],
[
28.5864039,
68.8556004
],
[
31.069314,
69.5191547
],
[
31.904253,
70.4368136
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "© MTBmap.no",
"url": "https://www.mtbmap.no/"
},
"country_code": "NO",
"description": "Norwegian mountain biking map from OSM (max zoom 14-16, varies per region)",
"icon": "",
"id": "mtbmap-no",
"max_zoom": 14,
"min_zoom": 3,
"name": "MTBmap.no",
"tile-size": "512",
"tile_size": 512,
"type": "tms",
"url": "https://mtbmap.no/tiles/osm/mtbmap/{zoom}/{x}/{y}.jpg",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,21 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"terms-of-use-text": "Maps © Thunderforest",
"terms-of-use-url": "https://thunderforest.com/terms/",
"text": "Data © OpenStreetMap contributors",
"url": "https://www.openstreetmap.org/copyright/"
},
"icon": "",
"id": "opencylemap",
"max_zoom": 22,
"name": "OpenCycleMap",
"type": "tms",
"url": "https://{switch:a,b,c}.tile.thunderforest.com/cycle/{zoom}/{x}/{y}.png?apikey=7bd5ed2197cf4da29fa26de0ba6530cc",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,21 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© OpenStreetMap contributors, CC-BY-SA",
"url": "http://openptmap.de/"
},
"icon": "http://openptmap.de/favicon_pt.png",
"id": "openpt_map",
"max_zoom": 17,
"min_zoom": 4,
"name": "OpenPT Map (overlay)",
"overlay": true,
"type": "tms",
"url": "http://openptmap.de/tiles/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,168 @@
{
"geometry": {
"coordinates": [
[
[
92.1023798,
20.8135629
],
[
93.5690546,
24.0975527
],
[
94.1733026,
23.9269484
],
[
95.1950312,
26.707274
],
[
96.7550898,
27.5286657
],
[
97.5845575,
28.5805966
],
[
98.738122,
27.514051
],
[
98.7436151,
25.8799151
],
[
97.6779413,
24.7577376
],
[
97.9635858,
24.042382
],
[
98.8205194,
24.1627239
],
[
99.5236444,
22.9593356
],
[
100.3695917,
21.5051376
],
[
101.7923212,
22.4830518
],
[
105.3628778,
23.3331079
],
[
106.8185663,
22.8480137
],
[
108.1973505,
21.3619661
],
[
107.4389505,
18.8539792
],
[
117.1453714,
7.4656173
],
[
119.6172953,
5.2875389
],
[
118.1231546,
4.0502277
],
[
117.2552347,
4.3624942
],
[
115.8654642,
4.3460623
],
[
115.5084085,
3.0249771
],
[
114.552598,
1.5100953
],
[
113.5418558,
1.2574836
],
[
112.9650736,
1.5704982
],
[
112.2454691,
1.5100953
],
[
111.67418,
1.0158321
],
[
110.4546976,
0.9004918
],
[
109.4988871,
1.9218969
],
[
103.2256937,
1.1256762
],
[
100.4626322,
3.2388904
],
[
97.6721048,
8.0588831
],
[
93.892808,
15.9398659
],
[
92.1023798,
20.8135629
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "© osm-tools.org & OpenStreetMap contributors, CC-BY-SA",
"url": "https://www.osm-tools.org/"
},
"id": "osm-cambodia_laos_thailand_vietnam-bilingual",
"max_zoom": 20,
"name": "Cambodia, Laos, Thailand, Vietnam, Malaysia, Myanmar bilingual",
"type": "tms",
"url": "https://{switch:a,b,c,d}.tile.osm-tools.org/osm/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,20 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© OpenStreetMap contributors, CC-BY-SA",
"url": "https://www.openstreetmap.org/"
},
"icon": "",
"id": "osm-mapnik-german_style",
"max_zoom": 18,
"mod-tile-features": true,
"name": "OpenStreetMap (German Style)",
"type": "tms",
"url": "https://{switch:a,b,c,d}.tile.openstreetmap.de/tiles/osmde/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

2023
layers/josm/osmbe-fr.geojson Normal file

File diff suppressed because it is too large Load diff

2023
layers/josm/osmbe-nl.geojson Normal file

File diff suppressed because it is too large Load diff

2023
layers/josm/osmbe.geojson Normal file

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,46 @@
{
"geometry": {
"coordinates": [
[
[
16.2075922,
62.7408449
],
[
16.2900797,
33.8453727
],
[
-10.7364447,
33.8043768
],
[
-10.8189321,
62.7182339
],
[
16.2075922,
62.7408449
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "Tiles © OpenStreetMap France, data © OpenStreetMap contributors, ODbL",
"url": "https://www.openstreetmap.org/"
},
"icon": "",
"id": "osmfr-basque",
"max_zoom": 20,
"mod-tile-features": true,
"name": "OpenStreetMap (Basque Style)",
"type": "tms",
"url": "https://tile.openstreetmap.bzh/eu/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,46 @@
{
"geometry": {
"coordinates": [
[
[
16.2075922,
62.7408449
],
[
16.2900797,
33.8453727
],
[
-10.7364447,
33.8043768
],
[
-10.8189321,
62.7182339
],
[
16.2075922,
62.7408449
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "Tiles © OpenStreetMap France, data © OpenStreetMap contributors, ODbL",
"url": "https://www.openstreetmap.org/"
},
"icon": "",
"id": "osmfr-breton",
"max_zoom": 20,
"mod-tile-features": true,
"name": "OpenStreetMap (Breton Style)",
"type": "tms",
"url": "https://tile.openstreetmap.bzh/br/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,46 @@
{
"geometry": {
"coordinates": [
[
[
16.2075922,
62.7408449
],
[
16.2900797,
33.8453727
],
[
-10.7364447,
33.8043768
],
[
-10.8189321,
62.7182339
],
[
16.2075922,
62.7408449
]
]
],
"type": "Polygon"
},
"properties": {
"attribution": {
"required": true,
"text": "Tiles © OpenStreetMap France, data © OpenStreetMap contributors, ODbL",
"url": "https://www.openstreetmap.org/"
},
"icon": "",
"id": "osmfr-occitan",
"max_zoom": 20,
"mod-tile-features": true,
"name": "OpenStreetMap (Occitan Style)",
"type": "tms",
"url": "https://tile.openstreetmap.bzh/oc/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

20
layers/josm/osmfr.geojson Normal file
View file

@ -0,0 +1,20 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "Tiles © cquest@Openstreetmap France, data © OpenStreetMap contributors, ODBL",
"url": "https://www.openstreetmap.org/"
},
"icon": "",
"id": "osmfr",
"max_zoom": 20,
"mod-tile-features": true,
"name": "OpenStreetMap (French Style)",
"type": "tms",
"url": "https://{switch:a,b,c}.tile.openstreetmap.fr/osmfr/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,21 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© OpenStreetMap contributors, CC-BY-SA",
"url": "https://www.openstreetmap.org/"
},
"default": true,
"icon": "",
"id": "standard",
"max_zoom": 19,
"mod-tile-features": true,
"name": "OpenStreetMap Carto (Standard)",
"type": "tms",
"url": "https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

View file

@ -0,0 +1,18 @@
{
"geometry": null,
"properties": {
"attribution": {
"required": true,
"text": "© OpenStreetMap contributors, CC-BY-SA",
"url": "https://www.openstreetmap.org/"
},
"id": "wikimedia-map",
"max_zoom": 18,
"name": "Wikimedia Map",
"type": "tms",
"url": "https://maps.wikimedia.org/osm-intl/{zoom}/{x}/{y}.png",
"valid-georeference": true,
"dataSource": "JOSM"
},
"type": "Feature"
}

209
locales/de.json Normal file
View file

@ -0,0 +1,209 @@
{
"about": {
"bug-reports": "Fehlerberichte und Funktionsanfragen:",
"bug-reports-back": "Server / Backend, Routing Engine, Android-App, Profile, brouter.de Website",
"bug-reports-front": "Web-Client / Frontend.",
"chat": "Chatte mit Benutzern und Entwicklern",
"contact": "Kontakt:",
"data": "Daten:",
"data-description": "Dies basiert auf <a href=\"https://www.openstreetmap.org\" target=\"_blank\">OpenStreetMap</a>. Es wird normalerweise einmal pro Woche aktualisiert, wenn ein neues Planet-File verfügbar ist. Siehe Datum der <a href=\"http://brouter.de/brouter/segments4/\" target=\"_blank\">Datendateien</a>.",
"description": "Online-Service der BRouter Routing Engine. Für die Offline-Android-App und weitere Informationen siehe <a href=\"http://brouter.de/\" target=\"_blank\">brouter.de</a>.",
"details": "<i><a href=\"http://brouter.de/privacypolicy.html\" target=\"_blank\">Datenschutz-Bestimmungen</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web#credits-and-licenses\" target=\"_blank\">Credits</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web/blob/master/CHANGELOG.md\" target=\"_blank\">Änderungsprotokoll</a></i> und\n<i><a href=\"https://github.com/nrenner/brouter-web#readme\" target=\"_blank\">weitere Infos</a></i> zum Client.",
"support": "Generelle Diskussionen/Fragen, Support",
"title": "Über"
},
"credits": {
"brouter": "BRouter",
"brouter-license": "<a target=\"_blank\" href=\"http://brouter.de/brouter\">BRouter</a> &copy; Arndt Brenschede",
"esri-license": "<a target=\"_blank\" href=\"http://goto.arcgisonline.com/maps/World_Imagery\">World Imagery</a> &copy; <a target=\"_blank\" href=\"https://www.esri.com/\">Esri</a>, Quellen: Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP und die GIS Benutzergemeinschaft",
"esri-tiles": "Esri World Imagery",
"map-data": "Kartendaten",
"map-tiles": "Kartendarstellung",
"nominatim": "Suchen mit <a href=\"https://wiki.openstreetmap.org/wiki/Nominatim\" target=\"_blank\" data-i18n=\"credits.nominatim\">Nominatim</a>",
"openstreetmap": "&copy; <a target=\"_blank\" href=\"https://www.openstreetmap.org/copyright\" >OpenStreetMap contributors</a> unter <a target=\"_blank\" href=\"https://opendatacommons.org/licenses/odbl/\" >ODbL</a>"
},
"export": {
"format": "Format",
"format_csv": "CSV",
"format_geojson": "GeoJSON",
"format_gpx": "GPX",
"format_kml": "KML",
"route-from-to": "{{from}} -> {{to}} ({{distance}} km)",
"route-loop": "{{from}} ({{distance}} km)",
"title": "Route exportieren",
"trackname": "Name"
},
"footer": {
"ascend": "Aufsteigend",
"cost": "Kosten",
"distance": "Entfernung",
"energy-per-100km": "Energie pro 100 km",
"hours": "Stunden",
"hours-abbrev": "h",
"kilometer": "Kilometer",
"kilometer-abbrev": "km",
"kilowatthour": "Kilowattstunden",
"kilowatthour-abbrev": "kWh",
"mean-cost-factor": "durchschnittlicher Kostenfaktor",
"meter": "Meter",
"meter-abbrev": "m",
"plain-ascend": "einfach aufsteigend",
"total-energy": "Gesamtenergie",
"travel-time": "Reisezeit"
},
"layers": {
"add-base": "Basisebene hinzufügen",
"add-overlay": "Overlay hinzufügen",
"customize": "Ebenen anpassen",
"placeholder-layer-name": "angepasster Ebenenname (Bsp: OpenStreetMap)",
"placeholder-layer-url": "angepasste Ebenen-URL (Bsp: https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png)",
"remove-selection": "Auswahl entfernen"
},
"loadNogos": {
"defaultProperties": "Standardeigenschaften",
"file": "Datei:",
"load": "Laden",
"nogoBuffer": "Puffer um No-go-Areas (in Metern):",
"nogoRadius": "No-go Radius (für Punkte):",
"nogoWeight": "No-go Gewicht:",
"source": "Quelle",
"title": "No-go-Areas laden",
"url": "URL:"
},
"map": {
"attribution-osm-long": "OpenStreetMap-Mitwirkende",
"attribution-osm-short": "OpenStreetMap",
"clear-route": "Route löschen",
"copyright": "Copyright",
"cycling": "Radfahren",
"delete-last-point": "Letzten Punkt löschen",
"delete-nogo-areas": "&nbsp;&nbsp;auch alle no-go areas löschen",
"delete-route": "Route löschen?",
"draw-route-start": "Route anzeigen (D Taste)",
"draw-route-stop": "Route entfernen (ESC Taste)",
"hikebike-hillshading": "Höhenschummerung",
"hiking": "Wandern",
"layer": {
"bing": "Bing Luftbild",
"cycle": "OpenCycleMap (Thunderf.)",
"cycling": "Radfahren (markierte Routen)",
"digitalglobe": "DigitalGlobe neueste Bilder",
"esri": "Esri Weltbilder",
"hikebike-hillshading": "Höhenschummerung (Hike & Bike Map)",
"hiking": "Wandern (markierte Routen)",
"osm": "OpenStreetMap",
"osmde": "OpenStreetMap.de",
"outdoors": "Outdoor (Thunderforest)",
"stamen-terrain": "Terrain (Stamen)",
"strava-segments": "Strava Segmente",
"topo": "OpenTopoMap"
},
"loading": "Lade...",
"locate-me": "Zeige meinen Standort",
"nogo": {
"cancel": "No-Go-Areas löschen",
"click-drag": "klicken und ziehen zum Zeichnen eines Kreises",
"draw": "No-Go-Area zeichnen (Kreis)",
"edit": "Klicken zum Bearbeiten",
"help": "&square; = Bewegen / Größe anpassen, <span class=\"fa fa-trash-o\"></span> = Löschen,<br>Kreis anklicken um Bearbeiten zu beenden"
},
"opacity-slider": "Transparenz von Route und Markern anpassen",
"privacy": "Datenschutz",
"reverse-route": "Route umkehren",
"strava-biking": "Zeige Strava Radfahrsegmente",
"strava-running": "Zeige Strava Läufersegmente",
"zoomInTitle": "Hineinzoomen",
"zoomOutTitle": "Herauszoomen"
},
"modal": {
"close": "Schließen"
},
"navbar": {
"about": "Über",
"alternative": {
"first": "1. Alternative",
"original": "Original",
"second": "2. Alternative",
"third": "3. Alternative"
},
"export": "Export",
"load": {
"nogos": "No-go-Areas",
"title": "Laden"
},
"profile": {
"car-eco": "Auto (effizient)",
"car-fast": "Auto (schnell)",
"car-test": "Auto (Test)",
"custom": "Benutzerdefiniert",
"fastbike": "Fahrrad (schnell)",
"fastbike-asia-pacific": "Fahrrad (schnell, Asien Pazifik)",
"fastbike-lowtraffic": "Fahrrad (schnell, Routen mit wenig Verkehr)",
"hiking-beta": "Wandern (beta)",
"moped": "Moped",
"rail": "Eisenbahn",
"river": "Fluss",
"safety": "Sicherste Route",
"shortest": "Kürzeste Route",
"trekking": "Trekkingrad",
"trekking-ignore-cr": "Trekkingrad (ignoriere Radrouten)",
"trekking-noferries": "Trekkingrad (keine Fähren)",
"trekking-nosteps": "Trekkingrad (keine Treppen)",
"trekking-steep": "Trekkingrad (steil)",
"vm-forum-liegerad-schnell": "Liegerad (schnell)",
"vm-forum-velomobil-schnell": "Velomobil (schnell)"
}
},
"sidebar": {
"custom-profile": {
"title": "Benutzerdefiniertes Profil"
},
"data": {
"title": "Daten"
},
"itinerary": {
"title": "Reiseroute"
},
"layers": {
"category": {
"base-layers": "Grundkarten",
"country": "Land",
"europe": "Europa",
"europe-monolingual": "Europa einsprachig",
"overlays": "Überlagerungen",
"worldwide": "Weltweit",
"worldwide-international": "Weltweit international",
"worldwide-monolingual": "Weltweit einsprachig"
},
"collapse": "Alle zusammenklappen",
"custom-layers": "Benutzerdefinierte Ebenen",
"customize": "Benutzerdefinierte Ebenen hinzufügen oder entfernen",
"expand": "Alle ausklappen",
"optional": "Optionale Ebenen hinzufügen oder entfernen",
"optional-layers": "Mehr",
"table": {
"URL": "URL",
"empty": "Noch keine benutzerdefinierte Ebene definiert.",
"name": "Name",
"type": "Type"
},
"title": "Ebenen"
},
"profile": {
"clear": "Löschen",
"help": "Hilfe",
"placeholder": "Erstelle benutzerdefiniertes Profil hier.",
"upload": "Hochladen"
}
},
"title": "BRouter Web Client",
"warning": {
"cannot-get-route": "Fehler beim Abrufen der Routen-URL",
"no-response": "Keine Antwort vom Server",
"no-route-found": "Fehler: kann für angegebene Punkte keine Route finden. Vielleicht die Punkte näher an Straßen verschieben?",
"profile-error": "Profil-Fehler: keine oder leere Antwort vom Server",
"strava-error": "Fehler beim Laden der Strava Segmente: {{error}}",
"temporary-profile": "<strong>Note:</strong> Hochgeladene benutzerdefinierte Profile nur verübergehend auf dem Server zwischengespeichert. <br/>Bitte Bearbeitungen auf dem lokalen PC speichern.",
"upload-error": "Fehler beim Hochladen: {{error}}"
}
}

209
locales/en.json Normal file
View file

@ -0,0 +1,209 @@
{
"about": {
"bug-reports": "Bug reports and feature requests:",
"bug-reports-back": "server / backend, routing engine, Android app, profiles, brouter.de site",
"bug-reports-front": "web client / frontend.",
"chat": "Chat with users and developers",
"contact": "Contact:",
"data": "Data:",
"data-description": "This is based on <a href=\"https://www.openstreetmap.org\" target=\"_blank\">OpenStreetMap</a>. It is usually updated once a week when a new Planet file is available, see dates of <a href=\"http://brouter.de/brouter/segments4/\" target=\"_blank\">data files</a>.",
"description": "Online service of the BRouter routing engine. For the offline Android app and more information see <a href=\"http://brouter.de/\" target=\"_blank\">brouter.de</a>",
"details": "<i><a href=\"http://brouter.de/privacypolicy.html\" target=\"_blank\">Privacy Policy</a></i>, \n<i><a href=\"https://github.com/nrenner/brouter-web#credits-and-licenses\" target=\"_blank\">Credits</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web/blob/master/CHANGELOG.md\" target=\"_blank\">Changelog</a></i> and\n<i><a href=\"https://github.com/nrenner/brouter-web#readme\" target=\"_blank\">more info</a></i> on the client.",
"support": "General discussions/questions, support",
"title": "About"
},
"credits": {
"brouter": "BRouter",
"brouter-license": "<a target=\"_blank\" href=\"http://brouter.de/brouter\">BRouter</a> &copy; Arndt Brenschede",
"esri-license": "<a target=\"_blank\" href=\"http://goto.arcgisonline.com/maps/World_Imagery\">World Imagery</a> &copy; <a target=\"_blank\" href=\"https://www.esri.com/\">Esri</a>, sources: Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP, and the GIS User Community",
"esri-tiles": "Esri World Imagery",
"map-data": "Map data",
"map-tiles": "Map tiles",
"nominatim": "Search by <a href=\"https://wiki.openstreetmap.org/wiki/Nominatim\" target=\"_blank\" data-i18n=\"credits.nominatim\">Nominatim</a>",
"openstreetmap": "&copy; <a target=\"_blank\" href=\"https://www.openstreetmap.org/copyright\" >OpenStreetMap contributors</a> under <a target=\"_blank\" href=\"https://opendatacommons.org/licenses/odbl/\" >ODbL</a>"
},
"export": {
"format": "Format",
"format_csv": "CSV",
"format_geojson": "GeoJSON",
"format_gpx": "GPX",
"format_kml": "KML",
"route-from-to": "{{from}} -> {{to}} ({{distance}}km)",
"route-loop": "{{from}} ({{distance}}km)",
"title": "Export route",
"trackname": "Name"
},
"footer": {
"ascend": "Ascend",
"cost": "Cost",
"distance": "Distance",
"energy-per-100km": "Energy per 100 km",
"hours": "hours",
"hours-abbrev": "h",
"kilometer": "kilometers",
"kilometer-abbrev": "km",
"kilowatthour": "kilowatt hours",
"kilowatthour-abbrev": "kWh",
"mean-cost-factor": "Mean cost factor",
"meter": "meters",
"meter-abbrev": "m",
"plain-ascend": "Plain ascend",
"total-energy": "Total Energy",
"travel-time": "Travel time"
},
"layers": {
"add-base": "Add base layer",
"add-overlay": "Add overlay",
"customize": "Customize layers",
"placeholder-layer-name": "Custom layer name. (ex: OpenStreetMap)",
"placeholder-layer-url": "Custom layer URL. (ex: https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png)",
"remove-selection": "Remove selection"
},
"loadNogos": {
"defaultProperties": "Default properties",
"file": "File: ",
"load": "Load",
"nogoBuffer": "Buffer no-go areas (in meters): ",
"nogoRadius": "No-go radius (for points): ",
"nogoWeight": "No-go weight: ",
"source": "Source",
"title": "Load no-go areas",
"url": "URL: "
},
"map": {
"attribution-osm-long": "OpenStreetMap contributors",
"attribution-osm-short": "OpenStreetMap",
"clear-route": "Clear route",
"copyright": "Copyright",
"cycling": "Cycling",
"delete-last-point": "Delete last point",
"delete-nogo-areas": "&nbsp;&nbsp;also delete all no-go areas",
"delete-route": "Delete route?",
"draw-route-start": "Draw route (D key)",
"draw-route-stop": "Stop drawing route (ESC key)",
"hikebike-hillshading": "Hillshading",
"hiking": "Hiking",
"layer": {
"bing": "Bing Aerial",
"cycle": "OpenCycleMap (Thunderf.)",
"cycling": "Cycling (Waymarked Trails)",
"digitalglobe": "DigitalGlobe Recent Imagery",
"esri": "Esri World Imagery",
"hikebike-hillshading": "Hillshading (Hike & Bike Map)",
"hiking": "Hiking (Waymarked Trails)",
"osm": "OpenStreetMap",
"osmde": "OpenStreetMap.de",
"outdoors": "Outdoors (Thunderforest)",
"stamen-terrain": "Terrain (Stamen)",
"strava-segments": "Strava segments",
"topo": "OpenTopoMap"
},
"loading": "Loading…",
"locate-me": "Show me where I am",
"nogo": {
"cancel": "Cancel drawing no-go area",
"click-drag": "Click and drag to draw circle",
"draw": "Draw no-go area (circle)",
"edit": "Click to edit",
"help": "&square; = move / resize, <span class=\"fa fa-trash-o\"></span> = delete,<br>click circle to quit editing"
},
"opacity-slider": "Set transparency of route track and markers",
"privacy": "Privacy",
"reverse-route": "Reverse route",
"strava-biking": "Show Strava biking segments",
"strava-running": "Show Strava running segments",
"zoomInTitle": "Zoom in",
"zoomOutTitle": "Zoom out"
},
"modal": {
"close": "Close"
},
"navbar": {
"about": "About",
"alternative": {
"first": "1st alternative",
"original": "Original",
"second": "2nd alternative",
"third": "3rd alternative"
},
"export": "Export",
"load": {
"nogos": "No-go areas",
"title": "Load"
},
"profile": {
"car-eco": "Car (economic)",
"car-fast": "Car (fast)",
"car-test": "Car (test)",
"custom": "Custom",
"fastbike": "Fastbike",
"fastbike-asia-pacific": "Fastbike (Asia Pacific)",
"fastbike-lowtraffic": "Fastbike (low traffic)",
"hiking-beta": "Hiking (beta)",
"moped": "Moped",
"rail": "Rail",
"river": "River",
"safety": "Safety",
"shortest": "Shortest",
"trekking": "Trekking bike",
"trekking-ignore-cr": "Trekking bike (ignore cycle routes)",
"trekking-noferries": "Trekking bike (no ferries)",
"trekking-nosteps": "Trekking bike (no steps)",
"trekking-steep": "Trekking bike (steep)",
"vm-forum-liegerad-schnell": "Recumbent bike (fast)",
"vm-forum-velomobil-schnell": "Velomobile (fast)"
}
},
"sidebar": {
"custom-profile": {
"title": "Custom profile"
},
"data": {
"title": "Data"
},
"itinerary": {
"title": "Itinerary"
},
"layers": {
"category": {
"base-layers": "Base layers",
"country": "Country",
"europe": "Europe",
"europe-monolingual": "Europe monolingual",
"overlays": "Overlays",
"worldwide": "Worldwide",
"worldwide-international": "Worldwide international",
"worldwide-monolingual": "Worldwide monolingual"
},
"collapse": "Collapse all",
"custom-layers": "Custom layers",
"customize": "Add or remove custom layers",
"expand": "Expand all",
"optional": "Add or remove optional layers",
"optional-layers": "More",
"table": {
"URL": "URL",
"empty": "No custom layer configured yet.",
"name": "Name",
"type": "Type"
},
"title": "Layers"
},
"profile": {
"clear": "Clear",
"help": "Help",
"placeholder": "Write your custom profile here.",
"upload": "Upload"
}
},
"title": "BRouter web client",
"warning": {
"cannot-get-route": "Error getting route URL",
"no-response": "no response from server",
"no-route-found": "Error: cannot find a route for given points. Maybe try to move them closer to roads?",
"profile-error": "Profile error: no or empty response from server",
"strava-error": "Error getting Strava segments: {{error}}",
"temporary-profile": "<strong>Note:</strong> Uploaded custom profiles are only cached temporarily on the server.<br/>Please save your edits to your local PC.",
"upload-error": "Upload error: {{error}}"
}
}

209
locales/fr.json Normal file
View file

@ -0,0 +1,209 @@
{
"about": {
"bug-reports": "Rapporter un problème ou une amélioration:",
"bug-reports-back": "serveur, moteur de routage, appli Android, profils, site brouter.de;",
"bug-reports-front": "client web.",
"chat": "Discuter avec d'autres utilisateurs et développeurs",
"contact": "Contact:",
"data": "Données:",
"data-description": "Les données reposent sur <a href=\"https://www.openstreetmap.org\" target=\"_blank\">OpenStreetMap</a>. La mise à jour est généralement hebdomadaire, voir <a href=\"http://brouter.de/brouter/segments4/\" target=\"_blank\">les fichiers de données</a>. ",
"description": "Service web pour le moteur de routage BRouter. Pour l'appli Android et de plus amples informations, voir <a href=\"http://brouter.de/\" target=\"_blank\">brouter.de</a>.",
"details": "<i><a href=\"http://brouter.de/privacypolicy.html\" target=\"_blank\">Respect de la vie privée</a></i>, \n<i><a href=\"https://github.com/nrenner/brouter-web#credits-and-licenses\" target=\"_blank\">Crédits</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web/blob/master/CHANGELOG.md\" target=\"_blank\">Changements</a></i> et\n<i><a href=\"https://github.com/nrenner/brouter-web#readme\" target=\"_blank\">plus d'infos</a></i> sur le client web.",
"support": "Discussions/Questions générales, support;",
"title": "À propos"
},
"credits": {
"brouter": "BRouter",
"brouter-license": "<a target=\"_blank\" href=\"http://brouter.de/brouter\">BRouter</a> &copy; Arndt Brenschede",
"esri-license": "<a target=\"_blank\" href=\"http://goto.arcgisonline.com/maps/World_Imagery\">World Imagery</a> &copy; <a target=\"_blank\" href=\"https://www.esri.com/\">Esri</a>, sources : Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP, et la communauté d'utilisateurs GIS",
"esri-tiles": "Imagerie mondiale Esri",
"map-data": "Données de carte",
"map-tiles": "Tuiles de carte",
"nominatim": "Recherche par <a href=\"https://wiki.openstreetmap.org/wiki/Nominatim\" target=\"_blank\" data-i18n=\"credits.nominatim\">Nominatim</a>",
"openstreetmap": "&copy; <a target=\"_blank\" href=\"https://www.openstreetmap.org/copyright\" >contributeurs OpenStreetMap</a> sous <a target=\"_blank\" href=\"https://opendatacommons.org/licenses/odbl/\" >ODbL</a>"
},
"export": {
"format": "Format",
"format_csv": "CSV",
"format_geojson": "GeoJSON",
"format_gpx": "GPX",
"format_kml": "KML",
"route-from-to": "{{from}} -> {{to}} ({{distance}} km)",
"route-loop": "{{from}} ({{distance}} km)",
"title": "Exporter l'itinéraire",
"trackname": "Nom"
},
"footer": {
"ascend": "Dénivelé cumulé",
"cost": "Coût",
"distance": "Distance",
"energy-per-100km": "pour 100km",
"hours": "heures",
"hours-abbrev": "h",
"kilometer": "kilomètres",
"kilometer-abbrev": "km",
"kilowatthour": "kilowattheures",
"kilowatthour-abbrev": "kWh",
"mean-cost-factor": "Facteur coût moyen",
"meter": "mètres",
"meter-abbrev": "m",
"plain-ascend": "global",
"total-energy": "Énergie totale",
"travel-time": "Temps de trajet"
},
"layers": {
"add-base": "Ajouter un fond de carte ",
"add-overlay": "Ajouter un calque superposable",
"customize": "Personnaliser les calques",
"placeholder-layer-name": "Nom du calque (p. ex. OpenStreetMap).",
"placeholder-layer-url": "URL du calque (p. ex. https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png).",
"remove-selection": "Supprimer la sélection"
},
"loadNogos": {
"defaultProperties": "Propriétés par défaut",
"file": "Fichier:",
"load": "Charger",
"nogoBuffer": "Tampon des zones interdites (en mètres):",
"nogoRadius": "Rayon des zones interdites (pour les points):",
"nogoWeight": "Poids des zones interdites:",
"source": "Source",
"title": "Charger les zones interdites",
"url": "URL:"
},
"map": {
"attribution-osm-long": "Contributeurs OpenStreetMap",
"attribution-osm-short": "OpenStreetMap",
"clear-route": "Supprimer l'itinéraire",
"copyright": "Copyright",
"cycling": "Vélo",
"delete-last-point": "Supprimer le dernier point",
"delete-nogo-areas": "&nbsp;&nbsp;supprimer aussi toutes les zones interdites",
"delete-route": "Supprimer l'itinéraire?",
"draw-route-start": "Dessiner l'itinéraire (touche D)",
"draw-route-stop": "Arrêter de dessiner l'itinéraire (touche ECH)",
"hikebike-hillshading": "Relief avec ombre",
"hiking": "Randonnée",
"layer": {
"bing": "Imagerie aérienne BING",
"cycle": "OpenCycleMap (Thunderf.)",
"cycling": "Cycling (Sentiers balisés)",
"digitalglobe": "Imagerie récente DigitalGlobe",
"esri": "Imagerie mondiale Esri",
"hikebike-hillshading": "Relief avec ombre (carte randonnée et vélo)",
"hiking": "Randonnée (Sentiers balisés)",
"osm": "OpenStreetMap",
"osmde": "OpenStreetMap.de",
"outdoors": "Extérieur (Thunderforest)",
"stamen-terrain": "Terrain (Stamen)",
"strava-segments": "Segments Strava",
"topo": "OpenTopoMap"
},
"loading": "Chargement…",
"locate-me": "Où suis-je ?",
"nogo": {
"cancel": "Annuler la zone interdite",
"click-drag": "Cliquer et faire glisser pour dessiner un cercle",
"draw": "Dessiner une zone interdite (cercle)",
"edit": "Cliquer pour éditer",
"help": "&square; = déplacer / redimensionner, <span class=\"fa fa-trash-o\"></span> = supprimer,<br>cliquer sur le cercle pour arrêter l'édition"
},
"opacity-slider": "Définie l'opacité de l'itinéraire et des marqueurs",
"privacy": "Vie privée",
"reverse-route": "Inverse l'itinéraire",
"strava-biking": "Afficher les segments Strava vélo",
"strava-running": "Afficher les segments Strava à pied",
"zoomInTitle": "Zoomer",
"zoomOutTitle": "Dézoomer"
},
"modal": {
"close": "Fermer"
},
"navbar": {
"about": "À propos",
"alternative": {
"first": "1ère alternative",
"original": "Originale",
"second": "2nde alternative",
"third": "3ème alternative"
},
"export": "Exporter",
"load": {
"nogos": "Zones interdites",
"title": "Charger"
},
"profile": {
"car-eco": "Voiture (économique)",
"car-fast": "Voiture (rapide)",
"car-test": "Voiture (test)",
"custom": "Personnalisé",
"fastbike": "Vélo de route",
"fastbike-asia-pacific": "Vélo de route (Asie Pacifique)",
"fastbike-lowtraffic": "Vélo de route (faible traffic)",
"hiking-beta": "Randonnée (beta)",
"moped": "Mobylette",
"rail": "Train",
"river": "Fluvial",
"safety": "Sécurité",
"shortest": "Le plus court",
"trekking": "Cyclotourisme",
"trekking-ignore-cr": "Cycloutourisme (ignore pistes cyclables)",
"trekking-noferries": "Cyclotourisme (pas de ferries)",
"trekking-nosteps": "Cyclotourisme (pas d'escaliers)",
"trekking-steep": "Cyclotourisme (escarpé)",
"vm-forum-liegerad-schnell": "Vélo couché (rapide)",
"vm-forum-velomobil-schnell": "Vélomobile (rapide)"
}
},
"sidebar": {
"custom-profile": {
"title": "Profil personnalisé"
},
"data": {
"title": "Données"
},
"itinerary": {
"title": "Itinéraire"
},
"layers": {
"category": {
"base-layers": "Fonds de carte",
"country": "Pays",
"europe": "Europe",
"europe-monolingual": "Europe monolingue",
"overlays": "Calques superposables",
"worldwide": "Mondial",
"worldwide-international": "Mondial international",
"worldwide-monolingual": "Mondial monolingue"
},
"collapse": "Tout réduire",
"custom-layers": "Calques personels",
"customize": "Ajouter ou supprimer des calques",
"expand": "Tout ouvrir",
"optional": "Ajouter ou supprimer des calques optionnels",
"optional-layers": "Plus",
"table": {
"URL": "URL",
"empty": "Aucun calque personnel trouvé.",
"name": "Nom",
"type": "Type"
},
"title": "Calques"
},
"profile": {
"clear": "Nettoyer",
"help": "Aide",
"placeholder": "Saisissez votre profil personnalisé ici.",
"upload": "Envoyer"
}
},
"title": "Client web BRouter",
"warning": {
"cannot-get-route": "Erreur lors de la réception de l'itinéraire",
"no-response": "aucune réponse du serveur",
"no-route-found": "Erreur: impossible de trouver un itinéraire correspondant aux points donnés. Essayez peut-être de rapprocher les points des routes?",
"profile-error": "Erreur de profil: pas de réponse ou réponse invalide du serveur",
"strava-error": "Impossible d'afficher les segments Strava: {{error}}",
"temporary-profile": "<strong>Note:</strong> Les profils personnalisés téléversés ne sont disponibles que temporairement sur le serveur.<br/>Merci de sauvegarder votre profil sur votre ordinateur.",
"upload-error": "Erreur d'envoi : {{error}}"
}
}

209
locales/gl.json Normal file
View file

@ -0,0 +1,209 @@
{
"about": {
"bug-reports": "Bug reports and feature requests:",
"bug-reports-back": "server / backend, routing engine, Android app, profiles, brouter.de site",
"bug-reports-front": "web client / frontend.",
"chat": "Chat with users and developers",
"contact": "Contacto:",
"data": "Datos:",
"data-description": "This is based on <a href=\"https://www.openstreetmap.org\" target=\"_blank\">OpenStreetMap</a>. It is usually updated once a week when a new Planet file is available, see dates of <a href=\"http://brouter.de/brouter/segments4/\" target=\"_blank\">data files</a>.",
"description": "Online service of the BRouter routing engine. For the offline Android app and more information see <a href=\"http://brouter.de/\" target=\"_blank\">brouter.de</a>",
"details": "<i><a href=\"http://brouter.de/privacypolicy.html\" target=\"_blank\">Privacy Policy</a></i>, \n<i><a href=\"https://github.com/nrenner/brouter-web#credits-and-licenses\" target=\"_blank\">Credits</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web/blob/master/CHANGELOG.md\" target=\"_blank\">Changelog</a></i> and\n<i><a href=\"https://github.com/nrenner/brouter-web#readme\" target=\"_blank\">more info</a></i> on the client.",
"support": "General discussions/questions, support",
"title": "Sobre nós"
},
"credits": {
"brouter": "BRouter",
"brouter-license": "<a target=\"_blank\" href=\"http://brouter.de/brouter\">BRouter</a> &copy; Arndt Brenschede",
"esri-license": "<a target=\"_blank\" href=\"http://goto.arcgisonline.com/maps/World_Imagery\">World Imagery</a> &copy; <a target=\"_blank\" href=\"https://www.esri.com/\">Esri</a>, sources: Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP, and the GIS User Community",
"esri-tiles": "Esri World Imagery",
"map-data": "Map data",
"map-tiles": "Map tiles",
"nominatim": "Search by <a href=\"https://wiki.openstreetmap.org/wiki/Nominatim\" target=\"_blank\" data-i18n=\"credits.nominatim\">Nominatim</a>",
"openstreetmap": "&copy; <a target=\"_blank\" href=\"https://www.openstreetmap.org/copyright\" >OpenStreetMap contributors</a> under <a target=\"_blank\" href=\"https://opendatacommons.org/licenses/odbl/\" >ODbL</a>"
},
"export": {
"format": "Format",
"format_csv": "CSV",
"format_geojson": "GeoJSON",
"format_gpx": "GPX",
"format_kml": "KML",
"route-from-to": "{{from}} -> {{to}} ({{distance}}km)",
"route-loop": "{{from}} ({{distance}}km)",
"title": "Export route",
"trackname": "Name"
},
"footer": {
"ascend": "Ascend",
"cost": "Cost",
"distance": "Distance",
"energy-per-100km": "Energy per 100 km",
"hours": "hours",
"hours-abbrev": "h",
"kilometer": "kilometers",
"kilometer-abbrev": "km",
"kilowatthour": "kilowatt hours",
"kilowatthour-abbrev": "kWh",
"mean-cost-factor": "Mean cost factor",
"meter": "meters",
"meter-abbrev": "m",
"plain-ascend": "Plain ascend",
"total-energy": "Total Energy",
"travel-time": "Travel time"
},
"layers": {
"add-base": "Add base layer",
"add-overlay": "Add overlay",
"customize": "Customize layers",
"placeholder-layer-name": "Custom layer name. (ex: OpenStreetMap)",
"placeholder-layer-url": "Custom layer URL. (ex: https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png)",
"remove-selection": "Remove selection"
},
"loadNogos": {
"defaultProperties": "Default properties",
"file": "File: ",
"load": "Load",
"nogoBuffer": "Buffer no-go areas (in meters): ",
"nogoRadius": "No-go radius (for points): ",
"nogoWeight": "No-go weight: ",
"source": "Source",
"title": "Load no-go areas",
"url": "URL: "
},
"map": {
"attribution-osm-long": "OpenStreetMap contributors",
"attribution-osm-short": "OpenStreetMap",
"clear-route": "Clear route",
"copyright": "Copyright",
"cycling": "Cycling",
"delete-last-point": "Delete last point",
"delete-nogo-areas": "&nbsp;&nbsp;also delete all no-go areas",
"delete-route": "Delete route?",
"draw-route-start": "Draw route (D key)",
"draw-route-stop": "Stop drawing route (ESC key)",
"hikebike-hillshading": "Hillshading",
"hiking": "Hiking",
"layer": {
"bing": "Bing Aerial",
"cycle": "OpenCycleMap (Thunderf.)",
"cycling": "Cycling (Waymarked Trails)",
"digitalglobe": "DigitalGlobe Recent Imagery",
"esri": "Esri World Imagery",
"hikebike-hillshading": "Hillshading (Hike & Bike Map)",
"hiking": "Hiking (Waymarked Trails)",
"osm": "OpenStreetMap",
"osmde": "OpenStreetMap.de",
"outdoors": "Outdoors (Thunderforest)",
"stamen-terrain": "Terrain (Stamen)",
"strava-segments": "Strava segments",
"topo": "OpenTopoMap"
},
"loading": "Loading…",
"locate-me": "Show me where I am",
"nogo": {
"cancel": "Cancel drawing no-go area",
"click-drag": "Click and drag to draw circle",
"draw": "Draw no-go area (circle)",
"edit": "Click to edit",
"help": "&square; = move / resize, <span class=\"fa fa-trash-o\"></span> = delete,<br>click circle to quit editing"
},
"opacity-slider": "Set transparency of route track and markers",
"privacy": "Privacy",
"reverse-route": "Reverse route",
"strava-biking": "Show Strava biking segments",
"strava-running": "Show Strava running segments",
"zoomInTitle": "Zoom in",
"zoomOutTitle": "Zoom out"
},
"modal": {
"close": "Close"
},
"navbar": {
"about": "About",
"alternative": {
"first": "1st alternative",
"original": "Original",
"second": "2nd alternative",
"third": "3rd alternative"
},
"export": "Export",
"load": {
"nogos": "No-go areas",
"title": "Load"
},
"profile": {
"car-eco": "Car (economic)",
"car-fast": "Car (fast)",
"car-test": "Car (test)",
"custom": "Custom",
"fastbike": "Fastbike",
"fastbike-asia-pacific": "Fastbike (Asia Pacific)",
"fastbike-lowtraffic": "Fastbike (low traffic)",
"hiking-beta": "Hiking (beta)",
"moped": "Moped",
"rail": "Rail",
"river": "River",
"safety": "Safety",
"shortest": "Shortest",
"trekking": "Trekking bike",
"trekking-ignore-cr": "Trekking bike (ignore cycle routes)",
"trekking-noferries": "Trekking bike (no ferries)",
"trekking-nosteps": "Trekking bike (no steps)",
"trekking-steep": "Trekking bike (steep)",
"vm-forum-liegerad-schnell": "Recumbent bike (fast)",
"vm-forum-velomobil-schnell": "Velomobile (fast)"
}
},
"sidebar": {
"custom-profile": {
"title": "Custom profile"
},
"data": {
"title": "Data"
},
"itinerary": {
"title": "Itinerary"
},
"layers": {
"category": {
"base-layers": "Base layers",
"country": "Country",
"europe": "Europe",
"europe-monolingual": "Europe monolingual",
"overlays": "Overlays",
"worldwide": "Worldwide",
"worldwide-international": "Worldwide international",
"worldwide-monolingual": "Worldwide monolingual"
},
"collapse": "Collapse all",
"custom-layers": "Custom layers",
"customize": "Add or remove custom layers",
"expand": "Expand all",
"optional": "Add or remove optional layers",
"optional-layers": "More",
"table": {
"URL": "URL",
"empty": "No custom layer configured yet.",
"name": "Name",
"type": "Type"
},
"title": "Layers"
},
"profile": {
"clear": "Clear",
"help": "Help",
"placeholder": "Write your custom profile here.",
"upload": "Upload"
}
},
"title": "BRouter web client",
"warning": {
"cannot-get-route": "Error getting route URL",
"no-response": "no response from server",
"no-route-found": "Error: cannot find a route for given points. Maybe try to move them closer to roads?",
"profile-error": "Profile error: no or empty response from server",
"strava-error": "Error getting Strava segments: {{error}}",
"temporary-profile": "<strong>Note:</strong> Uploaded custom profiles are only cached temporarily on the server.<br/>Please save your edits to your local PC.",
"upload-error": "Upload error: {{error}}"
}
}

209
locales/hu.json Normal file
View file

@ -0,0 +1,209 @@
{
"about": {
"bug-reports": "Hibák és kérések bejelentése:",
"bug-reports-back": "kiszolgálóoldal / back end, útvonaltervező motor, Android alkalmazás, profilok, brouter.de weboldal",
"bug-reports-front": "weboldali kliens / frontend.",
"chat": "Chat with users and developers",
"contact": "Kapcsolat:",
"data": "Adatok:",
"data-description": "Az adatok alapja az <a href=\"https://www.openstreetmap.org\" target=\"_blank\">OpenStreetMap</a>. Az oldal általában hetente frissül, amikor az új bolygófájl (Planet fájl) elérhetővé válik. A dátumok megtalálhatók az <a href=\"http://brouter.de/brouter/segments4/\" target=\"_blank\">adatfájlokban</a>.",
"description": "Ez itt a BRouter útvonaltervező motor online szolgáltatása. Az Android alkalmazásról és a további információkról a <a href=\"http://brouter.de/\" target=\"_blank\">brouter.de</a> oldalon olvashat.",
"details": "<i><a href=\"http://brouter.de/privacypolicy.html\" target=\"_blank\">Adatvédelmi irányelvek</a></i> \n<i><a href=\"https://github.com/nrenner/brouter-web#credits-and-licenses\" target=\"_blank\">stáblista</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web/blob/master/CHANGELOG.md\" target=\"_blank\">változásnapló</a></i> és\n<i><a href=\"https://github.com/nrenner/brouter-web#readme\" target=\"_blank\">további tájékoztatás</a></i> a kliensen.",
"support": "Általános viták/kérdések, támogatás",
"title": "Névjegy"
},
"credits": {
"brouter": "BRouter",
"brouter-license": "<a target=\"_blank\" href=\"http://brouter.de/brouter\">BRouter</a> &copy; Arndt Brenschede",
"esri-license": "<a target=\"_blank\" href=\"http://goto.arcgisonline.com/maps/World_Imagery\">Globális légi felvételek</a> &copy; <a target=\"_blank\" href=\"https://www.esri.com/\">Esri</a>, források: Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP valamint a GIS felhasználói közösség",
"esri-tiles": "Esri World Imagery",
"map-data": "Térképadatok",
"map-tiles": "Map tiles",
"nominatim": "Keresés: <a href=\"https://wiki.openstreetmap.org/wiki/Nominatim\" target=\"_blank\" data-i18n=\"credits.nominatim\">Nominatim</a>",
"openstreetmap": "&copy; <a target=\"_blank\" href=\"https://www.openstreetmap.org/copyright\" >OpenStreetMap közreműködők</a>, <a target=\"_blank\" href=\"https://opendatacommons.org/licenses/odbl/\" >ODbL</a> licenc szerint"
},
"export": {
"format": "Formátum",
"format_csv": "CSV",
"format_geojson": "GeoJSON",
"format_gpx": "GPX",
"format_kml": "KML",
"route-from-to": "{{from}} -> {{to}} ({{distance}} km)",
"route-loop": "{{from}} ({{distance}} km)",
"title": "Útvonal exportálása",
"trackname": "Név"
},
"footer": {
"ascend": "Összes szintkülönbség",
"cost": "Ráfordítás",
"distance": "Távolság",
"energy-per-100km": "Energia / 100 km",
"hours": "óra",
"hours-abbrev": "h",
"kilometer": "kilométer",
"kilometer-abbrev": "km",
"kilowatthour": "kilowattóra",
"kilowatthour-abbrev": "kWh",
"mean-cost-factor": "Közepes ráfordítástényező",
"meter": "méter",
"meter-abbrev": "m",
"plain-ascend": "Végpontok közötti szintkülönbség",
"total-energy": "Teljes energiafelhasználás",
"travel-time": "Utazási idő"
},
"layers": {
"add-base": "Alapréteg hozzáadása",
"add-overlay": "Rátétréteg hozzáadása",
"customize": "Rétegek személyre szabása",
"placeholder-layer-name": "Egyedi rétegnév (pl. OpenStreetMap)",
"placeholder-layer-url": "Egyedi réteg URL-je (pl. https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png)",
"remove-selection": "Kijelölés eltávolítása"
},
"loadNogos": {
"defaultProperties": "Alapértelmezett tulajdonságok",
"file": "Fájl:",
"load": "Betöltés",
"nogoBuffer": "Elkerülendő területek körüli távolság (méter)",
"nogoRadius": "Elkerülendő pontok körüli sugár:",
"nogoWeight": "Elkerülés súlyozása:",
"source": "Forrás",
"title": "Elkerülendő területek betöltése",
"url": "URL:"
},
"map": {
"attribution-osm-long": "OpenStreetMap közreműködők",
"attribution-osm-short": "OpenStreetMap",
"clear-route": "Útvonal törlése",
"copyright": "Szerzői jogok",
"cycling": "Kerékpározás",
"delete-last-point": "Utolsó pont törlése",
"delete-nogo-areas": "&nbsp;&nbsp;törlődjenek az elkerülendő területek is",
"delete-route": "Útvonal törlése?",
"draw-route-start": "Útvonal rajzolása (D billentyű)",
"draw-route-stop": "Útvonal rajzolásának megszakítása (ESC billentyű)",
"hikebike-hillshading": "Domborzatárnyékolás",
"hiking": "Túrázás",
"layer": {
"bing": "Bing Aerial",
"cycle": "OpenCycleMap (Thunderfores)",
"cycling": "Kerékpározás (Waymarked Trails)",
"digitalglobe": "DigitalGlobe Recent Imagery",
"esri": "Esri World Imagery",
"hikebike-hillshading": "Domborzatárnyékolás (Hike & Bike Map)",
"hiking": "Túrázás (Waymarked Trails)",
"osm": "OpenStreetMap",
"osmde": "OpenStreetMap.de",
"outdoors": "Outdoors (Thunderforest)",
"stamen-terrain": "Terrain (Stamen)",
"strava-segments": "Strava útvonalszakaszok",
"topo": "OpenTopoMap"
},
"loading": "Betöltés…",
"locate-me": "Mutassa a térképen, hol vagyok",
"nogo": {
"cancel": "Elkerülendő terület rajzolásának megszakítása",
"click-drag": "Kör rajzolása kattintással és húzással",
"draw": "Elkerülendő terület rajzolása (kör)",
"edit": "Szerkesztés kattintással",
"help": "&square; = áthelyezés / átméretezés, <span class=\"fa fa-trash-o\"></span> = törlés,<br>a szerkesztés befejezéséhez kattintson a körre"
},
"opacity-slider": "Nyomvonal és jelölők átlátszóságának beállítása",
"privacy": "Adatvédelem",
"reverse-route": "Útvonal megfordítása",
"strava-biking": "Strava motoros útvonalszakaszok megjelenítése",
"strava-running": "Strava futó útvonalszakaszok megjelenítése",
"zoomInTitle": "Nagyítás",
"zoomOutTitle": "Kicsinyítés"
},
"modal": {
"close": "Bezárás"
},
"navbar": {
"about": "Névjegy",
"alternative": {
"first": "1. alternatíva",
"original": "Eredeti",
"second": "2. alternatíva",
"third": "3. alternatíva"
},
"export": "Exportálás",
"load": {
"nogos": "Elkerülendő területek",
"title": "Betöltés"
},
"profile": {
"car-eco": "Személyautó (gazdaságos)",
"car-fast": "Személyautó (gyors)",
"car-test": "Személyautó (teszt)",
"custom": "Egyedi",
"fastbike": "Versenykerékpár",
"fastbike-asia-pacific": "Versenykerékpár (ázsiai-csendes-óceáni térség)",
"fastbike-lowtraffic": "Versenykerékpár (kis forgalom)",
"hiking-beta": "Túrázás (béta)",
"moped": "Robogó",
"rail": "Vasút",
"river": "Hajózás",
"safety": "Biztonság",
"shortest": "Legrövidebb",
"trekking": "Túrakerékpár",
"trekking-ignore-cr": "Túrakerékpár (kerékpárutak figyelmen kívül hagyása)",
"trekking-noferries": "Túrakerékpár (komp nélkül)",
"trekking-nosteps": "Túrakerékpár (lépcső nélkül)",
"trekking-steep": "Túrakerékpár (meredek)",
"vm-forum-liegerad-schnell": "Rekumbens kerékpár (gyors)",
"vm-forum-velomobil-schnell": "Velomobil (aerodinamikus kerékpár, gyors)"
}
},
"sidebar": {
"custom-profile": {
"title": "Egyedi profil"
},
"data": {
"title": "Adatok"
},
"itinerary": {
"title": "Útiterv"
},
"layers": {
"category": {
"base-layers": "Alaprétegek",
"country": "Ország",
"europe": "Európa",
"europe-monolingual": "Európa (egynyelvű)",
"overlays": "Rátétrétegek",
"worldwide": "Világ",
"worldwide-international": "Világ (nemzetközi)",
"worldwide-monolingual": "Világ (egynyelvű)"
},
"collapse": "Minden összecsukása",
"custom-layers": "Egyedi rétegek",
"customize": "Egyedi rétegek hozzáadása vagy eltávolítása",
"expand": "Minden kibontása",
"optional": "Rétegek hozzáadása vagy eltávolítása",
"optional-layers": "Több",
"table": {
"URL": "URL",
"empty": "Még nincs beállítva egyedi réteg",
"name": "Név",
"type": "Típus"
},
"title": "Rétegek"
},
"profile": {
"clear": "Törlés",
"help": "Súgó",
"placeholder": "Ide írhatja egyedi profilját",
"upload": "Feltöltés"
}
},
"title": "BRouter webkliens",
"warning": {
"cannot-get-route": "Hiba történt az útvonal URL-jének beolvasásakor",
"no-response": "a kiszolgáló nem válaszol",
"no-route-found": "Hiba: az adott pontokhoz nem sikerül útvonalat találni. Esetleg próbálja meg közelebb tenni őket az utakhoz.",
"profile-error": "Profilhiba: a kiszolgáló nem válaszol vagy üres választ küld",
"strava-error": "Hiba történt a Strava útvonalszakaszok lekérésénél: {{error}}",
"temporary-profile": "<strong>Figyelem:</strong> A kiszolgáló csak ideiglenesen gyorsítótárazza a feltöltött egyedi profilokat.<br/>Kérjük, szerkesztéseit mentse el saját számítógépére.",
"upload-error": "Feltöltési hiba: {{error}}"
}
}

36
locales/keys.js Normal file
View file

@ -0,0 +1,36 @@
// this file contains translatable keys that are dynamic / not visible by i18n extractor tool
i18next.t('navbar.profile.car-eco');
i18next.t('navbar.profile.car-fast');
i18next.t('navbar.profile.car-test');
i18next.t('navbar.profile.fastbike');
i18next.t('navbar.profile.fastbike-asia-pacific');
i18next.t('navbar.profile.fastbike-asia-pacific');
i18next.t('navbar.profile.fastbike-lowtraffic');
i18next.t('navbar.profile.hiking-beta');
i18next.t('navbar.profile.moped');
i18next.t('navbar.profile.rail');
i18next.t('navbar.profile.river');
i18next.t('navbar.profile.safety');
i18next.t('navbar.profile.shortest');
i18next.t('navbar.profile.trekking');
i18next.t('navbar.profile.trekking-ignore-cr');
i18next.t('navbar.profile.trekking-noferries');
i18next.t('navbar.profile.trekking-nosteps');
i18next.t('navbar.profile.trekking-steep');
i18next.t('navbar.profile.vm-forum-liegerad-schnell');
i18next.t('navbar.profile.vm-forum-velomobil-schnell');
i18next.t('sidebar.layers.category.base-layers', 'Base layers');
i18next.t(
'sidebar.layers.category.worldwide-international',
'Worldwide international'
);
i18next.t(
'sidebar.layers.category.worldwide-monolingual',
'Worldwide monolingual'
);
i18next.t('sidebar.layers.category.europe', 'Europe');
i18next.t('sidebar.layers.category.europe-monolingual', 'Europe monolingual');
i18next.t('sidebar.layers.category.country', 'Country');
i18next.t('sidebar.layers.category.overlays', 'Overlays');
i18next.t('sidebar.layers.category.worldwide', 'Worldwide');

209
locales/pl.json Normal file
View file

@ -0,0 +1,209 @@
{
"about": {
"bug-reports": "Zgłaszanie błędów i nowych funkcji:",
"bug-reports-back": "serwer / aplikacja, silnik wyznaczania tras, aplikacja na Androida, profile, strona brouter.de",
"bug-reports-front": "klient przeglądarkowy / interfejs.",
"chat": "Czat z użytkownikami i twórcami",
"contact": "Kontakt:",
"data": "Dane:",
"data-description": "Bazujemy na <a href=\"https://www.openstreetmap.org\" target=\"_blank\">OpenStreetMap</a>. Aktualizacje są zazwyczaj raz w tygodniu, kiedy są dostępne nowe pliki Planet, więc sprawdź daty <a href=\"http://brouter.de/brouter/segments4/\" target=\"_blank\">plików z danymi</a>.",
"description": "Dostęp online do silnika wyznaczania tras BRouter. Aplikację offline na Androida i więcej informacji znajdziesz na <a href=\"http://brouter.de/\" target=\"_blank\">brouter.de</a>",
"details": "<i><a href=\"http://brouter.de/privacypolicy.html\" target=\"_blank\">Polityka prywatności</a></i>, \n<i><a href=\"https://github.com/nrenner/brouter-web#credits-and-licenses\" target=\"_blank\">Autorzy</a></i>,\n<i><a href=\"https://github.com/nrenner/brouter-web/blob/master/CHANGELOG.md\" target=\"_blank\">Histroia zmian</a></i> i\n<i><a href=\"https://github.com/nrenner/brouter-web#readme\" target=\"_blank\">więcej informacji</a></i> na temat klienta.",
"support": "Ogólna dyskusja/pytania, wsparcie",
"title": "Informacje"
},
"credits": {
"brouter": "BRouter",
"brouter-license": "<a target=\"_blank\" href=\"http://brouter.de/brouter\">BRouter</a> &copy; Arndt Brenschede",
"esri-license": "<a target=\"_blank\" href=\"http://goto.arcgisonline.com/maps/World_Imagery\">World Imagery</a> &copy; <a target=\"_blank\" href=\"https://www.esri.com/\">Esri</a>, źródła: Esri, DigitalGlobe, Earthstar Geographics, CNES/Airbus DS, GeoEye, USDA FSA, USGS, Getmapping, Aerogrid, IGN, IGP oraz Społeczność Użytkowników GIS",
"esri-tiles": "Obrazowanie świata Esri",
"map-data": "Dane mapy",
"map-tiles": "Kafelki mapy",
"nominatim": "Wyszukaj przez <a href=\"https://wiki.openstreetmap.org/wiki/Nominatim\" target=\"_blank\" data-i18n=\"credits.nominatim\">Nominatim</a>",
"openstreetmap": "&copy; <a target=\"_blank\" href=\"https://www.openstreetmap.org/copyright\" >Współtwórcy OpenStreetMap</a> na licencji <a target=\"_blank\" href=\"https://opendatacommons.org/licenses/odbl/\" >ODbL</a>"
},
"export": {
"format": "Format",
"format_csv": "CSV",
"format_geojson": "GeoJSON",
"format_gpx": "GPX",
"format_kml": "KML",
"route-from-to": "{{from}} -> {{to}} ({{distance}}km)",
"route-loop": "{{from}} ({{distance}}km)",
"title": "Eksportuj trasę",
"trackname": "Nazwa"
},
"footer": {
"ascend": "Przewyższenie",
"cost": "Koszt",
"distance": "Odległość",
"energy-per-100km": "Energia na 100 km",
"hours": "godzin",
"hours-abbrev": "godz.",
"kilometer": "kilometrów",
"kilometer-abbrev": "km",
"kilowatthour": "kilowatogodziny",
"kilowatthour-abbrev": "kWh",
"mean-cost-factor": "Średni współczynnik kosztów",
"meter": "metrów",
"meter-abbrev": "m",
"plain-ascend": "Różnica wysokości",
"total-energy": "Całkowita energia",
"travel-time": "Czas podróży"
},
"layers": {
"add-base": "Dodaj warstwę bazową",
"add-overlay": "Dodaj nakładkę",
"customize": "Dostosuj warstwy",
"placeholder-layer-name": "Własna nazwa warstwy. (np. OpenStreetMap)",
"placeholder-layer-url": "URL niestandardowej warstwy. (np: https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png)",
"remove-selection": "Usuń zaznaczenie"
},
"loadNogos": {
"defaultProperties": "Domyślne właściwości",
"file": "Plik:",
"load": "Wczytaj",
"nogoBuffer": "Bufor stref no-go (w metrach):",
"nogoRadius": "Promień no-go (dla punktów):",
"nogoWeight": "Waga no-go:",
"source": "Źródło",
"title": "Wczytaj strefy no-go",
"url": "URL: "
},
"map": {
"attribution-osm-long": "Współtwórcy OpenStreetMap",
"attribution-osm-short": "OpenStreetMap",
"clear-route": "Wyczyść trasę",
"copyright": "Prawa autorskie",
"cycling": "Kolarstwo",
"delete-last-point": "Usuń ostatni punkt",
"delete-nogo-areas": "&nbsp;&nbsp;usunie również wszystkie strefy no-go",
"delete-route": "Usunąć trasę?",
"draw-route-start": "Rysuj trasę (klawisz D)",
"draw-route-stop": "Przestań rysować trasę (klawisz Esc)",
"hikebike-hillshading": "Cieniowanie",
"hiking": "Wspinaczka",
"layer": {
"bing": "Zdjęcia lotnicze Bing",
"cycle": "OpenCycleMap (Thunderf.)",
"cycling": "Jazda na rowerze (oznaczone szlaki)",
"digitalglobe": "Najnowsze obrazowanie DigitalGlobe",
"esri": "Obrazowanie świata Esri",
"hikebike-hillshading": "Cieniowanie (Mapa wspinaczkowo-rowerowa)",
"hiking": "Wspinaczka (oznaczone szlaki)",
"osm": "OpenStreetMap",
"osmde": "OpenStreetMap.de",
"outdoors": "Na zewnątrz (Thunderforest)",
"stamen-terrain": "Teren (Stamen)",
"strava-segments": "Segmenty Strava",
"topo": "OpenTopoMap"
},
"loading": "Wczytywanie…",
"locate-me": "Pokaż mi gdzie jestem",
"nogo": {
"cancel": "Anuluj rysowanie strefy no-go",
"click-drag": "Kliknij i przeciągnij, aby narysować okrąg",
"draw": "Rysuj strefę no-go (okrąg)",
"edit": "Kliknij, aby edytować",
"help": "&square; = przesuń / zmień rozmiar, <span class=\"fa fa-trash-o\"></span> = usuń,<br>kliknij okrąg, aby wyjść z edycji"
},
"opacity-slider": "Ustaw przezroczystość trasy i znaczników",
"privacy": "Prywatność",
"reverse-route": "Odwróć trasę",
"strava-biking": "Pokaż rowerowe segmenty Strava ",
"strava-running": "Pokaż biegowe segmenty Strava",
"zoomInTitle": "Przybliż",
"zoomOutTitle": "Oddal"
},
"modal": {
"close": "Zamknij"
},
"navbar": {
"about": "Informacje",
"alternative": {
"first": "Alternatywa nr 1",
"original": "Pierwotna",
"second": "Alternatywa nr 2",
"third": "Alternatywa nr 3"
},
"export": "Eksportuj",
"load": {
"nogos": "Stefy no-go",
"title": "Wczytaj"
},
"profile": {
"car-eco": "Samochód (ekonomicznie)",
"car-fast": "Samochód (szybko)",
"car-test": "Samochód (testowo)",
"custom": "Własne",
"fastbike": "Rower szosowy",
"fastbike-asia-pacific": "Rower szosowy (Azja Pacyfik)",
"fastbike-lowtraffic": "Rower szosowy (mały ruch)",
"hiking-beta": "Wspinaczka (beta)",
"moped": "Motorower",
"rail": "Kolej",
"river": "Rzeka",
"safety": "Bezpieczeństwo",
"shortest": "Najkrótsza",
"trekking": "Rower trekkingowy",
"trekking-ignore-cr": "Rower trekkingowy (ignoruj trasy rowerowe)",
"trekking-noferries": "Rower trekkingowy (bez promów)",
"trekking-nosteps": "Rower trekkingowy (bez schodów)",
"trekking-steep": "Rower trekkingowy (stromo)",
"vm-forum-liegerad-schnell": "Rower poziomy (szosowy)",
"vm-forum-velomobil-schnell": "Velomobile (szosowy)"
}
},
"sidebar": {
"custom-profile": {
"title": "Własny profil"
},
"data": {
"title": "Dane"
},
"itinerary": {
"title": "Plan podróży"
},
"layers": {
"category": {
"base-layers": "Warstwy bazowe",
"country": "Kraj",
"europe": "Europa",
"europe-monolingual": "Europa jeden język",
"overlays": "Nakładki",
"worldwide": "Cały świat",
"worldwide-international": "Świat wielojęzyczny",
"worldwide-monolingual": "Świat jeden język"
},
"collapse": "Zwiń wszystkie",
"custom-layers": "Własne warstwy",
"customize": "Dodaj lub usuń niestandardowe warstwy",
"expand": "Rozwiń wszystkie",
"optional": "Dodaj lub usuń opcjonalne warstwy",
"optional-layers": "Więcej",
"table": {
"URL": "URL",
"empty": "Nie skonfigurowano jeszcze niestandardowych warstw.",
"name": "Nazwa",
"type": "Rodzaj"
},
"title": "Warstwy"
},
"profile": {
"clear": "Wyczyść",
"help": "Pomoc",
"placeholder": "Zapisz tutaj swój własny profil.",
"upload": "Prześlij"
}
},
"title": "Klient przeglądarkowy BRouter",
"warning": {
"cannot-get-route": "Błąd pobierania URL trasy",
"no-response": "brak odpowiedzi z serwera",
"no-route-found": "Błąd: nie można znaleźć trasy dla zadanych punktów. Spróbuj przesunąć je bliżej dróg.",
"profile-error": "Błąd profilu: brak lub pusta odpowiedź z serwera",
"strava-error": "Błąd pobierania segmentów Strava: {{error}}",
"temporary-profile": "<strong>Uwaga:</strong> Przesłane własne profile są zapisywane na serwerze tylko tymczasowo.<br/>Zapisz zmiany na swoim komputerze.",
"upload-error": "Błąd przesyłania: {{error}}"
}
}

Some files were not shown because too many files have changed in this diff Show more