├── .eslintrc ├── .github ├── FUNDING.yml └── workflows │ └── publish.yml ├── .gitignore ├── API.md ├── CHANGELOG.md ├── LICENCE ├── README.md ├── dist ├── maplibre-gl-compare.css └── maplibre-gl-compare.js ├── example ├── index.html └── index.js ├── index.js ├── package-lock.json ├── package.json ├── style.css └── test └── index.js /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "node": true, 4 | "browser": true 5 | }, 6 | "globals": { 7 | "maplibregl": true 8 | }, 9 | "rules": { 10 | "strict": [0], 11 | "curly": [0], 12 | "no-new": [0], 13 | "no-alert": [0], 14 | "no-extra-bind": [0], 15 | "no-redeclare": [1], 16 | "new-cap": [0], 17 | "eol-last": [0], 18 | "no-mixed-requires": [0], 19 | "no-path-concat": [0], 20 | "camelcase": [0], 21 | "consistent-return": [0], 22 | "no-underscore-dangle": [0], 23 | "comma-spacing": [0], 24 | "key-spacing": [0], 25 | "no-use-before-define": [0] 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | github: [maplibre] 2 | open_collective: maplibre 3 | -------------------------------------------------------------------------------- /.github/workflows/publish.yml: -------------------------------------------------------------------------------- 1 | name: publish 2 | 3 | on: 4 | workflow_dispatch: 5 | push: 6 | tags: 7 | - v* 8 | 9 | jobs: 10 | publish: 11 | name: Publish 12 | runs-on: ubuntu-latest 13 | steps: 14 | - uses: actions/checkout@v2 15 | 16 | - name: Use Node.js 10 x64 17 | uses: actions/setup-node@v2 18 | with: 19 | node-version: 10 20 | architecture: x64 21 | registry-url: 'https://registry.npmjs.org' 22 | 23 | - name: Publish 24 | run: | 25 | npm publish --access=public --non-interactive 26 | env: 27 | NODE_AUTH_TOKEN: ${{ secrets.NPM_ORG_TOKEN }} 28 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | example/style.css 3 | .idea/ 4 | -------------------------------------------------------------------------------- /API.md: -------------------------------------------------------------------------------- 1 | 2 | 3 | ### Table of Contents 4 | 5 | * [Compare][1] 6 | * [Parameters][2] 7 | * [Examples][3] 8 | * [setSlider][4] 9 | * [Parameters][5] 10 | * [on][6] 11 | * [Parameters][7] 12 | * [fire][8] 13 | * [Parameters][9] 14 | * [off][10] 15 | * [Parameters][11] 16 | 17 | ## Compare 18 | 19 | * **See**: [Swipe between maps][12] 20 | 21 | ### Parameters 22 | 23 | * `a` **[Object][13]** The first MapLibre GL Map 24 | * `b` **[Object][13]** The second MapLibre GL Map 25 | * `container` **([string][14] | [HTMLElement][15])** An HTML Element, or an element selector string for the compare container. It should be a wrapper around the two map Elements. 26 | * `options` **[Object][13]** 27 | 28 | * `options.orientation` **[string][14]** The orientation of the compare slider. `vertical` creates a vertical slider bar to compare one map on the left (map A) with another map on the right (map B). `horizontal` creates a horizontal slider bar to compare on mop on the top (map A) and another map on the bottom (map B). (optional, default `vertical`) 29 | * `options.mousemove` **[boolean][16]** If `true` the compare slider will move with the cursor, otherwise the slider will need to be dragged to move. (optional, default `false`) 30 | 31 | ### Examples 32 | 33 | ```javascript 34 | var compare = new maplibregl.Compare(beforeMap, afterMap, '#wrapper', { 35 | orientation: 'vertical', 36 | mousemove: true 37 | }); 38 | ``` 39 | 40 | ### setSlider 41 | 42 | Set the position of the slider. 43 | 44 | #### Parameters 45 | 46 | * `x` **[number][17]** Slider position in pixels from left/top. 47 | 48 | ### on 49 | 50 | Adds a listener for events of a specified type. 51 | 52 | #### Parameters 53 | 54 | * `type` **[string][14]** The event type to listen for; one of `slideend`. 55 | * `fn` 56 | * `listener` **[Function][18]** The function to be called when the event is fired. 57 | 58 | Returns **[Compare][19]** `this` 59 | 60 | ### fire 61 | 62 | Fire an event of a specified type. 63 | 64 | #### Parameters 65 | 66 | * `type` **[string][14]** The event type to fire; one of `slideend`. 67 | * `data` **[Object][13]** Data passed to the event listener. 68 | 69 | Returns **[Compare][19]** `this` 70 | 71 | ### off 72 | 73 | Removes an event listener previously added with `Compare#on`. 74 | 75 | #### Parameters 76 | 77 | * `type` **[string][14]** The event type previously used to install the listener. 78 | * `fn` 79 | * `listener` **[Function][18]** The function previously installed as a listener. 80 | 81 | Returns **[Compare][19]** `this` 82 | 83 | [1]: #compare 84 | 85 | [2]: #parameters 86 | 87 | [3]: #examples 88 | 89 | [4]: #setslider 90 | 91 | [5]: #parameters-1 92 | 93 | [6]: #on 94 | 95 | [7]: #parameters-2 96 | 97 | [8]: #fire 98 | 99 | [9]: #parameters-3 100 | 101 | [10]: #off 102 | 103 | [11]: #parameters-4 104 | 105 | [12]: https://maplibre.org/maplibre-gl-js-docs/plugins/ 106 | 107 | [13]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object 108 | 109 | [14]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String 110 | 111 | [15]: https://developer.mozilla.org/docs/Web/HTML/Element 112 | 113 | [16]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean 114 | 115 | [17]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Number 116 | 117 | [18]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Statements/function 118 | 119 | [19]: #compare 120 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Change Log 2 | All notable changes to this project will be documented in this file. 3 | 4 | The format is based on [Keep a Changelog](http://keepachangelog.com/) 5 | and this project adheres to [Semantic Versioning](http://semver.org/). 6 | 7 | ## [Unreleased] 8 | 9 | ### Added 10 | 11 | - Forked from https://github.com/mapbox/mapbox-gl-compare 12 | 13 | ### Changed 14 | 15 | - Changed the text mapboxgl to maplibregl 16 | 17 | ### Deprecated 18 | 19 | ### Removed 20 | 21 | ### Fixed 22 | 23 | ### Security 24 | -------------------------------------------------------------------------------- /LICENCE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2021 MapLibre contributors 2 | 3 | Copyright (c) 2016, Mapbox 4 | 5 | Permission to use, copy, modify, and/or distribute this software for any 6 | purpose with or without fee is hereby granted, provided that the above 7 | copyright notice and this permission notice appear in all copies. 8 | 9 | THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 10 | WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 11 | MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 12 | ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 13 | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 14 | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 15 | OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 16 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ## MapLibre GL Compare 2 | 3 | Swipe and sync between two MapLibre maps. This plugin was originally developed for Mapbox (https://github.com/mapbox/mapbox-gl-compare) and was migrated to MapLibre after Mapbox changed its license. 4 | 5 | ## Code example 6 | 7 | ### Examples 8 | 9 | #### Full Example without the need of a token 10 | 11 | ```js 12 | 13 | 14 | 15 | 16 | Swipe between maps 17 | 21 | 22 | 23 | 27 | 40 | 41 | 42 | 43 | 63 | 64 | 65 |
66 |
67 |
68 |
69 | 93 | 94 | 95 | ``` 96 | 97 | #### Full Example with the need of a token 98 | 99 | ```js 100 | 101 | 102 | 103 | 104 | Swipe between maps 105 | 109 | 110 | 111 | 115 | 128 | 129 | 130 | 131 | 151 | 152 | 153 |
154 |
155 |
156 |
157 | 182 | 183 | 184 | ``` 185 | 186 | ### Usage 187 | 188 | ```js 189 | var beforeMap = new maplibregl.Map({ 190 | container: "before", 191 | style: "https://demotiles.maplibre.org/style.json", 192 | center: [7.221275, 50.326111], 193 | zoom: 5, 194 | }); 195 | 196 | var afterMap = new maplibregl.Map({ 197 | container: "after", 198 | style: 199 | "https://vectortiles.geo.admin.ch/styles/ch.swisstopo.leichte-basiskarte.vt/style.json", 200 | center: [7.221275, 50.326111], 201 | zoom: 5, 202 | }); 203 | 204 | // A selector or reference to HTML element 205 | var container = '#comparison-container'; 206 | 207 | var map = new maplibregl.Compare(beforeMap, afterMap, container, { 208 | mousemove: true, // Optional. Set to true to enable swiping during cursor movement. 209 | orientation: 'vertical' // Optional. Sets the orientation of swiper to horizontal or vertical, defaults to vertical 210 | }); 211 | ``` 212 | 213 | ### Methods 214 | 215 | ```js 216 | compare = new maplibregl.Compare(beforeMap, afterMap, container, { 217 | mousemove: true, // Optional. Set to true to enable swiping during cursor movement. 218 | orientation: 'vertical' // Optional. Sets the orientation of swiper to horizontal or vertical, defaults to vertical 219 | }); 220 | 221 | // Get Current position - this will return the slider's current position, in pixels 222 | compare.currentPosition; 223 | 224 | // Set Position - this will set the slider at the specified (x) number of pixels from the left-edge or top-edge of viewport based on swiper orientation 225 | compare.setSlider(x); 226 | 227 | //Listen to slider movement - and return current position on each slideend 228 | compare.on('slideend', (e) => { 229 | console.log(e.currentPosition); 230 | }); 231 | 232 | //Remove - this will remove the compare control from the DOM and stop synchronizing the two maps. 233 | compare.remove(); 234 | ``` 235 | 236 | ## Live examples 237 | 238 | [Compare hybrid with streets](https://rawcdn.githack.com/astridx/astridx.github.io/a9d7297a4fe1e3a4d7ebeb1e4e662fd1339ef3b5/maplibreexamples/plugins/maplibre-gl-compare-swipe-between-maps.html) 239 | 240 | [Compare swisstopo with demotiles](https://rawcdn.githack.com/astridx/astridx.github.io/a9d7297a4fe1e3a4d7ebeb1e4e662fd1339ef3b5/maplibreexamples/plugins/maplibre-gl-compare-swipe-between-maps_.html) 241 | 242 | ## Installation 243 | 244 | ### Native 245 | 246 | Add tags referencing `maplibre-gl-compare` after adding `maplibre-gl` to your website: 247 | 248 | ```html 249 | 250 | 251 | 252 | 253 | 254 | 255 | ``` 256 | 257 | ### Node 258 | 259 | 260 | 261 | ## Motivation 262 | 263 | This project makes it possible to easily compare two maps. 264 | 265 | ## API Reference 266 | 267 | [API Reference](API.md) 268 | 269 | ## Bug Reports & Feature Requests 270 | 271 | Please use the [issue tracker](https://github.com/maplibre/maplibre-gl-compare/issues) to report any bugs or file feature requests. 272 | 273 | ## Licence 274 | ISC © [MapLibre](https://github.com/maplibre) © [Mapbox](https://github.com/mapbox) 275 | -------------------------------------------------------------------------------- /dist/maplibre-gl-compare.css: -------------------------------------------------------------------------------- 1 | .maplibregl-compare { 2 | background-color: #fff; 3 | position: absolute; 4 | width: 2px; 5 | height: 100%; 6 | z-index: 1; 7 | } 8 | .maplibregl-compare .compare-swiper-vertical { 9 | background-color: #3887be; 10 | box-shadow: inset 0 0 0 2px #fff; 11 | display: inline-block; 12 | border-radius: 50%; 13 | position: absolute; 14 | width: 60px; 15 | height: 60px; 16 | top: 50%; 17 | left: -30px; 18 | margin: -30px 1px 0; 19 | color: #fff; 20 | cursor: ew-resize; 21 | background-image: url(); 22 | } 23 | 24 | .maplibregl-compare-horizontal { 25 | position: relative; 26 | width: 100%; 27 | height: 2px; 28 | } 29 | .maplibregl-compare .compare-swiper-horizontal { 30 | background-color: #3887be; 31 | box-shadow: inset 0 0 0 2px #fff; 32 | display: inline-block; 33 | border-radius: 50%; 34 | position: absolute; 35 | width: 60px; 36 | height: 60px; 37 | top: 50%; 38 | left: 50%; 39 | margin: -30px 1px 0; 40 | color: #fff; 41 | cursor: ns-resize; 42 | background-image: url(); 43 | transform: rotate(90deg); 44 | } 45 | -------------------------------------------------------------------------------- /dist/maplibre-gl-compare.js: -------------------------------------------------------------------------------- 1 | !function i(o,r,s){function h(t,e){if(!r[t]){if(!o[t]){var n="function"==typeof require&&require;if(!e&&n)return n(t,!0);if(u)return u(t,!0);throw(e=new Error("Cannot find module '"+t+"'")).code="MODULE_NOT_FOUND",e}n=r[t]={exports:{}},o[t][0].call(n.exports,function(e){return h(o[t][1][e]||e)},n,n.exports,i,o,r,s)}return r[t].exports}for(var u="function"==typeof require&&require,e=0;ethis._bounds.width?this._bounds.width:e},_getY:function(e){e=(e=e.touches?e.touches[0]:e).clientY-this._bounds.top;return e=(e=e<0?0:e)>this._bounds.height?this._bounds.height:e},setSlider:function(e){this._setPosition(e)},on:function(e,t){return this._ev.on(e,t),this},fire:function(e,t){return this._ev.emit(e,t),this},off:function(e,t){return this._ev.removeListener(e,t),this},remove:function(){this._clearSync(),this._mapB.off("resize",this._onResize);var e=this._mapA.getContainer(),e=(e&&(e.style.clip=null,e.removeEventListener("mousemove",this._onMove)),this._mapB.getContainer());e&&(e.style.clip=null,e.removeEventListener("mousemove",this._onMove)),this._swiper.removeEventListener("mousedown",this._onDown),this._swiper.removeEventListener("touchstart",this._onDown),this._controlContainer.remove()}},window.maplibregl?maplibregl.Compare=i:void 0!==t&&(t.exports=i)},{"@mapbox/mapbox-gl-sync-move":2,events:3}],2:[function(e,t,n){t.exports=function(){var e=arguments.length;if(1===e)t=arguments[0];else for(var t=[],n=0;no&&!r.warned&&(r.warned=!0,(i=new Error("Possible EventEmitter memory leak detected. "+r.length+" "+String(t)+" listeners added. Use emitter.setMaxListeners() to increase limit")).name="MaxListenersExceededWarning",i.emitter=e,i.type=t,i.count=r.length,n=i,console&&console.warn&&console.warn(n))),e}function h(e,t,n){e={fired:!1,wrapFn:void 0,target:e,type:t,listener:n},t=function(){if(!this.fired)return this.target.removeListener(this.type,this.wrapFn),this.fired=!0,0===arguments.length?this.listener.call(this.target):this.listener.apply(this.target,arguments)}.bind(e);return t.listener=n,e.wrapFn=t}function p(e,t,n){e=e._events;if(void 0===e)return[];e=e[t];{if(void 0===e)return[];if("function"==typeof e)return n?[e.listener||e]:[e];if(n){for(var i=e,o=new Array(i.length),r=0;r 2 | 3 | 4 | 5 | 6 | 10 | 11 | 15 | 16 | 41 | 42 | 43 |
44 |
45 |
46 |
47 | 48 |
49 |
50 | 51 | 52 | 53 | -------------------------------------------------------------------------------- /example/index.js: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | 3 | require("../"); 4 | 5 | // 'Before' style from https://github.com/lukasmartinelli/naturalearthtiles 6 | var before = new maplibregl.Map({ 7 | container: "before", 8 | style: "https://raw.githubusercontent.com/lukasmartinelli/naturalearthtiles/gh-pages/maps/natural_earth.vector.json", 9 | zoom: 2 10 | }); 11 | 12 | // 'After' style from https://github.com/maplibre/demotiles 13 | var after = new maplibregl.Map({ 14 | container: "after", 15 | style: "https://demotiles.maplibre.org/style.json", 16 | zoom: 2 17 | }); 18 | 19 | // Use either of these patterns to select a container for the compare widget 20 | var wrapperSelector = "#wrapper"; 21 | var wrapperElement = document.body.querySelectorAll("#wrapper")[0]; 22 | 23 | // available options 24 | var options = { 25 | mousemove: true, 26 | orientation: "horizontal", 27 | }; 28 | 29 | window.compare = new maplibregl.Compare( 30 | before, 31 | after, 32 | wrapperSelector 33 | // options 34 | ); 35 | 36 | var closeButton = document.getElementById("close-button"); 37 | 38 | closeButton.addEventListener("click", function (e) { 39 | after.getContainer().style.display = "none"; 40 | window.compare.remove(); 41 | after.remove(); 42 | }); 43 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | 3 | var syncMove = require("@mapbox/mapbox-gl-sync-move"); 4 | var EventEmitter = require("events").EventEmitter; 5 | 6 | /** 7 | * @param {Object} a The first MapLibre GL Map 8 | * @param {Object} b The second MapLibre GL Map 9 | * @param {string|HTMLElement} container An HTML Element, or an element selector string for the compare container. It should be a wrapper around the two map Elements. 10 | * @param {Object} options 11 | * @param {string} [options.orientation=vertical] The orientation of the compare slider. `vertical` creates a vertical slider bar to compare one map on the left (map A) with another map on the right (map B). `horizontal` creates a horizontal slider bar to compare on mop on the top (map A) and another map on the bottom (map B). 12 | * @param {boolean} [options.mousemove=false] If `true` the compare slider will move with the cursor, otherwise the slider will need to be dragged to move. 13 | * @example 14 | * var compare = new maplibregl.Compare(beforeMap, afterMap, '#wrapper', { 15 | * orientation: 'vertical', 16 | * mousemove: true 17 | * }); 18 | * @see [Swipe between maps](https://maplibre.org/maplibre-gl-js-docs/plugins/) 19 | */ 20 | function Compare(a, b, container, options) { 21 | this.options = options ? options : {}; 22 | this._mapA = a; 23 | this._mapB = b; 24 | this._horizontal = this.options.orientation === "horizontal"; 25 | this._onDown = this._onDown.bind(this); 26 | this._onMove = this._onMove.bind(this); 27 | this._onMouseUp = this._onMouseUp.bind(this); 28 | this._onTouchEnd = this._onTouchEnd.bind(this); 29 | this._ev = new EventEmitter(); 30 | this._swiper = document.createElement("div"); 31 | this._swiper.className = this._horizontal 32 | ? "compare-swiper-horizontal" 33 | : "compare-swiper-vertical"; 34 | 35 | this._controlContainer = document.createElement("div"); 36 | this._controlContainer.className = this._horizontal 37 | ? "maplibregl-compare maplibregl-compare-horizontal" 38 | : "maplibregl-compare"; 39 | this._controlContainer.className = this._controlContainer.className; 40 | this._controlContainer.appendChild(this._swiper); 41 | 42 | if (typeof container === "string" && document.body.querySelectorAll) { 43 | // get container with a selector 44 | var appendTarget = document.body.querySelectorAll(container)[0]; 45 | if (!appendTarget) { 46 | throw new Error("Cannot find element with specified container selector."); 47 | } 48 | appendTarget.appendChild(this._controlContainer); 49 | } else if (container instanceof Element && container.appendChild) { 50 | // get container directly 51 | container.appendChild(this._controlContainer); 52 | } else { 53 | throw new Error( 54 | "Invalid container specified. Must be CSS selector or HTML element." 55 | ); 56 | } 57 | 58 | this._bounds = b.getContainer().getBoundingClientRect(); 59 | var swiperPosition = 60 | (this._horizontal ? this._bounds.height : this._bounds.width) / 2; 61 | 62 | this._setPosition(swiperPosition); 63 | 64 | this._clearSync = syncMove(a, b); 65 | this._onResize = function () { 66 | this._bounds = b.getContainer().getBoundingClientRect(); 67 | if (this.currentPosition) this._setPosition(this.currentPosition); 68 | }.bind(this); 69 | 70 | b.on("resize", this._onResize); 71 | 72 | if (this.options && this.options.mousemove) { 73 | a.getContainer().addEventListener("mousemove", this._onMove); 74 | b.getContainer().addEventListener("mousemove", this._onMove); 75 | } 76 | 77 | this._swiper.addEventListener("mousedown", this._onDown); 78 | this._swiper.addEventListener("touchstart", this._onDown); 79 | } 80 | 81 | Compare.prototype = { 82 | _setPointerEvents: function (v) { 83 | this._controlContainer.style.pointerEvents = v; 84 | this._swiper.style.pointerEvents = v; 85 | }, 86 | 87 | _onDown: function (e) { 88 | if (e.touches) { 89 | document.addEventListener("touchmove", this._onMove); 90 | document.addEventListener("touchend", this._onTouchEnd); 91 | } else { 92 | document.addEventListener("mousemove", this._onMove); 93 | document.addEventListener("mouseup", this._onMouseUp); 94 | } 95 | }, 96 | 97 | _setPosition: function (x) { 98 | x = Math.min( 99 | x, 100 | this._horizontal ? this._bounds.height : this._bounds.width 101 | ); 102 | var pos = this._horizontal 103 | ? "translate(0, " + x + "px)" 104 | : "translate(" + x + "px, 0)"; 105 | this._controlContainer.style.transform = pos; 106 | this._controlContainer.style.WebkitTransform = pos; 107 | var clipA = this._horizontal 108 | ? "rect(0, 999em, " + x + "px, 0)" 109 | : "rect(0, " + x + "px, " + this._bounds.height + "px, 0)"; 110 | var clipB = this._horizontal 111 | ? "rect(" + x + "px, 999em, " + this._bounds.height + "px,0)" 112 | : "rect(0, 999em, " + this._bounds.height + "px," + x + "px)"; 113 | 114 | this._mapA.getContainer().style.clip = clipA; 115 | this._mapB.getContainer().style.clip = clipB; 116 | this.currentPosition = x; 117 | }, 118 | 119 | _onMove: function (e) { 120 | if (this.options && this.options.mousemove) { 121 | this._setPointerEvents(e.touches ? "auto" : "none"); 122 | } 123 | 124 | this._horizontal 125 | ? this._setPosition(this._getY(e)) 126 | : this._setPosition(this._getX(e)); 127 | }, 128 | 129 | _onMouseUp: function () { 130 | document.removeEventListener("mousemove", this._onMove); 131 | document.removeEventListener("mouseup", this._onMouseUp); 132 | this.fire("slideend", { currentPosition: this.currentPosition }); 133 | }, 134 | 135 | _onTouchEnd: function () { 136 | document.removeEventListener("touchmove", this._onMove); 137 | document.removeEventListener("touchend", this._onTouchEnd); 138 | this.fire("slideend", { currentPosition: this.currentPosition }); 139 | }, 140 | 141 | _getX: function (e) { 142 | e = e.touches ? e.touches[0] : e; 143 | var x = e.clientX - this._bounds.left; 144 | if (x < 0) x = 0; 145 | if (x > this._bounds.width) x = this._bounds.width; 146 | return x; 147 | }, 148 | 149 | _getY: function (e) { 150 | e = e.touches ? e.touches[0] : e; 151 | var y = e.clientY - this._bounds.top; 152 | if (y < 0) y = 0; 153 | if (y > this._bounds.height) y = this._bounds.height; 154 | return y; 155 | }, 156 | 157 | /** 158 | * Set the position of the slider. 159 | * 160 | * @param {number} x Slider position in pixels from left/top. 161 | */ 162 | setSlider: function (x) { 163 | this._setPosition(x); 164 | }, 165 | 166 | /** 167 | * Adds a listener for events of a specified type. 168 | * 169 | * @param {string} type The event type to listen for; one of `slideend`. 170 | * @param {Function} listener The function to be called when the event is fired. 171 | * @returns {Compare} `this` 172 | */ 173 | on: function (type, fn) { 174 | this._ev.on(type, fn); 175 | return this; 176 | }, 177 | 178 | /** 179 | * Fire an event of a specified type. 180 | * 181 | * @param {string} type The event type to fire; one of `slideend`. 182 | * @param {Object} data Data passed to the event listener. 183 | * @returns {Compare} `this` 184 | */ 185 | fire: function (type, data) { 186 | this._ev.emit(type, data); 187 | return this; 188 | }, 189 | 190 | /** 191 | * Removes an event listener previously added with `Compare#on`. 192 | * 193 | * @param {string} type The event type previously used to install the listener. 194 | * @param {Function} listener The function previously installed as a listener. 195 | * @returns {Compare} `this` 196 | */ 197 | off: function (type, fn) { 198 | this._ev.removeListener(type, fn); 199 | return this; 200 | }, 201 | 202 | remove: function () { 203 | this._clearSync(); 204 | this._mapB.off("resize", this._onResize); 205 | var aContainer = this._mapA.getContainer(); 206 | 207 | if (!!aContainer) { 208 | aContainer.style.clip = null; 209 | aContainer.removeEventListener("mousemove", this._onMove); 210 | } 211 | 212 | var bContainer = this._mapB.getContainer(); 213 | 214 | if (!!bContainer) { 215 | bContainer.style.clip = null; 216 | bContainer.removeEventListener("mousemove", this._onMove); 217 | } 218 | 219 | this._swiper.removeEventListener("mousedown", this._onDown); 220 | this._swiper.removeEventListener("touchstart", this._onDown); 221 | this._controlContainer.remove(); 222 | }, 223 | }; 224 | 225 | if (window.maplibregl) { 226 | maplibregl.Compare = Compare; 227 | } else if (typeof module !== "undefined") { 228 | module.exports = Compare; 229 | } 230 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@maplibre/maplibre-gl-compare", 3 | "version": "0.6.0-dev", 4 | "description": "Swipe and sync between two MapLibre maps", 5 | "funding": "https://github.com/maplibre/maplibre-gl-js?sponsor=1", 6 | "main": "index.js", 7 | "directories": { 8 | "example": "example" 9 | }, 10 | "scripts": { 11 | "start": "cp style.css example/style.css && budo example/index.js --serve example/bundle.js --dir example --live", 12 | "build": "cp style.css dist/maplibre-gl-compare.css && NODE_ENV=production && browserify index.js | uglifyjs -c -m > dist/maplibre-gl-compare.js", 13 | "test": "npm run lint && browserify -t envify test/index.js | smokestack -b firefox | tap-status", 14 | "lint": "eslint --no-eslintrc -c .eslintrc index.js example/index.js test/index.js", 15 | "docs": "documentation build index.js --format=md > API.md" 16 | }, 17 | "repository": { 18 | "type": "git", 19 | "url": "https://github.com/maplibre/maplibre-gl-compare.git" 20 | }, 21 | "keywords": [ 22 | "maplibre", 23 | "maplibregl", 24 | "maps", 25 | "gl", 26 | "ui" 27 | ], 28 | "license": "ISC", 29 | "devDependencies": { 30 | "browserify": "^17.0.0", 31 | "budo": "^11.6.4", 32 | "documentation": "^13.2.5", 33 | "envify": "^4.1.0", 34 | "eslint": "^8.9.0", 35 | "maplibre-gl": "^2.1.6", 36 | "smokestack": "^3.6.0", 37 | "tap-status": "^1.0.1", 38 | "tape": "^5.5.2", 39 | "uglify-js": "^3.15.1" 40 | }, 41 | "dependencies": { 42 | "@mapbox/mapbox-gl-sync-move": "^0.3.0" 43 | }, 44 | "peerDependencies": { 45 | "maplibre-gl": ">=2.1.6" 46 | } 47 | } 48 | -------------------------------------------------------------------------------- /style.css: -------------------------------------------------------------------------------- 1 | .maplibregl-compare { 2 | background-color: #fff; 3 | position: absolute; 4 | width: 2px; 5 | height: 100%; 6 | z-index: 1; 7 | } 8 | .maplibregl-compare .compare-swiper-vertical { 9 | background-color: #3887be; 10 | box-shadow: inset 0 0 0 2px #fff; 11 | display: inline-block; 12 | border-radius: 50%; 13 | position: absolute; 14 | width: 60px; 15 | height: 60px; 16 | top: 50%; 17 | left: -30px; 18 | margin: -30px 1px 0; 19 | color: #fff; 20 | cursor: ew-resize; 21 | background-image: url(); 22 | } 23 | 24 | .maplibregl-compare-horizontal { 25 | position: relative; 26 | width: 100%; 27 | height: 2px; 28 | } 29 | .maplibregl-compare .compare-swiper-horizontal { 30 | background-color: #3887be; 31 | box-shadow: inset 0 0 0 2px #fff; 32 | display: inline-block; 33 | border-radius: 50%; 34 | position: absolute; 35 | width: 60px; 36 | height: 60px; 37 | top: 50%; 38 | left: 50%; 39 | margin: -30px 1px 0; 40 | color: #fff; 41 | cursor: ns-resize; 42 | background-image: url(); 43 | transform: rotate(90deg); 44 | } 45 | -------------------------------------------------------------------------------- /test/index.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | var test = require('tape'); 4 | window.maplibregl = require('maplibre-gl'); 5 | require('../'); 6 | 7 | // Tests 8 | test('Compare', function (t) { 9 | var a = new maplibregl.Map({ 10 | container: document.createElement('div'), 11 | style: 'https://demotiles.maplibre.org/style.json', 12 | }); 13 | 14 | var b = new maplibregl.Map({ 15 | container: document.createElement('div'), 16 | style: 'https://demotiles.maplibre.org/style.json', 17 | }); 18 | 19 | // insert the container's into the document so compare.setSlider test works 20 | document.body.appendChild(a.getContainer()); 21 | document.body.appendChild(b.getContainer()); 22 | 23 | var container = document.createElement('div'); 24 | 25 | var compare = new maplibregl.Compare(a, b, container); 26 | 27 | t.ok(!!a.getContainer().style.clip, 'Map A is clipped'); 28 | t.ok(!!b.getContainer().style.clip, 'Map B is clipped'); 29 | 30 | b.jumpTo({ 31 | bearing: 20, 32 | center: { 33 | lat: 16, 34 | lng: -155, 35 | }, 36 | pitch: 20, 37 | zoom: 3, 38 | }); 39 | 40 | t.equals(a.getZoom(), 3, 'Zoom is synched'); 41 | t.equals(a.getPitch(), 20, 'Pitch is synched'); 42 | t.equals(a.getBearing(), 20, 'Bearing is synched'); 43 | t.equals(a.getCenter().lng, -155, 'Lng is synched'); 44 | t.equals(a.getCenter().lat, 16, 'Lat is synched'); 45 | 46 | compare.setSlider(20); 47 | 48 | t.equals(compare.currentPosition, 20, 'Slider has been moved'); 49 | 50 | compare.remove(); 51 | 52 | t.ok(!a.getContainer().style.clip, 'Map A is no longer clipped'); 53 | t.ok(!b.getContainer().style.clip, 'Map B is no longer clipped'); 54 | 55 | b.jumpTo({ 56 | bearing: 10, 57 | center: { 58 | lat: 26, 59 | lng: -105, 60 | }, 61 | pitch: 30, 62 | zoom: 5, 63 | }); 64 | 65 | t.equals(a.getZoom(), 3, 'Zoom is no longer synched'); 66 | t.equals(a.getPitch(), 20, 'Pitch is no longer synched'); 67 | t.equals(a.getBearing(), 20, 'Bearing is no longer synched'); 68 | t.equals(a.getCenter().lng, -155, 'Lng is no longer synched'); 69 | t.equals(a.getCenter().lat, 16, 'Lat is no longer synched'); 70 | 71 | t.end(); 72 | }); 73 | 74 | // close the smokestack window once tests are complete 75 | test('shutdown', function (t) { 76 | t.end(); 77 | setTimeout(function () { 78 | window.close(); 79 | }); 80 | }); 81 | --------------------------------------------------------------------------------