cesium-examples/map/libs/ol/4.6.5/ol-ext-2.0.6/ol-ext.js

15475 lines
530 KiB
JavaScript
Raw Normal View History

/**
* ol-ext - A set of cool extensions for OpenLayers (ol) in node modules structure
* @description ol3,openlayers,popup,menu,symbol,renderer,filter,canvas,interaction,split,statistic,charts,pie,LayerSwitcher,toolbar,animation
* @version v2.0.6
* @author Jean-Marc Viglino
* @see https://github.com/Viglino/ol-ext#,
* @license BSD-3-Clause
*/
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search Control.
* This is the base class for search controls. You can use it for simple custom search or as base to new class.
* @see ol.control.SearchFeature
* @see ol.control.SearchPhoton
*
* @constructor
* @extends {ol.control.Control}
* @fires select
* @fires change:input
* @param {Object=} options
* @param {string} options.className control class name
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {string | undefined} options.inputLabel label for the input, default none
* @param {string | undefined} options.noCollapse prevent collapsing on input blur, default false
* @param {number | undefined} options.typing a delay on each typing to start searching (ms) use -1 to prevent autocompletion, default 300.
* @param {integer | undefined} options.minLength minimum length to start searching, default 1
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
* @param {integer | undefined} options.maxHistory maximum number of items to display in history. Set -1 if you don't want history, default maxItems
* @param {function} options.getTitle a function that takes a feature and return the name to display in the index.
* @param {function} options.autocomplete a function that take a search string and callback function to send an array
*/
ol.control.Search = function(options) {
var self = this;
if (!options) options = {};
if (options.typing == undefined) options.typing = 300;
// Class name for history
this._classname = options.className || 'search';
var element = document.createElement("DIV");
var classNames = (options.className||"")+ " ol-search";
if (!options.target) {
classNames += " ol-unselectable ol-control ol-collapsed";
this.button = document.createElement("BUTTON");
this.button.setAttribute("type", "button");
this.button.setAttribute("title", options.label||"search");
this.button.addEventListener("click", function() {
element.classList.toggle("ol-collapsed");
if (!element.classList.contains("ol-collapsed")) {
element.querySelector("input.search").focus();
var listElements = element.querySelectorAll("li");
for (var i = 0; i < listElements.length; i++) {
listElements[i].classList.remove("select");
}
// Display history
if (!input.value) {
self.drawList_();
}
}
});
element.appendChild(this.button);
}
element.setAttribute('class', classNames);
// Input label
if (options.inputLabel) {
var label = document.createElement("LABEL");
label.innerText = options.inputLabel;
element.appendChild(label);
}
// Search input
var tout, cur="";
var input = this._input = document.createElement("INPUT");
input.setAttribute("type", "search");
input.setAttribute("class", "search");
input.setAttribute("placeholder", options.placeholder||"Search...");
input.addEventListener("change", function(e) {
self.dispatchEvent({ type:"change:input", input:e, value:input.value });
});
var doSearch = function(e) {
// console.log(e.type+" "+e.key)'
var li = element.querySelector("ul.autocomplete li.select");
var val = input.value;
// move up/down
if (e.key=='ArrowDown' || e.key=='ArrowUp' || e.key=='Down' || e.key=='Up') {
if (li) {
li.classList.remove("select");
li = (/Down/.test(e.key)) ? li.nextElementSibling : li.previousElementSibling;
if (li) li.classList.add("select");
}
else element.querySelector("ul.autocomplete li").classList.add("select");
}
// Clear input
else if (e.type=='input' && !val) {
self.drawList_();
}
// Select in the list
else if (li && (e.type=="search" || e.key =="Enter")) {
if (element.classList.contains("ol-control")) input.blur();
li.classList.remove("select");
cur = val;
self._handleSelect(self._list[li.getAttribute("data-search")]);
}
// Search / autocomplete
else if ( (e.type=="search" || e.key =='Enter')
|| (cur!=val && options.typing>=0)) {
// current search
cur = val;
if (cur) {
// prevent searching on each typing
if (tout) clearTimeout(tout);
tout = setTimeout(function() {
if (cur.length >= self.get("minLength")) {
var s = self.autocomplete (cur, function(auto) { self.drawList_(auto); });
if (s) self.drawList_(s);
}
else self.drawList_();
}, options.typing);
}
else self.drawList_();
}
// Clear list selection
else {
var li = element.querySelector("ul.autocomplete li");
if (li) li.classList.remove('select');
}
};
input.addEventListener("keyup", doSearch);
input.addEventListener("search", doSearch);
input.addEventListener("cut", doSearch);
input.addEventListener("paste", doSearch);
input.addEventListener("input", doSearch);
if (!options.noCollapse) {
input.addEventListener('blur', function() {
setTimeout(function(){ element.classList.add('ol-collapsed') }, 200);
});
input.addEventListener('focus', function() {
element.classList.remove('ol-collapsed');
});
}
element.appendChild(input);
// Autocomplete list
var ul = document.createElement('UL');
ul.classList.add('autocomplete');
element.appendChild(ul);
ol.control.Control.call(this, {
element: element,
target: options.target
});
if (typeof (options.getTitle)=='function') this.getTitle = options.getTitle;
if (typeof (options.autocomplete)=='function') this.autocomplete = options.autocomplete;
// Options
this.set('minLength', options.minLength || 1);
this.set('maxItems', options.maxItems || 10);
this.set('maxHistory', options.maxHistory || options.maxItems || 10);
// History
this.restoreHistory();
this.drawList_();
};
ol.inherits(ol.control.Search, ol.control.Control);
/** Returns the text to be displayed in the menu
* @param {any} f feature to be displayed
* @return {string} the text to be displayed in the index, default f.name
* @api
*/
ol.control.Search.prototype.getTitle = function (f) {
return f.name || "No title";
};
/** Force search to refresh
*/
ol.control.Search.prototype.search = function () {
var search = this.element.querySelector("input.search");
this._triggerCustomEvent('search', search);
};
/** Trigger custom event on elemebt
* @param {*} eventName
* @param {*} element
* @private
*/
ol.control.Search.prototype._triggerCustomEvent = function (eventName, element) {
var event;
if (window.CustomEvent) {
event = new CustomEvent(eventName);
} else {
event = document.createEvent("CustomEvent");
event.initCustomEvent(eventName, true, true, {});
}
element.dispatchEvent(event);
};
/** Set the input value in the form (for initialisation purpose)
* @param {string} value
* @param {boolean} search to start a search
* @api
*/
ol.control.Search.prototype.setInput = function (value, search) {
var input = this.element.querySelector("input.search");
input.value = value;
if (search) this._triggerCustomEvent("keyup", input);
};
/** A ligne has been clicked in the menu > dispatch event
* @param {any} f the feature, as passed in the autocomplete
* @api
*/
ol.control.Search.prototype.select = function (f) {
this.dispatchEvent({ type:"select", search:f });
};
/**
* Save history and select
* @param {*} f
* @private
*/
ol.control.Search.prototype._handleSelect = function (f) {
if (!f) return;
// Save input in history
var hist = this.get('history');
// Prevent error on stringify
try {
var fstr = JSON.stringify(f);
for (var i=hist.length-1; i>=0; i--) {
if (!hist[i] || JSON.stringify(hist[i]) === fstr) {
hist.splice(i,1);
}
}
} catch (e) {
for (var i=hist.length-1; i>=0; i--) {
if (hist[i] === f) {
hist.splice(i,1);
}
}
};
hist.unshift(f);
while (hist.length > (this.get('maxHistory')||10)) {
hist.pop();
}
this.saveHistory();
// Select feature
this.select(f);
//this.drawList_();
};
/** Save history (in the localstorage)
*/
ol.control.Search.prototype.saveHistory = function () {
if (this.get('maxHistory')>=0) {
try {
localStorage["ol@search-"+this._classname] = JSON.stringify(this.get('history'));
} catch (e) {};
} else {
localStorage.removeItem("ol@search-"+this._classname);
}
};
/** Restore history (from the localstorage)
*/
ol.control.Search.prototype.restoreHistory = function () {
try {
this.set('history', JSON.parse(localStorage["ol@search-"+this._classname]) );
} catch(e) {
this.set('history', []);
}
};
/**
* Remove previous history
*/
ol.control.Search.prototype.clearHistory = function () {
this.set('history', []);
this.saveHistory();
this.drawList_();
};
/**
* Get history table
*/
ol.control.Search.prototype.getHistory = function () {
return this.get('history');
};
/** Autocomplete function
* @param {string} s search string
* @param {function} cback a callback function that takes an array to display in the autocomplete field (for asynchronous search)
* @return {Array|false} an array of search solutions or false if the array is send with the cback argument (asnchronous)
* @api
*/
ol.control.Search.prototype.autocomplete = function (s, cback) {
cback ([]);
return false;
// or just return [];
};
/** Draw the list
* @param {Array} auto an array of search result
* @private
*/
ol.control.Search.prototype.drawList_ = function (auto) {
var self = this;
var ul = this.element.querySelector("ul.autocomplete");
ul.innerHTML = '';
this._list = [];
if (!auto) {
var input = this.element.querySelector("input.search");
var value = input.value;
if (!value) {
auto = this.get('history');
} else {
return;
}
ul.setAttribute('class', 'autocomplete history');
} else {
ul.setAttribute('class', 'autocomplete');
}
var max = Math.min (self.get("maxItems"),auto.length);
for (var i=0; i<max; i++) {
if (auto[i]) {
if (!i || !self.equalFeatures(auto[i], auto[i-1])) {
var li = document.createElement("LI");
li.setAttribute("data-search", i);
this._list.push(auto[i]);
li.addEventListener("click", function(e) {
self._handleSelect(self._list[e.currentTarget.getAttribute("data-search")]);
});
li.innerHTML = self.getTitle(auto[i]);
ul.appendChild(li);
}
}
}
if (max && this.get("copy")) {
var li = document.createElement("LI");
li.classList.add("copy");
li.innerHTML = this.get("copy");
ul.appendChild(li);
}
};
/** Test if 2 features are equal
* @param {any} f1
* @param {any} f2
* @return {boolean}
*/
ol.control.Search.prototype.equalFeatures = function (f1, f2) {
return false;
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* This is the base class for search controls that use a json service to search features.
* You can use it for simple custom search or as base to new class.
*
* @constructor
* @extends {ol.control.Search}
* @fires select
* @param {any} options extend ol.control.Search options
* @param {string} options.className control class name
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 1000.
* @param {integer | undefined} options.minLength minimum length to start searching, default 3
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
* @param {function | undefined} options.handleResponse Handle server response to pass the features array to the list
*
* @param {string|undefined} options.url Url of the search api
* @param {string | undefined} options.authentication: basic authentication for the search API as btoa("login:pwd")
*/
ol.control.SearchJSON = function(options)
{ options = options || {};
options.className = options.className || 'JSON';
delete options.autocomplete;
options.minLength = options.minLength || 3;
options.typing = options.typing || 800;
ol.control.Search.call(this, options);
// Handle Mix Content Warning
// If the current connection is an https connection all other connections must be https either
var url = options.url || "";
if (window.location.protocol === "https:") {
var parser = document.createElement('a');
parser.href = url;
parser.protocol = window.location.protocol;
url = parser.href;
}
this.set('url', url);
this._auth = options.authentication;
// Overwrite handleResponse
if (typeof(options.handleResponse)==='function') this.handleResponse = options.handleResponse;
};
ol.inherits(ol.control.SearchJSON, ol.control.Search);
/** Autocomplete function (ajax request to the server)
* @param {string} s search string
* @param {function} cback a callback function that takes an array of {name, feature} to display in the autocomplete field
*/
ol.control.SearchJSON.prototype.autocomplete = function (s, cback)
{ var data = this.requestData(s);
var url = encodeURI(this.get('url'));
var parameters = '';
for (var index in data) {
parameters += (parameters) ? '&' : '?';
if (data.hasOwnProperty(index)) parameters += index + '=' + data[index];
}
this.ajax(url + parameters,
function (resp) {
if (resp.status >= 200 && resp.status < 400) {
var data = JSON.parse(resp.response);
cback(this.handleResponse(data));
} else {
console.log(url + parameters, arguments);
}
}, function(){
console.log(url + parameters, arguments);
});
};
/** Send an ajax request (GET)
* @param {string} url
* @param {function} onsuccess callback
* @param {function} onerror callback
*/
ol.control.SearchJSON.prototype.ajax = function (url, onsuccess, onerror){
var self = this;
// Abort previous request
if (this._request) {
this._request.abort();
}
// New request
var ajax = this._request = new XMLHttpRequest();
ajax.open('GET', url, true);
if (this._auth) {
ajax.setRequestHeader("Authorization", "Basic " + this._auth);
}
this.element.classList.add('searching');
// Load complete
ajax.onload = function() {
self._request = null;
self.element.classList.remove('searching');
onsuccess.call(self, this);
};
// Oops, TODO do something ?
ajax.onerror = function() {
self._request = null;
self.element.classList.remove('searching');
if (onerror) onerror.call(self);
};
// GO!
ajax.send();
};
/**
* @param {string} s the search string
* @return {Object} request data (as key:value)
* @api
*/
ol.control.SearchJSON.prototype.requestData = function (s){
return { q: s };
};
/**
* Handle server response to pass the features array to the display list
* @param {any} response server response
* @return {Array<any>} an array of feature
* @api
*/
ol.control.SearchJSON.prototype.handleResponse = function (response) {
return response;
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search places using the photon API.
*
* @constructor
* @extends {ol.control.SearchJSON}
* @fires select
* @param {Object=} Control options.
* @param {string} options.className control class name
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 1000.
* @param {integer | undefined} options.minLength minimum length to start searching, default 3
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
* @param {function | undefined} options.handleResponse Handle server response to pass the features array to the list
*
* @param {string|undefined} options.url Url to photon api, default "http://photon.komoot.de/api/"
* @param {string|undefined} options.lang Force preferred language, default none
* @param {boolean} options.position Search, with priority to geo position, default false
* @param {function} options.getTitle a function that takes a feature and return the name to display in the index, default return street + name + contry
*/
ol.control.SearchPhoton = function(options)
{ options = options || {};
options.className = options.className || 'photon';
options.url = options.url || "http://photon.komoot.de/api/";
ol.control.SearchJSON.call(this, options);
this.set('lang', options.lang);
this.set('position', options.position);
this.set("copy","<a href='http://www.openstreetmap.org/copyright' target='new'>&copy; OpenStreetMap contributors</a>");
};
ol.inherits(ol.control.SearchPhoton, ol.control.SearchJSON);
/** Returns the text to be displayed in the menu
* @param {ol.Feature} f the feature
* @return {string} the text to be displayed in the index
* @api
*/
ol.control.SearchPhoton.prototype.getTitle = function (f)
{ var p = f.properties;
return (p.housenumber||"")
+ " "+(p.street || p.name || "")
+ "<i>"
+ " "+(p.postcode||"")
+ " "+(p.city||"")
+ " ("+p.country
+ ")</i>";
};
/**
* @param {string} s the search string
* @return {Object} request data (as key:value)
* @api
*/
ol.control.SearchPhoton.prototype.requestData = function (s)
{ var data =
{ q: s,
lang: this.get('lang'),
limit: this.get('maxItems')
}
// Handle position proirity
if (this.get('position'))
{ var view = this.getMap().getView();
var pt = new ol.geom.Point(view.getCenter());
pt = (pt.transform (view.getProjection(), "EPSG:4326")).getCoordinates();
data.lon = pt[0];
data.lat = pt[1];
}
return data;
};
/**
* Handle server response to pass the features array to the list
* @param {any} response server response
* @return {Array<any>} an array of feature
*/
ol.control.SearchPhoton.prototype.handleResponse = function (response, cback) {
return response.features;
};
/** Prevent same feature to be drawn twice: test equality
* @param {} f1 First feature to compare
* @param {} f2 Second feature to compare
* @return {boolean}
* @api
*/
ol.control.SearchPhoton.prototype.equalFeatures = function (f1, f2) {
return (this.getTitle(f1) === this.getTitle(f2)
&& f1.geometry.coordinates[0] === f2.geometry.coordinates[0]
&& f1.geometry.coordinates[1] === f2.geometry.coordinates[1]);
};
/** A ligne has been clicked in the menu > dispatch event
* @param {any} f the feature, as passed in the autocomplete
* @api
*/
ol.control.SearchPhoton.prototype.select = function (f)
{ var c = f.geometry.coordinates;
// Add coordinate to the event
try {
c = ol.proj.transform (f.geometry.coordinates, 'EPSG:4326', this.getMap().getView().getProjection());
} catch(e) {};
this.dispatchEvent({ type:"select", search:f, coordinate: c });
};
/** */
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search places using the French National Base Address (BAN) API.
*
* @constructor
* @extends {ol.control.SearchJSON}
* @fires select
* @param {any} options extend ol.control.SearchJSON options
* @param {string} options.className control class name
* @param {boolean | undefined} options.apiKey the service api key.
* @param {string | undefined} options.authentication: basic authentication for the service API as btoa("login:pwd")
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 500.
* @param {integer | undefined} options.minLength minimum length to start searching, default 3
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
*
* @param {StreetAddress|PositionOfInterest|CadastralParcel|Commune} options.type type of search. Using Commune will return the INSEE code, default StreetAddress,PositionOfInterest
* @see {@link https://geoservices.ign.fr/documentation/geoservices/geocodage.html}
*/
ol.control.SearchGeoportail = function(options) {
options = options || {};
options.className = options.className || 'IGNF';
options.typing = options.typing || 500;
options.url = "http://wxs.ign.fr/"+options.apiKey+"/ols/apis/completion";
ol.control.SearchJSON.call(this, options);
this.set("copy","<a href='https://www.geoportail.gouv.fr/' target='new'>&copy; IGN-Géoportail</a>");
this.set('type', options.type || 'StreetAddress,PositionOfInterest');
};
ol.inherits(ol.control.SearchGeoportail, ol.control.SearchJSON);
/** Returns the text to be displayed in the menu
* @param {ol.Feature} f the feature
* @return {string} the text to be displayed in the index
* @api
*/
ol.control.SearchGeoportail.prototype.getTitle = function (f) {
var title = f.fulltext;
return (title);
};
/**
* @param {string} s the search string
* @return {Object} request data (as key:value)
* @api
*/
ol.control.SearchGeoportail.prototype.requestData = function (s) {
return {
text: s,
type: this.get('type')==='Commune' ? 'PositionOfInterest' : this.get('type') || 'StreetAddress,PositionOfInterest',
maximumResponses: this.get('maxItems')
};
};
/**
* Handle server response to pass the features array to the display list
* @param {any} response server response
* @return {Array<any>} an array of feature
* @api
*/
ol.control.SearchGeoportail.prototype.handleResponse = function (response) {
var response = response.results;
if (this.get('type') === 'Commune') {
for (var i=response.length-1; i>=0; i--) {
if ( response[i].kind
&& (response[i].classification>5 || response[i].kind=="Département") ) {
response.splice(i,1);
}
}
}
return response;
};
/** A ligne has been clicked in the menu > dispatch event
* @param {any} f the feature, as passed in the autocomplete
* @api
*/
ol.control.SearchGeoportail.prototype.select = function (f){
if (f.x || f.y) {
var c = [Number(f.x), Number(f.y)];
// Add coordinate to the event
try {
c = ol.proj.transform (c, 'EPSG:4326', this.getMap().getView().getProjection());
} catch(e) {};
// Get insee commune ?
if (this.get('type')==='Commune') {
this.searchCommune(f, function () {
this.dispatchEvent({ type:"select", search:f, coordinate: c });
});
} else {
this.dispatchEvent({ type:"select", search:f, coordinate: c });
}
} else {
this.searchCommune(f);
}
};
/** Search if no position and get the INSEE code
* @param {string} s le nom de la commune
*/
ol.control.SearchGeoportail.prototype.searchCommune = function (f, cback) {
var request = '<?xml version="1.0" encoding="UTF-8"?>'
+'<XLS xmlns:xls="http://www.opengis.net/xls" xmlns:gml="http://www.opengis.net/gml" xmlns="http://www.opengis.net/xls" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="1.2" xsi:schemaLocation="http://www.opengis.net/xls http://schemas.opengis.net/ols/1.2/olsAll.xsd">'
+'<RequestHeader/>'
+'<Request requestID="1" version="1.2" methodName="LocationUtilityService">'
+'<GeocodeRequest returnFreeForm="false">'
+'<Address countryCode="PositionOfInterest">'
+'<freeFormAddress>'+f.fulltext+'+</freeFormAddress>'
+'</Address>'
+'</GeocodeRequest>'
+'</Request>'
+'</XLS>'
var url = this.get('url').replace('ols/apis/completion','geoportail/ols')+"?xls="+encodeURIComponent(request);
this.ajax (url, function(resp) {
var xml = resp.response;
if (xml) {
xml = xml.replace(/\n|\r/g,'');
var p = (xml.replace(/.*<gml:pos\>(.*)<\/gml:pos>.*/, "$1")).split(' ');
f.x = Number(p[1]);
f.y = Number(p[0]);
f.kind = (xml.replace(/.*<Place type="Nature">([^<]*)<\/Place>.*/, "$1"));
f.insee = (xml.replace(/.*<Place type="INSEE">([^<]*)<\/Place>.*/, "$1"));
if (f.x || f.y) {
if (cback) cback.call(this, [f]);
else this._handleSelect(f);
}
}
});
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc OpenLayers 3 Layer Switcher Control.
* @require jQuery
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options
* @param {function} displayInLayerSwitcher function that takes a layer and return a boolean if the layer is displayed in the switcher, default test the displayInLayerSwitcher layer attribute
* @param {boolean} options.show_progress show a progress bar on tile layers, default false
* @param {boolean} mouseover show the panel on mouseover, default false
* @param {boolean} reordering allow layer reordering, default true
* @param {boolean} trash add a trash button to delete the layer, default false
* @param {function} oninfo callback on click on info button, if none no info button is shown
* @param {boolean} extent add an extent button to zoom to the extent of the layer
* @param {function} onextent callback when click on extent, default fits view to extent
*
* Layers attributes that control the switcher
* - allwaysOnTop {boolean} true to force layer stay on top of the others while reordering, default false
* - displayInLayerSwitcher {boolean} display in switcher, default true
* - noSwitcherDelete {boolean} to prevent layer deletion (w. trash option = true), default false
*/
ol.control.LayerSwitcher = function(options)
{ options = options || {};
var self = this;
this.dcount = 0;
this.show_progress = options.show_progress;
this.oninfo = (typeof (options.oninfo) == "function" ? options.oninfo: null);
this.onextent = (typeof (options.onextent) == "function" ? options.onextent: null);
this.hasextent = options.extent || options.onextent;
this.hastrash = options.trash;
this.reordering = (options.reordering!==false);
// displayInLayerSwitcher
if (typeof(options.displayInLayerSwitcher) === 'function') {
this.displayInLayerSwitcher = options.displayInLayerSwitcher;
}
var element;
if (options.target)
{ element = $("<div>").addClass(options.switcherClass || "ol-layerswitcher");
}
else
{ element = $("<div>").addClass((options.switcherClass || 'ol-layerswitcher') +' ol-unselectable ol-control ol-collapsed');
this.button = $("<button>")
.attr('type','button')
.on("touchstart", function(e)
{ element.toggleClass("ol-collapsed");
e.preventDefault();
self.overflow();
})
.click (function()
{ element.toggleClass("ol-forceopen").addClass("ol-collapsed");
self.overflow();
})
.appendTo(element);
if (options.mouseover)
{ $(element).mouseleave (function(){ element.addClass("ol-collapsed"); })
.mouseover(function(){ element.removeClass("ol-collapsed"); });
}
this.topv = $("<div>").addClass("ol-switchertopdiv")
.click(function(){ self.overflow("+50%"); })
.appendTo(element);
this.botv = $("<div>").addClass("ol-switcherbottomdiv")
.click(function(){ self.overflow("-50%"); })
.appendTo(element);
}
this.panel_ = $("<ul>").addClass("panel")
.appendTo(element);
this.panel_.on ('mousewheel DOMMouseScroll onmousewheel', function(e)
{ if (self.overflow(Math.max(-1, Math.min(1, (e.originalEvent.wheelDelta || -e.originalEvent.detail)))))
{ e.stopPropagation();
e.preventDefault();
}
});
this.header_ = $("<li>").addClass("ol-header").appendTo(this.panel_);
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
// Enable jQuery dataTransfert
// $.event.props.push('dataTransfer');
this.target = options.target;
};
ol.inherits(ol.control.LayerSwitcher, ol.control.Control);
/** List of tips for internationalization purposes
*/
ol.control.LayerSwitcher.prototype.tip =
{ up: "up/down",
down: "down",
info: "informations...",
extent: "zoom to extent",
trash: "remove layer",
plus: "expand/shrink"
};
/** Test if a layer should be displayed in the switcher
* @param {ol.layer} layer
* @return {boolean} true if the layer is displayed
*/
ol.control.LayerSwitcher.prototype.displayInLayerSwitcher = function(layer) {
return (layer.get("displayInLayerSwitcher")!==false);
};
/**
* Set the map instance the control is associated with.
* @param {_ol_Map_} map The map instance.
*/
ol.control.LayerSwitcher.prototype.setMap = function(map)
{ ol.control.Control.prototype.setMap.call(this, map);
this.drawPanel();
if (this._listener) {
if (this._listener) ol.Observable.unByKey(this._listener.change);
if (this._listener) ol.Observable.unByKey(this._listener.moveend);
if (this._listener) ol.Observable.unByKey(this._listener.size);
}
this._listener = null;
this.map_ = map;
// Get change (new layer added or removed)
if (map)
{ this._listener = {
change: map.getLayerGroup().on('change', this.drawPanel.bind(this)),
moveend: map.on('moveend', this.viewChange.bind(this)),
size: map.on('change:size', this.overflow.bind(this))
}
}
};
/** Add a custom header
*/
ol.control.LayerSwitcher.prototype.setHeader = function(html)
{ this.header_.html(html);
};
/** Calculate overflow and add scrolls
* @param {Number} dir scroll direction -1|0|1|'+50%'|'-50%'
*/
ol.control.LayerSwitcher.prototype.overflow = function(dir)
{
if (this.button)
{ // Nothing to show
if (this.panel_.css('display')=='none')
{ $(this.element).css("height", "auto");
return;
}
// Calculate offset
var h = $(this.element).outerHeight();
var hp = this.panel_.outerHeight();
var dh = this.button.position().top + this.button.outerHeight(true);
var top = this.panel_.position().top-dh;
if (hp > h-dh)
{ // Bug IE: need to have an height defined
$(this.element).css("height", "100%");
switch (dir)
{ case 1: top += 2*$("li.visible .li-content",this.panel_).height(); break;
case -1: top -= 2*$("li.visible .li-content",this.panel_).height(); break;
case "+50%": top += Math.round(h/2); break;
case "-50%": top -= Math.round(h/2); break;
default: break;
}
// Scroll div
if (top+hp <= h-3*dh/2)
{ top = h-3*dh/2-hp;
this.botv.hide();
}
else
{ this.botv.css("display","");//show();
}
if (top >= 0)
{ top = 0;
this.topv.hide();
}
else
{ this.topv.css("display","");
}
// Scroll ?
this.panel_.css('top', top+"px");
return true;
}
else
{ $(this.element).css("height", "auto");
this.panel_.css('top', "0px");
this.botv.hide();
this.topv.hide();
return false;
}
}
else return false;
};
/**
* On view change hide layer depending on resolution / extent
* @param {ol.event} map The map instance.
* @private
*/
ol.control.LayerSwitcher.prototype.viewChange = function(e)
{
var map = this.map_;
var res = this.map_.getView().getResolution();
$("li", this.panel_).each(function()
{ var l = $(this).data('layer');
if (l)
{ if (l.getMaxResolution()<=res || l.getMinResolution()>=res) $(this).addClass("ol-layer-hidden");
else
{ var ex0 = l.getExtent();
if (ex0)
{ var ex = map.getView().calculateExtent(map.getSize());
if (!ol.extent.intersects(ex, ex0))
{ $(this).addClass("ol-layer-hidden");
}
else $(this).removeClass("ol-layer-hidden");
}
else $(this).removeClass("ol-layer-hidden");
}
}
});
};
/**
* Draw the panel control (prevent multiple draw due to layers manipulation on the map with a delay function)
*/
ol.control.LayerSwitcher.prototype.drawPanel = function(e)
{
if (!this.getMap()) return;
var self = this;
// Multiple event simultaneously / draw once => put drawing in the event queue
this.dcount++;
setTimeout (function(){ self.drawPanel_(); }, 0);
};
/** Delayed draw panel control
* @private
*/
ol.control.LayerSwitcher.prototype.drawPanel_ = function(e)
{ if (--this.dcount || this.dragging_) return;
$("li", this.panel_).not(".ol-header").remove();
this.drawList (this.panel_, this.getMap().getLayers());
};
/** Change layer visibility according to the baselayer option
* @param {ol.layer}
* @param {Array<ol.layer>} related layers
*/
ol.control.LayerSwitcher.prototype.switchLayerVisibility = function(l, layers)
{
if (!l.get('baseLayer')) l.setVisible(!l.getVisible());
else
{ if (!l.getVisible()) l.setVisible(true);
layers.forEach(function(li)
{ if (l!==li && li.get('baseLayer') && li.getVisible()) li.setVisible(false);
});
}
};
/** Check if layer is on the map (depending on zoom and extent)
* @param {ol.layer}
* @return {boolean}
*/
ol.control.LayerSwitcher.prototype.testLayerVisibility = function(layer)
{
if (this.map_)
{ var res = this.map_.getView().getResolution();
if (layer.getMaxResolution()<=res || layer.getMinResolution()>=res) return false;
else
{ var ex0 = layer.getExtent();
if (ex0)
{ var ex = this.map_.getView().calculateExtent(this.map_.getSize());
return ol.extent.intersects(ex, ex0);
}
return true;
}
}
return true;
};
/** Start ordering the list
* @param {event} e drag event
* @private
*/
ol.control.LayerSwitcher.prototype.dragOrdering_ = function(e)
{ var drag = e.data;
switch (e.type)
{ // Start ordering
case 'mousedown':
case 'touchstart':
{ e.stopPropagation();
e.preventDefault();
var pageY = e.pageY
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageY)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageY);
drag =
{ self: drag.self,
elt: $(e.currentTarget).closest("li"),
start: true,
element: drag.self.element,
panel: drag.self.panel_,
pageY: pageY
};
drag.elt.parent().addClass('drag');
$(document).on("mouseup mousemove touchend touchcancel touchmove", drag, drag.self.dragOrdering_);
break;
}
// Stop ordering
case 'touchcancel':
case 'touchend':
case 'mouseup':
{ if (drag.target)
{ // Get drag on parent
var drop = drag.layer;
var target = drag.target;
if (drop && target)
{ var collection ;
if (drag.group) collection = drag.group.getLayers();
else collection = drag.self.getMap().getLayers();
var layers = collection.getArray();
// Switch layers
for (var i=0; i<layers.length; i++)
{ if (layers[i]==drop)
{ collection.removeAt (i);
break;
}
}
for (var j=0; j<layers.length; j++)
{ if (layers[j]==target)
{ if (i>j) collection.insertAt (j,drop);
else collection.insertAt (j+1,drop);
break;
}
}
}
}
$("li",drag.elt.parent()).removeClass("dropover dropover-after dropover-before");
drag.elt.removeClass("drag");
drag.elt.parent().removeClass("drag");
$(drag.element).removeClass('drag');
if (drag.div) drag.div.remove();
$(document).off("mouseup mousemove touchend touchcancel touchmove", drag.self.dragOrdering_);
break;
}
// Ordering
case 'mousemove':
case 'touchmove':
{ // First drag (more than 2 px) => show drag element (ghost)
var pageY = e.pageY
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageY)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageY);
if (drag.start && Math.abs(drag.pageY - pageY) > 2)
{ drag.start = false;
drag.elt.addClass("drag");
drag.layer = drag.elt.data('layer');
drag.target = false;
drag.group = drag.elt.parent().parent().data('layer');
// Ghost div
drag.div = $("<li>").appendTo(drag.panel);
drag.div.css ({ position: "absolute", "z-index":10000, left:drag.elt.position().left, opacity:0.5 })
.html($(drag.elt).html())
.addClass("ol-dragover")
.width(drag.elt.outerWidth())
.height(drag.elt.height());
$(drag.element).addClass('drag');
}
if (!drag.start)
{ e.preventDefault();
e.stopPropagation();
// Ghost div
drag.div.css ({ top:pageY - drag.panel.offset().top + drag.panel.scrollTop() +5 });
var li;
if (!e.originalEvent.touches) li = $(e.target);
else li = $(document.elementFromPoint(e.originalEvent.touches[0].clientX, e.originalEvent.touches[0].clientY));
if (li.hasClass("ol-switcherbottomdiv"))
{ drag.self.overflow(-1);
console.log('bottom')
}
else if (li.hasClass("ol-switchertopdiv"))
{ drag.self.overflow(1);
}
if (!li.is("li")) li = li.closest("li");
if (!li.hasClass('dropover')) $("li", drag.elt.parent()).removeClass("dropover dropover-after dropover-before");
if (li.parent().hasClass('drag') && li.get(0) !== drag.elt.get(0))
{ var target = li.data("layer");
// Don't mix layer level
if (target && !target.get("allwaysOnTop") == !drag.layer.get("allwaysOnTop"))
{ li.addClass("dropover");
li.addClass((drag.elt.position().top < li.position().top)?"dropover-after":"dropover-before");
drag.target = target;
}
else
{ drag.target = false;
}
drag.div.show();
}
else
{ drag.target = false;
if (li.get(0) === drag.elt.get(0)) drag.div.hide();
else drag.div.show();
}
if (!drag.target) drag.div.addClass("forbidden");
else drag.div.removeClass("forbidden");
}
break;
}
default: break;
}
};
/** Change opacity on drag
* @param {event} e drag event
* @private
*/
ol.control.LayerSwitcher.prototype.dragOpacity_ = function(e)
{ var drag = e.data;
switch (e.type)
{ // Start opacity
case 'mousedown':
case 'touchstart':
{ e.stopPropagation();
e.preventDefault();
drag.start = e.pageX
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageX)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageX);
drag.elt = $(e.target);
drag.layer = drag.elt.closest("li").data('layer')
drag.self.dragging_ = true;
$(document).on("mouseup touchend mousemove touchmove touchcancel", drag, drag.self.dragOpacity_);
break;
}
// Stop opacity
case 'touchcancel':
case 'touchend':
case 'mouseup':
{ $(document).off("mouseup touchend mousemove touchmove touchcancel", drag.self.dragOpacity_);
drag.layer.setOpacity(drag.opacity);
drag.elt.parent().next().text(Math.round(drag.opacity*100));
drag.self.dragging_ = false;
drag = false;
break;
}
// Move opcaity
default:
{ var x = e.pageX
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageX)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageX);
var dx = Math.max ( 0, Math.min( 1, (x - drag.elt.parent().offset().left) / drag.elt.parent().width() ));
drag.elt.css("left", (dx*100)+"%");
drag.elt.parent().next().text(Math.round(drag.opacity*100));
drag.opacity = dx;
drag.layer.setOpacity(dx);
break;
}
}
}
/** Render a list of layer
* @param {elt} element to render
* @layers {Array{ol.layer}} list of layer to show
* @api stable
*/
ol.control.LayerSwitcher.prototype.drawList = function(ul, collection)
{ var self = this;
var layers = collection.getArray();
var setVisibility = function(e)
{ e.stopPropagation();
e.preventDefault();
var l = $(this).parent().parent().data("layer");
self.switchLayerVisibility(l,collection);
};
function moveLayer (l, layers, inc)
{
for (var i=0; i<layers.getLength(); i++)
{ if (layers.item(i) === l)
{ layers.remove(l);
layers.insertAt(i+inc, l);
return true;
}
if (layers.item(i).getLayers && moveLayer (l, layers.item(i).getLayers(), inc)) return true;
}
return false;
};
function moveLayerUp(e)
{ e.stopPropagation();
e.preventDefault();
moveLayer($(this).closest('li').data("layer"), self.map_.getLayers(), +1);
};
function moveLayerDown(e)
{ e.stopPropagation();
e.preventDefault();
moveLayer($(this).closest('li').data("layer"), self.map_.getLayers(), -1);
};
function onInfo(e)
{ e.stopPropagation();
e.preventDefault();
self.oninfo($(this).closest('li').data("layer"));
};
function zoomExtent(e)
{ e.stopPropagation();
e.preventDefault();
if (self.onextent) self.onextent($(this).closest('li').data("layer"));
else self.map_.getView().fit ($(this).closest('li').data("layer").getExtent(), self.map_.getSize());
};
function removeLayer(e)
{ e.stopPropagation();
e.preventDefault();
var li = $(this).closest("ul").parent();
if (li.data("layer"))
{ li.data("layer").getLayers().remove($(this).closest('li').data("layer"));
if (li.data("layer").getLayers().getLength()==0 && !li.data("layer").get('noSwitcherDelete'))
{ removeLayer.call($(".layerTrash", li), e);
}
}
else self.map_.removeLayer($(this).closest('li').data("layer"));
};
// Add the layer list
for (var i=layers.length-1; i>=0; i--)
{ var layer = layers[i];
if (!self.displayInLayerSwitcher(layer)) continue;
var li = $("<li>").addClass((layer.getVisible()?"visible ":" ")+(layer.get('baseLayer')?"baselayer":""))
.data("layer",layer).appendTo(ul);
var layer_buttons = $("<div>").addClass("ol-layerswitcher-buttons").appendTo(li);
var d = $("<div>").addClass('li-content').appendTo(li);
if (!this.testLayerVisibility(layer)) d.addClass("ol-layer-hidden");
// Visibility
$("<input>")
.attr('type', layer.get('baseLayer') ? 'radio' : 'checkbox')
.attr("checked",layer.getVisible())
.on ('click', setVisibility)
.appendTo(d);
// Label
$("<label>").text(layer.get("title") || layer.get("name"))
.attr('title', layer.get("title") || layer.get("name"))
.on ('click', setVisibility)
.attr('unselectable', 'on')
.css('user-select', 'none')
.on('selectstart', false)
.appendTo(d);
// up/down
if (this.reordering)
{ if ( (i<layers.length-1 && (layer.get("allwaysOnTop") || !layers[i+1].get("allwaysOnTop")) )
|| (i>0 && (!layer.get("allwaysOnTop") || layers[i-1].get("allwaysOnTop")) ) )
{ $("<div>").addClass("layerup")
.on ("mousedown touchstart", {self:this}, this.dragOrdering_ )
.attr("title", this.tip.up)
.appendTo(layer_buttons);
}
}
// Show/hide sub layers
if (layer.getLayers)
{ var nb = 0;
layer.getLayers().forEach(function(l)
{ if (self.displayInLayerSwitcher(l)) nb++;
});
if (nb)
{ $("<div>").addClass(layer.get("openInLayerSwitcher") ? "collapse-layers" : "expend-layers" )
.click(function()
{ var l = $(this).closest('li').data("layer");
l.set("openInLayerSwitcher", !l.get("openInLayerSwitcher") )
})
.attr("title", this.tip.plus)
.appendTo(layer_buttons);
}
}
// $("<div>").addClass("ol-separator").appendTo(layer_buttons);
// Info button
if (this.oninfo)
{ $("<div>").addClass("layerInfo")
.on ('click', onInfo)
.attr("title", this.tip.info)
.appendTo(layer_buttons);
}
// Layer remove
if (this.hastrash && !layer.get("noSwitcherDelete"))
{ $("<div>").addClass("layerTrash")
.on ('click', removeLayer)
.attr("title", this.tip.trash)
.appendTo(layer_buttons);
}
// Layer extent
if (this.hasextent && layers[i].getExtent())
{ var ex = layers[i].getExtent();
if (ex.length==4 && ex[0]<ex[2] && ex[1]<ex[3])
{ $("<div>").addClass("layerExtent")
.on ('click', zoomExtent)
.attr("title", this.tip.extent)
.appendTo(layer_buttons);
}
}
// Progress
if (this.show_progress && layer instanceof ol.layer.Tile)
{ var p = $("<div>")
.addClass("layerswitcher-progress")
.appendTo(d);
this.setprogress_(layer);
layer.layerswitcher_progress = $("<div>").appendTo(p);
}
// Opacity
var opacity = $("<div>").addClass("layerswitcher-opacity")
.on("click", function(e)
{ e.stopPropagation();
e.preventDefault();
var x = e.pageX
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageX)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageX);
var dx = Math.max ( 0, Math.min( 1, (x - $(this).offset().left) / $(this).width() ));
$(this).closest("li").data('layer').setOpacity(dx);
})
.appendTo(d);
$("<div>").addClass("layerswitcher-opacity-cursor")
.on("mousedown touchstart", { self: this }, self.dragOpacity_ )
.css ('left', (layer.getOpacity()*100)+"%")
.appendTo(opacity);
// Percent
$("<div>").addClass("layerswitcher-opacity-label")
.text(Math.round(layer.getOpacity()*100))
.appendTo(d);
// Layer group
if (layer.getLayers)
{ li.addClass('ol-layer-group');
if (layer.get("openInLayerSwitcher")===true)
{ this.drawList ($("<ul>").appendTo(li), layer.getLayers());
}
}
else if (layer instanceof ol.layer.Vector) li.addClass('ol-layer-vector');
else if (layer instanceof ol.layer.VectorTile) li.addClass('ol-layer-vector');
else if (layer instanceof ol.layer.Tile) li.addClass('ol-layer-tile');
else if (layer instanceof ol.layer.Image) li.addClass('ol-layer-image');
else if (layer instanceof ol.layer.Heatmap) li.addClass('ol-layer-heatmap');
}
if (ul==this.panel_) this.overflow();
};
/** Handle progress bar for a layer
* @private
*/
ol.control.LayerSwitcher.prototype.setprogress_ = function(layer)
{
if (!layer.layerswitcher_progress)
{ var loaded = 0;
var loading = 0;
function draw()
{ if (loading === loaded)
{ loading = loaded = 0;
layer.layerswitcher_progress.width(0);
}
else
{ layer.layerswitcher_progress.css('width', (loaded / loading * 100).toFixed(1) + '%');
}
}
layer.getSource().on('tileloadstart', function()
{ loading++;
draw();
});
layer.getSource().on('tileloadend', function()
{ loaded++;
draw();
});
layer.getSource().on('tileloaderror', function()
{ loaded++;
draw();
});
}
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Control bar for OL3
* The control bar is a container for other controls. It can be used to create toolbars.
* Control bars can be nested and combined with ol.control.Toggle to handle activate/deactivate.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {String} options.className class of the control
* @param {bool} options.group is a group, default false
* @param {bool} options.toggleOne only one toggle control is active at a time, default false
* @param {bool} options.autoDeactivate used with subbar to deactivate all control when top level control deactivate, default false
* @param {Array<_ol_control_>} options.controls a list of control to add to the bar
*/
ol.control.Bar = function(options)
{ if (!options) options={};
var element = $("<div>").addClass('ol-unselectable ol-control ol-bar');
if (options.className) element.addClass(options.className);
if (options.group) element.addClass('ol-group');
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
this.set('toggleOne', options.toggleOne);
this.set('autoDeactivate', options.autoDeactivate);
this.controls_ = [];
if (options.controls instanceof Array)
{ for (var i=0; i<options.controls.length; i++)
{ this.addControl(options.controls[i]);
}
}
};
ol.inherits(ol.control.Bar, ol.control.Control);
/** Set the control visibility
* @param {boolean} b
*/
ol.control.Bar.prototype.setVisible = function (val) {
if (val) $(this.element).show();
else $(this.element).hide();
}
/** Get the control visibility
* @return {boolean} b
*/
ol.control.Bar.prototype.getVisible = function ()
{ return ($(this.element).css('display') != 'none');
}
/**
* Set the map instance the control is associated with
* and add its controls associated to this map.
* @param {_ol_Map_} map The map instance.
*/
ol.control.Bar.prototype.setMap = function (map)
{ ol.control.Control.prototype.setMap.call(this, map);
for (var i=0; i<this.controls_.length; i++)
{ var c = this.controls_[i];
// map.addControl(c);
c.setMap(map);
}
};
/** Get controls in the panel
* @param {Array<_ol_control_>}
*/
ol.control.Bar.prototype.getControls = function ()
{ return this.controls_;
};
/** Set tool bar position
* @param {top|left|bottom|right} pos
*/
ol.control.Bar.prototype.setPosition = function (pos)
{ $(this.element).removeClass('ol-left ol-top ol-bottom ol-right');
pos=pos.split ('-');
for (var i=0; i<pos.length; i++)
{ switch (pos[i])
{ case 'top':
case 'left':
case 'bottom':
case 'right':
$(this.element).addClass ("ol-"+pos[i]);
break;
default: break;
}
}
};
/** Add a control to the bar
* @param {_ol_control_} c control to add
*/
ol.control.Bar.prototype.addControl = function (c)
{ this.controls_.push(c);
c.setTarget(this.element);
if (this.getMap())
{ this.getMap().addControl(c);
}
// Activate and toogleOne
c.on ('change:active', this.onActivateControl_.bind(this));
if (c.getActive && c.getActive())
{ c.dispatchEvent({ type:'change:active', key:'active', oldValue:false, active:true });
}
};
/** Deativate all controls in a bar
* @param {_ol_control_} except a control
*/
ol.control.Bar.prototype.deactivateControls = function (except)
{ for (var i=0; i<this.controls_.length; i++)
{ if (this.controls_[i] !== except && this.controls_[i].setActive)
{ this.controls_[i].setActive(false);
}
}
};
ol.control.Bar.prototype.getActiveControls = function ()
{ var active = [];
for (var i=0, c; c=this.controls_[i]; i++)
{ if (c.getActive && c.getActive()) active.push(c);
}
return active;
}
/** Auto activate/deactivate controls in the bar
* @param {boolean} b activate/deactivate
*/
ol.control.Bar.prototype.setActive = function (b)
{ if (!b && this.get("autoDeactivate"))
{ this.deactivateControls();
}
if (b)
{ var ctrls = this.getControls();
for (var i=0, sb; (sb = ctrls[i]); i++)
{ if (sb.get("autoActivate")) sb.setActive(true);
}
}
}
/** Post-process an activated/deactivated control
* @param {ol.event} e :an object with a target {_ol_control_} and active flag {bool}
*/
ol.control.Bar.prototype.onActivateControl_ = function (e) {
if (this.get('toggleOne'))
{ if (e.active)
{ var n;
var ctrl = e.target;
for (n=0; n<this.controls_.length; n++)
{ if (this.controls_[n]===ctrl) break;
}
// Not here!
if (n==this.controls_.length) return;
this.deactivateControls (this.controls_[n]);
}
else
{ // No one active > test auto activate
if (!this.getActiveControls().length)
{ for (var i=0, c; c=this.controls_[i]; i++)
{ if (c.get("autoActivate"))
{ c.setActive();
break;
}
}
}
}
}
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** A simple push button control
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {String} options.className class of the control
* @param {String} options.title title of the control
* @param {String} options.html html to insert in the control
* @param {function} options.handleClick callback when control is clicked (or use change:active event)
*/
ol.control.Button = function(options)
{ options = options || {};
var element = $("<div>").addClass((options.className||"") + ' ol-button ol-unselectable ol-control');
var self = this;
var bt = $("<button>").html(options.html || "")
.attr('type','button')
.attr('title', options.title)
.on("touchstart click", function(e)
{ if (e && e.preventDefault)
{ e.preventDefault();
e.stopPropagation();
}
if (options.handleClick) options.handleClick.call(self, e);
})
.appendTo(element);
// Try to get a title in the button content
if (!options.title) bt.attr("title", bt.children().first().attr('title'));
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
if (options.title) this.set("title", options.title);
};
ol.inherits(ol.control.Button, ol.control.Control);
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc
* OpenLayers 3 Attribution Control integrated in the canvas (for jpeg/png
* @see http://www.kreidefossilien.de/webgis/dokumentation/beispiele/export-map-to-png-with-scale
*
* @constructor
* @extends {ol.control.Attribution}
* @param {Object=} options extend the ol.control.Attribution options.
* @param {ol.style.Style} options.style option is usesd to draw the text.
*/
ol.control.CanvasAttribution = function(options)
{ if (!options) options = {};
ol.control.Attribution.call(this, options);
// Draw in canvas
this.isCanvas_ = !!options.canvas;
// Get style options
if (!options) options={};
if (!options.style) options.style = new ol.style.Style();
this.setStyle (options.style);
}
ol.inherits(ol.control.CanvasAttribution, ol.control.Attribution);
/**
* Draw attribution on canvas
* @param {boolean} b draw the attribution on canvas.
*/
ol.control.CanvasAttribution.prototype.setCanvas = function (b)
{ this.isCanvas_ = b;
$(this.element).css("visibility", b ? "hidden":"visible");
if (this.map_) this.map_.renderSync();
};
/**
* Change the control style
* @param {ol.style.Style} style
*/
ol.control.CanvasAttribution.prototype.setStyle = function (style)
{ var text = style.getText();
this.font_ = text ? text.getFont() : "10px Arial";
var stroke = text ? text.getStroke() : null;
var fill = text ? text.getFill() : null;
this.fontStrokeStyle_ = stroke ? ol.color.asString(stroke.getColor()) : "#fff";
this.fontFillStyle_ = fill ? ol.color.asString(fill.getColor()) : "#000";
this.fontStrokeWidth_ = stroke ? stroke.getWidth() : 3;
if (this.getMap()) this.getMap().render();
};
/**
* Remove the control from its current map and attach it to the new map.
* Subclasses may set up event handlers to get notified about changes to
* the map here.
* @param {ol.Map} map Map.
* @api stable
*/
ol.control.CanvasAttribution.prototype.setMap = function (map)
{ var oldmap = this.getMap();
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.ScaleLine.prototype.setMap.call(this, map);
if (oldmap) oldmap.renderSync();
// Get change (new layer added or removed)
if (map) {
this._listener = map.on('postcompose', this.drawAttribution_.bind(this));
}
this.map_ = map;
this.setCanvas (this.isCanvas_);
}
/**
* Draw attribution in the final canvas
* @private
*/
ol.control.CanvasAttribution.prototype.drawAttribution_ = function(e)
{ var ctx = e.context;
if (!this.isCanvas_) return;
var text = "";
$("li", this.element).each (function()
{ if ($(this).css("display")!="none") text += (text ? " - ":"") + $(this).text();
});
// Get size of the scale div
var position = $(this.element).position();
// Retina device
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.scale(ratio,ratio);
// Position if transform:scale()
var container = $(this.getMap().getViewport()).parent();
var scx = container.outerWidth() / container.get(0).getBoundingClientRect().width;
var scy = container.outerHeight() / container.get(0).getBoundingClientRect().height;
position.left *= scx;
position.top *= scy;
position.right = position.left + $(this.element).outerWidth();
position.bottom = position.top + $(this.element).outerHeight();
// Draw scale text
ctx.beginPath();
ctx.strokeStyle = this.fontStrokeStyle_;
ctx.fillStyle = this.fontFillStyle_;
ctx.lineWidth = this.fontStrokeWidth_;
ctx.textAlign = "right";
ctx.textBaseline ="bottom";
ctx.font = this.font_;
ctx.strokeText(text, position.right, position.bottom);
ctx.fillText(text, position.right, position.bottom);
ctx.closePath();
ctx.restore();
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc
* OpenLayers 3 Scale Line Control integrated in the canvas (for jpeg/png
* @see http://www.kreidefossilien.de/webgis/dokumentation/beispiele/export-map-to-png-with-scale
*
* @constructor
* @extends {ol.control.ScaleLine}
* @param {Object=} options extend the ol.control.ScaleLine options.
* @param {ol.style.Style} options.style used to draw the scale line (default is black/white, 10px Arial).
*/
ol.control.CanvasScaleLine = function(options)
{ ol.control.ScaleLine.call(this, options);
this.scaleHeight_ = 6;
// Get style options
if (!options) options={};
if (!options.style) options.style = new ol.style.Style();
this.setStyle(options.style);
}
ol.inherits(ol.control.CanvasScaleLine, ol.control.ScaleLine);
/**
* Remove the control from its current map and attach it to the new map.
* Subclasses may set up event handlers to get notified about changes to
* the map here.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.control.CanvasScaleLine.prototype.setMap = function (map)
{ var oldmap = this.getMap();
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.ScaleLine.prototype.setMap.call(this, map);
if (oldmap) oldmap.renderSync();
// Add postcompose on the map
if (map) {
this._listener = map.on('postcompose', this.drawScale_.bind(this));
}
// Hide the default DOM element
this.$element = $(this.element).css("visibility","hidden");
this.olscale = $(".ol-scale-line-inner", this.element);
}
/**
* Change the control style
* @param {_ol_style_Style_} style
*/
ol.control.CanvasScaleLine.prototype.setStyle = function (style)
{ var stroke = style.getStroke();
this.strokeStyle_ = stroke ? ol.color.asString(stroke.getColor()) : "#000";
this.strokeWidth_ = stroke ? stroke.getWidth() : 2;
var fill = style.getFill();
this.fillStyle_ = fill ? ol.color.asString(fill.getColor()) : "#fff";
var text = style.getText();
this.font_ = text ? text.getFont() : "10px Arial";
stroke = text ? text.getStroke() : null;
fill = text ? text.getFill() : null;
this.fontStrokeStyle_ = stroke ? ol.color.asString(stroke.getColor()) : this.fillStyle_;
this.fontStrokeWidth_ = stroke ? stroke.getWidth() : 3;
this.fontFillStyle_ = fill ? ol.color.asString(fill.getColor()) : this.strokeStyle_;
// refresh
if (this.getMap()) this.getMap().render();
}
/**
* Draw attribution in the final canvas
* @private
*/
ol.control.CanvasScaleLine.prototype.drawScale_ = function(e)
{ if ( this.$element.css("display")==="none" ) return;
var ctx = e.context;
// Get size of the scale div
var scalewidth = this.olscale.width();
if (!scalewidth) return;
var text = this.olscale.text();
var position = this.$element.position();
// Retina device
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.scale(ratio,ratio);
// Position if transform:scale()
var container = $(this.getMap().getViewport()).parent();
var scx = container.outerWidth() / container.get(0).getBoundingClientRect().width;
var scy = container.outerHeight() / container.get(0).getBoundingClientRect().height;
position.left *= scx;
position.top *= scy;
// On top
position.top += this.$element.height() - this.scaleHeight_;
// Draw scale text
ctx.beginPath();
ctx.strokeStyle = this.fontStrokeStyle_;
ctx.fillStyle = this.fontFillStyle_;
ctx.lineWidth = this.fontStrokeWidth_;
ctx.textAlign = "center";
ctx.textBaseline ="bottom";
ctx.font = this.font_;
ctx.strokeText(text, position.left+scalewidth/2, position.top);
ctx.fillText(text, position.left+scalewidth/2, position.top);
ctx.closePath();
// Draw scale bar
position.top += 2;
ctx.lineWidth = this.strokeWidth_;
ctx.strokeStyle = this.strokeStyle_;
var max = 4;
var n = parseInt(text);
while (n%10 === 0) n/=10;
if (n%5 === 0) max = 5;
for (var i=0; i<max; i++)
{ ctx.beginPath();
ctx.fillStyle = i%2 ? this.fillStyle_ : this.strokeStyle_;
ctx.rect(position.left+i*scalewidth/max, position.top, scalewidth/max, this.scaleHeight_);
ctx.stroke();
ctx.fill();
ctx.closePath();
}
ctx.restore();
}
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* OpenLayers 3 Title Control integrated in the canvas (for jpeg/png
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options extend the ol.control options.
* @param {ol.style.Style} options.style style usesd to draw the title.
*/
ol.control.CanvasTitle = function(options)
{ if (!options) options={};
// Get style options
if (!options.style) options.style = new ol.style.Style();
this.setStyle(options.style);
// Initialize parent
var elt = $("<div>").text(this.text_)
.addClass("ol-title ol-unselectable")
.css(
{ font: this.font_,
position: 'absolute',
top:0, left:0, right:0,
display: 'block',
visibility: 'hidden'
});
ol.control.Control.call(this,
{ element: elt.get(0),
target: options.target
});
}
ol.inherits(ol.control.CanvasTitle, ol.control.Control);
/**
* Remove the control from its current map and attach it to the new map.
* Subclasses may set up event handlers to get notified about changes to
* the map here.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.control.CanvasTitle.prototype.setMap = function (map)
{ var oldmap = this.getMap();
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (oldmap) oldmap.renderSync();
// Get change (new layer added or removed)
if (map) {
this._listener = map.on('postcompose', this.drawTitle_.bind(this));
}
}
/**
* Change the control style
* @param {ol.style.Style} style
*/
ol.control.CanvasTitle.prototype.setStyle = function (style)
{ var text = style.getText();
this.font_ = text ? text.getFont() || "20px Arial" : "20px Arial";
this.text_ = text ? text.getText() : "";
var stroke = text ? text.getStroke() : null;
var fill = text ? text.getFill() : null;
this.strokeStyle_ = stroke ? ol.color.asString(stroke.getColor()) : "#fff";
this.fillStyle_ = fill ? ol.color.asString(fill.getColor()) : "#000";
if (this.element)
{ $(this.element).text(this.text_).css ({font: this.font_});
}
// refresh
if (this.getMap()) this.getMap().render();
}
/**
* Set the map title
* @param {string} map title.
* @api stable
*/
ol.control.CanvasTitle.prototype.setTitle = function (title)
{ this.text_ = title;
$(this.element).text(title);
if (this.getMap()) this.getMap().renderSync();
}
/**
* Get the map title
* @param {string} map title.
* @api stable
*/
ol.control.CanvasTitle.prototype.getTitle = function (title)
{ return this.text_;
}
/**
* Set control visibility
* @param {bool} b
* @api stable
*/
ol.control.CanvasTitle.prototype.setVisible = function (b)
{ if (b) $(this.element).show();
else $(this.element).hide();
if (this.getMap()) this.getMap().renderSync();
}
/**
* Get control visibility
* @return {bool}
* @api stable
*/
ol.control.CanvasTitle.prototype.getVisible = function (b)
{ return ($(this.element).css('display') !== 'none');
}
/** Draw scale line in the final canvas
*/
ol.control.CanvasTitle.prototype.drawTitle_ = function(e)
{ if (!this.getVisible()) return;
var ctx = e.context;
// Retina device
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.scale(ratio,ratio);
var w = ctx.canvas.width/ratio;
var h = $(this.element).height();
var position = { top:0, left:w/2 };
ctx.beginPath();
ctx.fillStyle = this.strokeStyle_;
ctx.rect(0,0, w, h);
ctx.fill();
ctx.closePath();
ctx.beginPath();
ctx.fillStyle = this.fillStyle_;
ctx.textAlign = "center";
ctx.textBaseline = "middle";
ctx.font = this.font_;
ctx.fillText(this.text_, position.left, position.top +h/2);
ctx.closePath();
ctx.restore();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** ol.control.Cloud adds an old map effect on a canvas renderer.
* It colors the map, adds a parchment texture and compass onto the map.
* @constructor
* @param {Object} options
* @param {_ol_color_} options.hue color to set hue of the map, default #963
* @param {Number} options.saturation saturation of the hue color, default 0.6
* @param {Number} options.opacity opacity of the overimpose image, default 0.7
* @todo add effects on pan / zoom change
*/
ol.control.Cloud = function(options)
{ options = options || {};
var div = document.createElement('div');
div.className = "ol-cloud ol-unselectable ol-control";
ol.control.Control.call(this,
{ element: div
});
// Defaut cloud image
this.cloud = options.img;
if (!this.cloud)
{ this.cloud = new Image();
this.cloud.src = "
}
this.bird = options.bird;
if (!this.bird)
{ this.bird = new Image();
this.bird.src = "";
}
// Parameters
this.set('opacity', options.opacity||0.3);
this.set('density', options.density||0.5);
this.setWind(options);
};
ol.inherits(ol.control.Cloud, ol.control.Control);
/**
* Remove the control from its current map and attach it to the new map.
* Subclasses may set up event handlers to get notified about changes to
* the map here.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.control.Cloud.prototype.setMap = function (map) {
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (map)
{ this._listener = map.on('postcompose', this.drawCloud_.bind(this));
}
};
/** Set wind direction / force
*/
ol.control.Cloud.prototype.setWind= function (options)
{ options = options || {};
var rnd = Math.random;
var a = options.windAngle || rnd()*Math.PI;
var speed = options.windSpeed || rnd();
this.wind = { angle: a, cos: Math.cos(a), sin: Math.sin(a), speed: speed };
}
/**
* @private
*/
ol.control.Cloud.prototype.drawCloud_ = function (event)
{ if (!this.getMap()) return;
var ctx = event.context;
var canvas = ctx.canvas;
var ratio = event.frameState.pixelRatio;
// var m = Math.max(canvas.width, canvas.height);
// var res = view.getResolution()/ratio;
// var rot = view.getRotation();
// Not ready !
if (!this.cloud.width) return;
// Go!
var p = this.particules;
var rnd = Math.random;
var w = this.cloud.width;
var h = this.cloud.height;
var w2 = this.cloud.width/2;
var h2 = this.cloud.height/2;
var d = (this.get('density')*10*canvas.width*canvas.height/w/h) << 0;
function addClouds (nb, reset)
{ for (var i=0; i<nb; i++)
{ p.push({ x: rnd()*canvas.width-w2, y: rnd()*canvas.height-h2 });
}
}
// First time: init clouds
if (!p)
{ p = this.particules = [];
addClouds(d);
// Wind
var a = rnd()*Math.PI;
this.width = canvas.width;
this.height = canvas.height;
// Birds
this.birds = [];
for (var i=0; i<5; i++)
{ var b = { angle: rnd()*2*Math.PI, x: rnd()*canvas.width, y: rnd()*canvas.height, rot:0, fly:0 };
b.cos = Math.cos(b.angle);
b.sin = Math.cos(b.angle);
this.birds.push (b);
}
}
// Parameters changed
else if (d != p.length)
{ if (this.width !== canvas.width || this.height !== canvas.height)
{ p = this.particules = [];
addClouds(d);
this.width = canvas.width;
this.height = canvas.height;
}
else if (d > p.length)
{ addClouds(1);
}
else if (d < p.length)
{ p.splice ( (p.length*rnd()) << 0, 1);
}
}
// Draw clouds
var dx = this.wind.cos * this.wind.speed;
var dy = this.wind.sin * this.wind.speed;
for (var i=0; i<p.length; i++)
{ p[i].x += dx + rnd()*2-1;
p[i].y += dy + rnd()*2-1;
// out!
if (p[i].x < -w)
{ p[i].x = canvas.width;
p[i].y = rnd()*canvas.height-h2;
}
else if (p[i].x > canvas.width)
{ p[i].x = -w;
p[i].y = rnd()*canvas.height-h2;
}
if (p[i].y < -h)
{ p[i].y = canvas.height;
p[i].x = rnd()*canvas.width-w2;
}
else if (p[i].y > canvas.height)
{ p[i].y = -h;
p[i].x = rnd()*canvas.width-w2;
}
}
// Draw clouds
ctx.globalAlpha = this.get('opacity');
for (var i=0; i<p.length; i++)
{ ctx.drawImage(this.cloud, p[i].x,p[i].y);
}
ctx.globalAlpha = 1;
// Draw birds
var w = this.bird.width/2;
var h = this.bird.height/2;
var sc = 0.5;
var dw = canvas.width+w;
var dh = canvas.height+h;
for (var i=0; i<this.birds.length; i++)
{ var bi = this.birds[i];
// Animate birds
var sx = 0;
if (bi.fly)
{ bi.fly = (++bi.fly%5)
sx = -0.1
}
else if (rnd()<0.01) bi.fly=1;
// Rotate birds
if (bi.rot)
{ bi.angle += bi.rot;
bi.cos = Math.cos(bi.angle);
bi.sin = Math.sin(bi.angle);
}
if (rnd()<0.01)
{ bi.rot = bi.rot ? 0 : rnd()*Math.PI/200-Math.PI/400;
bi.cos = Math.cos(bi.angle);
bi.sin = Math.sin(bi.angle);
}
// Move birds
bi.x += bi.sin;
if (bi.x>dw) bi.x = -w;
if (bi.x<-w) bi.x = dw;
bi.y -= bi.cos;
if (bi.y>dh) bi.y = -h;
if (bi.y<-h) bi.y = dh;
// Draw birds
ctx.save();
ctx.translate (bi.x, bi.y);
ctx.rotate(bi.angle)
ctx.scale(sc+sx,sc);
ctx.drawImage( this.bird, -w,-h );
ctx.restore();
}
// Continue animation
this.getMap().render();
}
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Draw a compass on the map. The position/size of the control is defined in the css.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options. The style {_ol_style_Stroke_} option is usesd to draw the text.
* @param {string} options.className class name for the control
* @param {Image} options.image an image, default use the src option or a default image
* @param {string} options.src image src, default use the image option or a default image
* @param {boolean} options.rotateVithView rotate vith view (false to show watermark), default true
* @param {_ol_style_Stroke_} options.style style to draw the lines, default draw no lines
*/
ol.control.Compass = function(options)
{ var self = this;
if (!options) options = {};
// Initialize parent
var elt = document.createElement("div");
elt.className = "ol-compassctrl ol-unselectable ol-hidden" + (options.className ? " "+options.className : "");
elt.style.position = "absolute";
elt.style.visibility = "hidden";
ol.control.Control.call(this, { element: elt });
this.set('rotateVithView', options.rotateWithView!==false);
// Style to draw the lines
this.style = options.style;
// The image
if (options.image)
{ this.img_ = options.image;
}
else if (options.src)
{ this.img_ = new Image();
this.img_.onload = function(){ if (self.getMap()) self.getMap().renderSync(); }
this.img_.src = options.src;
}
else this.img_ = this.defaultCompass_($(this.element).width(), this.style ? this.style.getColor():"");
// 8 angles
this.da_ = [];
for (var i=0; i<8; i++) this.da_[i] = [ Math.cos(Math.PI*i/8), Math.sin(Math.PI*i/8) ];
};
ol.inherits(ol.control.Compass, ol.control.Control);
/**
* Remove the control from its current map and attach it to the new map.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.control.Compass.prototype.setMap = function (map)
{ var oldmap = this.getMap();
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (oldmap) oldmap.renderSync();
// Get change (new layer added or removed)
if (map) this._listener = map.on('postcompose', this.drawCompass_.bind(this));
};
/**
* Create a default image.
* @param {number} s the size of the compass
* @private
*/
ol.control.Compass.prototype.defaultCompass_ = function (s, color)
{ var canvas = document.createElement('canvas');
var ctx = canvas.getContext("2d");
var s = canvas.width = canvas.height;
var r = s/2;
var r2 = 0.22*r;
function draw (r, r2)
{ ctx.fillStyle = color ||"#963";
ctx.beginPath();
ctx.moveTo (0,0);
ctx.lineTo (r,0); ctx.lineTo (r2,r2); ctx.moveTo (0,0);
ctx.lineTo (-r,0); ctx.lineTo (-r2,-r2); ctx.moveTo (0,0);
ctx.lineTo (0,r); ctx.lineTo (-r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,-r); ctx.lineTo (r2,-r2); ctx.moveTo (0,0);
ctx.fill();
ctx.stroke();
};
function draw2 (r, r2)
{ ctx.globalCompositeOperation = "destination-out";
ctx.fillStyle = "#fff";
ctx.beginPath();
ctx.moveTo (0,0);
ctx.lineTo (r,0); ctx.lineTo (r2,-r2); ctx.moveTo (0,0);
ctx.lineTo (-r,0); ctx.lineTo (-r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,r); ctx.lineTo (r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,-r); ctx.lineTo (-r2,-r2); ctx.moveTo (0,0);
ctx.fill();
ctx.globalCompositeOperation="source-over";
ctx.beginPath();
ctx.moveTo (0,0);
ctx.lineTo (r,0); ctx.lineTo (r2,-r2); ctx.moveTo (0,0);
ctx.lineTo (-r,0); ctx.lineTo (-r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,r); ctx.lineTo (r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,-r); ctx.lineTo (-r2,-r2); ctx.moveTo (0,0);
ctx.stroke();
};
ctx.translate(r,r);
ctx.strokeStyle = color || "#963";
ctx.lineWidth = 1.5;
ctx.beginPath();
ctx.arc (0,0, s*0.41, 0, 2*Math.PI);
ctx.arc (0,0, s*0.44, 0, 2*Math.PI);
ctx.stroke();
ctx.rotate(Math.PI/4)
draw (r*0.9, r2*0.8);
draw2 (r*0.9, r2*0.8);
ctx.rotate(-Math.PI/4)
draw (r, r2);
draw2 (r, r2);
return canvas;
};
/** Draw compass
* @param {ol.event} e postcompose event
* @private
*/
ol.control.Compass.prototype.drawCompass_ = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
// Retina device
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.scale(ratio,ratio);
var w = $(this.element).width();
var h = $(this.element).height();
var pos = $(this.element).position();
var compass = this.img_;
var rot = e.frameState.viewState.rotation;
ctx.beginPath();
ctx.translate(pos.left+w/2, pos.top+h/2);
if (this.get('rotateVithView')) ctx.rotate(rot);
/*
ctx.globalCompositeOperation = "multiply";
ctx.globalAlpha = this.opacity || 1;
*/
if (this.style)
{ ctx.beginPath();
ctx.strokeStyle = this.style.getColor();
ctx.lineWidth = this.style.getWidth();
var m = Math.max(canvas.width, canvas.height);
for (var i=0; i<8; i++)
{ ctx.moveTo (-this.da_[i][0]*m, -this.da_[i][1]*m);
ctx.lineTo (this.da_[i][0]*m, this.da_[i][1]*m);
}
ctx.stroke();
}
if (compass.width)
{ ctx.drawImage (compass, -w/2, -h/2, w, h);
}
ctx.closePath();
ctx.restore();
};
/** A simple control to disable all actions on the map.
* The control will create an invisible div over the map.
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {String} options.class class of the control
* @param {String} options.html html code to insert in the control
* @param {bool} options.on the control is on
* @param {function} options.toggleFn callback when control is clicked
*/
ol.control.Disable = function(options)
{ var options = options||{};
var element = $("<div>").addClass((options.calssName||"")+' ol-disable ol-unselectable ol-control');
element.css({ top:0, left:0, right:0, bottom:0, "z-index":10000, background:"none", display:"none" });
ol.control.Control.call(this,
{ element: element.get(0)
});
}
ol.inherits(ol.control.Disable, ol.control.Control);
/** Test if the control is on
* @return {bool}
* @api stable
*/
ol.control.Disable.prototype.isOn = function()
{ return $(this.element).hasClass("ol-disable");
}
/** Disable all action on the map
* @param {bool} b, default false
* @api stable
*/
ol.control.Disable.prototype.disableMap = function(b)
{ if (b)
{ $(this.element).addClass("ol-enable").show();
}
else
{ $(this.element).removeClass("ol-enable").hide();
}
}
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** A simple gauge control to display level information on the map.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {String} options.className class of the control
* @param {String} options.title title of the control
* @param {number} options.max maximum value, default 100;
* @param {number} options.val the value, default 0
*/
ol.control.Gauge = function(options)
{ options = options || {};
var element = $("<div>").addClass((options.className||"") + ' ol-gauge ol-unselectable ol-control');
this.title_ = $("<span>").appendTo(element);
this.gauge_ = $("<button>").attr('type','button').appendTo($("<div>").appendTo(element)).width(0);
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
this.setTitle(options.title);
this.val(options.val);
this.set("max", options.max||100);
};
ol.inherits(ol.control.Gauge, ol.control.Control);
/** Set the control title
* @param {string} title
*/
ol.control.Gauge.prototype.setTitle = function(title)
{ this.title_.html(title||"");
if (!title) this.title_.hide();
else this.title_.show();
};
/** Set/get the gauge value
* @param {number|undefined} v the value or undefined to get it
* @return {number} the value
*/
ol.control.Gauge.prototype.val = function(v)
{ if (v!==undefined)
{ this.val_ = v;
this.gauge_.css("width", (v/this.get('max')*100)+"%");
}
return this.val_;
};
/** Bookmark positions on ol maps.
*
* @constructor
* @extends {ol.control.Control}
* @fires add
* @fires remove
* @param {} options Geobookmark's options
* @param {string} options.className default ol-bookmark
* @param {string} options.placeholder input placeholder, default Add a new geomark...
* @param {bool} options.editable enable modification, default true
* @param {string} options.namespace a namespace to save the boolmark (if more than one on a page), default ol
* @param {Array<any>} options.marks a list of default bookmarks:
* @see [Geobookmark example](../../examples/map.control.geobookmark.html)
* @example
var bm = new GeoBookmark ({
marks: {
"Paris": {pos:_ol_proj_.transform([2.351828, 48.856578], 'EPSG:4326', 'EPSG:3857'), zoom:11, permanent: true },
"London": {pos:_ol_proj_.transform([-0.1275,51.507222], 'EPSG:4326', 'EPSG:3857'), zoom:12}
}
});
*/
ol.control.GeoBookmark = function(options) {
options = options || {};
var self = this;
var element = document.createElement('div');
if (options.target) {
element.className = options.className || "ol-bookmark";
} else {
element.className = (options.className || "ol-bookmark") +
" ol-unselectable ol-control ol-collapsed";
element.addEventListener("mouseleave", function() {
if (input !== document.activeElement) {
menu.style.display = 'none';
};
});
// Show bookmarks on click
this.button = document.createElement('button');
this.button.setAttribute('type', 'button');
this.button.addEventListener('click', function(e) {
menu.style.display = (menu.style.display === '' || menu.style.display === 'none' ? 'block': 'none');
});
element.appendChild(this.button);
}
// The menu
var menu = document.createElement('div');
element.appendChild(menu);
var ul = document.createElement('ul');
menu.appendChild(ul);
var input = document.createElement('input');
input.setAttribute("placeholder", options.placeholder || "Add a new geomark...")
input.addEventListener("change", function(e) {
var title = this.value;
if (title) {
self.addBookmark(title);
this.value = '';
self.dispatchEvent({
type: "add",
name: title
});
}
menu.style.display = 'none';
});
input.addEventListener("blur", function() {
menu.style.display = 'none';
});
menu.appendChild(input);
// Init
ol.control.Control.call(this, {
element: element,
target: options.target
});
this.on("propertychange", function(e) {
if (e.key==='editable') {
element.className = element.className.replace(" ol-editable","");
if (this.get('editable')) {
element.className += " ol-editable";
}
}
// console.log(e);
}.bind(this));
this.set("namespace", options.namespace || 'ol');
this.set("editable", options.editable !== false);
// Set default bmark
this.setBookmarks(localStorage[this.get('namespace')+"@bookmark"] ? null:options.marks);
};
ol.inherits(ol.control.GeoBookmark, ol.control.Control);
/** Set bookmarks
* @param {} bmark a list of bookmarks, default retreave in the localstorage
* @example
bm.setBookmarks({
"Paris": {pos:_ol_proj_.transform([2.351828, 48.856578], 'EPSG:4326', 'EPSG:3857'), zoom:11, permanent: true },
"London": {pos:_ol_proj_.transform([-0.1275,51.507222], 'EPSG:4326', 'EPSG:3857'), zoom:12}
});
*/
ol.control.GeoBookmark.prototype.setBookmarks = function(bmark) {
if (!bmark) bmark = JSON.parse(localStorage[this.get('namespace')+"@bookmark"] || "{}");
var modify = this.get("editable");
var ul = this.element.querySelector("ul");
var menu = this.element.querySelector("div");
var self = this;
ul.innerHTML = '';
for (var b in bmark) {
var li = document.createElement('li');
li.textContent = b;
li.setAttribute('data-bookmark', JSON.stringify(bmark[b]));
li.addEventListener('click', function() {
var bm = JSON.parse(this.getAttribute("data-bookmark"));
self.getMap().getView().setCenter(bm.pos);
self.getMap().getView().setZoom(bm.zoom);
menu.style.display = 'none';
});
ul.appendChild(li);
if (modify && !bmark[b].permanent) {
var button = document.createElement('button');
button.setAttribute('data-name', b);
button.setAttribute("title", "Suppr.");
button.addEventListener('click', function(e) {
self.removeBookmark(this.getAttribute("data-name"));
self.dispatchEvent({ type: "remove", name: this.getAttribute("data-name") });
e.stopPropagation();
});
li.appendChild(button);
}
}
localStorage[this.get('namespace')+"@bookmark"] = JSON.stringify(bmark);
};
/** Get Geo bookmarks
* @return {any} a list of bookmarks : { BM1:{pos:ol.coordinates, zoom: integer}, BM2:{pos:ol.coordinates, zoom: integer} }
*/
ol.control.GeoBookmark.prototype.getBookmarks = function() {
return JSON.parse(localStorage[this.get('namespace')+"@bookmark"] || "{}");
};
/** Remove a Geo bookmark
* @param {string} name
*/
ol.control.GeoBookmark.prototype.removeBookmark = function(name) {
if (!name) {
return;
};
var bmark = this.getBookmarks();
delete bmark[name];
this.setBookmarks(bmark);
};
/** Add a new Geo bookmark (replace existing one if any)
* @param {string} name name of the bookmark (display in the menu)
* @param {_ol_coordinate_} position default current position
* @param {number} zoom default current map zoom
* @param {bool} permanent prevent from deletion, default false
*/
ol.control.GeoBookmark.prototype.addBookmark = function(name, position, zoom, permanent)
{
if (!name) return;
var bmark = this.getBookmarks();
// Don't override permanent bookmark
if (bmark[name] && bmark[name].permanent) return;
// Create or override
bmark[name] = {
pos: position || this.getMap().getView().getCenter(),
zoom: zoom || this.getMap().getView().getZoom(),
permanent: !!permanent
};
this.setBookmarks(bmark);
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Control bar for OL3
* The control bar is a container for other controls. It can be used to create toolbars.
* Control bars can be nested and combined with ol.control.Toggle to handle activate/deactivate.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {String} options.className class of the control
* @param {String} options.centerLabel label for center button, default center
*/
ol.control.GeolocationBar = function(options) {
if (!options) options = {};
options.className = options.className || 'ol-geobar';
ol.control.Bar.call(this, options);
this.setPosition(options.position || 'bottom-right');
var element = $(this.element);
// Geolocation draw interaction
var interaction = new ol.interaction.GeolocationDraw({
source: options.source,
zoom: options.zoom,
followTrack: true,
minAccuracy: options.minAccuracy || 10000
});
this._geolocBt = new ol.control.Toggle ({
className: 'geolocBt',
interaction: interaction,
onToggle: function(b) {
interaction.pause(true);
interaction.setFollowTrack(true);
element.removeClass('pauseTrack');
}
});
this.addControl(this._geolocBt);
this._geolocBt.setActive(false);
// Buttons
var bar = new ol.control.Bar();
this.addControl(bar);
var centerBt = new ol.control.TextButton ({
className: 'centerBt',
html: options.centerLabel ||'center',
handleClick: function(b) {
interaction.setFollowTrack('auto');
}
});
bar.addControl(centerBt);
var startBt = new ol.control.Button ({
className: 'startBt',
handleClick: function(){
interaction.pause(false);
interaction.setFollowTrack('auto');
element.addClass('pauseTrack');
}
});
bar.addControl(startBt);
var pauseBt = new ol.control.Button ({
className: 'pauseBt',
handleClick: function(){
interaction.pause(true);
interaction.setFollowTrack('auto');
element.removeClass('pauseTrack');
}
});
bar.addControl(pauseBt);
interaction.on('follow', function(e) {
if (e.following) {
element.removeClass('centerTrack');
} else {
element.addClass('centerTrack');
}
});
// Activate
this._geolocBt.on('change:active', function(e) {
if (e.active) {
element.addClass('ol-active');
} else {
element.removeClass('ol-active');
}
});
};
ol.inherits(ol.control.GeolocationBar, ol.control.Bar);
/** Get the ol.interaction.GeolocationDraw associatedwith the bar
*
*/
ol.control.GeolocationBar.prototype.getInteraction = function () {
return this._geolocBt.getInteraction();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* OpenLayers 3 lobe Overview Control.
* The globe can rotate with map (follow.)
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {boolean} follow follow the map when center change, default false
* @param {top|bottom-left|right} align position as top-left, etc.
* @param {Array<ol.layer>} layers list of layers to display on the globe
* @param {ol.style.Style | Array.<ol.style.Style> | undefined} style style to draw the position on the map , default a marker
*/
ol.control.Globe = function(opt_options)
{ var options = opt_options || {};
var self = this;
// API
var element;
if (options.target)
{ element = $("<div>");
this.panel_ = $(options.target);
}
else
{ element = $("<div>").addClass('ol-globe ol-unselectable ol-control');
if (/top/.test(options.align)) element.addClass('ol-control-top');
if (/right/.test(options.align)) element.addClass('ol-control-right');
this.panel_ = $("<div>").addClass("panel")
.appendTo(element);
this.pointer_ = $("<div>").addClass("ol-pointer")
.appendTo(element);
}
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
// http://openlayers.org/en/latest/examples/sphere-mollweide.html ???
// Create a globe map
this.ovmap_ = new ol.Map(
{ controls: new ol.Collection(),
interactions: new ol.Collection(),
target: this.panel_.get(0),
view: new ol.View
({ zoom: 0,
center: [0,0]
}),
layers: options.layers
});
setTimeout (function()
{ self.ovmap_.updateSize();
}, 0);
this.set('follow', options.follow || false);
// Cache extent
this.extentLayer = new ol.layer.Vector(
{ name: 'Cache extent',
source: new ol.source.Vector(),
style: options.style || [new ol.style.Style(
{ image: new ol.style.Circle(
{ fill: new ol.style.Fill({
color: 'rgba(255,0,0, 1)'
}),
stroke: new ol.style.Stroke(
{ width: 7,
color: 'rgba(255,0,0, 0.8)'
}),
radius: 5
})
}
)]
})
this.ovmap_.addLayer(this.extentLayer);
};
ol.inherits(ol.control.Globe, ol.control.Control);
/**
* Set the map instance the control associated with.
* @param {ol.Map} map The map instance.
*/
ol.control.Globe.prototype.setMap = function(map) {
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
// Get change (new layer added or removed)
if (map)
{ this._listener = map.getView().on('propertychange', this.setView.bind(this));
this.setView();
}
};
/** Set the globe center with the map center
*/
ol.control.Globe.prototype.setView = function()
{ if (this.getMap() && this.get('follow'))
{ this.setCenter(this.getMap().getView().getCenter());
}
}
/** Get globe map
* @return {ol.Map}
*/
ol.control.Globe.prototype.getGlobe = function()
{ return this.ovmap_;
}
/** Show/hide the globe
*/
ol.control.Globe.prototype.show = function(b)
{ if (b!==false) $(this.element).removeClass("ol-collapsed");
else $(this.element).addClass("ol-collapsed");
this.ovmap_.updateSize();
}
/** Set position on the map
* @param {top|bottom-left|right} align
*/
ol.control.Globe.prototype.setPosition = function(align)
{ if (/top/.test(align)) $(this.element).addClass("ol-control-top");
else $(this.element).removeClass("ol-control-top");
if (/right/.test(align)) $(this.element).addClass("ol-control-right");
else $(this.element).removeClass("ol-control-right");
}
/** Set the globe center
* @param {_ol_coordinate_} center the point to center to
* @param {boolean} show true to show a pointer
*/
ol.control.Globe.prototype.setCenter = function (center, show)
{ var self = this;
this.pointer_.addClass("hidden");
if (center)
{ var map = this.ovmap_;
var p = map.getPixelFromCoordinate(center);
var h = $(this.element).height();
setTimeout(function() {
self.pointer_.css({ 'top': Math.min(Math.max(p[1],0),h) , 'left': "50%" } )
.removeClass("hidden");
}, 800);
map.getView().animate({ center: [center[0],0] });
}
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Draw a graticule on the map.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} _ol_control_ options.
* - projection {ol.projectionLike} projection to use for the graticule, default EPSG:4326
* - maxResolution {number} max resolution to display the graticule
* - style {ol.style.Style} Style to use for drawing the graticule, default black.
* - step {number} step beetween lines (in proj units), default 1
* - stepCoord {number} show a coord every stepCoord, default 1
* - spacing {number} spacing beetween lines (in px), default 40px
* - borderWidth {number} width of the border (in px), default 5px
* - margin {number} margin of the border (in px), default 0px
*/
ol.control.Graticule = function(options)
{ var self = this;
if (!options) options = {};
// Initialize parent
var elt = document.createElement("div");
elt.className = "ol-graticule ol-unselectable ol-hidden";
ol.control.Control.call(this, { element: elt });
this.set('projection', options.projection || 'EPSG:4326');
// Use to limit calculation
var p = new ol.proj.Projection({code:this.get('projection')});
var m = p.getMetersPerUnit();
this.fac = 1;
while (m/this.fac>10)
{ this.fac *= 10;
}
this.fac = 10000/this.fac;
this.set('maxResolution', options.maxResolution || Infinity);
this.set('step', options.step || 0.1);
this.set('stepCoord', options.stepCoord || 1);
this.set('spacing', options.spacing || 40);
this.set('margin', options.margin || 0);
this.set('borderWidth', options.borderWidth || 5);
this.set('stroke', options.stroke!==false);
this.formatCoord = options.formatCoord || function(c){return c;};
if (options.style instanceof ol.style.Style) this.style = options.style;
else this.style = new ol.style.Style(
{ stroke: new ol.style.Stroke({ color:"#000", width:1 }),
fill: new ol.style.Fill({ color: "#fff" }),
text: new ol.style.Text(
{ stroke: new ol.style.Stroke({ color:"#fff", width:2 }),
fill: new ol.style.Fill({ color:"#000" }),
})
});
};
ol.inherits(ol.control.Graticule, ol.control.Control);
/**
* Remove the control from its current map and attach it to the new map.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.control.Graticule.prototype.setMap = function (map) {
var oldmap = this.getMap();
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (oldmap) oldmap.renderSync();
// Get change (new layer added or removed)
if (map) {
this._listener = map.on('postcompose', this.drawGraticule_.bind(this));
}
};
ol.control.Graticule.prototype.setStyle = function (style)
{ this.style = style;
};
ol.control.Graticule.prototype.getStyle = function (style)
{ return style;
};
ol.control.Graticule.prototype.drawGraticule_ = function (e)
{ if (this.get('maxResolution')<e.frameState.viewState.resolution) return;
var ctx = e.context;
var canvas = ctx.canvas;
var ratio = e.frameState.pixelRatio;
var w = canvas.width/ratio;
var h = canvas.height/ratio;
var proj = this.get('projection');
var map = this.getMap();
var bbox =
[ map.getCoordinateFromPixel([0,0]),
map.getCoordinateFromPixel([w,0]),
map.getCoordinateFromPixel([w,h]),
map.getCoordinateFromPixel([0,h])
];
var xmax = -Infinity;
var xmin = Infinity;
var ymax = -Infinity;
var ymin = Infinity;
for (var i=0, c; c=bbox[i]; i++)
{ bbox[i] = ol.proj.transform (c, map.getView().getProjection(), proj);
xmax = Math.max (xmax, bbox[i][0]);
xmin = Math.min (xmin, bbox[i][0]);
ymax = Math.max (ymax, bbox[i][1]);
ymin = Math.min (ymin, bbox[i][1]);
}
var spacing = this.get('spacing');
var step = this.get('step');
var step2 = this.get('stepCoord');
var borderWidth = this.get('borderWidth');
var margin = this.get('margin');
// Limit max line draw
var ds = (xmax-xmin)/step*spacing;
if (ds>w)
{ var dt = Math.round((xmax-xmin)/w*spacing /step);
step *= dt;
if (step>this.fac) step = Math.round(step/this.fac)*this.fac;
}
xmin = (Math.floor(xmin/step))*step -step;
ymin = (Math.floor(ymin/step))*step -step;
xmax = (Math.floor(xmax/step))*step +2*step;
ymax = (Math.floor(ymax/step))*step +2*step;
var extent = ol.proj.get(proj).getExtent();
if (extent)
{ if (xmin < extent[0]) xmin = extent[0];
if (ymin < extent[1]) ymin = extent[1];
if (xmax > extent[2]) xmax = extent[2]+step;
if (ymax > extent[3]) ymax = extent[3]+step;
}
var hasLines = this.style.getStroke() && this.get("stroke");
var hasText = this.style.getText();
var hasBorder = this.style.getFill();
ctx.save();
ctx.scale(ratio,ratio);
ctx.beginPath();
ctx.rect(margin, margin, w-2*margin, h-2*margin);
ctx.clip();
ctx.beginPath();
var txt = {top:[],left:[],bottom:[], right:[]};
for (var x=xmin; x<xmax; x += step)
{ var p0 = ol.proj.transform ([x, ymin], proj, map.getView().getProjection());
p0 = map.getPixelFromCoordinate(p0);
if (hasLines) ctx.moveTo(p0[0], p0[1]);
var p = p0;
for (var y=ymin+step; y<=ymax; y+=step)
{ var p1 = ol.proj.transform ([x, y], proj, map.getView().getProjection());
p1 = map.getPixelFromCoordinate(p1);
if (hasLines) ctx.lineTo(p1[0], p1[1]);
if (p[1]>0 && p1[1]<0) txt.top.push([x, p]);
if (p[1]>h && p1[1]<h) txt.bottom.push([x,p]);
p = p1;
}
}
for (var y=ymin; y<ymax; y += step)
{ var p0 = ol.proj.transform ([xmin, y], proj, map.getView().getProjection());
p0 = map.getPixelFromCoordinate(p0);
if (hasLines) ctx.moveTo(p0[0], p0[1]);
var p = p0;
for (var x=xmin+step; x<=xmax; x+=step)
{ var p1 = ol.proj.transform ([x, y], proj, map.getView().getProjection());
p1 = map.getPixelFromCoordinate(p1);
if (hasLines) ctx.lineTo(p1[0], p1[1]);
if (p[0]<0 && p1[0]>0) txt.left.push([y,p]);
if (p[0]<w && p1[0]>w) txt.right.push([y,p]);
p = p1;
}
}
if (hasLines)
{ ctx.strokeStyle = this.style.getStroke().getColor();
ctx.lineWidth = this.style.getStroke().getWidth();
ctx.stroke();
}
// Draw text
if (hasText)
{
ctx.fillStyle = this.style.getText().getFill().getColor();
ctx.strokeStyle = this.style.getText().getStroke().getColor();
ctx.lineWidth = this.style.getText().getStroke().getWidth();
ctx.textAlign = 'center';
ctx.textBaseline = 'hanging';
var tf;
var offset = (hasBorder ? borderWidth : 0) + margin + 2;
for (var i=0, t; t = txt.top[i]; i++) if (!(Math.round(t[0]/this.get('step'))%step2))
{ tf = this.formatCoord(t[0]);
ctx.strokeText(tf, t[1][0], offset);
ctx.fillText(tf, t[1][0], offset);
}
ctx.textBaseline = 'alphabetic';
for (var i=0, t; t = txt.bottom[i]; i++) if (!(Math.round(t[0]/this.get('step'))%step2))
{ tf = this.formatCoord(t[0]);
ctx.strokeText(tf, t[1][0], h-offset);
ctx.fillText(tf, t[1][0], h-offset);
}
ctx.textBaseline = 'middle';
ctx.textAlign = 'left';
for (var i=0, t; t = txt.left[i]; i++) if (!(Math.round(t[0]/this.get('step'))%step2))
{ tf = this.formatCoord(t[0]);
ctx.strokeText(tf, offset, t[1][1]);
ctx.fillText(tf, offset, t[1][1]);
}
ctx.textAlign = 'right';
for (var i=0, t; t = txt.right[i]; i++) if (!(Math.round(t[0]/this.get('step'))%step2))
{ tf = this.formatCoord(t[0]);
ctx.strokeText(tf, w-offset, t[1][1]);
ctx.fillText(tf, w-offset, t[1][1]);
}
}
// Draw border
if (hasBorder)
{ var fillColor = this.style.getFill().getColor();
var color, stroke;
if (stroke = this.style.getStroke())
{ color = this.style.getStroke().getColor();
}
else
{ color = fillColor;
fillColor = "#fff";
}
ctx.strokeStyle = color;
ctx.lineWidth = stroke ? stroke.getWidth() : 1;
//
for (var i=1; i<txt.top.length; i++)
{ ctx.beginPath();
ctx.rect(txt.top[i-1][1][0], margin, txt.top[i][1][0]-txt.top[i-1][1][0], borderWidth);
ctx.fillStyle = Math.round(txt.top[i][0]/step)%2 ? color: fillColor;
ctx.fill();
ctx.stroke();
}
for (var i=1; i<txt.bottom.length; i++)
{ ctx.beginPath();
ctx.rect(txt.bottom[i-1][1][0], h-borderWidth-margin, txt.bottom[i][1][0]-txt.bottom[i-1][1][0], borderWidth);
ctx.fillStyle = Math.round(txt.bottom[i][0]/step)%2 ? color: fillColor;
ctx.fill();
ctx.stroke();
}
for (var i=1; i<txt.left.length; i++)
{ ctx.beginPath();
ctx.rect(margin, txt.left[i-1][1][1], borderWidth, txt.left[i][1][1]-txt.left[i-1][1][1]);
ctx.fillStyle = Math.round(txt.left[i][0]/step)%2 ? color: fillColor;
ctx.fill();
ctx.stroke();
}
for (var i=1; i<txt.right.length; i++)
{ ctx.beginPath();
ctx.rect(w-borderWidth-margin, txt.right[i-1][1][1], borderWidth, txt.right[i][1][1]-txt.right[i-1][1][1]);
ctx.fillStyle = Math.round(txt.right[i][0]/step)%2 ? color: fillColor;
ctx.fill();
ctx.stroke();
}
ctx.beginPath();
ctx.fillStyle = color;
ctx.rect(margin,margin, borderWidth, borderWidth);
ctx.rect(margin,h-borderWidth-margin, borderWidth,borderWidth);
ctx.rect(w-borderWidth-margin,margin, borderWidth, borderWidth);
ctx.rect(w-borderWidth-margin,h-borderWidth-margin, borderWidth,borderWidth);
ctx.fill();
}
ctx.restore();
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Draw a grid reference on the map and add an index.
*
* @constructor
* @extends {ol.control.Control}
* @fires select
* @param {Object=} Control options.
* - style {ol.style.Style} Style to use for drawing the grid (stroke and text), default black.
* - maxResolution {number} max resolution to display the graticule
* - extent {ol.extent} extent of the grid, required
* - size {ol.size} number of lines and cols, required
* - margin {number} margin to display text (in px), default 0px
* - source {ol.source.Vector} source to use for the index, default none (use setIndex to reset the index)
* - property {string | function} a property to display in the index or a function that takes a feature and return the name to display in the index, default 'name'.
* - sortFeatures {function|undefined} sort function to sort 2 features in the index, default sort on property option
* - indexTitle {function|undefined} a function that takes a feature and return the title to display in the index, default the first letter of property option
* - filterLabel {string} label to display in the search bar, default 'filter'
*/
ol.control.GridReference = function(options)
{ var self = this;
if (!options) options = {};
// Initialize parent
var elt = document.createElement("div");
elt.className = (!options.target ? "ol-control ":"") +"ol-gridreference ol-unselectable "+(options.className||"");
ol.control.Control.call(this,
{ element: elt,
target: options.target
});
if (typeof (options.property)=='function') this.getFeatureName = options.property;
if (typeof (options.sortFeatures)=='function') this.sortFeatures = options.sortFeatures;
if (typeof (options.indexTitle)=='function') this.indexTitle = options.indexTitle;
// Set index using the source
this.source_ = options.source;
if (options.source)
{ this.setIndex(options.source.getFeatures(), options);
// reload on ready
options.source.once('change',function(e)
{ if (options.source.getState() === 'ready')
{ this.setIndex(options.source.getFeatures(), options);
}
}.bind(this));
};
// Options
this.set('maxResolution', options.maxResolution || Infinity);
this.set('extent', options.extent);
this.set('size', options.size);
this.set('margin', options.margin || 0);
this.set('property', options.property || 'name');
this.set('filterLabel', options.filterLabel || 'filter');
if (options.style instanceof ol.style.Style) this.style = options.style;
else this.style = new ol.style.Style(
{ stroke: new ol.style.Stroke({ color:"#000", width:1 }),
text: new ol.style.Text(
{ font: "bold 14px Arial",
stroke: new ol.style.Stroke({ color:"#fff", width:2 }),
fill: new ol.style.Fill({ color:"#000" }),
})
});
};
ol.inherits(ol.control.GridReference, ol.control.Control);
/** Returns the text to be displayed in the index
* @param {ol.Feature} f the feature
* @return {string} the text to be displayed in the index
* @api
*/
ol.control.GridReference.prototype.getFeatureName = function (f)
{ return f.get(this.get('property')||'name');
};
/** Sort function
* @param {ol.Feature} a first feature
* @param {ol.Feature} b second feature
* @return {Number} 0 if a==b, -1 if a<b, 1 if a>b
* @api
*/
ol.control.GridReference.prototype.sortFeatures = function (a,b)
{ return (this.getFeatureName(a) == this.getFeatureName(b)) ? 0 : (this.getFeatureName(a) < this.getFeatureName(b)) ? -1 : 1;
};
/** Get the feature title
* @param {ol.Feature} f
* @return the first letter of the eature name (getFeatureName)
* @api
*/
ol.control.GridReference.prototype.indexTitle = function (f)
{ return this.getFeatureName(f).charAt(0);
};
/** Display features in the index
* @param { Array<ol.Feature> | ol.Collection<ol.Feature> } features
*/
ol.control.GridReference.prototype.setIndex = function (features)
{ if (!this.getMap()) return;
var self = this;
if (features.getArray) features = features.getArray();
features.sort ( function(a,b) { return self.sortFeatures(a,b); } );
var elt = $(this.element).html("");
var search = $("<input>").attr('type', 'search')
.attr('placeholder', this.get('filterLabel') || 'filter')
.on('search keyup', function()
{ var v = $(this).val().replace(/^\*/,'');
// console.log(v)
var r = new RegExp (v, 'i');
$('li',ul).each(function()
{ var self = $(this);
if (self.hasClass('ol-title')) self.show();
else
{ if (r.test($('.ol-name',self).text())) self.show();
else self.hide();
}
});
$("li.ol-title", ul).each(function()
{ var nextVisible = $(this).nextAll("li:visible").first()
if (nextVisible.length && !nextVisible.hasClass('ol-title')) $(this).show();
else $(this).hide();
});
})
.appendTo(elt);
var ul = $("<ul>").appendTo(elt);
var r, title;
for (var i=0, f; f=features[i]; i++)
{ r = this.getReference(f.getGeometry().getFirstCoordinate());
if (r)
{ var name = this.getFeatureName(f);
var c = this.indexTitle(f);
if (c != title)
{ $("<li>").addClass('ol-title').text(c).appendTo(ul);
}
title = c;
$("<li>").append($("<span>").addClass("ol-name").text(name))
.append($("<span>").addClass("ol-ref").text(r))
.data ('feature', f)
.click(function()
{ self.dispatchEvent({ type:"select", feature:$(this).data('feature') });
})
.appendTo(ul);
}
}
};
/** Get reference for a coord
* @param {ol.coordinate} coords
* @return {string} the reference
*/
ol.control.GridReference.prototype.getReference = function (coords)
{ if (!this.getMap()) return;
var extent = this.get('extent');
var size = this.get('size');
var dx = Math.floor ( (coords[0] - extent[0]) / (extent[2]- extent[0]) * size[0] );
if (dx<0 || dx>=size[0]) return "";
var dy = Math.floor ( (extent[3] - coords[1]) / (extent[3]- extent[1]) * size[1] );
if (dy<0 || dy>=size[1]) return "";
return String.fromCharCode(65+dx)+dy;
};
/**
* Remove the control from its current map and attach it to the new map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.control.GridReference.prototype.setMap = function (map)
{ var oldmap = this.getMap();
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (oldmap) oldmap.renderSync();
// Get change (new layer added or removed)
if (map)
{ this._listener = map.on('postcompose', this.drawGrid_.bind(this));
if (this.source_) this.setIndex(this.source_.getFeatures());
}
};
/** Set style
* @param {ol.style.Style} style
*/
ol.control.GridReference.prototype.setStyle = function (style)
{ this.style = style;
};
/** Get style
* @return {ol.style.Style} style
*/
ol.control.GridReference.prototype.getStyle = function ()
{ return style;
};
/** Draw the grid
* @param {ol.event} e postcompose event
* @private
*/
ol.control.GridReference.prototype.drawGrid_ = function (e)
{ if (this.get('maxResolution')<e.frameState.viewState.resolution) return;
var ctx = e.context;
var canvas = ctx.canvas;
var ratio = e.frameState.pixelRatio;
var w = canvas.width/ratio;
var h = canvas.height/ratio;
var extent = this.get('extent');
var size = this.get('size');
var map = this.getMap();
var ex = ol.extent.boundingExtent([map.getPixelFromCoordinate([extent[0],extent[1]]), map.getPixelFromCoordinate([extent[2],extent[3]])]);
var p0 = [ex[0],ex[1]];
var p1 = [ex[2],ex[3]];
var dx = (p1[0]-p0[0])/size[0];
var dy = (p1[1]-p0[1])/size[1];
ctx.save();
var margin = this.get('margin');
ctx.scale(ratio,ratio);
ctx.strokeStyle = this.style.getStroke().getColor();
ctx.lineWidth = this.style.getStroke().getWidth();
// Draw grid
ctx.beginPath();
for (var i=0; i<=size[0]; i++)
{ ctx.moveTo(p0[0]+i*dx, p0[1]);
ctx.lineTo(p0[0]+i*dx, p1[1]);
}
for (var i=0; i<=size[1]; i++)
{ ctx.moveTo(p0[0], p0[1]+i*dy);
ctx.lineTo(p1[0], p0[1]+i*dy);
}
ctx.stroke();
// Draw text
ctx.font = this.style.getText().getFont();
ctx.fillStyle = this.style.getText().getFill().getColor();
ctx.strokeStyle = this.style.getText().getStroke().getColor();
var lw = ctx.lineWidth = this.style.getText().getStroke().getWidth();
var spacing = margin +lw;
ctx.textAlign = 'center';
var letter, x, y;
for (var i=0; i<size[0]; i++)
{ letter = String.fromCharCode(65+i);
x = p0[0]+i*dx+dx/2;
y = p0[1]-spacing;
if (y<0)
{ y = spacing;
ctx.textBaseline = 'hanging';
}
else ctx.textBaseline = 'alphabetic';
ctx.strokeText(letter, x, y);
ctx.fillText(letter, x, y);
y = p1[1]+spacing;
if (y>h)
{ y = h-spacing;
ctx.textBaseline = 'alphabetic';
}
else ctx.textBaseline = 'hanging';
ctx.strokeText(letter, x, y);
ctx.fillText(letter, x, y);
}
ctx.textBaseline = 'middle';
for (var i=0; i<size[0]; i++)
{ y = p0[1]+i*dy+dy/2;
ctx.textAlign = 'right';
x = p0[0] - spacing;
if (x<0)
{ x = spacing;
ctx.textAlign = 'left';
}
else ctx.textAlign = 'right';
ctx.strokeText(i, x, y);
ctx.fillText(i, x, y);
x = p1[0] + spacing;
if (x>w)
{ x = w-spacing;
ctx.textAlign = 'right';
}
else ctx.textAlign = 'left';
ctx.strokeText(i, x, y);
ctx.fillText(i, x, y);
}
ctx.restore();
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* OpenLayers 3 Layer Switcher Control.
* @require jQuery
*
* @constructor
* @extends {ol.control.LayerSwitcher}
* @param {Object=} options Control options.
*/
ol.control.LayerPopup = function(options)
{ options = options || {};
options.switcherClass="ol-layerswitcher-popup";
if (options.mouseover!==false) options.mouseover=true;
ol.control.LayerSwitcher.call(this, options);
};
ol.inherits(ol.control.LayerPopup, ol.control.LayerSwitcher);
/** Disable overflow
*/
ol.control.LayerPopup.prototype.overflow = function(){};
/** Render a list of layer
* @param {elt} element to render
* @layers {Array{ol.layer}} list of layer to show
* @api stable
*/
ol.control.LayerPopup.prototype.drawList = function(ul, layers)
{ var self=this;
var setVisibility = function(e)
{ e.preventDefault();
var l = $(this).data("layer");
self.switchLayerVisibility(l,layers);
};
layers.forEach(function(layer)
{ if (self.displayInLayerSwitcher(layer))
{ var d = $("<li>").text(layer.get("title") || layer.get("name"))
.data ('layer', layer)
.click (setVisibility)
.on ("touchstart", setVisibility)
.appendTo(ul);
if (self.testLayerVisibility(layer)) d.addClass("ol-layer-hidden");
if (layer.getVisible()) d.addClass("select");
}
});
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc OpenLayers 3 Layer Switcher Control.
* @require layer.getPreview
* @require jQuery
*
* @constructor
* @extends {ol.control.LayerSwitcher}
* @param {Object=} options Control options.
*/
ol.control.LayerSwitcherImage = function(options)
{ options = options || {};
options.switcherClass="ol-layerswitcher-image";
if (options.mouseover!==false) options.mouseover=true;
ol.control.LayerSwitcher.call(this, options);
};
ol.inherits(ol.control.LayerSwitcherImage, ol.control.LayerSwitcher);
/** Render a list of layer
* @param {elt} element to render
* @layers {Array{ol.layer}} list of layer to show
* @api stable
*/
ol.control.LayerSwitcherImage.prototype.drawList = function(ul, layers)
{ var self = this;
var setVisibility = function(e)
{ e.preventDefault();
var l = $(this).data("layer");
self.switchLayerVisibility(l,layers);
if (e.type=="touchstart") $(self.element).addClass("ol-collapsed");
};
ul.css("height","auto");
layers.forEach(function(layer)
{ if (self.displayInLayerSwitcher(layer))
{ var prev = layer.getPreview ? layer.getPreview() : ["none"];
var d = $("<li>").addClass("ol-imgcontainer")
.data ('layer', layer)
.click (setVisibility)
.on ("touchstart", setVisibility);
if (layer.getVisible()) d.addClass("select");
for (var k=0; k<prev.length; k++)
{ $("<img>").attr('src', prev[k])
.appendTo(d);
}
$("<p>").text(layer.get("title") || layer.get("name")).appendTo(d);
if (self.testLayerVisibility(layer)) d.addClass("ol-layer-hidden");
d.appendTo(ul);
}
});
};
/** Disable overflow
*/
ol.control.LayerSwitcherImage.prototype.overflow = function(){};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Control overlay for OL3
* The overlay control is a control that display an overlay over the map
*
* @constructor
* @extends {ol.control.Control}
* @fire change:visible
* @param {Object=} options Control options.
* - className {String} class of the control
* - hideOnClick {bool} hide the control on click, default false
* - closeBox {bool} add a closeBox to the control, default false
*/
ol.control.Overlay = function(options)
{ if (!options) options={};
var element = $("<div>").addClass('ol-unselectable ol-overlay');
if (options.className) element.addClass(options.className);
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
var self = this;
if (options.hideOnClick) element.click(function(){self.hide();});
this.set("closeBox", options.closeBox);
this._timeout = false;
this.setContent (options.content);
};
ol.inherits(ol.control.Overlay, ol.control.Control);
/** Set the content of the overlay
* @param {string} html the html to display in the control (or a jQuery object)
*/
ol.control.Overlay.prototype.setContent = function (html)
{ var self = this;
if (html)
{ var elt = $(this.element);
elt.html(html);
if (this.get("closeBox"))
{ var cb = $("<div>").addClass("ol-closebox")
.click(function(){self.hide();});
elt.prepend(cb);
}
};
};
/** Set the control visibility
* @param {string} html the html to display in the control (or a jQuery object)
* @param {ol.coordinate} coord coordinate of the top left corner of the control to start from
*/
ol.control.Overlay.prototype.show = function (html, coord)
{ var self = this;
var elt = $(this.element).show();
if (coord)
{ this.center_ = this.getMap().getPixelFromCoordinate(coord);
elt.css({"top":this.center_[1], "left":this.center_[0] });
}
else
{
//TODO: Do fix from hkollmann pull request
this.center_ = false;
elt.css({"top":"", "left":"" });
}
this.setContent(html);
if (this._timeout) clearTimeout(this._timeout);
this._timeout = setTimeout(function()
{ elt.addClass("ol-visible")
.css({ "top":"", "left":"" });
self.dispatchEvent({ type:'change:visible', visible:true, element: self.element });
}, 10);
self.dispatchEvent({ type:'change:visible', visible:false, element: self.element });
};
/** Set the control visibility hidden
*/
ol.control.Overlay.prototype.hide = function ()
{ var elt = $(this.element).removeClass("ol-visible");
if (this.center_)
{ elt.css({"top":this.center_[1], "left":this.center_[0] })
this.center_ = false;
}
if (this._timeout) clearTimeout(this._timeout);
this._timeout = setTimeout(function(){ elt.hide(); }, 500);
this.dispatchEvent({ type:'change:visible', visible:false, element: this.element });
};
/** Toggle control visibility
*/
ol.control.Overlay.prototype.toggle = function ()
{ if (this.getVisible()) this.hide();
else this.show();
}
/** Get the control visibility
* @return {boolean} b
*/
ol.control.Overlay.prototype.getVisible = function ()
{ return ($(this.element).css('display') != 'none');
};
/** Change class name
* @param {String} className
*/
ol.control.Overlay.prototype.setClass = function (className)
{ var vis = $(this.element).hasClass("ol-visible");
$(this.element).removeClass().addClass('ol-unselectable ol-overlay'+(vis?" ol-visible ":" ")+className);
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* OpenLayers 3 Layer Overview Control.
* The overview can rotate with map.
* Zoom levels are configurable.
* Click on the overview will center the map.
* Change width/height of the overview trough css.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options Control options.
* @param {ol.ProjectionLike} options.projection The projection. Default is EPSG:3857 (Spherical Mercator).
* @param {Number} options.minZoom default 0
* @param {Number} options.maxZoom default 18
* @param {boolean} options.rotation enable rotation, default false
* @param {top|bottom-left|right} options.align position
* @param {Array<ol.layer>} options.layers list of layers
* @param {ol.style.Style | Array.<ol.style.Style> | undefined} options.style style to draw the map extent on the overveiw
* @param {bool|elastic} options.panAnimation use animation to center map on click, default true
*/
ol.control.Overview = function(options)
{ options = options || {};
var self = this;
// API
this.minZoom = options.minZoom || 0;
this.maxZoom = options.maxZoom || 18;
this.rotation = options.rotation;
var element;
if (options.target)
{ element = $("<div>");
this.panel_ = $(options.target);
}
else
{ element = $("<div>").addClass('ol-overview ol-unselectable ol-control ol-collapsed');
if (/top/.test(options.align)) element.addClass('ol-control-top');
if (/right/.test(options.align)) element.addClass('ol-control-right');
$("<button>").on("touchstart", function(e){ self.toggleMap(); e.preventDefault(); })
.attr('type','button')
.click (function(){self.toggleMap()})
.appendTo(element);
this.panel_ = $("<div>").addClass("panel")
.appendTo(element);
}
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
// Create a overview map
this.ovmap_ = new ol.Map(
{ controls: new ol.Collection(),
interactions: new ol.Collection(),
target: this.panel_.get(0),
view: new ol.View
({ zoom: 14,
center: [270148, 6247782],
projection: options.projection
}),
layers: options.layers
});
this.oview_ = this.ovmap_.getView();
// Cache extent
this.extentLayer = new ol.layer.Vector(
{ name: 'Cache extent',
source: new ol.source.Vector(),
style: options.style || [new ol.style.Style(
{ image: new ol.style.Circle(
{ fill: new ol.style.Fill({
color: 'rgba(255,0,0, 1)'
}),
stroke: new ol.style.Stroke(
{ width: 7,
color: 'rgba(255,0,0, 0.8)'
}),
radius: 5
}),
stroke: new ol.style.Stroke(
{ width: 5,
color: "rgba(255,0,0,0.8)"
})
}
)]
})
this.ovmap_.addLayer(this.extentLayer);
/** Elastic bounce
* @param {Int} bounce number of bounce
* @param {Number} amplitude amplitude of the bounce [0,1]
* @return {Number}
*/
ol.easing.bounceFn = function (bounce, amplitude)
{ var a = (2*bounce+1) * Math.PI/2;
var b = amplitude>0 ? -1/amplitude : -100;
var c = - Math.cos(a) * Math.pow(2, b);
return function(t)
{ t = 1-Math.cos(t*Math.PI/2);
return 1 + Math.abs( Math.cos(a*t) ) * Math.pow(2, b*t) + c*t;
}
}
/** Elastic bounce
* @param {Int} bounce number of bounce
* @param {Number} amplitude amplitude of the bounce [0,1]
* @return {Number}
*/
ol.easing.elasticFn = function (bounce, amplitude)
{ var a = 3*bounce * Math.PI/2;
var b = amplitude>0 ? -1/amplitude : -100;
var c = Math.cos(a) * Math.pow(2, b);
return function(t)
{ t = 1-Math.cos(t*Math.PI/2);
return 1 - Math.cos(a*t) * Math.pow(2, b*t) + c*t;
}
}
// Click on the preview center the map
this.ovmap_.addInteraction (new ol.interaction.Pointer({
handleDownEvent: function(evt) {
if (options.panAnimation !==false) {
if (options.panAnimation=="elastic" || options.elasticPan) {
self.getMap().getView().animate({
center: evt.coordinate,
easing: ol.easing.elasticFn(2,0.3),
duration: 1000
});
} else {
self.getMap().getView().animate({
center: evt.coordinate,
duration: 300
});
}
}
else self.getMap().getView().setCenter(evt.coordinate);
return false;
}
}));
};
ol.inherits(ol.control.Overview, ol.control.Control);
/** Get overview map
* @return {ol.Map}
*/
ol.control.Overview.prototype.getOverviewMap = function()
{ return this.ovmap_;
};
/** Toggle overview map
*/
ol.control.Overview.prototype.toggleMap = function()
{ $(this.element).toggleClass("ol-collapsed");
this.ovmap_.updateSize();
};
/** Set overview map position
* @param {top|bottom-left|right}
*/
ol.control.Overview.prototype.setPosition = function(align)
{ if (/top/.test(align)) $(this.element).addClass("ol-control-top");
else $(this.element).removeClass("ol-control-top");
if (/right/.test(align)) $(this.element).addClass("ol-control-right");
else $(this.element).removeClass("ol-control-right");
};
/**
* Set the map instance the control associated with.
* @param {ol.Map} map The map instance.
*/
ol.control.Overview.prototype.setMap = function(map) {
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (map)
{ this._listener = map.getView().on('propertychange', this.setView.bind(this));
this.setView();
}
};
/** Calculate the extent of the map and draw it on the overview
*/
ol.control.Overview.prototype.calcExtent_ = function(extent)
{ var map = this.getMap();
if (!map) return;
var source = this.extentLayer.getSource();
source.clear();
var f = new ol.Feature();
var size = map.getSize();
var resolution = map.getView().getResolution();
var rotation = map.getView().getRotation();
var center = map.getView().getCenter();
if (!resolution) return;
var dx = resolution * size[0] / 2;
var dy = resolution * size[1] / 2;
var res2 = this.oview_.getResolution();
if (dx/res2>5 || dy/res2>5)
{ var cos = Math.cos(rotation);
var sin = Math.sin(rotation);
var i, x, y;
extent=[[-dx,-dy],[-dx,dy],[dx,dy],[dx,-dy]];
for (i = 0; i < 4; ++i)
{ x = extent[i][0];
y = extent[i][1];
extent[i][0] = center[0] + x * cos - y * sin;
extent[i][1] = center[1] + x * sin + y * cos;
}
f.setGeometry (new ol.geom.Polygon( [ extent ]));
}
else
{ f.setGeometry (new ol.geom.Point( center ));
}
source.addFeature(f);
};
/**
* @private
*/
ol.control.Overview.prototype.setView = function(e)
{ if (!e)
{ // refresh all
this.setView({key:'rotation'});
this.setView({key:'resolution'});
this.setView({key:'center'});
return;
}
// Set the view params
switch (e.key)
{ case 'rotation':
if (this.rotation) this.oview_.setRotation(this.getMap().getView().getRotation());
else if (this.oview_.getRotation()) this.oview_.setRotation(0);
break;
case 'center':
{ var mapExtent = this.getMap().getView().calculateExtent(this.getMap().getSize());
var extent = this.oview_.calculateExtent(this.ovmap_.getSize());
if (mapExtent[0]<extent[0] || mapExtent[1]<extent[1]
|| mapExtent[2]>extent[2] || mapExtent[3]>extent[3])
{ this.oview_.setCenter(this.getMap().getView().getCenter());
}
break;
}
case 'resolution':
{ var z = Math.round(this.getMap().getView().getZoom()/2)*2-4;
z = Math.min ( this.maxZoom, Math.max(this.minZoom, z) );
this.oview_.setZoom(z);
break;
}
default: break;
}
this.calcExtent_();
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Permalink Control.
*
* Add a `permalink`property to layers to be handled by the control (and added in the url).
* The layer's permalink property is used to name the layer in the url.
* The control must be added after all layer are inserted in the map to take them into acount.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} options
* @param {bool} options.urlReplace replace url or not, default true
* @param {integer} options.fixed number of digit in coords, default 6
* @param {bool} options.anchor use "#" instead of "?" in href
* @param {function} options.onclick a function called when control is clicked
*/
ol.control.Permalink = function(opt_options)
{ var options = opt_options || {};
var self = this;
var button = document.createElement('button');
this.replaceState_ = (options.urlReplace!==false);
this.fixed_ = options.fixed || 6;
this.hash_ = options.anchor ? "#" : "?";
function linkto()
{ if (typeof(options.onclick) == 'function') options.onclick(self.getLink());
else self.setUrlReplace(!self.replaceState_);
}
button.addEventListener('click', linkto, false);
button.addEventListener('touchstart', linkto, false);
var element = document.createElement('div');
element.className = (options.className || "ol-permalink") + " ol-unselectable ol-control";
element.appendChild(button);
ol.control.Control.call(this,
{ element: element,
target: options.target
});
this.on ('change', this.viewChange_.bind(this));
// Save search params
this.search_ = {};
var hash = document.location.hash || document.location.search;
if (hash)
{ hash = hash.replace(/(^#|^\?)/,"").split("&");
for (var i=0; i<hash.length; i++)
{ var t = hash[i].split("=");
switch(t[0])
{ case 'lon':
case 'lat':
case 'z':
case 'r':
case 'l': break;
default: this.search_[t[0]] = t[1];
}
}
}
// Decode permalink
this.setPosition();
};
ol.inherits(ol.control.Permalink, ol.control.Control);
/**
* Set the map instance the control associated with.
* @param {ol.Map} map The map instance.
*/
ol.control.Permalink.prototype.setMap = function(map) {
if (this._listener) {
ol.Observable.unByKey(this._listener.change);
ol.Observable.unByKey(this._listener.moveend);
}
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
// Get change
if (map)
{ this._listener = {
change: map.getLayerGroup().on('change', this.layerChange_.bind(this)),
moveend: map.on('moveend', this.viewChange_.bind(this))
};
this.setPosition();
}
};
/** Get layer given a permalink name (permalink propertie in the layer)
* @param {string} the permalink to search for
* @param {Array<ol.layer>|undefined} an array of layer to search in
* @return {ol.layer|false}
*/
ol.control.Permalink.prototype.getLayerByLink = function (id, layers)
{ if (!layers && this.getMap()) layers = this.getMap().getLayers().getArray();
for (var i=0; i<layers.length; i++)
{ if (layers[i].get('permalink') == id) return layers[i];
// Layer Group
if (layers[i].getLayers)
{ var li = this.getLayerByLink ( id, layers[i].getLayers().getArray() );
if (li) return li;
}
}
return false;
}
/** Set map position according to the current link
*/
ol.control.Permalink.prototype.setPosition = function()
{ var map = this.getMap();
if (!map) return;
var hash = document.location.hash || document.location.search;
if (!hash) return;
var param = {};
hash = hash.replace(/(^#|^\?)/,"").split("&");
for (var i=0; i<hash.length; i++)
{ var t = hash[i].split("=");
param[t[0]] = t[1];
}
var c = ol.proj.transform([Number(param.lon),Number(param.lat)], 'EPSG:4326', map.getView().getProjection());
if (c[0] && c[1]) map.getView().setCenter(c);
if (param.z) map.getView().setZoom(Number(param.z));
if (param.r) map.getView().setRotation(Number(param.r));
if (param.l)
{ var l = param.l.split("|");
// Reset layers
function resetLayers(layers)
{ if (!layers) layers = map.getLayers().getArray();
for (var i=0; i<layers.length; i++)
{ if (layers[i].get('permalink'))
{ layers[i].setVisible(false);
// console.log("hide "+layers[i].get('permalink'));
}
if (layers[i].getLayers)
{ resetLayers (layers[i].getLayers().getArray());
}
}
}
resetLayers();
for (var i=0; i<l.length; i++)
{ var t = l[i].split(":");
var li = this.getLayerByLink(t[0]);
var op = Number(t[1]);
if (li)
{ li.setOpacity(op);
li.setVisible(true);
}
}
}
}
/**
* Get the parameters added to the url. The object can be changed to add new values.
* @return {Object} a key value object added to the url as &key=value
* @api stable
*/
ol.control.Permalink.prototype.getUrlParams = function()
{ return this.search_;
}
/**
* Get the permalink
* @return {permalink}
*/
ol.control.Permalink.prototype.getLink = function()
{ var map = this.getMap();
var c = ol.proj.transform(map.getView().getCenter(), map.getView().getProjection(), 'EPSG:4326');
var z = map.getView().getZoom();
var r = map.getView().getRotation();
var l = this.layerStr_;
// Change anchor
var anchor = "lon="+c[0].toFixed(this.fixed_)+"&lat="+c[1].toFixed(this.fixed_)+"&z="+z+(r?"&r="+(Math.round(r*10000)/10000):"")+(l?"&l="+l:"");
for (var i in this.search_) anchor += "&"+i+"="+this.search_[i];
//return document.location.origin+document.location.pathname+this.hash_+anchor;
return document.location.protocol+"//"+document.location.host+document.location.pathname+this.hash_+anchor;
}
/**
* Enable / disable url replacement (replaceSate)
* @param {bool}
*/
ol.control.Permalink.prototype.setUrlReplace = function(replace)
{ try{
this.replaceState_ = replace;
if (!replace)
{ var s = "";
for (var i in this.search_)
{ s += (s==""?"?":"&") + i+"="+this.search_[i];
}
window.history.replaceState (null,null, document.location.origin+document.location.pathname+s);
}
else window.history.replaceState (null,null, this.getLink());
}catch(e){}
}
/**
* On view change refresh link
* @param {ol.event} The map instance.
* @private
*/
ol.control.Permalink.prototype.viewChange_ = function()
{ try{
if (this.replaceState_) window.history.replaceState (null,null, this.getLink());
}catch(e){}
}
/**
* Layer change refresh link
* @param {ol.event} The map instance.
* @private
*/
ol.control.Permalink.prototype.layerChange_ = function(e)
{ // Get layers
var l = "";
function getLayers(layers)
{ for (var i=0; i<layers.length; i++)
{ if (layers[i].getVisible() && layers[i].get("permalink"))
{ if (l) l += "|";
l += layers[i].get("permalink")+":"+layers[i].get("opacity");
}
// Layer Group
if (layers[i].getLayers) getLayers(layers[i].getLayers().getArray());
}
}
getLayers(this.getMap().getLayers().getArray());
this.layerStr_ = l;
this.viewChange_();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc OpenLayers 3 Profil Control.
* Draw a profil of a feature (with a 3D geometry)
*
* @constructor
* @extends {ol.control.Control}
* @fires over, out, show
* @param {Object=} _ol_control_ opt_options.
*
*/
ol.control.Profil = function(opt_options)
{ var options = opt_options || {};
this.info = options.info || ol.control.Profil.prototype.info;
var self = this;
var element;
if (options.target)
{ element = $("<div>").addClass(options.className || "ol-profil");
}
else
{ element = $("<div>").addClass((options.className || 'ol-profil') +' ol-unselectable ol-control ol-collapsed');
this.button = $("<button>")
.attr('type','button')
.on("click touchstart", function(e)
{ self.toggle();
e.preventDefault();
})
.appendTo(element);
}
var div = $("<div>").addClass("ol-inner").appendTo(element);
div = $("<div>").css("position","relative").appendTo(div);
var ratio = this.ratio = 2;
this.canvas_ = document.createElement('canvas');
this.canvas_.width = (options.width || 300)*ratio;
this.canvas_.height = (options.height || 150)*ratio;
$(this.canvas_).css({
"transform":"scale(0.5,0.5)", "transform-origin":"0 0",
"-ms-transform":"scale(0.5,0.5)", "-ms-transform-origin":"0 0",
"-webkit-transform":"scale(0.5,0.5)", "-webkit-transform-origin":"0 0",
"transform":"scale(0.5,0.5)", "transform-origin":"0 0"
});
$("<div>").appendTo(div)
.width (this.canvas_.width/ratio)
.height (this.canvas_.height/ratio)
.append(this.canvas_)
.on("click mousemove", function(e){ self.onMove(e); });
ol.control.Control.call(this,
{ element: element.get(0),
target: options.target
});
// Offset in px
this.margin_ = { top:10*ratio, left:40*ratio, bottom:30*ratio, right:10*ratio };
if (!this.info.ytitle) this.margin_.left -= 20*ratio;
if (!this.info.xtitle) this.margin_.bottom -= 20*ratio;
// Cursor
this.bar_ = $("<div>").addClass("ol-profilbar")
.css({top:(this.margin_.top/ratio)+"px", height:(this.canvas_.height-this.margin_.top-this.margin_.bottom)/ratio+"px" })
.appendTo(div);
this.cursor_ = $("<div>").addClass("ol-profilcursor")
.appendTo(div);
this.popup_ = $("<div>").addClass("ol-profilpopup")
.appendTo(this.cursor_);
// Track information
var t = $("<table cellpadding='0' cellspacing='0'>").appendTo(div).width(this.canvas_.width/ratio);
var tr = $("<tr>").addClass("track-info").appendTo(t);
$("<td>").html((this.info.zmin||"Zmin")+': <span class="zmin">').appendTo(tr);
$("<td>").html((this.info.zmax||"Zmax")+': <span class="zmax">').appendTo(tr);
$("<td>").html((this.info.distance||"Distance")+': <span class="dist">').appendTo(tr);
$("<td>").html((this.info.time||"Time")+': <span class="time">').appendTo(tr);
tr = $("<tr>").addClass("point-info").appendTo(t);
$("<td>").html((this.info.altitude||"Altitude")+': <span class="z">').appendTo(tr);
$("<td>").html((this.info.distance||"Distance")+': <span class="dist">').appendTo(tr);
$("<td>").html((this.info.time||"Time")+': <span class="time">').appendTo(tr);
// Array of data
this.tab_ = [];
// Show feature
if (options.feature)
{ this.setGeometry (options.feature);
}
};
ol.inherits(ol.control.Profil, ol.control.Control);
/** Custom infos list
* @api stable
*/
ol.control.Profil.prototype.info =
{ "zmin": "Zmin",
"zmax": "Zmax",
"ytitle": "Altitude (m)",
"xtitle": "Distance (km)",
"time": "Time",
"altitude": "Altitude",
"distance": "Distance"
};
/** Show popup info
* @param {string} info to display as a popup
* @api stable
*/
ol.control.Profil.prototype.popup = function(info)
{ this.popup_.html(info);
}
/** Mouse move over canvas
*/
ol.control.Profil.prototype.onMove = function(e)
{ if (!this.tab_.length) return;
var pos = $(this.canvas_).offset();
var dx = e.pageX -pos.left;
var dy = e.pageY -pos.top;
var ratio = this.ratio;
if (dx>this.margin_.left/ratio && dx<(this.canvas_.width-this.margin_.right)/ratio
&& dy>this.margin_.top/ratio && dy<(this.canvas_.height-this.margin_.bottom)/ratio)
{ this.bar_.css("left", dx+"px").show();
var d = (dx*ratio-this.margin_.left)/this.scale_[0];
var p0 = this.tab_[0];
for (var i=1, p; p=this.tab_[i]; i++)
{ if (p[0]>=d)
{ if (d < (p[0]+p0[0])/2) p = p0;
break;
}
}
if (p) this.cursor_.css({
left:dx+"px",
top:(this.canvas_.height-this.margin_.bottom+p[1]*this.scale_[1]+this.dy_)/ratio+"px"
}).show();
else this.cursor_.hide();
this.bar_.parent().addClass("over");
$(".point-info .z", this.element).text(p[1]+"m");
$(".point-info .dist", this.element).text((p[0]/1000).toFixed(1)+"km");
$(".point-info .time", this.element).text(p[2]);
if (dx>this.canvas_.width/ratio/2) this.popup_.addClass('ol-left');
else this.popup_.removeClass('ol-left');
this.dispatchEvent({ type:'over', click:e.type=="click", coord: p[3], time: p[2], distance: p[0] });
}
else
{ if (this.bar_.parent().hasClass("over"))
{ this.bar_.hide();
this.cursor_.hide();
this.bar_.parent().removeClass("over");
this.dispatchEvent({ type:'out' });
}
}
}
/** Show panel
* @api stable
*/
ol.control.Profil.prototype.show = function()
{ $(this.element).removeClass("ol-collapsed");
this.dispatchEvent({ type:'show', show: true });
}
/** Hide panel
* @api stable
*/
ol.control.Profil.prototype.hide = function()
{ $(this.element).addClass("ol-collapsed");
this.dispatchEvent({ type:'show', show: false });
}
/** Toggle panel
* @api stable
*/
ol.control.Profil.prototype.toggle = function()
{ var b = $(this.element).toggleClass("ol-collapsed").hasClass("ol-collapsed");
this.dispatchEvent({ type:'show', show: !b });
}
/** Is panel visible
*/
ol.control.Profil.prototype.isShown = function()
{ return (!$(this.element).hasClass("ol-collapsed"));
}
/**
* Set the geometry to draw the profil.
* @param {ol.Feature|ol.geom} f the feature.
* @param {Object=} options
* - projection {ol.ProjectionLike} feature projection, default projection of the map
* - zunit {m|km} default m
* - unit {m|km} default km
* - zmin {Number|undefined} default 0
* - zmax {Number|undefined} default max Z of the feature
* - graduation {Number|undefined} z graduation default 100
* - amplitude {integer|undefined} amplitude of the altitude, default zmax-zmin
* @api stable
*/
ol.control.Profil.prototype.setGeometry = function(g, options)
{ if (!options) options = {};
if (g instanceof ol.Feature) g = g.getGeometry();
var canvas = this.canvas_;
var ctx = canvas.getContext('2d');
var w = canvas.width;
var h = canvas.height;
ctx.setTransform(1, 0, 0, 1, 0, 0);
ctx.clearRect(0,0, w, h);
// No Z
if (!/Z/.test(g.getLayout())) return;
// No time
if(/M/.test(g.getLayout())) $(".time", this.element).parent().show();
else $(".time", this.element).parent().hide();
// Coords
var c = g.getCoordinates();
switch (g.getType())
{ case "LineString": break;
case "MultiLineString": c = c[0]; break;
default: return;
}
// Distance beetween 2 coords
var wgs84Sphere = new ol.Sphere(6378137);
var proj = options.projection || this.getMap().getView().getProjection();
function dist2d(p1,p2)
{ return wgs84Sphere.haversineDistance(
ol.proj.transform(p1, proj, 'EPSG:4326'),
ol.proj.transform(p2, proj, 'EPSG:4326'));
}
function getTime(t0, t1)
{ if (!t0 || !t1) return "-"
var dt = (t1-t0) / 60; // mn
var ti = Math.trunc(dt/60);
var mn = Math.trunc(dt-ti*60);
return ti+"h"+(mn<10?"0":"")+mn+"mn";
}
// Margin
ctx.setTransform(1, 0, 0, 1, this.margin_.left, h-this.margin_.bottom);
var ratio = this.ratio;
w -= this.margin_.right + this.margin_.left;
h -= this.margin_.top + this.margin_.bottom;
// Draw axes
ctx.strokeStyle = "#000";
ctx.lineWidth = 0.5*ratio;
ctx.beginPath();
ctx.moveTo(0,0); ctx.lineTo(0,-h);
ctx.moveTo(0,0); ctx.lineTo(w, 0);
ctx.stroke();
//
var zmin=Infinity, zmax=-Infinity;
var d, z, ti, t = this.tab_ = [];
for (var i=0, p; p=c[i]; i++)
{ z = p[2];
if (z<zmin) zmin=z;
if (z>zmax) zmax=z;
if (i==0) d = 0;
else d += dist2d(c[i-1], p);
ti = getTime(c[0][3],p[3]);
t.push ([d, z, ti, p]);
}
// Info
$(".track-info .zmin", this.element).text(zmin.toFixed(2)+"m");
$(".track-info .zmax", this.element).text(zmax.toFixed(2)+"m");
if (d>1000)
{ $(".track-info .dist", this.element).text((d/1000).toFixed(1)+"km");
}
else
{ $(".track-info .dist", this.element).text((d).toFixed(1)+"m");
}
$(".track-info .time", this.element).text(ti);
// Set graduation
var grad = options.graduation || 100;
while (true)
{ zmax = Math.ceil(zmax/grad)*grad;
zmin = Math.floor(zmin/grad)*grad;
var nbgrad = (zmax-zmin)/grad;
if (h/nbgrad < 15*ratio)
{ grad *= 2;
}
else break;
}
// Set amplitude
if (typeof(options.zmin)=='number' && zmin > options.zmin) zmin = options.zmin;
if (typeof(options.zmax)=='number' && zmax < options.zmax) zmax = options.zmax;
var amplitude = options.amplitude;
if (amplitude)
{ zmax = Math.max (zmin + amplitude, zmax);
}
// Scales lines
var scx = w/d;
var scy = -h/(zmax-zmin);
var dy = this.dy_ = -zmin*scy;
this.scale_ = [scx,scy];
// Draw
ctx.font = (10*ratio)+"px arial";
ctx.textAlign = "right";
ctx.textBaseline = "middle";
ctx.fillStyle="#000";
// Scale Z
ctx.beginPath();
for (var i=zmin; i<=zmax; i+=grad)
{ if (options.zunit!="km") ctx.fillText(i, -4*ratio, i*scy+dy);
else ctx.fillText((i/1000).toFixed(1), -4*ratio, i*scy+dy);
ctx.moveTo (-2*ratio, i*scy+dy);
if (i!=0) ctx.lineTo (d*scx, i*scy+dy);
else ctx.lineTo (0, i*scy+dy);
}
// Scale X
ctx.textAlign = "center";
ctx.textBaseline = "top";
ctx.setLineDash([ratio,3*ratio]);
var unit = options.unit ||"km";
var step;
if (d>1000)
{ step = Math.round(d/1000)*100;
if (step > 1000) step = Math.ceil(step/1000)*1000;
}
else
{ unit = "m";
if (d>100) step = Math.round(d/100)*10;
else if (d>10) step = Math.round(d/10);
else if (d>1) step = Math.round(d)/10;
else step = d;
}
for (var i=0; i<=d; i+=step)
{ var txt = (unit=="m") ? i : (i/1000);
//if (i+step>d) txt += " "+ (options.zunits || "km");
ctx.fillText(Math.round(txt*10)/10, i*scx, 4*ratio);
ctx.moveTo (i*scx, 2*ratio); ctx.lineTo (i*scx, 0);
}
ctx.font = (12*ratio)+"px arial";
ctx.fillText(this.info.xtitle.replace("(km)","("+unit+")"), w/2, 18*ratio);
ctx.save();
ctx.rotate(-Math.PI/2);
ctx.fillText(this.info.ytitle, h/2, -this.margin_.left);
ctx.restore();
ctx.stroke();
//
ctx.strokeStyle = "#369";
ctx.lineWidth = 1;
ctx.setLineDash([]);
ctx.beginPath();
for (var i=0, p; p=t[i]; i++)
{ if (i==0) ctx.moveTo(p[0]*scx,p[1]*scy+dy);
else ctx.lineTo(p[0]*scx,p[1]*scy+dy);
}
ctx.stroke();
};
/** Get profil image
* @param {string|undefined} type image format or 'canvas' to get the canvas image, default image/png.
* @param {Number|undefined} encoderOptions between 0 and 1 indicating image quality image/jpeg or image/webp, default 0.92.
* @return {string} requested data uri
* @api stable
*/
ol.control.Profil.prototype.getImage = function(type, encoderOptions)
{ if (type==="canvas") return this.canvas_;
return this.canvas_.toDataURL(type, encoderOptions);
}
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search places using the French National Base Address (BAN) API.
*
* @constructor
* @extends {ol.control.Search}
* @fires select
* @param {Object=} Control options.
* @param {string} options.className control class name
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 500.
* @param {integer | undefined} options.minLength minimum length to start searching, default 3
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
*
* @param {string|undefined} options.url Url to BAN api, default "https://api-adresse.data.gouv.fr/search/"
* @param {boolean} options.position Search, with priority to geo position, default false
* @param {function} options.getTitle a function that takes a feature and return the text to display in the menu, default return label attribute
* @see {@link https://adresse.data.gouv.fr/api/}
*/
ol.control.SearchBAN = function(options)
{ options = options || {};
options.typing = options.typing || 500;
options.url = options.url || "https://api-adresse.data.gouv.fr/search/";
options.className = options.className || 'BAN';
ol.control.SearchPhoton.call(this, options);
this.set("copy","<a href='https://adresse.data.gouv.fr/' target='new'>&copy; BAN-data.gouv.fr</a>");
};
ol.inherits(ol.control.SearchBAN, ol.control.SearchPhoton);
/** Returns the text to be displayed in the menu
* @param {ol.Feature} f the feature
* @return {string} the text to be displayed in the index
* @api
*/
ol.control.SearchBAN.prototype.getTitle = function (f) {
var p = f.properties;
return (p.label);
};
/** A ligne has been clicked in the menu > dispatch event
* @param {any} f the feature, as passed in the autocomplete
* @api
*/
ol.control.SearchBAN.prototype.select = function (f){
var c = f.geometry.coordinates;
// Add coordinate to the event
try {
c = ol.proj.transform (f.geometry.coordinates, 'EPSG:4326', this.getMap().getView().getProjection());
} catch(e) {};
this.dispatchEvent({ type:"select", search:f, coordinate: c });
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search features.
*
* @constructor
* @extends {ol.control.Search}
* @fires select
* @param {Object=} Control options.
* @param {string} options.className control class name
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 300.
* @param {integer | undefined} options.minLength minimum length to start searching, default 1
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
*
* @param {string | undefined} options.property a property to display in the index, default 'name'.
* @param {function} options.getTitle a function that takes a feature and return the name to display in the index, default return the property
* @param {function | undefined} options.getSearchString a function that take a feature and return a text to be used as search string, default geTitle() is used as search string
*/
ol.control.SearchFeature = function(options) {
if (!options) options = {};
options.className = options.className || 'feature';
ol.control.Search.call(this, options);
if (typeof(options.getSearchString)=="function") this.getSearchString = options.getSearchString;
this.set('property', options.property || 'name');
this.source_ = options.source;
};
ol.inherits(ol.control.SearchFeature, ol.control.Search);
/** Returns the text to be displayed in the menu
* @param {ol.Feature} f the feature
* @return {string} the text to be displayed in the index
* @api
*/
ol.control.SearchFeature.prototype.getTitle = function (f) {
return f.get(this.get('property')||'name');
};
/** Return the string to search in
* @param {ol.Feature} f the feature
* @return {string} the text to be used as search string
* @api
*/
ol.control.SearchFeature.prototype.getSearchString = function (f) {
return this.getTitle(f);
};
/** Get the source
* @return {ol.source.Vector}
* @api
*/
ol.control.SearchFeature.prototype.getSource = function () {
return this.source_;
}
/** Get the source
* @param {ol.source.Vector} source
* @api
*/
ol.control.SearchFeature.prototype.setSource = function (source) {
this.source_ = source;
};
/** Autocomplete function
* @param {string} s search string
* @param {int} max max
* @param {function} cback a callback function that takes an array to display in the autocomplete field (for asynchronous search)
* @return {Array<any>|false} an array of search solutions or false if the array is send with the cback argument (asnchronous)
* @api
*/
ol.control.SearchFeature.prototype.autocomplete = function (s) {
var result = [];
if (this.source_) {
// regexp
s = s.replace(/^\*/,'');
var rex = new RegExp(s, 'i');
// The source
var features = this.source_.getFeatures();
var max = this.get('maxItems')
for (var i=0, f; f=features[i]; i++) {
var att = this.getSearchString(f);
if (att !== undefined && rex.test(att)) {
result.push(f);
if ((--max)<=0) break;
}
}
}
return result;
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search places using the French National Base Address (BAN) API.
*
* @constructor
* @extends {ol.control.SearchJSON}
* @fires select
* @param {any} options extend ol.control.SearchJSON options
* @param {string} options.className control class name
* @param {boolean | undefined} options.apiKey the service api key.
* @param {string | undefined} options.authentication: basic authentication for the service API as btoa("login:pwd")
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 500.
* @param {integer | undefined} options.minLength minimum length to start searching, default 3
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
*
* @param {Number} options.pageSize item per page for parcelle list paging, use -1 for no paging, default 5
* @see {@link https://geoservices.ign.fr/documentation/geoservices/geocodage.html}
*/
ol.control.SearchGeoportailParcelle = function(options) {
var self = this;
options.type = "Commune";
options.className = options.className ? options.className+" IGNF-parcelle" : "IGNF-parcelle";
options.inputLabel = "Commune";
options.noCollapse = true;
options.placeholder = options.placeholder || "Choisissez une commune...";
ol.control.SearchGeoportail.call(this, options);
this.set('copy', null);
var element = this.element;
// Add parcel form
var div = document.createElement("DIV");
element.appendChild(div);
var label = document.createElement("LABEL");
label.innerText = 'Préfixe'
div.appendChild(label);
var label = document.createElement("LABEL");
label.innerText = 'Section'
div.appendChild(label);
var label = document.createElement("LABEL");
label.innerText = 'Numéro'
div.appendChild(label);
div.appendChild(document.createElement("BR"));
// Input
this._inputParcelle = {
prefix: document.createElement("INPUT"),
section: document.createElement("INPUT"),
numero: document.createElement("INPUT")
};
this._inputParcelle.prefix.setAttribute('maxlength',3);
this._inputParcelle.section.setAttribute('maxlength',2);
this._inputParcelle.numero.setAttribute('maxlength',4);
// Delay search
var tout;
var doSearch = function(e) {
if (tout) clearTimeout(tout);
tout = setTimeout(function() {
self.autocompleteParcelle();
}, options.typing || 0);
}
// Add inputs
for (var i in this._inputParcelle) {
div.appendChild(this._inputParcelle[i]);
this._inputParcelle[i].addEventListener("keyup", doSearch);
}
this.activateParcelle(false);
// Autocomplete list
var ul = document.createElement('UL');
ul.classList.add('autocomplete-parcelle');
element.appendChild(ul);
ul = document.createElement('UL');
ul.classList.add('autocomplete-page');
element.appendChild(ul);
// Show/hide list on fcus/blur
this._input.addEventListener('blur', function() {
setTimeout(function(){ element.classList.add('ol-collapsed-list') }, 200);
});
this._input.addEventListener('focus', function() {
element.classList.remove('ol-collapsed-list');
self._listParcelle([]);
if (self._commune) {
self._commune = null;
self._input.value = '';
self.drawList_();
}
self.activateParcelle(false);
});
this.on('select', this.selectCommune.bind(this));
this.set('pageSize', options.pageSize || 5);
};
ol.inherits(ol.control.SearchGeoportailParcelle, ol.control.SearchGeoportail);
/** Select a commune => start searching parcelle
* @param {any} e
* @private
*/
ol.control.SearchGeoportailParcelle.prototype.selectCommune = function(e) {
this._commune = e.search.insee;
this._input.value = e.search.insee + ' - ' + e.search.fulltext;
this.activateParcelle(true);
this._inputParcelle.numero.focus();
this.autocompleteParcelle();
};
/** Activate parcelle inputs
* @param {bolean} b
*/
ol.control.SearchGeoportailParcelle.prototype.activateParcelle = function(b) {
for (var i in this._inputParcelle) {
this._inputParcelle[i].readOnly = !b;
}
if (b) {
this._inputParcelle.section.parentElement.classList.add('ol-active');
} else {
this._inputParcelle.section.parentElement.classList.remove('ol-active');
}
};
/** Send search request for the parcelle
* @param {any} e
* @private
*/
ol.control.SearchGeoportailParcelle.prototype.autocompleteParcelle = function(e) {
var self = this;
// Add 0 to fit the format
function complete (s, n, c)
{ if (!s) return s;
c = c || "0";
while (s.length < n) s = c+s;
return s.replace(/\*/g,'_');
}
// The selected commune
var commune = this._commune;
var prefix = complete (this._inputParcelle.prefix.value, 3);
if (prefix === '000') {
prefix = '___';
}
// Get parcelle number
var section = complete (this._inputParcelle.section.value, 2);
var numero = complete (this._inputParcelle.numero.value, 4, "0");
var search = commune + (prefix||'___') + (section||"__") + (numero ? numero : section ? "____":"0001");
// Request
var request = '<?xml version="1.0" encoding="UTF-8"?>'
+'<XLS xmlns:xls="http://www.opengis.net/xls" xmlns:gml="http://www.opengis.net/gml" xmlns="http://www.opengis.net/xls" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="1.2" xsi:schemaLocation="http://www.opengis.net/xls http://schemas.opengis.net/ols/1.2/olsAll.xsd">'
+'<RequestHeader/>'
+'<Request requestID="1" version="1.2" methodName="LocationUtilityService">'
+'<GeocodeRequest returnFreeForm="false">'
+'<Address countryCode="CadastralParcel">'
+'<freeFormAddress>'+search+'+</freeFormAddress>'
+'</Address>'
+'</GeocodeRequest>'
+'</Request>'
+'</XLS>'
var url = this.get('url').replace('ols/apis/completion','geoportail/ols?xls=')+encodeURIComponent(request);
// Geocode
this.ajax(url, function(resp) {
// XML to JSON
var parser = new DOMParser();
var xmlDoc = parser.parseFromString(resp.response,"text/xml");
var parcelles = xmlDoc.getElementsByTagName('GeocodedAddress');
var jsonResp = []
for (var i=0, parc; parc= parcelles[i]; i++) {
var node = parc.getElementsByTagName('gml:pos')[0] || parc.getElementsByTagName('pos')[0];
var p = node.childNodes[0].nodeValue.split(' ');
var att = parc.getElementsByTagName('Place');
var json = {
lon: Number(p[1]),
lat: Number(p[0])
};
for (var k=0, a; a=att[k]; k++) {
json[a.attributes.type.value] = a.childNodes[0].nodeValue;
}
jsonResp.push(json);
}
self._listParcelle(jsonResp);
}, function() {
console.log('oops')
});
};
/**
* Draw the autocomplete list
* @param {*} resp
* @private
*/
ol.control.SearchGeoportailParcelle.prototype._listParcelle = function(resp) {
var self = this;
var ul = this.element.querySelector("ul.autocomplete-parcelle");
ul.innerHTML='';
var page = this.element.querySelector("ul.autocomplete-page");
page.innerHTML='';
this._listParc = [];
// Show page i
function showPage(i) {
var l = ul.children;
var visible = "ol-list-"+i;
for (k=0; k<l.length; k++) {
l[k].style.display = (l[k].className===visible) ? '' : 'none';
}
l = page.children;
for (k=0; k<l.length; k++) {
l[k].className = (l[k].innerText==i) ? 'selected' : '';
}
page.style.display = l.length>1 ? '' : 'none';
}
// Sort table
resp.sort(function(a,b) {
var na = a.INSEE+a.CommuneAbsorbee+a.Section+a.Numero;
var nb = b.INSEE+b.CommuneAbsorbee+b.Section+b.Numero;
return na===nb ? 0 : na<nb ? -1 : 1;
});
// Show list
var n = this.get('pageSize');
for (var i=0, r; r = resp[i]; i++) {
var li = document.createElement("LI");
li.setAttribute("data-search", i);
if (n>0) li.classList.add("ol-list-"+Math.floor(i/n));
this._listParc.push(r);
li.addEventListener("click", function(e) {
self._handleParcelle(self._listParc[e.currentTarget.getAttribute("data-search")]);
});
li.innerHTML = r.INSEE+r.CommuneAbsorbee+r.Section+r.Numero;
ul.appendChild(li);
//
if (n>0 && !(i%n)) {
li = document.createElement("LI");
li.innerText = Math.floor(i/n);
li.addEventListener("click", function(e) {
showPage(e.currentTarget.innerText);
});
page.appendChild(li);
}
}
if (n>0) showPage(0);
};
/**
* Handle parcelle section
* @param {*} parc
* @private
*/
ol.control.SearchGeoportailParcelle.prototype._handleParcelle = function(parc) {
this.dispatchEvent({
type:"parcelle",
search: parc,
coordinate: ol.proj.fromLonLat([parc.lon, parc.lat], this.getMap().getView().getProjection())
});
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Search places using the French National Base Address (BAN) API.
*
* @constructor
* @extends {ol.control.Search}
* @fires select
* @param {Object=} Control options.
* @param {string} options.className control class name
* @param {boolean | undefined} options.polygon To get output geometry of results (in geojson format), default false.
* @param {Element | string | undefined} options.target Specify a target if you want the control to be rendered outside of the map's viewport.
* @param {string | undefined} options.label Text label to use for the search button, default "search"
* @param {string | undefined} options.placeholder placeholder, default "Search..."
* @param {number | undefined} options.typing a delay on each typing to start searching (ms), default 500.
* @param {integer | undefined} options.minLength minimum length to start searching, default 3
* @param {integer | undefined} options.maxItems maximum number of items to display in the autocomplete list, default 10
*
* @param {string|undefined} options.url Url to BAN api, default "https://api-adresse.data.gouv.fr/search/"
* @see {@link https://wiki.openstreetmap.org/wiki/Nominatim}
*/
ol.control.SearchNominatim = function(options)
{ options = options || {};
options.className = options.className || 'nominatim';
options.typing = options.typing || 500;
options.url = options.url || "https://nominatim.openstreetmap.org/search";
ol.control.SearchJSON.call(this, options);
this.set("copy","<a href='http://www.openstreetmap.org/copyright' target='new'>&copy; OpenStreetMap contributors</a>");
this.set("polygon", options.polygon);
};
ol.inherits(ol.control.SearchNominatim, ol.control.SearchJSON);
/** Returns the text to be displayed in the menu
* @param {ol.Feature} f the feature
* @return {string} the text to be displayed in the index
* @api
*/
ol.control.SearchNominatim.prototype.getTitle = function (f) {
var title = f.display_name+"<i>"+f.class+" - "+f.type+"</i>";
if (f.icon) title = "<img src='"+f.icon+"' />" + title;
return (title);
};
/**
* @param {string} s the search string
* @return {Object} request data (as key:value)
* @api
*/
ol.control.SearchNominatim.prototype.requestData = function (s) {
return {
format: "json",
addressdetails: 1,
q: s,
polygon_geojson: this.get('polygon') ? 1:0,
limit: this.get('maxItems')
};
};
/** A ligne has been clicked in the menu > dispatch event
* @param {any} f the feature, as passed in the autocomplete
* @api
*/
ol.control.SearchNominatim.prototype.select = function (f){
var c = [Number(f.lon), Number(f.lat)];
// Add coordinate to the event
try {
c = ol.proj.transform (c, 'EPSG:4326', this.getMap().getView().getProjection());
} catch(e) {};
this.dispatchEvent({ type:"select", search:f, coordinate: c });
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc OpenLayers 3 swipe Control.
*
* @constructor
* @extends {ol.control.Control}
* @param {Object=} Control opt_options.
* - layers {ol.layer} layer to swipe
* - rightLayer {ol.layer} layer to swipe on right side
* - className {string} control class name
* - position {number} position propertie of the swipe [0,1], default 0.5
* - orientation {vertical|horizontal} orientation propertie, default vertical
*/
ol.control.Swipe = function(opt_options)
{ var options = opt_options || {};
var self = this;
var button = document.createElement('button');
var element = document.createElement('div');
element.className = (options.className || "ol-swipe") + " ol-unselectable ol-control";
element.appendChild(button);
$(element).on ("mousedown touchstart", this, this.move );
ol.control.Control.call(this,
{ element: element
});
// An array of listener on layer postcompose
this._listener = [];
this.layers = [];
if (options.layers) this.addLayer(options.layers, false);
if (options.rightLayers) this.addLayer(options.rightLayers, true);
this.on('propertychange', function()
{ if (this.getMap()) this.getMap().renderSync();
if (this.get('orientation') === "horizontal")
{ $(this.element).css("top", this.get('position')*100+"%");
$(this.element).css("left", "");
}
else
{ if (this.get('orientation') !== "vertical") this.set('orientation', "vertical");
$(this.element).css("left", this.get('position')*100+"%");
$(this.element).css("top", "");
}
$(this.element).removeClass("horizontal vertical");
$(this.element).addClass(this.get('orientation'));
}.bind(this));
this.set('position', options.position || 0.5);
this.set('orientation', options.orientation || 'vertical');
};
ol.inherits(ol.control.Swipe, ol.control.Control);
/**
* Set the map instance the control associated with.
* @param {_ol_Map_} map The map instance.
*/
ol.control.Swipe.prototype.setMap = function(map)
{
for (var i=0; i<this._listener.length; i++) {
ol.Observable.unByKey(this._listener[i]);
}
this._listener = [];
if (this.getMap()) {
this.getMap().renderSync();
}
ol.control.Control.prototype.setMap.call(this, map);
if (map)
{ this._listener = [];
for (var i=0; i<this.layers.length; i++)
{ var l = this.layers[i];
if (l.right) this._listener.push (l.layer.on('precompose', this.precomposeRight.bind(this)));
else this._listener.push (l.layer.on('precompose', this.precomposeLeft.bind(this)));
this._listener.push(l.layer.on('postcompose', this.postcompose.bind(this)));
}
map.renderSync();
}
};
/** @private
*/
ol.control.Swipe.prototype.isLayer_ = function(layer)
{ for (var k=0; k<this.layers.length; k++)
{ if (this.layers[k].layer === layer) return k;
}
return -1;
};
/** Add a layer to clip
* @param {ol.layer|Array<ol.layer>} layer to clip
* @param {bool} add layer in the right part of the map, default left.
*/
ol.control.Swipe.prototype.addLayer = function(layers, right)
{ if (!(layers instanceof Array)) layers = [layers];
for (var i=0; i<layers.length; i++) {
var l = layers[i];
if (this.isLayer_(l)<0)
{ this.layers.push({ layer:l, right:right });
if (this.getMap())
{ if (right) this._listener.push (l.on('precompose', this.precomposeRight.bind(this)));
else this._listener.push (l.on('precompose', this.precomposeLeft.bind(this)));
this._listener.push(l.on('postcompose', this.postcompose.bind(this)));
this.getMap().renderSync();
}
}
}
};
/** Remove a layer to clip
* @param {ol.layer|Array<ol.layer>} layer to clip
*/
ol.control.Swipe.prototype.removeLayer = function(layers)
{ if (!(layers instanceof Array)) layers = [layers];
for (var i=0; i<layers.length; i++)
{ var k = this.isLayer_(layers[i]);
if (k >=0 && this.getMap())
{ if (this.layers[k].right) layers[i].un('precompose', this.precomposeRight, this);
else layers[i].un('precompose', this.precomposeLeft, this);
layers[i].un('postcompose', this.postcompose, this);
this.layers.splice(k,1);
this.getMap().renderSync();
}
}
};
/** @private
*/
ol.control.Swipe.prototype.move = function(e)
{ var self = e.data;
switch (e.type)
{ case 'touchcancel':
case 'touchend':
case 'mouseup':
{ self.isMoving = false;
$(document).off ("mouseup mousemove touchend touchcancel touchmove", self.move );
break;
}
case 'mousedown':
case 'touchstart':
{ self.isMoving = true;
$(document).on ("mouseup mousemove touchend touchcancel touchmove", self, self.move );
}
case 'mousemove':
case 'touchmove':
{ if (self.isMoving)
{ if (self.get('orientation') === "vertical")
{ var pageX = e.pageX
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageX)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageX);
if (!pageX) break;
pageX -= $(self.getMap().getTargetElement()).offset().left;
var l = self.getMap().getSize()[0];
l = Math.min(Math.max(0, 1-(l-pageX)/l), 1);
self.set('position', l);
}
else
{ var pageY = e.pageY
|| (e.originalEvent.touches && e.originalEvent.touches.length && e.originalEvent.touches[0].pageY)
|| (e.originalEvent.changedTouches && e.originalEvent.changedTouches.length && e.originalEvent.changedTouches[0].pageY);
if (!pageY) break;
pageY -= $(self.getMap().getTargetElement()).offset().top;
var l = self.getMap().getSize()[1];
l = Math.min(Math.max(0, 1-(l-pageY)/l), 1);
self.set('position', l);
}
}
break;
}
default: break;
}
};
/** @private
*/
ol.control.Swipe.prototype.precomposeLeft = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
ctx.save();
ctx.beginPath();
if (this.get('orientation') === "vertical") ctx.rect (0,0, canvas.width*this.get('position'), canvas.height);
else ctx.rect (0,0, canvas.width, canvas.height*this.get('position'));
ctx.clip();
};
/** @private
*/
ol.control.Swipe.prototype.precomposeRight = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
ctx.save();
ctx.beginPath();
if (this.get('orientation') === "vertical") ctx.rect (canvas.width*this.get('position'), 0, canvas.width, canvas.height);
else ctx.rect (0,canvas.height*this.get('position'), canvas.width, canvas.height);
ctx.clip();
};
/** @private
*/
ol.control.Swipe.prototype.postcompose = function(e)
{ e.context.restore();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** ol.control.Target draw a target at the center of the map.
* @constructor
* @param {Object} options
* - style {ol.style.Style|Array<ol.style.Style>} ol.style.Stroke: draw a cross on the map, ol.style.Image: draw the image on the map
* - composite {string} composite operation : difference|multiply|xor|screen|overlay|darken|lighter|lighten|...
*/
ol.control.Target = function(options)
{ options = options || {};
this.style = options.style ||
[ new ol.style.Style({ image: new ol.style.RegularShape ({ points: 4, radius: 11, radius1: 0, radius2: 0, snapToPixel:true, stroke: new ol.style.Stroke({ color: "#fff", width:3 }) }) }),
new ol.style.Style({ image: new ol.style.RegularShape ({ points: 4, radius: 11, radius1: 0, radius2: 0, snapToPixel:true, stroke: new ol.style.Stroke({ color: "#000", width:1 }) }) })
];
if (!(this.style instanceof Array)) this.style = [this.style];
this.composite = options.composite || '';
var div = document.createElement('div');
div.className = "ol-target ol-unselectable ol-control";
ol.control.Control.call(this,
{ element: div,
target: options.target
});
this.setVisible(options.visible!==false);
};
ol.inherits(ol.control.Target, ol.control.Control);
/**
* Remove the control from its current map and attach it to the new map.
* Subclasses may set up event handlers to get notified about changes to
* the map here.
* @param {ol.Map} map Map.
* @api stable
*/
ol.control.Target.prototype.setMap = function (map)
{ if (this.getMap())
{ if (this.getVisible()) this.getMap().renderSync();
}
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.control.Control.prototype.setMap.call(this, map);
if (map)
{ this._listener = map.on('postcompose', this.drawTarget_.bind(this));
}
};
/** Set the control visibility
* @paraam {boolean} b
*/
ol.control.Target.prototype.setVisible = function (b)
{ this.set("visible",b);
if (this.getMap()) this.getMap().renderSync();
};
/** Get the control visibility
* @return {boolean} b
*/
ol.control.Target.prototype.getVisible = function ()
{ return this.get("visible");
};
/** Draw the target
* @private
*/
ol.control.Target.prototype.drawTarget_ = function (e)
{ if (!this.getMap() || !this.getVisible()) return;
var ctx = e.context;
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.scale(ratio,ratio);
var cx = ctx.canvas.width/(2*ratio);
var cy = ctx.canvas.height/(2*ratio);
var geom = new ol.geom.Point (this.getMap().getCoordinateFromPixel([cx,cy]));
if (this.composite) ctx.globalCompositeOperation = this.composite;
for (var i=0; i<this.style.length; i++)
{ var style = this.style[i];
if (style instanceof ol.style.Style)
{ var sc=0;
// OL < v4.3 : setImageStyle don't check retina
var imgs = ol.Map.prototype.getFeaturesAtPixel ? false : style.getImage();
if (imgs)
{ sc = imgs.getScale();
imgs.setScale(ratio*sc);
}
e.vectorContext.setStyle(style);
e.vectorContext.drawGeometry(geom);
if (imgs) imgs.setScale(sc);
}
}
/*
for (var i=0; i<this.style.length; i++)
{ var style = this.style[i];
if (style.stroke instanceof ol.style.Stroke)
{ ctx.lineWidth = style.stroke.getWidth();
ctx.strokeStyle = ol.color.asString(style.stroke.getColor());
var m = style.radius || 10;
var dx = cx + ctx.lineWidth/2;
var dy = cy + ctx.lineWidth/2;
ctx.beginPath();
ctx.moveTo (dx-m, dy);
ctx.lineTo (dx+m, dy);
ctx.moveTo (dx, dy-m);
ctx.lineTo( dx, dy+m);
ctx.stroke();
}
else if (style instanceof ol.style.Image)
{ var img = style.getImage();
ctx.drawImage(img, cx-img.width/2, cy-img.height/2);
}
else if (style instanceof ol.style.Text)
{ ctx.font = style.getFont();
ctx.textBaseline = "middle";
ctx.textAlign = "center";
var fill = style.getFill();
if (fill)
{ ctx.fillStyle = ol.color.asString(fill.getColor());
ctx.fillText(style.getText(), cx, cy);
}
var stroke = style.getStroke();
if (stroke)
{ ctx.lineWidth = stroke.getWidth();
ctx.strokeStyle = ol.color.asString(stroke.getColor());
ctx.strokeText(style.getText(), cx, cy);
}
}
}
*/
ctx.restore();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** A simple push button control drawn as text
* @constructor
* @extends {ol.control.Button}
* @param {Object=} options Control options.
* @param {String} options.className class of the control
* @param {String} options.title title of the control
* @param {String} options.html html to insert in the control
* @param {function} options.handleClick callback when control is clicked (or use change:active event)
*/
ol.control.TextButton = function(options)
{ options = options || {};
options.className = (options.className||"") + " ol-text-button";
ol.control.Button.call(this, options);
};
ol.inherits(ol.control.TextButton, ol.control.Button);
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** A simple toggle control
* The control can be created with an interaction to control its activation.
*
* @constructor
* @extends {ol.control.Control}
* @fires change:active, change:disable
* @param {Object=} options Control options.
* @param {String} options.className class of the control
* @param {String} options.title title of the control
* @param {String} options.html html to insert in the control
* @param {ol.interaction} options.interaction interaction associated with the control
* @param {bool} options.active the control is created active, default false
* @param {bool} options.disable the control is created disabled, default false
* @param {ol.control.Bar} options.bar a subbar associated with the control (drawn when active if control is nested in a ol.control.Bar)
* @param {bool} options.autoActive the control will activate when shown in an ol.control.Bar, default false
* @param {function} options.onToggle callback when control is clicked (or use change:active event)
*/
ol.control.Toggle = function(options)
{ options = options || {};
var self = this;
this.interaction_ = options.interaction;
if (this.interaction_)
{ this.interaction_.on("change:active", function(e)
{ self.setActive(!e.oldValue);
});
}
if (options.toggleFn) options.onToggle = options.toggleFn; // compat old version
options.handleClick = function()
{ self.toggle();
if (options.onToggle) options.onToggle.call(self, self.getActive());
};
options.className = (options.className||"") + " ol-toggle";
ol.control.Button.call(this, options);
this.set("title", options.title);
this.set ("autoActivate", options.autoActivate);
if (options.bar)
{ this.subbar_ = options.bar;
this.subbar_.setTarget(this.element);
$(this.subbar_.element).addClass("ol-option-bar");
}
this.setActive (options.active);
this.setDisable (options.disable);
};
ol.inherits(ol.control.Toggle, ol.control.Button);
/**
* Set the map instance the control is associated with
* and add interaction attached to it to this map.
* @param {_ol_Map_} map The map instance.
*/
ol.control.Toggle.prototype.setMap = function(map)
{ if (!map && this.getMap())
{ if (this.interaction_)
{ this.getMap().removeInteraction (this.interaction_);
}
if (this.subbar_) this.getMap().removeControl (this.subbar_);
}
ol.control.Control.prototype.setMap.call(this, map);
if (map)
{ if (this.interaction_) map.addInteraction (this.interaction_);
if (this.subbar_) map.addControl (this.subbar_);
}
};
/** Get the subbar associated with a control
* @return {ol.control.Bar}
*/
ol.control.Toggle.prototype.getSubBar = function ()
{ return this.subbar_;
};
/**
* Test if the control is disabled.
* @return {bool}.
* @api stable
*/
ol.control.Toggle.prototype.getDisable = function()
{ return $("button", this.element).prop("disabled");
};
/** Disable the control. If disable, the control will be deactivated too.
* @param {bool} b disable (or enable) the control, default false (enable)
*/
ol.control.Toggle.prototype.setDisable = function(b)
{ if (this.getDisable()==b) return;
$("button", this.element).prop("disabled", b);
if (b && this.getActive()) this.setActive(false);
this.dispatchEvent({ type:'change:disable', key:'disable', oldValue:!b, disable:b });
};
/**
* Test if the control is active.
* @return {bool}.
* @api stable
*/
ol.control.Toggle.prototype.getActive = function()
{ return $(this.element).hasClass("ol-active");
};
/** Toggle control state active/deactive
*/
ol.control.Toggle.prototype.toggle = function()
{ if (this.getActive()) this.setActive(false);
else this.setActive(true);
};
/** Change control state
* @param {bool} b activate or deactivate the control, default false
*/
ol.control.Toggle.prototype.setActive = function(b)
{ if (this.getActive()==b) return;
if (b) $(this.element).addClass("ol-active");
else $(this.element).removeClass("ol-active");
if (this.interaction_) this.interaction_.setActive (b);
if (this.subbar_) this.subbar_.setActive(b);
this.dispatchEvent({ type:'change:active', key:'active', oldValue:!b, active:b });
};
/** Set the control interaction
* @param {_ol_interaction_} i interaction to associate with the control
*/
ol.control.Toggle.prototype.setInteraction = function(i)
{ this.interaction_ = i;
};
/** Get the control interaction
* @return {_ol_interaction_} interaction associated with the control
*/
ol.control.Toggle.prototype.getInteraction = function()
{ return this.interaction_;
};
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Feature animation base class
* Use the {@link _ol_Map_#animateFeature} or {@link _ol_layer_Vector_#animateFeature} to animate a feature
* on postcompose in a map or a layer
* @constructor
* @fires animationstart|animationend
* @param {ol.featureAnimationOptions} options
* @param {Number} options.duration duration of the animation in ms, default 1000
* @param {bool} options.revers revers the animation direction
* @param {Number} options.repeat number of time to repeat the animation, default 0
* @param {oo.style.Style} options.hiddenStyle a style to display the feature when playing the animation
* to be used to make the feature selectable when playing animation
* (@see {@link ../examples/map.featureanimation.select.html}), default the feature
* will be hidden when playing (and niot selectable)
* @param {ol.easing.Function} options.fade an easing function used to fade in the feature, default none
* @param {ol.easing.Function} options.easing an easing function for the animation, default ol.easing.linear
*/
ol.featureAnimation = function(options)
{ options = options || {};
this.duration_ = typeof (options.duration)=='number' ? (options.duration>=0 ? options.duration : 0) : 1000;
this.fade_ = typeof(options.fade) == 'function' ? options.fade : null;
this.repeat_ = Number(options.repeat);
var easing = typeof(options.easing) =='function' ? options.easing : ol.easing.linear;
if (options.revers) this.easing_ = function(t) { return (1 - easing(t)); };
else this.easing_ = easing;
this.hiddenStyle = options.hiddenStyle;
ol.Object.call(this);
};
ol.inherits(ol.featureAnimation, ol.Object);
/** Draw a geometry
* @param {olx.animateFeatureEvent} e
* @param {ol.geom} geom geometry for shadow
* @param {ol.geom} shadow geometry for shadow (ie. style with zIndex = -1)
* @private
*/
ol.featureAnimation.prototype.drawGeom_ = function (e, geom, shadow)
{ if (this.fade_)
{ e.context.globalAlpha = this.fade_(1-e.elapsed);
}
var style = e.style;
for (var i=0; i<style.length; i++)
{ var sc=0;
// OL < v4.3 : setImageStyle doesn't check retina
var imgs = ol.Map.prototype.getFeaturesAtPixel ? false : style[i].getImage();
if (imgs)
{ sc = imgs.getScale();
imgs.setScale(e.frameState.pixelRatio*sc);
}
// Prevent crach if the style is not ready (image not loaded)
try{
e.vectorContext.setStyle(style[i]);
if (style[i].getZIndex()<0) e.vectorContext.drawGeometry(shadow||geom);
else e.vectorContext.drawGeometry(geom);
} catch(e) {};
if (imgs) imgs.setScale(sc);
}
};
/** Function to perform manipulations onpostcompose.
* This function is called with an ol.featureAnimationEvent argument.
* The function will be overridden by the child implementation.
* Return true to keep this function for the next frame, false to remove it.
* @param {ol.featureAnimationEvent} e
* @return {bool} true to continue animation.
* @api
*/
ol.featureAnimation.prototype.animate = function (e)
{ return false;
};
/** An animation controler object an object to control animation with start, stop and isPlaying function.
* To be used with {@link olx.Map#animateFeature} or {@link ol.layer.Vector#animateFeature}
* @typedef {Object} ol.animationControler
* @property {function} start - start animation.
* @property {function} stop - stop animation option arguments can be passed in animationend event.
* @property {function} isPlaying - return true if animation is playing.
*/
/** Animate feature on a map
* @function
* @fires animationstart, animationend
* @param {ol.Feature} feature Feature to animate
* @param {ol.featureAnimation|Array<ol.featureAnimation>} fanim the animation to play
* @return {olx.animationControler} an object to control animation with start, stop and isPlaying function
*/
ol.Map.prototype.animateFeature =
/** Animate feature on a vector layer
* @fires animationstart, animationend
* @param {ol.Feature} feature Feature to animate
* @param {ol.featureAnimation|Array<ol.featureAnimation>} fanim the animation to play
* @return {olx.animationControler} an object to control animation with start, stop and isPlaying function
*/
ol.layer.Vector.prototype.animateFeature = function(feature, fanim)
{ var self = this;
var listenerKey;
// Save style
var style = feature.getStyle();
var flashStyle = style || (this.getStyleFunction ? this.getStyleFunction()(feature) : null);
if (!flashStyle) flashStyle=[];
if (!(flashStyle instanceof Array)) flashStyle = [flashStyle];
// Hide feature while animating
feature.setStyle(fanim.hiddenStyle || []);
// Structure pass for animating
var event =
{ // Frame context
vectorContext: null,
frameState: null,
start: 0,
time: 0,
elapsed: 0,
extent: false,
// Feature information
feature: feature,
geom: feature.getGeometry(),
typeGeom: feature.getGeometry().getType(),
bbox: feature.getGeometry().getExtent(),
coord: ol.extent.getCenter(feature.getGeometry().getExtent()),
style: flashStyle
};
if (!(fanim instanceof Array)) fanim = [fanim];
// Remove null animations
for (var i=fanim.length-1; i>=0; i--)
{ if (fanim[i].duration_===0) fanim.splice(i,1);
}
var nb=0, step = 0;
function animate(e)
{ event.vectorContext = e.vectorContext;
event.frameState = e.frameState;
if (!event.extent)
{ event.extent = e.frameState.extent;
event.start = e.frameState.time;
event.context = e.context;
}
event.time = e.frameState.time - event.start;
event.elapsed = event.time / fanim[step].duration_;
if (event.elapsed > 1) event.elapsed = 1;
// Stop animation?
if (!fanim[step].animate(event))
{ nb++;
// Repeat animation
if (nb < fanim[step].repeat_)
{ event.extent = false;
}
// newt step
else if (step < fanim.length-1)
{ fanim[step].dispatchEvent({ type:'animationend', feature: feature });
step++;
nb=0;
event.extent = false;
}
// the end
else
{ stop();
}
}
// tell OL3 to continue postcompose animation
e.frameState.animate = true;
}
// Stop animation
function stop(options)
{ ol.Observable.unByKey(listenerKey);
listenerKey = null;
feature.setStyle(style);
// Send event
var event = { type:'animationend', feature: feature };
if (options)
{ for (var i in options) if (options.hasOwnProperty(i))
{ event[i] = options[i];
}
}
fanim[step].dispatchEvent(event);
self.dispatchEvent(event);
}
// Launch animation
function start(options)
{ if (fanim.length && !listenerKey)
{ listenerKey = self.on('postcompose', animate.bind(self));
// map or layer?
if (self.renderSync) self.renderSync();
else self.changed();
// Send event
var event = { type:'animationstart', feature: feature };
if (options)
{ for (var i in options) if (options.hasOwnProperty(i))
{ event[i] = options[i];
}
}
fanim[step].dispatchEvent(event);
self.dispatchEvent(event);
}
}
start();
// Return animation controler
return {
start: start,
stop: stop,
isPlaying: function() { return (!!listenerKey); }
};
};
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Bounce animation:
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationBounceOptions} options
* @param {Integer} options.bounce number of bounce, default 3
* @param {Integer} options.amplitude bounce amplitude,default 40
* @param {ol.easing} options.easing easing used for decaying amplitude, use function(){return 0} for no decay, default ol.easing.linear
* @param {Integer} options.duration duration in ms, default 1000
*/
ol.featureAnimation.Bounce = function(options)
{ options = options || {};
ol.featureAnimation.call(this, options);
this.amplitude_ = options.amplitude || 40;
this.bounce_ = -Math.PI*(options.bounce || 3);
}
ol.inherits(ol.featureAnimation.Bounce, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Bounce.prototype.animate = function (e)
{ // Animate
var flashGeom = e.geom.clone();
/*
var t = this.easing_(e.elapsed)
t = Math.abs(Math.sin(this.bounce_*t)) * this.amplitude_ * (1-t) * e.frameState.viewState.resolution;
*/
var t = Math.abs(Math.sin(this.bounce_*e.elapsed)) * this.amplitude_ * (1-this.easing_(e.elapsed)) * e.frameState.viewState.resolution;
flashGeom.translate(0, t);
this.drawGeom_(e, flashGeom, e.geom);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Drop animation: drop a feature on the map
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationDropOptions} options
* @param {Number} options.speed speed of the feature if 0 the duration parameter will be used instead, default 0
* @param {Number} options.side top or bottom, default top
*/
ol.featureAnimation.Drop = function(options)
{ options = options || {};
this.speed_ = options.speed || 0;
ol.featureAnimation.call(this, options);
this.side_ = options.side || 'top';
}
ol.inherits(ol.featureAnimation.Drop, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Drop.prototype.animate = function (e)
{ // First time > calculate duration / speed
if (!e.time)
{ var angle = e.frameState.viewState.rotation;
var s = e.frameState.size[1] * e.frameState.viewState.resolution;
if (this.side_!='top') s *= -1;
this.dx = -Math.sin(angle)*s;
this.dy = Math.cos(angle)*s;
if (this.speed_)
{ this.duration_ = s/this.speed_/e.frameState.viewState.resolution;
}
}
// Animate
var flashGeom = e.geom.clone();
flashGeom.translate(
this.dx*(1-this.easing_(e.elapsed)),
this.dy*(1-this.easing_(e.elapsed))
);
this.drawGeom_(e, flashGeom, e.geom);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Fade animation: feature fade in
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationOptions} options
*/
ol.featureAnimation.Fade = function(options)
{ options = options || {};
this.speed_ = options.speed || 0;
ol.featureAnimation.call(this, options);
}
ol.inherits(ol.featureAnimation.Fade, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Fade.prototype.animate = function (e)
{ e.context.globalAlpha = this.easing_(e.elapsed);
this.drawGeom_(e, e.geom);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Do nothing for a given duration
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationShowOptions} options
*
*/
ol.featureAnimation.None = function(options)
{ ol.featureAnimation.call(this, options);
};
ol.inherits(ol.featureAnimation.None, ol.featureAnimation);
/** Animate: do nothing during the laps time
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.None.prototype.animate = function (e)
{
return (e.time <= this.duration_);
};
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Do nothing
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationShowOptions} options
*/
ol.featureAnimation.Null = function(options)
{ ol.featureAnimation.call(this, { duration:0 });
};
ol.inherits(ol.featureAnimation.Null, ol.featureAnimation);
/*
Copyright (c) 2016-2018 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Path animation: feature follow a path
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationPathOptions} options extend ol.featureAnimation options
* @param {Number} options.speed speed of the feature, if 0 the duration parameter will be used instead, default 0
* @param {Number|boolean} options.rotate rotate the symbol when following the path, true or the initial rotation, default false
* @param {ol.geom.LineString|ol.Feature} options.path the path to follow
*/
ol.featureAnimation.Path = function(options)
{ options = options || {};
ol.featureAnimation.call(this, options);
this.speed_ = options.speed || 0;
this.path_ = options.path;
switch (options.rotate) {
case true:
case 0:
this.rotate_ = 0;
break;
default:
this.rotate_ = options.rotate || false;
break;
}
if (this.path_ && this.path_.getGeometry) this.path_ = this.path_.getGeometry();
if (this.path_ && this.path_.getLineString) this.path_ = this.path_.getLineString();
if (this.path_.getLength)
{ this.dist_ = this.path_.getLength()
if (this.path_ && this.path_.getCoordinates) this.path_ = this.path_.getCoordinates();
}
else this.dist_ = 0;
if (this.speed_>0) this.duration_ = this.dist_/this.speed_;
}
ol.inherits(ol.featureAnimation.Path, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Path.prototype.animate = function (e)
{ // First time
if (!e.time)
{ if (!this.dist_) return false;
}
var dmax = this.dist_*this.easing_(e.elapsed);
var p0, p, dx,dy, dl, d = 0;
p = this.path_[0];
// Linear interpol
for (var i = 1; i<this.path_.length; i++)
{ p0 = p;
p = this.path_[i];
dx = p[0]-p0[0];
dy = p[1]-p0[1];
dl = Math.sqrt(dx*dx+dy*dy);
if (dl && d+dl>=dmax)
{ var s = (dmax-d)/dl;
p = [ p0[0] + (p[0]-p0[0])*s, p0[1] + (p[1]-p0[1])*s];
break;
}
d += dl;
}
// Rotate symbols
if (this.rotate_!==false) {
var angle = this.rotate_ - Math.atan2(p0[1] - p[1], p0[0] - p[0]);
for (var k=0, s; s=e.style[k]; k++) {
if (s.getImage()) {
s.getImage().setRotation(angle)
}
}
}
e.geom.setCoordinates(p);
// Animate
this.drawGeom_(e, e.geom);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Shakee animation:
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationShakeOptions} options
* @param {Integer} options.bounce number o bounds, default 6
* @param {Integer} options.amplitude amplitude of the animation, default 40
* @param {bool} options.horizontal shake horizontally default false (vertical)
*/
ol.featureAnimation.Shake = function(options)
{ options = options || {};
ol.featureAnimation.call(this, options);
// this.easing_ = options.easing_ || function(t){return (0.5+t)*t -0.5*t ;};
this.amplitude_ = options.amplitude || 40;
this.bounce_ = -Math.PI*(options.bounce || 6);
this.horizontal_ = options.horizontal;
}
ol.inherits(ol.featureAnimation.Shake, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Shake.prototype.animate = function (e)
{ // Animate
var flashGeom = e.geom.clone();
var shadow = e.geom.clone();
var t = this.easing_(e.elapsed)
t = Math.sin(this.bounce_*t) * this.amplitude_ * (1-t) * e.frameState.viewState.resolution;
if (this.horizontal_)
{ flashGeom.translate(t, 0);
shadow.translate(t, 0);
}
else flashGeom.translate(0, t);
this.drawGeom_(e, flashGeom, shadow);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Show an object for a given duration
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationOptions} options
*/
ol.featureAnimation.Show = function(options)
{ ol.featureAnimation.call(this, options);
}
ol.inherits(ol.featureAnimation.Show, ol.featureAnimation);
/** Animate: just show the object during the laps time
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Show.prototype.animate = function (e)
{
this.drawGeom_(e, e.geom);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Slice animation: feature enter from left
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationSlideOptions} options
* @param {Number} options.speed speed of the animation, if 0 the duration parameter will be used instead, default 0
*/
ol.featureAnimation.Slide = function(options)
{ options = options || {};
this.speed_ = options.speed || 0;
ol.featureAnimation.call(this, options);
this.side_ = options.side || 'left';
}
ol.inherits(ol.featureAnimation.Slide, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Slide.prototype.animate = function (e)
{ // First time > calculate duration / speed
if (!e.time)
{ if (this.side_=='left') this.dx = (e.extent[0]-e.bbox[2])
else this.dx = (e.extent[2]-e.bbox[0])
if (this.speed_) this.duration_ = Math.abs(this.dx)/this.speed_/e.frameState.viewState.resolution;
}
// Animate
var flashGeom = e.geom.clone();
flashGeom.translate(this.dx*(1-this.easing_(e.elapsed)), 0);
this.drawGeom_(e, flashGeom);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Teleport a feature at a given place
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationOptions} options
*/
ol.featureAnimation.Teleport = function(options)
{ ol.featureAnimation.call(this, options);
}
ol.inherits(ol.featureAnimation.Teleport, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Teleport.prototype.animate = function (e)
{ var sc = this.easing_(e.elapsed);
if (sc)
{ e.context.save()
var ratio = e.frameState.pixelRatio;
e.context.globalAlpha = sc;
e.context.scale(sc,1/sc);
var m = e.frameState.coordinateToPixelTransform;
var dx = (1/sc-1) * ratio * (m[0]*e.coord[0] + m[1]*e.coord[1] +m[4]);
var dy = (sc-1) * ratio * (m[2]*e.coord[0] + m[3]*e.coord[1] +m[5]);
e.context.translate(dx,dy);
this.drawGeom_(e, e.geom);
e.context.restore()
}
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Slice animation: feature enter from left
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationThrowOptions} options
* @param {left|right} options.side side of the animation, default left
*/
ol.featureAnimation.Throw = function(options)
{ options = options || {};
ol.featureAnimation.call(this, options);
this.speed_ = options.speed || 0;
this.side_ = options.side || 'left';
}
ol.inherits(ol.featureAnimation.Throw, ol.featureAnimation);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Throw.prototype.animate = function (e)
{ // First time > calculate duration / speed
if (!e.time && this.speed_)
{ var dx, dy;
if (this.side_=='left')
{ dx = this.dx = e.extent[0]-e.bbox[2];
dy = this.dy = e.extent[3]-e.bbox[1];
}
else
{ dx = this.dx = e.extent[2]-e.bbox[0];
dy = this.dy = e.extent[3]-e.bbox[1];
}
this.duration_ = Math.sqrt(dx*dx+dy*dy)/this.speed_/e.frameState.viewState.resolution;
}
// Animate
var flashGeom = e.geom.clone();
var shadow = e.geom.clone();
flashGeom.translate(this.dx*(1-this.easing_(e.elapsed)),
this.dy*Math.cos(Math.PI/2*this.easing_(e.elapsed)));
shadow.translate(this.dx*(1-this.easing_(e.elapsed)), 0);
this.drawGeom_(e, flashGeom, shadow);
return (e.time <= this.duration_);
}
/*
Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL license (http://www.cecill.info/).
*/
/** Zoom animation: feature zoom in (for points)
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationZoomOptions} options
* @param {bool} options.zoomOut to zoom out
*/
ol.featureAnimation.Zoom = function(options)
{ options = options || {};
ol.featureAnimation.call(this, options);
this.set('zoomout', options.zoomOut);
}
ol.inherits(ol.featureAnimation.Zoom, ol.featureAnimation);
/** Zoom animation: feature zoom out (for points)
* @constructor
* @extends {ol.featureAnimation}
* @param {ol.featureAnimationZoomOptions} options
*/
ol.featureAnimation.ZoomOut = function(options)
{ options = options || {};
options.zoomOut = true;
ol.featureAnimation.Zoom.call(this, options);
}
ol.inherits(ol.featureAnimation.ZoomOut, ol.featureAnimation.Zoom);
/** Animate
* @param {ol.featureAnimationEvent} e
*/
ol.featureAnimation.Zoom.prototype.animate = function (e)
{ var fac = this.easing_(e.elapsed);
if (fac)
{ if (this.get('zoomout')) fac = 1/fac;
var style = e.style;
var imgs, sc=[]
for (var i=0; i<style.length; i++)
{ imgs = style[i].getImage();
if (imgs)
{ sc[i] = imgs.getScale();
imgs.setScale(sc[i]*fac);
}
}
e.context.save()
var ratio = e.frameState.pixelRatio;
var m = e.frameState.coordinateToPixelTransform;
var dx = (1/fac-1)* ratio * (m[0]*e.coord[0] + m[1]*e.coord[1] +m[4]);
var dy = (1/fac-1)* ratio * (m[2]*e.coord[0] + m[3]*e.coord[1] +m[5]);
e.context.scale(fac,fac);
e.context.translate(dx,dy);
this.drawGeom_(e, e.geom);
e.context.restore()
for (var i=0; i<style.length; i++)
{ imgs = style[i].getImage();
if (imgs) imgs.setScale(sc[i]);
}
}
/*
var sc = this.easing_(e.elapsed);
if (sc)
{ e.context.save()
console.log(e)
var ratio = e.frameState.pixelRatio;
var m = e.frameState.coordinateToPixelTransform;
var dx = (1/(sc)-1)* ratio * (m[0]*e.coord[0] + m[1]*e.coord[1] +m[4]);
var dy = (1/(sc)-1)*ratio * (m[2]*e.coord[0] + m[3]*e.coord[1] +m[5]);
e.context.scale(sc,sc);
e.context.translate(dx,dy);
this.drawGeom_(e, e.geom);
e.context.restore()
}
*/
return (e.time <= this.duration_);
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
ol.filter = {};
/**
* @classdesc
* Abstract base class; normally only used for creating subclasses and not instantiated in apps.
* Used to create filters
* Use {@link _ol_Map_#addFilter}, {@link _ol_Map_#removeFilter} or {@link _ol_Map_#getFilters} to handle filters on a map.
* Use {@link ol.layer.Base#addFilter}, {@link ol.layer.Base#removeFilter} or {@link ol.layer.Base#getFilters}
* to handle filters on layers.
*
* @constructor
* @extends {ol.Object}
* @param {} options Extend {@link _ol_control_Control_} options.
* @param {bool} options.active
*/
ol.filter.Base = function(options) {
ol.Object.call(this);
// Array of postcompose listener
this._listener = [];
if (options && options.active===false) this.set('active', false);
else this.set('active', true);
};
ol.inherits(ol.filter.Base, ol.Object);
/** Activate / deactivate filter
* @param {bool} b
*/
ol.filter.Base.prototype.setActive = function (b) {
this.set('active', b===true);
};
/** Get filter active
* @return {bool}
*/
ol.filter.Base.prototype.getActive = function (b) {
return this.get('active');
};
(function(){
/** Internal function
* @scoop {ol.filter} this the filter
* @private
*/
function precompose_(e)
{ if (this.get('active')) this.precompose(e);
}
/** Internal function
* @scoop {ol.filter} this the filter
* @private
*/
function postcompose_(e) {
if (this.get('active')) this.postcompose(e);
}
/** Force filter redraw / Internal function
* @scoop {ol.map||ol.layer} this: the map or layer the filter is added to
* @private
*/
function filterRedraw_(e) {
if (this.renderSync) this.renderSync();
else this.changed();
}
/** Add a filter to an ol object
* @scoop {ol.map||ol.layer} this: the map or layer the filter is added to
* @private
*/
function addFilter_(filter) {
if (!this.filters_) this.filters_ = [];
this.filters_.push(filter);
if (filter.precompose) filter._listener.push ( { listener: this.on('precompose', precompose_.bind(filter)), target: this });
if (filter.postcompose) filter._listener.push ( { listener: this.on('postcompose', postcompose_.bind(filter)), target: this });
filter._listener.push ( { listener: filter.on('propertychange', filterRedraw_.bind(this)), target: this });
filterRedraw_.call (this);
};
/** Remove a filter to an ol object
* @scoop {ol.map||ol.layer} this: the map or layer the filter is added to
* @private
*/
function removeFilter_(filter) {
if (!this.filters_) this.filters_ = [];
for (var i=this.filters_.length-1; i>=0; i--) {
if (this.filters_[i]===filter) this.filters_.splice(i,1);
}
for (var i=filter._listener.length-1; i>=0; i--) {
// Remove listener on this object
if (filter._listener[i].target === this) {
ol.Observable.unByKey(filter._listener[i].listener);
filter._listener.splice(i,1);
}
}
filterRedraw_.call (this);
};
/** Add a filter to an ol.Map
* @param {ol.filter}
*/
ol.Map.prototype.addFilter = function (filter) {
addFilter_.call (this, filter);
};
/** Remove a filter to an ol.Map
* @param {ol.filter}
*/
ol.Map.prototype.removeFilter = function (filter)
{ removeFilter_.call (this, filter);
};
/** Get filters associated with an ol.Map
* @return {Array<ol.filter>}
*/
ol.Map.prototype.getFilters = function ()
{ return this.filters_;
};
/** Add a filter to an ol.Layer
* @param {ol.filter}
*/
ol.layer.Base.prototype.addFilter = function (filter)
{ addFilter_.call (this, filter);
};
/** Remove a filter to an ol.Layer
* @param {ol.filter}
*/
ol.layer.Base.prototype.removeFilter = function (filter)
{ removeFilter_.call (this, filter);
};
/** Get filters associated with an ol.Map
* @return {Array<ol.filter>}
*/
ol.layer.Base.prototype.getFilters = function ()
{ return this.filters_;
};
})();
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Mask drawing using an ol.Feature
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @param {ol.filter.CropOptions} options
* - feature {ol.Feature} feature to mask with
* - fill {ol.style.Fill} style to fill with
* - inner {bool} mask inner, default false
*/
ol.filter.Mask = function(options)
{ options = options || {};
ol.filter.Base.call(this, options);
if (options.feature)
{ switch (options.feature.getGeometry().getType())
{ case "Polygon":
case "MultiPolygon":
this.feature_ = options.feature;
break;
default: break;
}
}
this.set("inner", options.inner);
this.fillColor_ = options.fill ? ol.color.asString(options.fill.getColor()) || "rgba(0,0,0,0.2)" : "rgba(0,0,0,0.2)";
}
ol.inherits(ol.filter.Mask, ol.filter.Base);
/** Draw the feature into canvas
*/
ol.filter.Mask.prototype.drawFeaturePath_ = function(e, out)
{ var ctx = e.context;
var canvas = ctx.canvas;
var ratio = e.frameState.pixelRatio;
// Transform
var m = e.frameState.coordinateToPixelTransform;
function tr(pt)
{ return [
(pt[0]*m[0]+pt[1]*m[1]+m[4])*ratio,
(pt[0]*m[2]+pt[1]*m[3]+m[5])*ratio
];
}
// Old version
if (!m)
{ m = e.frameState.coordinateToPixelMatrix;
tr = function(pt)
{ return [
(pt[0]*m[0]+pt[1]*m[1]+m[12])*ratio,
(pt[0]*m[4]+pt[1]*m[5]+m[13])*ratio
];
}
}
// Geometry
var ll = this.feature_.getGeometry().getCoordinates();
if (this.feature_.getGeometry().getType()=="Polygon") ll = [ll];
ctx.beginPath();
if (out)
{ ctx.moveTo (0,0);
ctx.lineTo (canvas.width, 0);
ctx.lineTo (canvas.width, canvas.height);
ctx.lineTo (0, canvas.height);
ctx.lineTo (0, 0);
}
for (var l=0; l<ll.length; l++)
{ var c = ll[l];
for (var i=0; i<c.length; i++)
{ var pt = tr(c[i][0]);
ctx.moveTo (pt[0], pt[1]);
for (var j=1; j<c[i].length; j++)
{ pt = tr(c[i][j]);
ctx.lineTo (pt[0], pt[1]);
}
}
}
}
ol.filter.Mask.prototype.postcompose = function(e)
{ if (!this.feature_) return;
var ctx = e.context;
ctx.save();
this.drawFeaturePath_(e, !this.get("inner"));
ctx.fillStyle = this.fillColor_;
ctx.fill("evenodd");
ctx.restore();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Clip layer or map
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @param {ol.filter.ClipOptions} options
* - coords {Array<ol.Coordinate>}
* - extent {ol.Extent}
* - units {%|px} coords units percent or pixel
* - keepAspectRatio {boolean} keep aspect ratio
* - color {string} backgroundcolor
*/
ol.filter.Clip = function(options)
{ options = options || {};
ol.filter.Base.call(this, options);
this.set("coords", options.coords);
this.set("units", options.units);
this.set("keepAspectRatio", options.keepAspectRatio);
this.set("extent", options.extent || [0,0,1,1]);
this.set("color", options.color);
if (!options.extent && options.units!="%" && options.coords)
{ var xmin = Infinity;
var ymin = Infinity;
var xmax = -Infinity;
var ymax = -Infinity;
for (var i=0, p; p=options.coords[i]; i++)
{ if (xmin > p[0]) xmin = p[0];
if (xmax < p[0]) xmax = p[0];
if (ymin > p[1]) ymin = p[1];
if (ymax < p[1]) ymax = p[1];
}
options.extent = [xmin,ymin,xmax,ymax];
}
}
ol.inherits(ol.filter.Clip, ol.filter.Base);
ol.filter.Clip.prototype.clipPath_ = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
var coords = this.get("coords");
if (!coords) return;
var ex = this.get('extent');
var scx = 1, scy = 1;
if (this.get("units")=="%")
{ scx = canvas.width/(ex[2]-ex[0]);
scy = canvas.height/(ex[3]-ex[1]);
}
if (this.get("keepAspectRatio"))
{ scx = scy = Math.min (scx, scy);
}
var pos = this.get('position');
var dx=0, dy=0;
if (/left/.test(pos))
{ dx = -ex[0]*scx;
}
else if (/center/.test(pos))
{ dx = canvas.width/2 - (ex[2]-ex[0])*scx/2;
}
else if (/right/.test(pos))
{ dx = canvas.width - (ex[2]-ex[0])*scx;
}
var fx = function(x) { return x*scx + dx };
if (/top/.test(pos))
{ dy = -ex[1]*scy;
}
else if (/middle/.test(pos))
{ dy = canvas.height/2 - (ex[3]-ex[1])*scy/2;
}
else if (/bottom/.test(pos))
{ dy = canvas.height - (ex[3]-ex[1])*scy;
}
var fy = function(y) { return y*scy + dy; };
ctx.moveTo ( fx(coords[0][0]), fy(coords[0][1]) );
for (var i=1; p=coords[i]; i++)
{ ctx.lineTo ( fx(p[0]), fy(p[1]) );
}
ctx.lineTo ( fx(coords[0][0]), fy(coords[0][1]) );
};
ol.filter.Clip.prototype.precompose = function(e)
{ if (!this.get("color"))
{ e.context.save();
e.context.beginPath();
this.clipPath_(e);
e.context.clip();
}
}
ol.filter.Clip.prototype.postcompose = function(e)
{ if (this.get("color"))
{ var ctx = e.context;
var canvas = e.context.canvas;
ctx.save();
ctx.beginPath();
ctx.moveTo(0,0);
ctx.lineTo(0,canvas.height);
ctx.lineTo(canvas.width, canvas.height);
ctx.lineTo(canvas.width, canvas.height);
ctx.lineTo(canvas.width, 0);
ctx.lineTo(0, 0);
this.clipPath_(e);
ctx.fillStyle = this.get("color");
ctx.fill("evenodd");
};
e.context.restore();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Colorize map or layer
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @author Thomas Tilak https://github.com/thhomas
* @author Jean-Marc Viglino https://github.com/viglino
* @param {ol.filter.ColorizeOptions} options
* - feature {ol.Feature} feature to mask with
* - color {Array<integer>} style to fill with
* - inner {bool} mask inner, default false
*/
ol.filter.Colorize = function(options)
{ ol.filter.Base.call(this, options);
this.setFilter(options);
}
ol.inherits(ol.filter.Colorize, ol.filter.Base);
ol.filter.Colorize.prototype.setFilter = function(options)
{ options = options || {};
switch (options)
{ case "grayscale": options = { operation:'hue', red:0, green:0, blue:0, value:1 }; break;
case "invert": options = { operation:'difference', red:255, green:255, blue:255, value:1 }; break;
case "sepia": options = { operation:'color', red:153, green:102, blue:51, value:0.6 }; break;
default: break;
}
var color = options.color ? ol.color.asArray(options.color) : [ options.red, options.green, options.blue, options.value];
this.set('color', ol.color.asString(color))
this.set ('value', color[3]||1);
switch (options.operation)
{ case 'color':
case 'hue':
case 'difference':
case 'color-dodge':
case 'enhance':
this.set ('operation', options.operation);
break;
case 'saturation':
var v = 255*(options.value || 0);
this.set('color', ol.color.asString([0,0,v,v||1]));
this.set ('operation', options.operation);
break;
case 'luminosity':
var v = 255*(options.value || 0);
this.set('color', ol.color.asString([v,v,v,255]));
//this.set ('operation', 'luminosity')
this.set ('operation', 'hard-light');
break;
case 'contrast':
var v = 255*(options.value || 0);
this.set('color', ol.color.asString([v,v,v,255]));
this.set('operation', 'soft-light');
break;
default:
this.set ('operation', 'color');
break;
}
}
ol.filter.Colorize.prototype.setValue = function(v)
{ this.set ('value', v);
var c = ol.color.asArray(this.get("color"));
c[3] = v;
this.set("color", ol.color.asString(c));
}
ol.filter.Colorize.prototype.setColor = function(c)
{ c = ol.color.asArray(c);
if (c)
{ c[3] = this.get("value");
this.set("color", ol.color.asString(c));
}
}
ol.filter.Colorize.prototype.precompose = function(e)
{}
ol.filter.Colorize.prototype.postcompose = function(e)
{ // Set back color hue
var ctx = e.context;
var canvas = ctx.canvas;
ctx.save();
if (this.get('operation')=='enhance')
{ var v = this.get('value');
if (v)
{ var w = canvas.width;
var h = canvas.height;
ctx.globalCompositeOperation = 'color-burn'
ctx.globalAlpha = v;
ctx.drawImage (canvas, 0, 0, w, h);
ctx.drawImage (canvas, 0, 0, w, h);
ctx.drawImage (canvas, 0, 0, w, h);
}
}
else
{ ctx.globalCompositeOperation = this.get('operation');
ctx.fillStyle = this.get('color');
ctx.fillRect(0,0,canvas.width,canvas.height);
}
ctx.restore();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Colorize map or layer
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @param {ol.filter.CompositeOptions} options
* - operation {string} composite operation
*/
ol.filter.Composite = function(options)
{ ol.filter.Base.call(this, options);
this.set("operation", options.operation || "source-over");
}
ol.inherits(ol.filter.Composite, ol.filter.Base);
/** Change the current operation
* @param {string} operation composite function
*/
ol.filter.Composite.prototype.setOperation = function(operation)
{ this.set('operation', operation || "source-over");
}
ol.filter.Composite.prototype.precompose = function(e)
{ var ctx = e.context;
ctx.save();
ctx.globalCompositeOperation = this.get('operation');
}
ol.filter.Composite.prototype.postcompose = function(e)
{ e.context.restore();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Crop drawing using an ol.Feature
* @constructor
* @requires ol.filter
* @requires ol.filter.Mask
* @extends {ol.filter.Mask}
* @param {ol.filter.CropOptions}
* - feature {_ol_Feature_} feature to crop with
* - inner {bool} crop inner, default false
*/
ol.filter.Crop = function(options)
{ options = options || {};
ol.filter.Mask.call(this, options);
}
ol.inherits(ol.filter.Crop, ol.filter.Mask);
ol.filter.Crop.prototype.precompose = function(e)
{ if (!this.feature_) return;
var ctx = e.context;
ctx.save();
this.drawFeaturePath_(e, this.get("inner"));
ctx.clip("evenodd");
}
ol.filter.Crop.prototype.postcompose = function(e)
{ if (this.feature_) e.context.restore();
}
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Fold filer map
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @param {ol.filter.FoldOptions}
* - fold {Array<int>} number of fold (horizontal and vertical)
* - margin {Number} margin in px, default 8
* - padding {Number} padding in px, default 8
* - fsize {integer|Array<integer>} fold size in px, default 8,10
*/
ol.filter.Fold = function(options)
{ options = options || {};
ol.filter.Base.call(this, options);
this.set("fold", options.fold || [8,4]);
this.set("margin", options.margin || 8);
this.set("padding", options.padding || 8);
if (typeof options.fsize == "number") options.fsize = [options.fsize,options.fsize];
this.set("fsize", options.fsize || [8,10]);
}
ol.inherits(ol.filter.Fold, ol.filter.Base);
ol.filter.Fold.prototype.drawLine_ = function(ctx, d, m)
{ var canvas = ctx.canvas;
var fold = this.get("fold");
var w = canvas.width;
var h = canvas.height;
ctx.beginPath();
ctx.moveTo ( m, m );
for (var i=1; i<=fold[0]; i++)
{ x = i*w/fold[0] - (i==fold[0] ? m : 0);
y = d[1]*(i%2) +m;
ctx.lineTo ( x, y );
}
for (var i=1; i<=fold[1]; i++)
{ x = w - d[0]*(i%2) - m;
y = i*h/fold[1] - (i==fold[1] ? d[0]*(fold[0]%2) + m : 0);
ctx.lineTo ( x, y );
}
for (var i=fold[0]; i>0; i--)
{ x = i*w/fold[0] - (i==fold[0] ? d[0]*(fold[1]%2) + m : 0);
y = h - d[1]*(i%2) -m;
ctx.lineTo ( x, y );
}
for (var i=fold[1]; i>0; i--)
{ x = d[0]*(i%2) + m;
y = i*h/fold[1] - (i==fold[1] ? m : 0);
ctx.lineTo ( x, y );
}
ctx.closePath();
}
ol.filter.Fold.prototype.precompose = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
var fold = this.get("fold");
var w = canvas.width;
var h = canvas.height;
ctx.save();
ctx.shadowColor = "rgba(0,0,0,0.3)";
ctx.shadowBlur = 8;
ctx.shadowOffsetX = 2;
ctx.shadowOffsetY = 3;
this.drawLine_(ctx, this.get("fsize"), this.get("margin"));
ctx.fillStyle="#fff";
ctx.fill();
ctx.strokeStyle = "rgba(0,0,0,0.1)";
ctx.stroke();
ctx.restore();
ctx.save();
this.drawLine_(ctx, this.get("fsize"), this.get("margin") + this.get("padding"));
ctx.clip();
}
ol.filter.Fold.prototype.postcompose = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
ctx.restore();
ctx.save();
this.drawLine_(ctx, this.get("fsize"), this.get("margin"));
ctx.clip();
var fold = this.get("fold");
var w = canvas.width/fold[0];
var h = canvas.height/fold[1];
var grd = ctx.createRadialGradient(5*w/8,5*w/8,w/4,w/2,w/2,w);
grd.addColorStop(0,"transparent");
grd.addColorStop(1,"rgba(0,0,0,0.2)");
ctx.fillStyle = grd;
ctx.scale (1,h/w);
for (var i=0; i<fold[0]; i++) for (var j=0; j<fold[1]; j++)
{ ctx.save()
ctx.translate(i*w, j*w);
ctx.fillRect(0,0,w,w);
ctx.restore()
}
ctx.restore();
}
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Make a map or layer look like made of a set of Lego bricks.
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @param {ol.filter.LegoOptions}
* - brickSize {Number} size of te brick, default 30
* - crossOrigin {null | string | undefined} crossOrigin attribute for loaded images.
*/
ol.filter.Lego = function(options)
{ if (!options) options = {};
ol.filter.Base.call(this, options);
var img = new Image();
// Default image
img.src = this.img[options.img] || this.img.ol3;
img.crossOrigin = options.crossOrigin || null;
// and pattern
this.pattern =
{ canvas: document.createElement('canvas')
};
this.setBrick (options.brickSize, img);
this.internal_ = document.createElement('canvas');
}
ol.inherits(ol.filter.Lego, ol.filter.Base);
/** Image definition
*/
ol.filter.Lego.prototype.img =
{ brick: "",
ol3: "",
lego: ""
};
/** Overwrite to handle brickSize
* @param {string} key
* @param {} val
*/
ol.filter.Lego.prototype.set = function (key, val)
{ ol.filter.Base.prototype.set.call(this, key, val);
if (key=="brickSize" && this.pattern.canvas.width!=val)
{ this.setBrick(val);
}
}
/** Set the current brick
* @param {Number} width the pattern width, default 30
* @param {brick|ol3|lego|undefined} img the pattern, default ol3
* @param {string} crossOrigin
*/
ol.filter.Lego.prototype.setBrick = function (width, img, crossOrigin)
{ width = Number(width) || 30;
if (typeof(img) === 'string')
{ var i = new Image;
i.src = this.img[img] || this.img.ol3;
i.crossOrigin = crossOrigin || null;
img = i;
}
if (img) this.pattern.img = img;
if (!this.pattern.img.width)
{ var self = this;
this.pattern.img.onload = function()
{ self.setBrick(width,img);
}
return;
}
this.pattern.canvas.width = this.pattern.canvas.height = width;
this.pattern.ctx = this.pattern.canvas.getContext("2d");
this.pattern.ctx.fillStyle = this.pattern.ctx.createPattern (this.pattern.img, 'repeat');
this.set("brickSize", width);
this.set("img", img.src);
};
/** Get translated pattern
* @param {number} offsetX x offset
* @param {number} offsetY y offset
*/
ol.filter.Lego.prototype.getPattern = function (offsetX, offsetY)
{
if (!this.pattern.ctx) return "transparent";
//return this.pattern.ctx.fillStyle
var c = this.pattern.canvas;
var ctx = this.pattern.ctx;
var sc = c.width / this.pattern.img.width;
ctx.save();
ctx.clearRect(0,0,c.width,c.height);
ctx.scale(sc,sc);
offsetX /= sc;
offsetY /= sc;
ctx.translate(offsetX, offsetY);
ctx.beginPath();
ctx.clearRect(-2*c.width, -2*c.height, 4*c.width, 4*c.height);
ctx.rect(-offsetX, -offsetY, 2*c.width/sc, 2*c.height/sc);
ctx.fill();
ctx.restore();
return ctx.createPattern(c, 'repeat');
};
/** Postcompose operation
*/
ol.filter.Lego.prototype.postcompose = function(e)
{ // Set back color hue
var ctx = e.context;
var canvas = ctx.canvas;
var ratio = e.frameState.pixelRatio;
ctx.save();
// resize
var step = this.pattern.canvas.width*ratio, step2 = step/2;
var p = e.frameState.extent;
var res = e.frameState.viewState.resolution/ratio;
var offset = [ -Math.round((p[0]/res)%step), Math.round((p[1]/res)%step) ];
var ctx2 = this.internal_.getContext("2d");
var w = this.internal_.width = canvas.width;
var h = this.internal_.height = canvas.height;
// No smoothing please
ctx2.webkitImageSmoothingEnabled =
ctx2.mozImageSmoothingEnabled =
ctx2.msImageSmoothingEnabled =
ctx2.imageSmoothingEnabled = false;
var w2 = Math.floor((w-offset[0])/step);
var h2 = Math.floor((h-offset[1])/step);
ctx2.drawImage (canvas, offset[0], offset[1], w2*step, h2*step, 0, 0, w2, h2);
//
ctx.webkitImageSmoothingEnabled =
ctx.mozImageSmoothingEnabled =
ctx.msImageSmoothingEnabled =
ctx.imageSmoothingEnabled = false;
ctx.clearRect (0, 0, w,h);
ctx.drawImage (this.internal_, 0,0, w2,h2, offset[0],offset[1], w2*step, h2*step);
/*
for (var x=offset[0]; x<w; x+=step) for (var y=offset[1]; y<h; y+=step)
{ if (x>=0 && y<h) ctx2.drawImage (canvas, x, y, 1, 1, x, y, step, step);
}
ctx.clearRect (0, 0, w,h);
ctx.drawImage (c, 0, 0);
*/
// Draw brick stud
ctx.scale(ratio,ratio);
ctx.fillStyle = this.getPattern (offset[0]/ratio, offset[1]/ratio);
ctx.rect(0,0, w, h);
ctx.fill();
ctx.restore();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Add texture effects on maps or layers
* @constructor
* @requires ol.filter
* @extends {ol.filter.Base}
* @param {ol.filter.TextureOptions} options
* - feature {_ol_Feature_} feature to mask with
* - fill {_ol_style_Fill_} style to fill with
* - inner {bool} mask inner, default false
*/
ol.filter.Texture = function(options)
{ ol.filter.Base.call(this, options);
this.setFilter(options);
}
ol.inherits(ol.filter.Texture, ol.filter.Base);
/** Set texture
* @option {ol.filter.TextureOptions}
* - img {Image | undefined} Image object for the texture
* - src {string} Image source URI
* - scale {number} scale to draw the image. Default 1.
* - rotateWithView {bool} Whether to rotate the texture with the view (may cause animation lags on mobile or slow devices). Default is true.
* - crossOrigin {null | string | undefined} The crossOrigin attribute for loaded images.
*/
ol.filter.Texture.prototype.setFilter = function(options)
{ var img;
options = options || {};
if (options.img) img = option.img;
else
{ img = new Image();
if (options.src) {
// Look for a texture stored in ol.filter.Texture.Image
if (ol.filter.Texture.Image && ol.filter.Texture.Image[options.src]) {
img.src = ol.filter.Texture.Image[options.src];
}
// default source
else {
if (!img.src) img.src = options.src;
}
}
img.crossOrigin = options.crossOrigin || null;
}
this.set('rotateWithView', options.rotateWithView !== false);
this.set('opacity', typeof(options.opacity)=='number' ? options.opacity : 1);
this.set('ready', false);
var self = this;
function setPattern(img)
{ self.pattern = {};
self.pattern.scale = options.scale || 1;
self.pattern.canvas = document.createElement('canvas');
self.pattern.canvas.width = img.width * self.pattern.scale;
self.pattern.canvas.height = img.height * self.pattern.scale;
self.pattern.canvas.width = img.width;// * self.pattern.scale;
self.pattern.canvas.height = img.height;// * self.pattern.scale;
self.pattern.ctx = self.pattern.canvas.getContext("2d");
self.pattern.ctx.fillStyle = self.pattern.ctx.createPattern(img, 'repeat');
// Force refresh
self.set('ready', true);
};
if (img.width)
{ setPattern(img);
}
else
{ img.onload = function()
{ setPattern(img);
}
}
}
/** Get translated pattern
* @param {number} x offset
* @param {number} y offset
*/
ol.filter.Texture.prototype.getPattern = function (offsetX, offsetY)
{ var c = this.pattern.canvas;
var ctx = this.pattern.ctx;
ctx.save();
/*
offsetX /= this.pattern.scale;
offsetY /= this.pattern.scale;
ctx.scale(this.pattern.scale,this.pattern.scale);
*/
ctx.translate(-offsetX, offsetY);
ctx.beginPath();
ctx.rect(offsetX, -offsetY, c.width, c.height);
ctx.fill();
ctx.restore();
return ctx.createPattern(c, 'repeat');
}
/** Draw pattern over the map on postcompose
*/
ol.filter.Texture.prototype.postcompose = function(e)
{ // not ready
if (!this.pattern) return;
// Set back color hue
var ctx = e.context;
var canvas = ctx.canvas;
var m = 1.5 * Math.max(canvas.width, canvas.height);
var mt = e.frameState.pixelToCoordinateTransform;
// Old version (matrix)
if (!mt)
{ mt = e.frameState.pixelToCoordinateMatrix,
mt[2] = mt[4];
mt[3] = mt[5];
mt[4] = mt[12];
mt[5] = mt[13];
}
var ratio = e.frameState.pixelRatio;
var res = e.frameState.viewState.resolution;
var w = canvas.width/2,
h = canvas.height/2;
ctx.save();
ctx.globalCompositeOperation = "multiply";
//ctx.globalCompositeOperation = "overlay";
//ctx.globalCompositeOperation = "color";
ctx.globalAlpha = this.get('opacity');
ctx.scale(ratio*this.pattern.scale,ratio*this.pattern.scale);
if (this.get('rotateWithView'))
{ // Translate pattern
res *= this.pattern.scale
ctx.fillStyle = this.getPattern ((w*mt[0] + h*mt[1] + mt[4])/res, (w*mt[2] + h*mt[3] + mt[5])/res);
// Rotate on canvas center and fill
ctx.translate(w/this.pattern.scale, h/this.pattern.scale);
ctx.rotate(e.frameState.viewState.rotation);
ctx.beginPath();
ctx.rect(-w-m, -h-m, 2*m, 2*m);
ctx.fill();
}
else
{
/**/
var dx = -(w*mt[0] + h*mt[1] + mt[4])/res;
var dy = (w*mt[2] + h*mt[3] + mt[5])/res;
var cos = Math.cos(e.frameState.viewState.rotation);
var sin = Math.sin(e.frameState.viewState.rotation);
var offsetX = (dx*cos - dy*sin) / this.pattern.scale;
var offsetY = (dx*sin + dy*cos) / this.pattern.scale;
ctx.translate(offsetX, offsetY);
ctx.beginPath();
ctx.fillStyle = this.pattern.ctx.fillStyle;
ctx.rect(-offsetX -m , -offsetY -m, 2*m, 2*m);
ctx.fill();
/* //old version without centered rotation
var offsetX = -(e.frameState.extent[0]/res) % this.pattern.canvas.width;
var offsetY = (e.frameState.extent[1]/res) % this.pattern.canvas.height;
ctx.rotate(e.frameState.viewState.rotation);
ctx.translate(offsetX, offsetY);
ctx.beginPath();
ctx.fillStyle = this.pattern.ctx.fillStyle
ctx.rect(-offsetX -m , -offsetY -m, 2*m, 2*m);
ctx.fill();
*/
}
ctx.restore();
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Handles coordinates on the center of the viewport.
* It can be used as abstract base class used for creating subclasses.
* The CenterTouch interaction modifies map browser event coordinate and pixel properties to force pointer on the viewport center to any interaction that them.
* Only pointermove pointerup are concerned with it.
* @constructor
* @extends {ol.interaction.Interaction}
* @param {olx.interaction.InteractionOptions} options Options
* - targetStyle {ol.style.Style|Array<ol.style.Style>} a style to draw the target point, default cross style
* - composite {string} composite operation : difference|multiply|xor|screen|overlay|darken|lighter|lighten|...
*/
ol.interaction.CenterTouch = function(options)
{ options = options || {};
// LIst of listerner on the object
this._listener = {};
// Filter event
var rex = /^pointermove$|^pointerup$/;
// Default style = cross
this.targetStyle = options.targetStyle ||
[ new ol.style.Style({ image: new ol.style.RegularShape ({ points: 4, radius: 11, radius1: 0, radius2: 0, snapToPixel:true, stroke: new ol.style.Stroke({ color: "#fff", width:3 }) }) }),
new ol.style.Style({ image: new ol.style.RegularShape ({ points: 4, radius: 11, radius1: 0, radius2: 0, snapToPixel:true, stroke: new ol.style.Stroke({ color: "#000", width:1 }) }) })
];
if (!(this.targetStyle instanceof Array)) this.targetStyle = [this.targetStyle];
this.composite = options.composite || '';
// Interaction to defer center on top of the interaction
// this is done to enable other coordinates manipulation inserted after the interaction (snapping)
this.ctouch = new ol.interaction.Interaction(
{ handleEvent: function(e)
{ if (rex.test(e.type) && this.getMap())
{ e.coordinate = this.getMap().getView().getCenter();
e.pixel = this.getMap().getSize();
e.pixel = [ e.pixel[0]/2, e.pixel[1]/2 ];
}
return true;
}
});
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ if (rex.test(e.type)) this.pos_ = e.coordinate;
if (options.handleEvent) return options.handleEvent.call (this,e);
return true;
}
});
};
ol.inherits(ol.interaction.CenterTouch, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.interaction.CenterTouch.prototype.setMap = function(map)
{ if (this.getMap())
{ this.getMap().removeInteraction(this.ctouch);
}
if (this._listener.drawtarget) ol.Observable.unByKey(this._listener.drawtarget);
this._listener.drawtarget = null;
ol.interaction.Interaction.prototype.setMap.call (this, map);
if (this.getMap())
{ if (this.getActive()) this.getMap().addInteraction(this.ctouch);
this._listener.drawtarget = this.getMap().on('postcompose', this.drawTarget_.bind(this));
}
};
/**
* Activate or deactivate the interaction.
* @param {boolean} active Active.
* @observable
* @api
*/
ol.interaction.CenterTouch.prototype.setActive = function(b)
{ ol.interaction.Interaction.prototype.setActive.call (this, b);
this.pos_ = null;
if (this.getMap())
{ if (this.getActive())
{ this.getMap().addInteraction(this.ctouch);
}
else this.getMap().removeInteraction(this.ctouch);
}
};
/** Get the position of the target
*/
ol.interaction.CenterTouch.prototype.getPosition = function (e)
{ if (!this.pos_)
{ var px =this.getMap().getSize();
px = [ px[0]/2, px[1]/2 ];
this.pos_ = this.getMap().getCoordinateFromPixel(px);
}
return this.pos_;
};
/** Draw the target
* @private
*/
ol.interaction.CenterTouch.prototype.drawTarget_ = function (e)
{ if (!this.getMap() || !this.getActive()) return;
var ctx = e.context;
var ratio = e.frameState.pixelRatio;
ctx.save();
var cx = ctx.canvas.width/(2*ratio);
var cy = ctx.canvas.height/(2*ratio);
var geom = new ol.geom.Point (this.getMap().getCoordinateFromPixel([cx,cy]));
if (this.composite) ctx.globalCompositeOperation = this.composite;
for (var i=0; i<this.targetStyle.length; i++)
{ var style = this.targetStyle[i];
if (style instanceof ol.style.Style)
{ var sc=0;
// OL < v4.3 : setImageStyle doesn't check retina
var imgs = ol.Map.prototype.getFeaturesAtPixel ? false : style.getImage();
if (imgs)
{ sc = imgs.getScale();
imgs.setScale(ratio*sc);
}
e.vectorContext.setStyle(style);
e.vectorContext.drawGeometry(geom);
if (imgs) imgs.setScale(sc);
}
}
ctx.restore();
};
/** Clip interaction to clip layers in a circle
* @constructor
* @extends {ol.interaction.Pointer}
* @param {ol.interaction.Clip.options} options flashlight param
* - radius {number} radius of the clip, default 100
* - layers {ol.layer|Array<ol.layer>} layers to clip
*/
ol.interaction.Clip = function(options) {
this.layers_ = [];
ol.interaction.Pointer.call(this,
{ handleDownEvent: this.setPosition,
handleMoveEvent: this.setPosition
});
// Default options
options = options || {};
this.pos = false;
this.radius = (options.radius||100);
if (options.layers) this.addLayer(options.layers);
};
ol.inherits(ol.interaction.Clip, ol.interaction.Pointer);
/** Set the map > start postcompose
*/
ol.interaction.Clip.prototype.setMap = function(map) {
if (this.getMap()) {
for (var i=0; i<this.layers_.length; i++) {
if (this.layers_[i].precompose) ol.Observable.unByKey(this.layers_[i].precompose);
if (this.layers_[i].postcompose) ol.Observable.unByKey(this.layers_[i].postcompose);
this.layers_[i].precompose = this.layers_[i].postcompose = null;
}
this.getMap().renderSync();
}
ol.interaction.Pointer.prototype.setMap.call(this, map);
if (map) {
for (var i=0; i<this.layers_.length; i++) {
this.layers_[i].precompose = this.layers_[i].layer.on('precompose', this.precompose_.bind(this));
this.layers_[i].postcompose = this.layers_[i].layer.on('postcompose', this.postcompose_.bind(this));
}
map.renderSync();
}
}
/** Set clip radius
* @param {integer} radius
*/
ol.interaction.Clip.prototype.setRadius = function(radius)
{ this.radius = radius;
if (this.getMap()) this.getMap().renderSync();
}
/** Add a layer to clip
* @param {ol.layer|Array<ol.layer>} layer to clip
*/
ol.interaction.Clip.prototype.addLayer = function(layers)
{ if (!(layers instanceof Array)) layers = [layers];
for (var i=0; i<layers.length; i++) {
var l = { layer: layers[i] }
if (this.getMap()) {
l.precompose = layers[i].on('precompose', this.precompose_.bind(this));
l.postcompose = layers[i].on('postcompose', this.postcompose_.bind(this));
this.getMap().renderSync();
}
this.layers_.push(l);
}
}
/** Remove a layer to clip
* @param {ol.layer|Array<ol.layer>} layer to clip
*/
ol.interaction.Clip.prototype.removeLayer = function(layers)
{ if (!(layers instanceof Array)) layers = [layers];
for (var i=0; i<layers.length; i++)
{ var k;
for (k=0; k<this.layers_.length; k++)
{ if (this.layers_[k].layer===layers[i])
{ break;
}
}
if (k!=this.layers_.length && this.getMap())
{ if (this.layers_[k].precompose) ol.Observable.unByKey(this.layers_[k].precompose);
if (this.layers_[k].postcompose) ol.Observable.unByKey(this.layers_[k].postcompose);
this.layers_.splice(k,1);
this.getMap().renderSync();
}
}
}
/** Set position of the clip
* @param {ol.Pixel|ol.MapBrowserEvent}
*/
ol.interaction.Clip.prototype.setPosition = function(e)
{ if (e.pixel) this.pos = e.pixel;
else
{ if (e && e instanceof Array) this.pos = e;
else e = [-10000000,-10000000];
}
if (this.getMap()) this.getMap().renderSync();
}
/* @private
*/
ol.interaction.Clip.prototype.precompose_ = function(e)
{ var ctx = e.context;
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.beginPath();
ctx.arc (this.pos[0]*ratio, this.pos[1]*ratio, this.radius*ratio, 0, 2*Math.PI);
ctx.clip();
}
/* @private
*/
ol.interaction.Clip.prototype.postcompose_ = function(e)
{ e.context.restore();
};
/**
* Activate or deactivate the interaction.
* @param {boolean} active Active.
* @observable
* @api
*/
ol.interaction.Clip.prototype.setActive = function(b)
{ ol.interaction.Pointer.prototype.setActive.call (this, b);
if(b) {
for(var i=0; i<this.layers_.length; i++) {
this.layers_[i].precompose = this.layers_[i].layer.on('precompose', this.precompose_.bind(this));
this.layers_[i].postcompose = this.layers_[i].layer.on('postcompose', this.postcompose_.bind(this));
}
} else {
for(var i=0; i<this.layers_.length; i++) {
if (this.layers_[i].precompose) ol.Observable.unByKey(this.layers_[i].precompose);
if (this.layers_[i].postcompose) ol.Observable.unByKey(this.layers_[i].postcompose);
this.layers_[i].precompose = this.layers_[i].postcompose = null;
}
}
if (this.getMap()) this.getMap().renderSync();
}
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction draw hole
* @constructor
* @extends {ol.interaction.Interaction}
* @fires drawstart, drawend
* @param {olx.interaction.DrawHoleOptions} options extend olx.interaction.DrawOptions
* @param {Array<ol.layer.Vector> | undefined} options.layers A list of layers from which polygons should be selected. Alternatively, a filter function can be provided. default: all visible layers
*/
ol.interaction.DrawHole = function(options)
{ if (!options) options = {};
var self = this;
// Select interaction for the current feature
this._select = new ol.interaction.Select();
this._select.setActive(false);
// Geometry function that test points inside the current
var geometryFn, geomFn = options.geometryFunction;
if (geomFn)
{ geometryFn = function(c,g)
{ g = self._geometryFn (c, g);
return geomFn (c,g);
}
}
else
{ geometryFn = function(c,g) { return self._geometryFn (c, g); }
}
// Create draw interaction
options.type = "Polygon";
options.geometryFunction = geometryFn;
ol.interaction.Draw.call(this, options);
// Layer filter function
if (options.layers)
{ if (typeof (options.layers) === 'function') this.layers_ = options.layers;
else if (options.layers.indexOf)
{ this.layers_ = function(l)
{ return (options.layers.indexOf(l) >= 0);
};
}
}
// Start drawing if inside a feature
this.on('drawstart', this._startDrawing.bind(this));
// End drawing add the hole to the current Polygon
this.on('drawend', this._finishDrawing.bind(this));
};
ol.inherits(ol.interaction.DrawHole, ol.interaction.Draw);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.DrawHole.prototype.setMap = function(map)
{ if (this.getMap()) this.getMap().removeInteraction(this._select);
if (map) map.addInteraction(this._select);
ol.interaction.Draw.prototype.setMap.call (this, map);
};
/**
* Activate/deactivate the interaction
* @param {boolean}
* @api stable
*/
ol.interaction.DrawHole.prototype.setActive = function(b)
{ this._select.getFeatures().clear();
ol.interaction.Draw.prototype.setActive.call (this, b);
};
/**
* Remove last point of the feature currently being drawn
* (test if points to remove before).
*/
ol.interaction.DrawHole.prototype.removeLastPoint = function()
{ if (this._feature && this._feature.getGeometry().getCoordinates()[0].length>2)
{ ol.interaction.Draw.prototype.removeLastPoint.call(this);
}
};
/**
* Get the current polygon to hole
* @return {ol.Feature}
*/
ol.interaction.DrawHole.prototype.getPolygon = function()
{ return this._polygon;
// return this._select.getFeatures().item(0).getGeometry();
};
/**
* Get current feature to add a hole and start drawing
* @param {ol.interaction.Draw.Event} e
* @private
*/
ol.interaction.DrawHole.prototype._startDrawing = function(e)
{ var map = this.getMap();
var layersFilter = this.layers_;
this._feature = e.feature;
var coord = e.feature.getGeometry().getCoordinates()[0][0];
// Check object under the pointer
var features = map.getFeaturesAtPixel(
map.getPixelFromCoordinate(coord),
{ layerFilter: layersFilter
}
);
var current = null;
if (features)
{ var poly = features[0].getGeometry();
if (poly.getType() === "Polygon"
&& poly.intersectsCoordinate(coord)) {
this._polygonIndex = false;
this._polygon = poly;
current = features[0];
}
else if (poly.getType() === "MultiPolygon"
&& poly.intersectsCoordinate(coord)) {
for (var i=0, p; p=poly.getPolygon(i); i++) {
if (p.intersectsCoordinate(coord)) {
this._polygonIndex = i;
this._polygon = p;
current = features[0];
break;
}
}
}
else current = null;
}
console.log(this._polygonIndex)
this._select.getFeatures().clear();
if (!current)
{ this.setActive(false);
this.setActive(true);
}
else
{ this._select.getFeatures().push(current);
}
};
/**
* Stop drawing and add the sketch feature to the target feature.
* @param {ol.interaction.Draw.Event} e
* @private
*/
ol.interaction.DrawHole.prototype._finishDrawing = function(e)
{ // The feature is the hole
e.hole = e.feature;
// Get the current feature
e.feature = this._select.getFeatures().item(0);
// Create the hole
var c = e.hole.getGeometry().getCoordinates()[0];
if (c.length > 3) {
if (this._polygonIndex!==false) {
var geom = e.feature.getGeometry();
var newGeom = new ol.geom.MultiPolygon();
for (var i=0, pi; pi=geom.getPolygon(i); i++) {
if (i===this._polygonIndex) {
pi.appendLinearRing(new ol.geom.LinearRing(c));
newGeom.appendPolygon(pi);
}
else newGeom.appendPolygon(pi);
}
e.feature.setGeometry(newGeom);
} else {
this.getPolygon().appendLinearRing(new ol.geom.LinearRing(c));
}
}
// reset
this._feature = null;
this._select.getFeatures().clear();
};
/**
* Function that is called when a geometry's coordinates are updated.
* @param {Array<ol.coordinate>} coordinates
* @param {ol.geom.Polygon} geometry
* @return {ol.geom.Polygon}
* @private
*/
ol.interaction.DrawHole.prototype._geometryFn = function(coordinates, geometry)
{ var coord = coordinates[0].pop();
if (!this.getPolygon() || this.getPolygon().intersectsCoordinate(coord))
{ this.lastOKCoord = [coord[0],coord[1]];
}
coordinates[0].push([this.lastOKCoord[0],this.lastOKCoord[1]]);
if (geometry)
{ geometry.setCoordinates([coordinates[0].concat([coordinates[0][0]])]);
}
else
{ geometry = new ol.geom.Polygon(coordinates);
}
return geometry;
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction rotate
* @constructor
* @extends {ol.interaction.Interaction}
* @fires drawstart, drawing, drawend, drawcancel
* @param {olx.interaction.TransformOptions} options
* @param {Array<ol.Layer>} source Destination source for the drawn features
* @param {ol.Collection<ol.Feature>} features Destination collection for the drawn features
* @param {ol.style.Style | Array.<ol.style.Style> | ol.style.StyleFunction | undefined} style style for the sketch
* @param {integer} sides number of sides, default 0 = circle
* @param { ol.events.ConditionType | undefined } squareCondition A function that takes an ol.MapBrowserEvent and returns a boolean to draw square features.
* @param { ol.events.ConditionType | undefined } centerCondition A function that takes an ol.MapBrowserEvent and returns a boolean to draw centered features.
* @param { bool } canRotate Allow rotation when centered + square, default: true
* @param { number } clickTolerance click tolerance on touch devices, default: 6
* @param { number } maxCircleCoordinates Maximum number of point on a circle, default: 100
*/
ol.interaction.DrawRegular = function(options)
{ if (!options) options={};
var self = this;
this.squaredClickTolerance_ = options.clickTolerance ? options.clickTolerance * options.clickTolerance : 36;
this.maxCircleCoordinates_ = options.maxCircleCoordinates || 100;
// Collection of feature to transform
this.features_ = options.features;
// List of layers to transform
this.source_ = options.source;
// Square condition
this.squareFn_ = options.squareCondition;
// Centered condition
this.centeredFn_ = options.centerCondition;
// Allow rotation when centered + square
this.canRotate_ = (options.canRotate !== false);
// Specify custom geometry name
this.geometryName_ = options.geometryName
// Number of sides (default=0: circle)
this.setSides(options.sides);
// Style
var white = [255, 255, 255, 1];
var blue = [0, 153, 255, 1];
var width = 3;
var defaultStyle = [
new ol.style.Style({
stroke: new ol.style.Stroke({ color: white, width: width + 2 })
}),
new ol.style.Style({
image: new ol.style.Circle({
radius: width * 2,
fill: new ol.style.Fill({ color: blue }),
stroke: new ol.style.Stroke({ color: white, width: width / 2 })
}),
stroke: new ol.style.Stroke({ color: blue, width: width }),
fill: new ol.style.Fill({
color: [255, 255, 255, 0.5]
})
})
];
// Create a new overlay layer for the sketch
this.sketch_ = new ol.Collection();
this.overlayLayer_ = new ol.layer.Vector(
{ source: new ol.source.Vector({
features: this.sketch_,
useSpatialIndex: false
}),
name:'DrawRegular overlay',
displayInLayerSwitcher: false,
style: options.style || defaultStyle
});
ol.interaction.Interaction.call(this,
{
/*
handleDownEvent: this.handleDownEvent_,
handleMoveEvent: this.handleMoveEvent_,
handleUpEvent: this.handleUpEvent_,
*/
handleEvent: this.handleEvent_
});
};
ol.inherits(ol.interaction.DrawRegular, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.DrawRegular.prototype.setMap = function(map)
{ if (this.getMap()) this.getMap().removeLayer(this.overlayLayer_);
ol.interaction.Interaction.prototype.setMap.call (this, map);
this.overlayLayer_.setMap(map);
};
/**
* Activate/deactivate the interaction
* @param {boolean}
* @api stable
*/
ol.interaction.DrawRegular.prototype.setActive = function(b)
{ this.reset();
ol.interaction.Interaction.prototype.setActive.call (this, b);
}
/**
* Reset the interaction
* @api stable
*/
ol.interaction.DrawRegular.prototype.reset = function()
{ this.overlayLayer_.getSource().clear();
this.started_ = false;
}
/**
* Set the number of sides.
* @param {int} number of sides.
* @api stable
*/
ol.interaction.DrawRegular.prototype.setSides = function (nb)
{ nb = parseInt(nb);
this.sides_ = nb>2 ? nb : 0;
}
/**
* Allow rotation when centered + square
* @param {bool}
* @api stable
*/
ol.interaction.DrawRegular.prototype.canRotate = function (b)
{ if (b===true || b===false) this.canRotate_ = b;
return this.canRotate_;
}
/**
* Get the number of sides.
* @return {int} number of sides.
* @api stable
*/
ol.interaction.DrawRegular.prototype.getSides = function ()
{ return this.sides_;
}
/** Default start angle array for each sides
*/
ol.interaction.DrawRegular.prototype.startAngle =
{ 'default':Math.PI/2,
3: -Math.PI/2,
4: Math.PI/4
};
/** Get geom of the current drawing
* @return {ol.geom.Polygon | ol.geom.Point}
*/
ol.interaction.DrawRegular.prototype.getGeom_ = function ()
{ this.overlayLayer_.getSource().clear();
if (!this.center_) return false;
var g;
if (this.coord_)
{ var center = this.center_;
var coord = this.coord_;
// Special case: circle
if (!this.sides_ && this.square_ && !this.centered_){
center = [(coord[0] + center[0])/2, (coord[1] + center[1])/2];
var d = [coord[0] - center[0], coord[1] - center[1]];
var r = Math.sqrt(d[0]*d[0]+d[1]*d[1]);
var circle = new ol.geom.Circle(center, r, 'XY');
// Optimize points on the circle
var centerPx = this.getMap().getPixelFromCoordinate(center);
var dmax = Math.max (100, Math.abs(centerPx[0]-this.coordPx_[0]), Math.abs(centerPx[1]-this.coordPx_[1]));
dmax = Math.min ( this.maxCircleCoordinates_, Math.round(dmax / 3 ));
return ol.geom.Polygon.fromCircle (circle, dmax, 0);
}
else {
var hasrotation = this.canRotate_ && this.centered_ && this.square_;
var d = [coord[0] - center[0], coord[1] - center[1]];
if (this.square_ && !hasrotation)
{ //var d = [coord[0] - center[0], coord[1] - center[1]];
var dm = Math.max (Math.abs(d[0]), Math.abs(d[1]));
coord[0] = center[0] + (d[0]>0 ? dm:-dm);
coord[1] = center[1] + (d[1]>0 ? dm:-dm);
}
var r = Math.sqrt(d[0]*d[0]+d[1]*d[1]);
if (r>0)
{ var circle = new ol.geom.Circle(center, r, 'XY');
var a;
if (hasrotation) a = Math.atan2(d[1], d[0]);
else a = this.startAngle[this.sides_] || this.startAngle['default'];
if (this.sides_) g = ol.geom.Polygon.fromCircle (circle, this.sides_, a);
else
{ // Optimize points on the circle
var centerPx = this.getMap().getPixelFromCoordinate(this.center_);
var dmax = Math.max (100, Math.abs(centerPx[0]-this.coordPx_[0]), Math.abs(centerPx[1]-this.coordPx_[1]));
dmax = Math.min ( this.maxCircleCoordinates_, Math.round(dmax / (this.centered_ ? 3:5) ));
g = ol.geom.Polygon.fromCircle (circle, dmax, 0);
}
if (hasrotation) return g;
// Scale polygon to fit extent
var ext = g.getExtent();
if (!this.centered_) center = this.center_;
else center = [ 2*this.center_[0]-this.coord_[0], 2*this.center_[1]-this.coord_[1] ];
var scx = (center[0] - coord[0]) / (ext[0] - ext[2]);
var scy = (center[1] - coord[1]) / (ext[1] - ext[3]);
if (this.square_)
{ var sc = Math.min(Math.abs(scx),Math.abs(scy));
scx = Math.sign(scx)*sc;
scy = Math.sign(scy)*sc;
}
var t = [ center[0] - ext[0]*scx, center[1] - ext[1]*scy ];
g.applyTransform(function(g1, g2, dim)
{ for (var i=0; i<g1.length; i+=dim)
{ g2[i] = g1[i]*scx + t[0];
g2[i+1] = g1[i+1]*scy + t[1];
}
return g2;
});
return g;
}
}
}
// No geom => return a point
return new ol.geom.Point(this.center_);
};
/** Draw sketch
* @return {ol.Feature} The feature being drawn.
*/
ol.interaction.DrawRegular.prototype.drawSketch_ = function(evt)
{ this.overlayLayer_.getSource().clear();
if (evt)
{ this.square_ = this.squareFn_ ? this.squareFn_(evt) : evt.originalEvent.shiftKey;
this.centered_ = this.centeredFn_ ? this.centeredFn_(evt) : evt.originalEvent.metaKey || evt.originalEvent.ctrlKey;
var g = this.getGeom_();
if (g)
{ var f = this.feature_;
if (this.geometryName_) f.setGeometryName(this.geometryName_)
f.setGeometry (g);
this.overlayLayer_.getSource().addFeature(f);
if (this.coord_ && this.square_ && ((this.canRotate_ && this.centered_ && this.coord_) || (!this.sides_ && !this.centered_)))
{ this.overlayLayer_.getSource().addFeature(new ol.Feature(new ol.geom.LineString([this.center_,this.coord_])));
}
return f;
}
}
};
/** Draw sketch (Point)
*/
ol.interaction.DrawRegular.prototype.drawPoint_ = function(pt, noclear)
{ if (!noclear) this.overlayLayer_.getSource().clear();
this.overlayLayer_.getSource().addFeature(new ol.Feature(new ol.geom.Point(pt)));
};
/**
* @param {ol.MapBrowserEvent} evt Map browser event.
*/
ol.interaction.DrawRegular.prototype.handleEvent_ = function(evt)
{ switch (evt.type)
{ case "pointerdown": {
this.downPx_ = evt.pixel;
this.start_(evt);
}
break;
case "pointerup":
// Started and fisrt move
if (this.started_ && this.coord_)
{ var dx = this.downPx_[0] - evt.pixel[0];
var dy = this.downPx_[1] - evt.pixel[1];
if (dx*dx + dy*dy <= this.squaredClickTolerance_)
{ // The pointer has moved
if ( this.lastEvent == "pointermove" )
{ this.end_(evt);
}
// On touch device there is no move event : terminate = click on the same point
else
{ dx = this.upPx_[0] - evt.pixel[0];
dy = this.upPx_[1] - evt.pixel[1];
if ( dx*dx + dy*dy <= this.squaredClickTolerance_)
{ this.end_(evt);
}
else
{ this.handleMoveEvent_(evt);
this.drawPoint_(evt.coordinate,true);
}
}
}
}
this.upPx_ = evt.pixel;
break;
case "pointerdrag":
if (this.started_)
{ var centerPx = this.getMap().getPixelFromCoordinate(this.center_);
var dx = centerPx[0] - evt.pixel[0];
var dy = centerPx[1] - evt.pixel[1];
if (dx*dx + dy*dy <= this.squaredClickTolerance_)
{ this.reset();
}
}
break;
case "pointermove":
if (this.started_)
{ var dx = this.downPx_[0] - evt.pixel[0];
var dy = this.downPx_[1] - evt.pixel[1];
if (dx*dx + dy*dy > this.squaredClickTolerance_)
{ this.handleMoveEvent_(evt);
this.lastEvent = evt.type;
}
}
break;
default:
this.lastEvent = evt.type;
// Prevent zoom in on dblclick
if (this.started_ && evt.type==='dblclick')
{ //evt.stopPropagation();
return false;
}
break;
}
return true;
}
/** Stop drawing.
*/
ol.interaction.DrawRegular.prototype.finishDrawing = function()
{ if (this.started_ && this.coord_)
{ this.end_({ pixel: this.upPx_, coordinate: this.coord_});
}
};
/**
* @param {ol.MapBrowserEvent} evt Event.
*/
ol.interaction.DrawRegular.prototype.handleMoveEvent_ = function(evt)
{ if (this.started_)
{ this.coord_ = evt.coordinate;
this.coordPx_ = evt.pixel;
var f = this.drawSketch_(evt);
this.dispatchEvent({
type:'drawing',
feature: f,
pixel: evt.pixel,
startCoordinate: this.center_,
coordinate: evt.coordinate,
square: this.square_,
centered: this.centered_
});
}
else
{ this.drawPoint_(evt.coordinate);
}
};
/** Start an new draw
* @param {ol.MapBrowserEvent} evt Map browser event.
* @return {boolean} `false` to stop the drag sequence.
*/
ol.interaction.DrawRegular.prototype.start_ = function(evt)
{ if (!this.started_)
{ this.started_ = true;
this.center_ = evt.coordinate;
this.coord_ = null;
var f = this.feature_ = new ol.Feature();
this.drawSketch_(evt);
this.dispatchEvent({ type:'drawstart', feature: f, pixel: evt.pixel, coordinate: evt.coordinate });
}
else
{ this.coord_ = evt.coordinate;
}
};
/** End drawing
* @param {ol.MapBrowserEvent} evt Map browser event.
* @return {boolean} `false` to stop the drag sequence.
*/
ol.interaction.DrawRegular.prototype.end_ = function(evt)
{ this.coord_ = evt.coordinate;
this.started_ = false;
// Add new feature
if (this.coord_ && this.center_[0]!=this.coord_[0] && this.center_[1]!=this.coord_[1])
{ var f = this.feature_;
if (this.geometryName_) f.setGeometryName(this.geometryName_)
f.setGeometry(this.getGeom_());
if (this.source_) this.source_.addFeature(f);
else if (this.features_) this.features_.push(f);
this.dispatchEvent({ type:'drawend', feature: f, pixel: evt.pixel, coordinate: evt.coordinate, square: this.square_, centered: this.centered_ });
}
else
{ this.dispatchEvent({ type:'drawcancel', feature: null, pixel: evt.pixel, coordinate: evt.coordinate, square: this.square_, centered: this.centered_ });
}
this.center_ = this.coord_ = null;
this.drawSketch_();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction DrawTouch :
* @constructor
* @extends {ol.interaction.CenterTouch}
* @param {olx.interaction.DrawOptions} options
* - source {ol.source.Vector | undefined} Destination source for the drawn features.
* - type {ol.geom.GeometryType} Drawing type ('Point', 'LineString', 'Polygon') not ('MultiPoint', 'MultiLineString', 'MultiPolygon' or 'Circle'). Required.
* - tap {boolean} enable on tap, default true
* Inherited params
* - targetStyle {ol.style.Style|Array<ol.style.Style>} a style to draw the target point, default cross style
* - composite {string} composite operation : difference|multiply|xor|screen|overlay|darken|lighter|lighten|...
*/
ol.interaction.DrawTouch = function(options)
{ var options = options||{};
var self = this;
options.handleEvent = function(e)
{ if (this.get("tap"))
{ switch (e.type)
{ case "singleclick":
this.addPoint();
break;
case "dblclick":
this.addPoint();
this.finishDrawing();
return false;
break;
default: break;
}
}
return true;
}
ol.interaction.CenterTouch.call(this, options);
this.typeGeom_ = options.type;
this.source_ = options.source;
this.set("tap", (options.tap!==false));
// Style
var white = [255, 255, 255, 1];
var blue = [0, 153, 255, 1];
var width = 3;
var defaultStyle = [
new ol.style.Style({
stroke: new ol.style.Stroke({ color: white, width: width + 2 })
}),
new ol.style.Style({
image: new ol.style.Circle({
radius: width * 2,
fill: new ol.style.Fill({ color: blue }),
stroke: new ol.style.Stroke({ color: white, width: width / 2 })
}),
stroke: new ol.style.Stroke({ color: blue, width: width }),
fill: new ol.style.Fill({
color: [255, 255, 255, 0.5]
})
})
];
this.overlay_ = new ol.layer.Vector(
{ source: new ol.source.Vector({useSpatialIndex: false }),
style: defaultStyle
});
this.geom_ = [];
};
ol.inherits(ol.interaction.DrawTouch, ol.interaction.CenterTouch);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.DrawTouch.prototype.setMap = function(map) {
if (this._listener.drawSketch) ol.Observable.unByKey(this._listener.drawSketch);
this._listener.drawSketch = null;
ol.interaction.CenterTouch.prototype.setMap.call (this, map);
this.overlay_.setMap(map);
if (this.getMap())
{ this._listener.drawSketch = this.getMap().on("postcompose", this.drawSketchLink_.bind(this));
}
};
/** Start drawing and add the sketch feature to the target layer.
* The ol.interaction.Draw.EventType.DRAWSTART event is dispatched before inserting the feature.
*/
ol.interaction.DrawTouch.prototype.startDrawing = function()
{ this.geom_ = [];
this.addPoint();
};
/** Get geometry type
* @return {ol.geom.GeometryType}
*/
ol.interaction.DrawTouch.prototype.getGeometryType = function()
{ return this.typeGeom_;
};
/** Start drawing and add the sketch feature to the target layer.
* The ol.interaction.Draw.EventType.DRAWEND event is dispatched before inserting the feature.
*/
ol.interaction.DrawTouch.prototype.finishDrawing = function()
{ if (!this.getMap()) return;
var f;
switch (this.typeGeom_)
{ case "LineString":
if (this.geom_.length > 1) f = new ol.Feature(new ol.geom.LineString(this.geom_));
break;
case "Polygon":
// Close polygon
if (this.geom_[this.geom_.length-1] != this.geom_[0])
{ this.geom_.push(this.geom_[0]);
}
// Valid ?
if (this.geom_.length > 3)
{ f = new ol.Feature(new ol.geom.Polygon([ this.geom_ ]));
}
break;
default: break;
}
if (f) this.source_.addFeature (f);
// reset
this.geom_ = [];
this.drawSketch_();
}
/** Add a new Point to the drawing
*/
ol.interaction.DrawTouch.prototype.addPoint = function()
{ if (!this.getMap()) return;
this.geom_.push(this.getPosition());
switch (this.typeGeom_)
{ case "Point":
var f = new ol.Feature( new ol.geom.Point (this.geom_.pop()));
this.source_.addFeature(f);
break;
case "LineString":
case "Polygon":
this.drawSketch_();
break;
default: break;
}
}
/** Remove last point of the feature currently being drawn.
*/
ol.interaction.DrawTouch.prototype.removeLastPoint = function()
{ if (!this.getMap()) return;
this.geom_.pop();
this.drawSketch_();
}
/** Draw sketch
* @private
*/
ol.interaction.DrawTouch.prototype.drawSketch_ = function()
{ if (!this.overlay_) return;
this.overlay_.getSource().clear();
if (this.geom_.length)
{ var f;
if (this.typeGeom_ == "Polygon")
{ f = new ol.Feature(new ol.geom.Polygon([this.geom_]));
this.overlay_.getSource().addFeature(f);
}
var geom = new ol.geom.LineString(this.geom_);
f = new ol.Feature(geom);
this.overlay_.getSource().addFeature(f);
f = new ol.Feature( new ol.geom.Point (this.geom_.slice(-1).pop()) );
this.overlay_.getSource().addFeature(f);
}
}
/** Draw contruction lines on postcompose
* @private
*/
ol.interaction.DrawTouch.prototype.drawSketchLink_ = function(e)
{ if (!this.getActive() || !this.getPosition()) return;
var ctx = e.context;
ctx.save();
var p, pt = this.getMap().getPixelFromCoordinate(this.getPosition());
var ratio = e.frameState.pixelRatio || 1;
ctx.scale(ratio,ratio);
ctx.strokeStyle = "rgba(0, 153, 255, 1)";
ctx.lineWidth = 1;
ctx.beginPath();
ctx.arc (pt[0],pt[1], 5, 0, 2*Math.PI);
ctx.stroke();
if (this.geom_.length)
{ p = this.getMap().getPixelFromCoordinate(this.geom_[this.geom_.length-1]);
ctx.beginPath();
ctx.moveTo(p[0],p[1]);
ctx.lineTo(pt[0],pt[1]);
if (this.typeGeom_ == "Polygon")
{ p = this.getMap().getPixelFromCoordinate(this.geom_[0]);
ctx.lineTo(p[0],p[1]);
}
ctx.stroke();
}
ctx.restore();
}
/**
* Activate or deactivate the interaction.
* @param {boolean} active Active.
* @observable
* @api
*/
ol.interaction.DrawTouch.prototype.setActive = function(b)
{ ol.interaction.CenterTouch.prototype.setActive.call (this, b);
if (!b) this.geom_ = [];
this.drawSketch_();
}
/** Extend DragAndDrop choose drop zone + fires loadstart, loadend
* @require jQuery
*
* @constructor
* @extends {ol.interaction.DragAndDrop}
* @fires loadstart, loadend, addfeatures
* @param {ol.dropfile.options} flashlight options param
* - zone {string} selector for the drop zone, default document
* - projection {ol.projection} default projection of the map
* - formatConstructors {Array<function(new:ol.format.Feature)>|undefined} Format constructors, default [ ol.format.GPX, ol.format.GeoJSON, ol.format.IGC, ol.format.KML, ol.format.TopoJSON ]
* - accept {Array<string>|undefined} list of eccepted format, default ["gpx","json","geojson","igc","kml","topojson"]
*/
ol.interaction.DropFile = function(options)
{ options = options||{};
ol.interaction.DragAndDrop.call(this, {});
var zone = options.zone || document;
$(zone).on('dragenter', this.onstop );
$(zone).on('dragover', this.onstop );
$(zone).on('dragleave', this.onstop );
// Options
this.formatConstructors_ = options.formatConstructors || [ ol.format.GPX, ol.format.GeoJSON, ol.format.IGC, ol.format.KML, ol.format.TopoJSON ];
this.projection_ = options.projection;
this.accept_ = options.accept || ["gpx","json","geojson","igc","kml","topojson"];
var self = this;
$(zone).on('drop', function(e){ return self.ondrop(e.originalEvent); });
};
ol.inherits(ol.interaction.DropFile, ol.interaction.DragAndDrop);
/** Set the map
*/
ol.interaction.DropFile.prototype.setMap = function(map)
{ ol.interaction.Interaction.prototype.setMap.call(this, map);
};
/** Do somthing when over
*/
ol.interaction.DropFile.prototype.onstop = function(e)
{ e.preventDefault();
e.stopPropagation();
return false;
}
/** Do somthing when over
*/
ol.interaction.DropFile.prototype.ondrop = function(e)
{ if (e.dataTransfer && e.dataTransfer.files.length)
{ var self = this;
e.preventDefault();
e.stopPropagation();
// fetch FileList object
var files = e.dataTransfer.files; // e.originalEvent.target.files ?
// process all File objects
var file;
var pat = /\.([0-9a-z]+)(?=[?#])|(\.)(?:[\w]+)$/;
for (var i=0; file=files[i]; i++)
{ var ex = file.name.match(pat)[0];
self.dispatchEvent({ type:'loadstart', file: file, filesize: file.size, filetype: file.type, fileextension: ex, projection: projection, target: self });
// Load file
features = [];
var reader = new FileReader();
var projection = this.projection_ || this.getMap().getView().getProjection();
var formatConstructors = this.formatConstructors_
if (!projection) return;
function tryReadFeatures (format, result, options)
{ try
{ return format.readFeatures(result, options);
} catch (e) {}
}
var theFile = file;
reader.onload = function(e)
{ var result = e.target.result;
var features = [];
var i, ii;
for (i = 0, ii = formatConstructors.length; i < ii; ++i)
{ var formatConstructor = formatConstructors[i];
var format = new formatConstructor();
features = tryReadFeatures(format, result, { featureProjection: projection });
if (features && features.length > 0)
{ self.dispatchEvent({ type:'addfeatures', features: features, file: theFile, projection: projection, target: self });
self.dispatchEvent({ type:'loadend', features: features, file: theFile, projection: projection, target: self });
return;
}
}
self.dispatchEvent({ type:'loadend', file: theFile, target: self });
};
reader.readAsText(file);
};
}
else {}
return false;
};
/**
* @constructor
* @extends {ol.interaction.Pointer}
* @param {ol.flashlight.options} flashlight options param
* - color {ol.Color} light color, default transparent
* - fill {ol.Color} fill color, default rgba(0,0,0,0.8)
* - radius {number} radius of the flash
*/
ol.interaction.Flashlight = function(options) {
ol.interaction.Pointer.call(this,
{ handleDownEvent: this.setPosition,
handleMoveEvent: this.setPosition
});
// Default options
options = options||{};
this.pos = false;
this.radius = (options.radius||100);
this.setColor(options);
};
ol.inherits(ol.interaction.Flashlight, ol.interaction.Pointer);
/** Set the map > start postcompose
*/
ol.interaction.Flashlight.prototype.setMap = function(map) {
if (this.getMap()) {
this.getMap().render();
}
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.interaction.Pointer.prototype.setMap.call(this, map);
if (map) {
this._listener = map.on('postcompose', this.postcompose_.bind(this));
}
}
/** Set flashlight radius
* @param {integer} radius
*/
ol.interaction.Flashlight.prototype.setRadius = function(radius)
{ this.radius = radius
if (this.getMap()) this.getMap().renderSync();
}
/** Set flashlight color
* @param {ol.flashlight.options} flashlight options param
* - color {ol.Color} light color, default transparent
* - fill {ol.Color} fill color, default rgba(0,0,0,0.8)
*/
ol.interaction.Flashlight.prototype.setColor = function(options)
{ // Backcolor
var color = (options.fill ? options.fill : [0,0,0,0.8]);
var c = ol.color.asArray(color);
this.startColor = ol.color.asString(c);
// Halo color
var endColor;
if (options.color)
{ c = this.endColor = ol.color.asString(ol.color.asArray(options.color)||options.color);
}
else
{ c[3] = 0
this.endColor = ol.color.asString(c);
}
c[3] = 0.1;
this.midColor = ol.color.asString(c);
if (this.getMap()) this.getMap().renderSync();
}
/** Set position of the flashlight
* @param {ol.Pixel|ol.MapBrowserEvent}
*/
ol.interaction.Flashlight.prototype.setPosition = function(e)
{ if (e.pixel) this.pos = e.pixel;
else this.pos = e;
if (this.getMap())
{ this.getMap().renderSync();
}
}
/** Postcompose function
*/
ol.interaction.Flashlight.prototype.postcompose_ = function(e)
{ var ctx = e.context;
var ratio = e.frameState.pixelRatio;
var w = ctx.canvas.width;
var h = ctx.canvas.height;
ctx.save();
ctx.scale(ratio,ratio);
if (!this.pos)
{ ctx.fillStyle = this.startColor;
ctx.fillRect( 0,0,w,h );
}
else
{ var d = Math.max(w, h);
// reveal wherever we drag
var radGrd = ctx.createRadialGradient( this.pos[0], this.pos[1], w*this.radius/d, this.pos[0], this.pos[1], h*this.radius/d );
radGrd.addColorStop( 0, this.startColor );
radGrd.addColorStop( 0.8, this.midColor );
radGrd.addColorStop( 1, this.endColor );
ctx.fillStyle = radGrd;
ctx.fillRect( this.pos[0] - d, this.pos[1] - d, 2*d, 2*d );
}
ctx.restore();
};
/** Interaction to draw on the current geolocation
* It combines a draw with a ol.Geolocation
* @constructor
* @extends {ol.interaction.Interaction}
* @fires drawstart, drawend, drawing, tracking, follow
* @param {any} options
* @param { ol.Collection.<ol.Feature> | undefined } option.features Destination collection for the drawn features.
* @param { ol.source.Vector | undefined } options.source Destination source for the drawn features.
* @param {ol.geom.GeometryType} options.type Drawing type ('Point', 'LineString', 'Polygon'), default LineString.
* @param {Number | undefined} options.minAccuracy minimum accuracy underneath a new point will be register (if no condition), default 20
* @param {function | undefined} options.condition a function that take a ol.Geolocation object and return a boolean to indicate whether location should be handled or not, default return true if accuraty < minAccuraty
* @param {Object} options.attributes a list of attributes to register as Point properties: {accuracy:true,accuracyGeometry:true,heading:true,speed:true}, default none.
* @param {Number} options.tolerance tolerance to add a new point (in projection unit), use ol.geom.LineString.simplify() method, default 5
* @param {Number} options.zoom zoom for tracking, default 16
* @param {boolean|auto|position|visible} options.followTrack true if you want the interaction to follow the track on the map, default true
* @param { ol.style.Style | Array.<ol.style.Style> | ol.StyleFunction | undefined } options.style Style for sketch features.
*/
ol.interaction.GeolocationDraw = function(options)
{ if (!options) options={};
var self = this;
// Geolocation
var geoloc = this.geolocation = new ol.Geolocation(
({ projection: "EPSG:4326",
trackingOptions:
{ maximumAge: 10000,
enableHighAccuracy: true,
timeout: 600000
}
}));
this.geolocation.on('change', this.draw_.bind(this));
// Current path
this.path_ = [];
this.lastPosition_ = false;
// Default style
var white = [255, 255, 255, 1];
var blue = [0, 153, 255, 1];
var width = 3;
var circle = new ol.style.Circle(
{ radius: width * 2,
fill: new ol.style.Fill({ color: blue }),
stroke: new ol.style.Stroke({ color: white, width: width / 2 })
});
var style =
[ new ol.style.Style(
{ stroke: new ol.style.Stroke({ color: white, width: width + 2 })
}),
new ol.style.Style(
{ stroke: new ol.style.Stroke({ color: blue, width: width }),
fill: new ol.style.Fill({
color: [255, 255, 255, 0.5]
})
})
];
var triangle = new ol.style.RegularShape(
{ radius: width * 3.5,
points: 3,
rotation: 0,
fill: new ol.style.Fill({ color: blue }),
stroke: new ol.style.Stroke({ color: white, width: width / 2 })
});
// stretch the symbol
var c = triangle.getImage();
var ctx = c.getContext("2d");
var c2 = document.createElement('canvas');
c2.width = c2.height = c.width;
c2.getContext("2d").drawImage(c, 0,0);
ctx.clearRect(0,0,c.width,c.height);
ctx.drawImage(c2, 0,0, c.width, c.height, width, 0, c.width-2*width, c.height);
var defaultStyle = function(f)
{ if (f.get('heading')===undefined)
{ style[1].setImage(circle);
}
else
{ style[1].setImage(triangle);
triangle.setRotation( f.get('heading') || 0);
}
return style;
}
// Style for the accuracy geometry
this.locStyle =
{ error: new ol.style.Style({ fill: new ol.style.Fill({ color: [255, 0, 0, 0.2] }) }),
warn: new ol.style.Style({ fill: new ol.style.Fill({ color: [255, 192, 0, 0.2] }) }),
ok: new ol.style.Style({ fill: new ol.style.Fill({ color: [0, 255, 0, 0.2] }) }),
};
// Create a new overlay layer for the sketch
this.overlayLayer_ = new ol.layer.Vector(
{ source: new ol.source.Vector(),
name:'GeolocationDraw overlay',
style: options.style || defaultStyle
});
this.sketch_ = [new ol.Feature(), new ol.Feature(), new ol.Feature()];
this.overlayLayer_.getSource().addFeatures(this.sketch_);
this.features_ = options.features;
this.source_ = options.source;
this.condition_ = options.condition || function(loc) { return loc.getAccuracy() < this.get("minAccuracy") };
// Prevent interaction when tracking
ol.interaction.Interaction.call(this,
{ handleEvent: function()
{ return (!this.get('followTrack') || this.get('followTrack')=='auto');// || !geoloc.getTracking());
}
});
this.set("type", options.type||"LineString");
this.set("attributes", options.attributes||{});
this.set("minAccuracy", options.minAccuracy||20);
this.set("tolerance", options.tolerance||5);
this.set("zoom", options.zoom);
this.setFollowTrack (options.followTrack===undefined ? true : options.followTrack);
this.setActive(false);
};
ol.inherits(ol.interaction.GeolocationDraw, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.GeolocationDraw.prototype.setMap = function(map)
{ if (this.getMap()) this.getMap().removeLayer(this.overlayLayer_);
ol.interaction.Pointer.prototype.setMap.call (this, map);
this.overlayLayer_.setMap(map);
if (map) this.geolocation.setProjection(map.getView().getProjection());
};
/** Activate or deactivate the interaction.
* @param {boolean} active
*/
ol.interaction.GeolocationDraw.prototype.setActive = function(active)
{ ol.interaction.Interaction.prototype.setActive.call(this, active);
this.overlayLayer_.setVisible(active);
if (this.getMap())
{ this.geolocation.setTracking(active);
this.getMap().renderSync();
}
this.pause(!active);
if (active)
{ // Start drawing
this.reset();
this.dispatchEvent({ type:'drawstart', feature: this.sketch_[1]});
}
else
{ var f = this.sketch_[1].clone();
if (f.getGeometry())
{ if (this.features_) this.features_.push(f);
if (this.source_) this.source_.addFeature(f);
this.dispatchEvent({ type:'drawend', feature: f});
}
}
};
/** Reset drawing
*/
ol.interaction.GeolocationDraw.prototype.reset = function()
{ this.sketch_[1].setGeometry();
this.path_ = [];
this.lastPosition_ = false;
};
/** Start tracking = setActive(true)
*/
ol.interaction.GeolocationDraw.prototype.start = function()
{ this.setActive(true);
};
/** Stop tracking = setActive(false)
*/
ol.interaction.GeolocationDraw.prototype.stop = function()
{ this.setActive(false);
};
/** Pause drawing
* @param {boolean} b
*/
ol.interaction.GeolocationDraw.prototype.pause = function(b)
{ this.pause_ = b!==false;
};
/** Is paused
* @return {boolean} b
*/
ol.interaction.GeolocationDraw.prototype.isPaused = function()
{ return this.pause_;
};
/** Enable following the track on the map
* @param {boolean|auto|position|visible} follow,
* false: don't follow,
* true: follow (position+zoom),
* 'position': follow only position,
* 'auto': start following until user move the map,
* 'visible': center when position gets out of the visible extent
*/
ol.interaction.GeolocationDraw.prototype.setFollowTrack = function(follow)
{ this.set('followTrack', follow);
var map = this.getMap();
// Center if wanted
if (follow !== false && !this.lastPosition_ && map)
{ var pos = this.path_[this.path_.length-1];
if (pos)
{ map.getView().animate({
center: pos,
zoom: (follow!="position" ? this.get("zoom") : undefined)
})
}
}
this.lastPosition_ = false;
this.dispatchEvent({ type:'follow', following: follow!==false });
};
/** Add a new point to the current path
* @private
*/
ol.interaction.GeolocationDraw.prototype.draw_ = function(active)
{ var map = this.getMap();
if (!map) return;
// Current location
var loc = this.geolocation;
var accu = loc.getAccuracy();
var pos = loc.getPosition();
pos.push (Math.round((loc.getAltitude()||0)*100)/100);
pos.push (Math.round((new Date()).getTime()/1000));
var p = loc.getAccuracyGeometry();
// Center on point
// console.log(this.get('followTrack'))
switch (this.get('followTrack'))
{ // Follow center + zoom
case true:
// modify zoom
if (this.get('followTrack') == true)
{ map.getView().setZoom( this.get("zoom") || 16 );
if (!ol.extent.containsExtent(map.getView().calculateExtent(map.getSize()), p.getExtent()))
{ map.getView().fit(p.getExtent());
}
}
// Follow position
case 'position':
// modify center
map.getView().setCenter( pos );
break;
// Keep on following
case 'auto':
if (this.lastPosition_)
{ var center = map.getView().getCenter();
// console.log(center,this.lastPosition_)
if (center[0]!=this.lastPosition_[0] || center[1]!=this.lastPosition_[1])
{ //this.dispatchEvent({ type:'follow', following: false });
this.setFollowTrack (false);
}
else
{ map.getView().setCenter( pos );
this.lastPosition_ = pos;
}
}
else
{ map.getView().setCenter( pos );
if (this.get("zoom")) map.getView().setZoom( this.get("zoom") );
this.lastPosition_ = pos;
}
break;
// Force to stay on the map
case 'visible':
if (!ol.extent.containsCoordinate(map.getView().calculateExtent(map.getSize()), pos))
{ map.getView().setCenter (pos);
}
break;
// Don't follow
default: break;
}
// Draw occuracy
var f = this.sketch_[0];
f.setGeometry(p);
if (accu < this.get("minAccuracy")/2) f.setStyle(this.locStyle.ok);
else if (accu < this.get("minAccuracy")) f.setStyle(this.locStyle.warn);
else f.setStyle(this.locStyle.error);
var geo;
if (!this.pause_ && this.condition_.call(this, loc))
{ f = this.sketch_[1];
this.path_.push(pos);
switch (this.get("type"))
{ case "Point":
this.path_ = [pos];
f.setGeometry(new ol.geom.Point(pos, 'XYZM'));
var attr = this.get('attributes');
if (attr.heading) f.set("heading",loc.getHeading());
if (attr.accuracy) f.set("accuracy",loc.getAccuracy());
if (attr.altitudeAccuracy) f.set("altitudeAccuracy",loc.getAltitudeAccuracy());
if (attr.speed) f.set("speed",loc.getSpeed());
break;
case "LineString":
if (this.path_.length>1)
{ geo = new ol.geom.LineString(this.path_, 'XYZM');
geo.simplify (this.get("tolerance"));
f.setGeometry(geo);
}
else f.setGeometry();
break;
case "Polygon":
if (this.path_.length>2)
{ geo = new ol.geom.Polygon([this.path_], 'XYZM');
geo.simplify (this.get("tolerance"));
f.setGeometry(geo);
}
else f.setGeometry();
break;
}
this.dispatchEvent({ type:'drawing', feature: this.sketch_[1], geolocation: loc });
}
this.sketch_[2].setGeometry(new ol.geom.Point(pos));
this.sketch_[2].set("heading",loc.getHeading());
// Drawing
this.dispatchEvent({ type:'tracking', feature: this.sketch_[1], geolocation: loc });
};
/** Interaction hover do to something when hovering a feature
* @constructor
* @extends {ol.interaction.Interaction}
* @fires hover, enter, leave
* @param {olx.interaction.HoverOptions}
* - cursor { string | undefined } css cursor propertie or a function that gets a feature, default: none
* - featureFilter {function | undefined} filter a function with two arguments, the feature and the layer of the feature. Return true to select the feature
* - layerFilter {function | undefined} filter a function with one argument, the layer to test. Return true to test the layer
* - handleEvent { function | undefined } Method called by the map to notify the interaction that a browser event was dispatched to the map. The function may return false to prevent the propagation of the event to other interactions in the map's interactions chain.
*/
ol.interaction.Hover = function(options)
{ if (!options) options={};
var self = this;
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ if (e.type=="pointermove") { self.handleMove_(e); };
if (options.handleEvent) return options.handleEvent(e);
return true;
}
});
this.setFeatureFilter (options.featureFilter);
this.setLayerFilter (options.layerFilter);
this.setCursor (options.cursor);
};
ol.inherits(ol.interaction.Hover, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.Hover.prototype.setMap = function(map)
{ if (this.previousCursor_!==undefined && this.getMap())
{ this.getMap().getTargetElement().style.cursor = this.previousCursor_;
this.previousCursor_ = undefined;
}
ol.interaction.Interaction.prototype.setMap.call (this, map);
};
/**
* Set cursor on hover
* @param { string } cursor css cursor propertie or a function that gets a feature, default: none
* @api stable
*/
ol.interaction.Hover.prototype.setCursor = function(cursor)
{ if (!cursor && this.previousCursor_!==undefined && this.getMap())
{ this.getMap().getTargetElement().style.cursor = this.previousCursor_;
this.previousCursor_ = undefined;
}
this.cursor_ = cursor;
};
/** Feature filter to get only one feature
* @param {function} filter a function with two arguments, the feature and the layer of the feature. Return true to select the feature
*/
ol.interaction.Hover.prototype.setFeatureFilter = function(filter)
{ if (typeof (filter) == 'function') this.featureFilter_ = filter;
else this.featureFilter_ = function(){ return true; };
};
/** Feature filter to get only one feature
* @param {function} filter a function with one argument, the layer to test. Return true to test the layer
*/
ol.interaction.Hover.prototype.setLayerFilter = function(filter)
{ if (typeof (filter) == 'function') this.layerFilter_ = filter;
else this.layerFilter_ = function(){ return true; };
};
/** Get features whenmove
* @param {ol.event} e "move" event
*/
ol.interaction.Hover.prototype.handleMove_ = function(e)
{ var map = this.getMap();
if (map)
{ //var b = map.hasFeatureAtPixel(e.pixel);
var feature, layer;
var self = this;
var b = map.forEachFeatureAtPixel(e.pixel,
function(f, l)
{ if (self.layerFilter_.call(null, l)
&& self.featureFilter_.call(null,f,l))
{ feature = f;
layer = l;
return true;
}
else
{ feature = layer = null;
return false;
}
});
if (b) this.dispatchEvent({ type:"hover", feature:feature, layer:layer, coordinate:e.coordinate, pixel: e.pixel, map: e.map, dragging:e.dragging });
if (this.feature_===feature && this.layer_===layer)
{
}
else
{ this.feature_ = feature;
this.layer_ = layer;
if (feature) this.dispatchEvent({ type:"enter", feature:feature, layer:layer, coordinate:e.coordinate, pixel: e.pixel, map: e.map, dragging:e.dragging });
else this.dispatchEvent({ type:"leave", coordinate:e.coordinate, pixel: e.pixel, map: e.map, dragging:e.dragging });
}
if (this.cursor_)
{ var style = map.getTargetElement().style;
if (b)
{ if (style.cursor != this.cursor_)
{ this.previousCursor_ = style.cursor;
style.cursor = this.cursor_;
}
}
else if (this.previousCursor_ !== undefined)
{ style.cursor = this.previousCursor_;
this.previousCursor_ = undefined;
}
}
}
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction to handle longtouch events
* @constructor
* @extends {ol.interaction.Interaction}
* @param {olx.interaction.LongTouchOptions}
* @param {function | undefined} options.handleLongTouchEvent Function handling "longtouch" events, it will receive a mapBrowserEvent.
* @param {interger | undefined} options.delay The delay for a long touch in ms, default is 1000
*/
ol.interaction.LongTouch = function(options)
{ if (!options) options = {};
this.delay_ = options.delay || 1000;
var ltouch = options.handleLongTouchEvent || function(){};
var _timeout = null;
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ if (this.getActive())
{ switch (e.type)
{ case 'pointerdown':
if (_timeout) clearTimeout(_timeout);
_timeout = setTimeout (function()
{ e.type = "longtouch";
ltouch(e)
}, this.delay_);
break;
/* case 'pointermove': */
case 'pointerdrag':
case 'pointerup':
if (_timeout)
{ clearTimeout(_timeout);
_timeout = null;
}
break;
default: break;;
}
}
else
{ if (_timeout)
{ clearTimeout(_timeout);
_timeout = null;
}
}
return true;
}
});
};
ol.inherits(ol.interaction.LongTouch, ol.interaction.Interaction);
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Modify interaction with a popup to delet a point on touch device
* @constructor
* @fires showpopup
* @fires hidepopup
* @extends {ol.interaction.Modify}
* @param {olx.interaction.ModifyOptions} options
* @param {String|undefined} options.title title to display, default "remove point"
* @param {Boolean|undefined} options.usePopup use a popup, default true
*/
ol.interaction.ModifyTouch = function(options) {
var self = this;
if (!options) options = {};
this._popup = new ol.Overlay.Popup ({
popupClass: options.calssName || 'modifytouch',
positioning: 'bottom-rigth',
offsetBox: 10
});
this._source = options.source;
this._features = options.features;
// popup content
var a = document.createElement('a');
a.appendChild(document.createTextNode(options.title || "remove point"));
a.onclick = function() {
self.removePoint();
};
this.setPopupContent(a);
var pixelTolerance = options.pixelTolerance || 0;
var searchDist = pixelTolerance +5;
// Check if there is a feature to select
options.condition = function(e) {
var features = this.getMap().getFeaturesAtPixel(e.pixel,{
hitTolerance: searchDist
});
var found = false;
if (features) {
var search = this._features;
if (!search) {
var p0 = [e.pixel[0] - searchDist, e.pixel[1] - searchDist]
var p1 = [e.pixel[0] + searchDist, e.pixel[1] + searchDist]
p0 = this.getMap().getCoordinateFromPixel(p0);
p1 = this.getMap().getCoordinateFromPixel(p1);
var ext = ol.extent.boundingExtent([p0,p1]);
search = this._source.getFeaturesInExtent(ext);
}
if (search.getArray) search = search.getArray();
for (var i=0, f; f=features[i]; i++) {
if (search.indexOf(f) >= 0) break;
}
if (f) {
var p0 = e.pixel;
var p1 = f.getGeometry().getClosestPoint(e.coordinate);
p1 = this.getMap().getPixelFromCoordinate(p1);
var dx = p0[0] - p1[0];
var dy = p0[1] - p1[1];
found = (Math.sqrt(dx*dx+dy*dy) < searchDist);
}
}
// Show popup if any
this.showDeleteBt(found ? { type:'show', feature:f, coordinate: e.coordinate } : { type:'hide' });
// Prevent click on the popup
e.preventDefault();
e.stopPropagation();
return true;
};
// Hide popup on insert
options.insertVertexCondition = function(e) {
this.showDeleteBt({ type:'hide' });
return true;
}
ol.interaction.Modify.call(this, options);
this.on(['modifystart','modifyend'], function(){
this.showDeleteBt({ type:'hide', modifying: true });
});
// Use a popup ?
this.set('usePopup', options.usePopup !== false);
};
ol.inherits(ol.interaction.ModifyTouch, ol.interaction.Modify);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.ModifyTouch.prototype.setMap = function(map) {
if (this.getMap()) {
this.getMap().removeOverlay(this._popup);
}
ol.interaction.Modify.prototype.setMap.call (this, map);
if (this.getMap()) {
this.getMap().addOverlay(this._popup);
}
};
/** Activate the interaction and remove popup
* @param {Boolean} b
*/
ol.interaction.ModifyTouch.prototype.setActive = function(b) {
ol.interaction.Modify.prototype.setActive.call (this, b);
this.showDeleteBt({ type:'hide' });
};
/**
* Remove the current point
*/
ol.interaction.ModifyTouch.prototype.removePoint = function() {
// Prevent touch + click on popup
if (new Date() - this._timeout < 200) return;
// Remove point
ol.interaction.Modify.prototype.removePoint.call (this);
this.showDeleteBt({ type:'hide' });
}
/**
* Show the delete button (menu)
* @param {Event} e
* @api stable
*/
ol.interaction.ModifyTouch.prototype.showDeleteBt = function(e) {
if (this.get('usePopup') && e.type==='show') {
this._popup.show(e.coordinate, this._menu);
} else {
this._popup.hide();
}
e.type += 'popup';
this.dispatchEvent(e);
// Date if popup start a timeout to prevent touch + click on the popup
this._timeout = new Date();
};
/**
* Change the popup content
* @param {DOMElement} html
*/
ol.interaction.ModifyTouch.prototype.setPopupContent = function(html) {
this._menu = html;
}
/**
* Get the popup content
* @return {DOMElement}
*/
ol.interaction.ModifyTouch.prototype.getPopupContent = function() {
return this._menu;
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Offset interaction for offseting feature geometry
* @constructor
* @extends {ol.interaction.Pointer}
* @fires offsetstart
* @fires offsetting
* @fires offsetend
* @param {any} options
* @param {ol.layer.Vector | Array<ol.layer.Vector>} options.layers list of feature to transform
* @param {ol.Collection.<ol.Feature>} options.features collection of feature to transform
* @param {ol.source.Vector | undefined} options.source source to duplicate feature when ctrl key is down
* @param {boolean} options.duplicate force feature to duplicate (source must be set)
*/
ol.interaction.Offset = function(options)
{ if (!options) options = {};
// Extend pointer
ol.interaction.Pointer.call(this, {
handleDownEvent: this.handleDownEvent_,
handleDragEvent: this.handleDragEvent_,
handleMoveEvent: this.handleMoveEvent_,
handleUpEvent: this.handleUpEvent_
});
// Collection of feature to transform
this.features_ = options.features;
// List of layers to transform
this.layers_ = options.layers ? (options.layers instanceof Array) ? options.layers:[options.layers] : null;
// duplicate
this.set('duplicate', options.duplicate);
this.source_ = options.source;
// init
this.previousCursor_ = false;
};
ol.inherits(ol.interaction.Offset, ol.interaction.Pointer);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.Offset.prototype.setMap = function(map) {
ol.interaction.Pointer.prototype.setMap.call (this, map);
};
/** Get Feature at pixel
* @param {ol.MapBrowserEvent} evt Map browser event.
* @return {any} a feature and the hit point
* @private
*/
ol.interaction.Offset.prototype.getFeatureAtPixel_ = function(e) {
var self = this;
return this.getMap().forEachFeatureAtPixel(e.pixel,
function(feature, layer) {
var current;
// feature belong to a layer
if (self.layers_) {
for (var i=0; i<self.layers_.length; i++) {
if (self.layers_[i]===layer) {
current = feature;
break;
}
}
}
// feature in the collection
else if (self.features_) {
self.features_.forEach (function(f) {
if (f===feature) {
current = feature
}
});
}
// Others
else {
current = feature;
}
// Only poygon or linestring
var typeGeom = current.getGeometry().getType();
if (current && /Polygon|LineString/.test(typeGeom)) {
if (typeGeom==='Polygon' && current.getGeometry().getCoordinates().length>1) return false;
// test distance
var p = current.getGeometry().getClosestPoint(e.coordinate);
var dx = p[0]-e.coordinate[0];
var dy = p[1]-e.coordinate[1];
var d = Math.sqrt(dx*dx+dy*dy) / e.frameState.viewState.resolution;
if (d<5) {
return {
feature: current,
hit: p,
coordinates: current.getGeometry().getCoordinates(),
geom: current.getGeometry().clone(),
geomType: typeGeom
}
} else {
return false;
}
} else {
return false;
}
}, { hitTolerance: 5 });
};
/**
* @param {ol.MapBrowserEvent} e Map browser event.
* @return {boolean} `true` to start the drag sequence.
* @private
*/
ol.interaction.Offset.prototype.handleDownEvent_ = function(e) {
this.current_ = this.getFeatureAtPixel_(e);
if (this.source_ && (this.get('duplicate') ||e.originalEvent.ctrlKey)) {
this.current_.feature = this.current_.feature.clone();
this.source_.addFeature(this.current_.feature);
}
if (this.current_) {
this.dispatchEvent({ type:'offsetstart', feature: this.current_.feature, offset: 0 });
return true;
} else {
return false;
}
};
/**
* @param {ol.MapBrowserEvent} e Map browser event.
* @private
*/
ol.interaction.Offset.prototype.handleDragEvent_ = function(e) {
var p = this.current_.geom.getClosestPoint(e.coordinate);
var d = ol.coordinate.dist2d(p, e.coordinate);
switch (this.current_.geomType) {
case 'Polygon': {
var seg = ol.coordinate.findSegment(p, this.current_.coordinates[0]).segment;
if (seg) {
var v1 = [ seg[1][0]-seg[0][0], seg[1][1]-seg[0][1] ];
var v2 = [ e.coordinate[0]-p[0], e.coordinate[1]-p[1] ];
if (v1[0]*v2[1] - v1[1]*v2[0] > 0) {
d = -d;
}
var offset = [];
for (var i=0; i<this.current_.coordinates.length; i++) {
offset.push( ol.coordinate.offsetCoords(this.current_.coordinates[i], i==0 ? d : -d) );
}
this.current_.feature.setGeometry(new ol.geom.Polygon(offset));
}
break;
}
case 'LineString': {
var seg = ol.coordinate.findSegment(p, this.current_.coordinates).segment;
if (seg) {
var v1 = [ seg[1][0]-seg[0][0], seg[1][1]-seg[0][1] ];
var v2 = [ e.coordinate[0]-p[0], e.coordinate[1]-p[1] ];
if (v1[0]*v2[1] - v1[1]*v2[0] > 0) {
d = -d;
}
var offset = ol.coordinate.offsetCoords(this.current_.coordinates, d);
this.current_.feature.setGeometry(new ol.geom.LineString(offset));
}
break;
}
default: {
break;
}
}
this.dispatchEvent({ type:'offsetting', feature: this.current_.feature, offset: d, segment: [p, e.coordinate], coordinate: e.coordinate });
};
/**
* @param {ol.MapBrowserEvent} e Map browser event.
* @private
*/
ol.interaction.Offset.prototype.handleUpEvent_ = function(e) {
this.dispatchEvent({ type:'offsetend', feature: this.current_.feature, coordinate: e.coordinate });
this.current_ = false;
};
/**
* @param {ol.MapBrowserEvent} e Event.
* @private
*/
ol.interaction.Offset.prototype.handleMoveEvent_ = function(e) {
var f = this.getFeatureAtPixel_(e);
if (f) {
if (this.previousCursor_ === false) {
this.previousCursor_ = e.map.getTargetElement().style.cursor;
}
e.map.getTargetElement().style.cursor = 'pointer';
} else {
e.map.getTargetElement().style.cursor = this.previousCursor_;
this.previousCursor_ = false;
}
};
/*
Water ripple effect.
Original code (Java) by Neil Wallis
@link http://www.neilwallis.com/java/water.html
Original code (JS) by Sergey Chikuyonok (serge.che@gmail.com)
@link http://chikuyonok.ru
@link http://media.chikuyonok.ru/ripple/
Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
@link https://github.com/Viglino
*/
/**
* @constructor
* @extends {ol.interaction.Pointer}
* @param {ol.flashlight.options} flashlight options param
* - color {ol.Color} light color, default transparent
* - fill {ol.Color} fill color, default rgba(0,0,0,0.8)
* - radius {number} radius of the flash
*/
ol.interaction.Ripple = function(options)
{
ol.interaction.Pointer.call(this,
{ handleDownEvent: this.rainDrop,
handleMoveEvent: this.rainDrop
});
// Default options
options = options||{};
this.riprad = options.radius || 3;
this.ripplemap = [];
this.last_map = [];
// Generate random ripples
this.interval = options.interval;
this.rains (this.interval);
};
ol.inherits(ol.interaction.Ripple, ol.interaction.Pointer);
/** Set the map > start postcompose
*/
ol.interaction.Ripple.prototype.setMap = function(map)
{ if (this.oncompose)
{ ol.Observable.unByKey(oncompose);
if (this.getMap()) this.getMap().render();
}
ol.interaction.Pointer.prototype.setMap.call(this, map);
if (map)
{ this.oncompose = map.on('postcompose', this.postcompose_.bind(this));
}
}
/** Generate random rain drop
* @param {integer} interval
*/
ol.interaction.Ripple.prototype.rains = function(interval)
{ if (this.onrain) clearTimeout (this.onrain);
var self = this;
vdelay = (typeof(interval)=="number" ? interval : 1000)/2;
delay = 3*vdelay/2;
var rnd = Math.random;
function rain()
{ if (self.width) self.rainDrop([rnd() * self.width, rnd() * self.height]);
self.onrain = setTimeout (rain, rnd()*vdelay + delay);
};
// Start raining
if (delay) rain();
}
/** Disturb water at specified point
* @param {ol.Pixel|ol.MapBrowserEvent}
*/
ol.interaction.Ripple.prototype.rainDrop = function(e)
{ if (!this.width) return;
var dx,dy;
if (e.pixel)
{ dx = e.pixel[0]*this.ratio;
dy = e.pixel[1]*this.ratio;
}
else
{ dx = e[0]*this.ratio;
dy = e[1]*this.ratio;
}
dx <<= 0;
dy <<= 0;
for (var j = dy - this.riprad*this.ratio; j < dy + this.riprad*this.ratio; j++)
{ for (var k = dx - this.riprad*this.ratio; k < dx + this.riprad*this.ratio; k++)
{ this.ripplemap[this.oldind + (j * this.width) + k] += 128;
}
}
}
/** Postcompose function
*/
ol.interaction.Ripple.prototype.postcompose_ = function(e)
{ var ctx = e.context;
var canvas = ctx.canvas;
// Initialize when canvas is ready / modified
if (this.width != canvas.width || this.height != canvas.height)
{ this.width = canvas.width;
this.height = canvas.height;
this.ratio = e.frameState.pixelRatio;
this.half_width = this.width >> 1;
this.half_height = this.height >> 1;
this.size = this.width * (this.height + 2) * 2;
this.oldind = this.width;
this.newind = this.width * (this.height + 3);
for (var i = 0; i < this.size; i++) {
this.last_map[i] = this.ripplemap[i] = 0;
}
}
this.texture = ctx.getImageData(0, 0, this.width, this.height);
this.ripple = ctx.getImageData(0, 0, this.width, this.height);
// Run animation
var a, b, data, cur_pixel, new_pixel;
var t = this.oldind; this.oldind = this.newind; this.newind = t;
var i = 0;
var _rd = this.ripple.data,
_td = this.texture.data;
for (var y = 0; y < this.height; y++) {
for (var x = 0; x < this.width; x++) {
var _newind = this.newind + i,
_mapind = this.oldind + i;
data = (
this.ripplemap[_mapind - this.width] +
this.ripplemap[_mapind + this.width] +
this.ripplemap[_mapind - 1] +
this.ripplemap[_mapind + 1]) >> 1;
data -= this.ripplemap[_newind];
data -= data >> 5;
this.ripplemap[_newind] = data;
//where data=0 then still, where data>0 then wave
data = 1024 - data;
if (this.last_map[i] != data)
{ this.last_map[i] = data;
//offsets
a = (((x - this.half_width) * data / 1024) << 0) + this.half_width;
b = (((y - this.half_height) * data / 1024) << 0) + this.half_height;
//bounds check
if (a >= this.width) a = this.width - 1;
if (a < 0) a = 0;
if (b >= this.height) b = this.height - 1;
if (b < 0) b = 0;
new_pixel = (a + (b * this.width)) * 4;
cur_pixel = i * 4;
/**/
_rd[cur_pixel] = _td[new_pixel];
_rd[cur_pixel + 1] = _td[new_pixel + 1];
_rd[cur_pixel + 2] = _td[new_pixel + 2];
/*/
// only in blue pixels
if (_td[new_pixel + 2]>_td[new_pixel + 1]
&& _td[new_pixel + 2]>_td[new_pixel])
{
_rd[cur_pixel] = _td[new_pixel];
_rd[cur_pixel + 1] = _td[new_pixel + 1];
_rd[cur_pixel + 2] = _td[new_pixel + 2];
}
else this.ripplemap[_newind] = 0;
/**/
}
++i;
}
}
ctx.putImageData(this.ripple, 0, 0);
// tell OL3 to continue postcompose animation
this.getMap().render();
};
/*
Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (http://www.cecill.info/).
ol.interaction.SelectCluster is an interaction for selecting vector features in a cluster.
*/
/**
* @classdesc
* Interaction for selecting vector features in a cluster.
* It can be used as an ol.interaction.Select.
* When clicking on a cluster, it springs apart to reveal the features in the cluster.
* Revealed features are selectable and you can pick the one you meant.
* Revealed features are themselves a cluster with an attribute features that contain the original feature.
*
* @constructor
* @extends {ol.interaction.Select}
* @param {olx.interaction.SelectOptions=} options SelectOptions.
* @param {ol.style} options.featureStyle used to style the revealed features as options.style is used by the Select interaction.
* @param {boolean} options.selectCluster false if you don't want to get cluster selected
* @param {Number} options.PointRadius to calculate distance between the features
* @param {bool} options.spiral means you want the feature to be placed on a spiral (or a circle)
* @param {Number} options.circleMaxObject number of object that can be place on a circle
* @param {Number} options.maxObjects number of object that can be drawn, other are hidden
* @param {bool} options.animation if the cluster will animate when features spread out, default is false
* @param {Number} options.animationDuration animation duration in ms, default is 500ms
* @fires ol.interaction.SelectEvent
* @api stable
*/
ol.interaction.SelectCluster = function(options)
{ options = options || {};
this.pointRadius = options.pointRadius || 12;
this.circleMaxObjects = options.circleMaxObjects || 10;
this.maxObjects = options.maxObjects || 60;
this.spiral = (options.spiral !== false);
this.animate = options.animate;
this.animationDuration = options.animationDuration || 500;
this.selectCluster_ = (options.selectCluster !== false);
// Create a new overlay layer for
var overlay = this.overlayLayer_ = new ol.layer.Vector(
{ source: new ol.source.Vector({
features: new ol.Collection(),
useSpatialIndex: true
}),
name:'Cluster overlay',
updateWhileAnimating: true,
updateWhileInteracting: true,
displayInLayerSwitcher: false,
style: options.featureStyle
});
// Add the overlay to selection
if (options.layers)
{ if (typeof(options.layers) == "function")
{ var fn = options.layers;
options.layers = function(layer)
{ return (layer===overlay || fn(layer));
};
}
else if (options.layers.push)
{ options.layers.push(this.overlayLayer_);
}
}
// Don't select links
if (options.filter)
{ var fn = options.filter;
options.filter = function(f,l)
{ //if (l===overlay && f.get("selectclusterlink")) return false;
if (!l && f.get("selectclusterlink")) return false;
else return fn(f,l);
};
}
else options.filter = function(f,l)
{ //if (l===overlay && f.get("selectclusterlink")) return false;
if (!l && f.get("selectclusterlink")) return false;
else return true;
};
this.filter_ = options.filter;
ol.interaction.Select.call(this, options);
this.on("select", this.selectCluster.bind(this));
};
ol.inherits(ol.interaction.SelectCluster, ol.interaction.Select);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.SelectCluster.prototype.setMap = function(map) {
if (this.getMap()) {
this.getMap().removeLayer(this.overlayLayer_);
}
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.interaction.Select.prototype.setMap.call (this, map);
this.overlayLayer_.setMap(map);
// map.addLayer(this.overlayLayer_);
if (map && map.getView()) {
this._listener = map.getView().on('change:resolution', this.clear.bind(this));
}
};
/**
* Clear the selection, close the cluster and remove revealed features
* @api stable
*/
ol.interaction.SelectCluster.prototype.clear = function()
{ this.getFeatures().clear();
this.overlayLayer_.getSource().clear();
};
/**
* Get the layer for the revealed features
* @api stable
*/
ol.interaction.SelectCluster.prototype.getLayer = function()
{ return this.overlayLayer_;
};
/**
* Select a cluster
* @param {ol.Feature} a cluster feature ie. a feature with a 'features' attribute.
* @api stable
*/
ol.interaction.SelectCluster.prototype.selectCluster = function (e)
{ // Nothing selected
if (!e.selected.length)
{ this.clear();
return;
}
// Get selection
var feature = e.selected[0];
// It's one of ours
if (feature.get('selectclusterfeature')) return;
// Clic out of the cluster => close it
var source = this.overlayLayer_.getSource();
source.clear();
var cluster = feature.get('features');
// Not a cluster (or just one feature)
if (!cluster || cluster.length==1) return;
// Remove cluster from selection
if (!this.selectCluster_) this.getFeatures().clear();
var center = feature.getGeometry().getCoordinates();
// Pixel size in map unit
var pix = this.getMap().getView().getResolution();
var r = pix * this.pointRadius * (0.5 + cluster.length / 4);
// Draw on a circle
if (!this.spiral || cluster.length <= this.circleMaxObjects)
{ var max = Math.min(cluster.length, this.circleMaxObjects);
for (var i=0; i<max; i++)
{ var a = 2*Math.PI*i/max;
if (max==2 || max == 4) a += Math.PI/4;
var p = [ center[0]+r*Math.sin(a), center[1]+r*Math.cos(a) ];
var cf = new ol.Feature({ 'selectclusterfeature':true, 'features':[cluster[i]], geometry: new ol.geom.Point(p) });
cf.setStyle(cluster[i].getStyle());
source.addFeature(cf);
var lk = new ol.Feature({ 'selectclusterlink':true, geometry: new ol.geom.LineString([center,p]) });
source.addFeature(lk);
};
}
// Draw on a spiral
else
{ // Start angle
var a = 0;
var r;
var d = 2*this.pointRadius;
var features = new Array();
var links = new Array();
var max = Math.min (this.maxObjects, cluster.length);
// Feature on a spiral
for (var i=0; i<max; i++)
{ // New radius => increase d in one turn
r = d/2 + d*a/(2*Math.PI);
// Angle
a = a + (d+0.1)/r;
var dx = pix*r*Math.sin(a)
var dy = pix*r*Math.cos(a)
var p = [ center[0]+dx, center[1]+dy ];
var cf = new ol.Feature({ 'selectclusterfeature':true, 'features':[cluster[i]], geometry: new ol.geom.Point(p) });
cf.setStyle(cluster[i].getStyle());
source.addFeature(cf);
var lk = new ol.Feature({ 'selectclusterlink':true, geometry: new ol.geom.LineString([center,p]) });
source.addFeature(lk);
}
}
if (this.animate) this.animateCluster_(center);
};
/**
* Animate the cluster and spread out the features
* @param {ol.Coordinates} the center of the cluster
*/
ol.interaction.SelectCluster.prototype.animateCluster_ = function(center)
{ // Stop animation (if one is running)
if (this.listenerKey_)
{ this.overlayLayer_.setVisible(true);
ol.Observable.unByKey(this.listenerKey_);
}
// Features to animate
var features = this.overlayLayer_.getSource().getFeatures();
if (!features.length) return;
this.overlayLayer_.setVisible(false);
var style = this.overlayLayer_.getStyle();
var stylefn = (typeof(style) == 'function') ? style : style.length ? function(){ return style; } : function(){ return [style]; } ;
var duration = this.animationDuration || 500;
var start = new Date().getTime();
// Animate function
function animate(event)
{ var vectorContext = event.vectorContext;
// Retina device
var ratio = event.frameState.pixelRatio;
var res = event.target.getView().getResolution();
var e = ol.easing.easeOut((event.frameState.time - start) / duration);
for (var i=0, feature; feature = features[i]; i++) if (feature.get('features'))
{ var pt = feature.getGeometry().getCoordinates();
pt[0] = center[0] + e * (pt[0]-center[0]);
pt[1] = center[1] + e * (pt[1]-center[1]);
var geo = new ol.geom.Point(pt);
// Image style
var st = stylefn(feature, res);
for (var s=0; s<st.length; s++)
{ var sc;
// OL < v4.3 : setImageStyle doesn't check retina
var imgs = ol.Map.prototype.getFeaturesAtPixel ? false : st[s].getImage();
if (imgs)
{ sc = imgs.getScale();
imgs.setScale(ratio);
}
// OL3 > v3.14
if (vectorContext.setStyle)
{ vectorContext.setStyle(st[s]);
vectorContext.drawGeometry(geo);
}
// older version
else
{ vectorContext.setImageStyle(imgs);
vectorContext.drawPointGeometry(geo);
}
if (imgs) imgs.setScale(sc);
}
}
// Stop animation and restore cluster visibility
if (e > 1.0)
{ ol.Observable.unByKey(this.listenerKey_);
this.overlayLayer_.setVisible(true);
this.overlayLayer_.changed();
return;
}
// tell OL3 to continue postcompose animation
event.frameState.animate = true;
}
// Start a new postcompose animation
this.listenerKey_ = this.getMap().on('postcompose', animate.bind(this));
//select.getMap().renderSync();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction to snap to guidelines
* @constructor
* @extends {ol.interaction.Interaction}
* @param {olx.interaction.SnapGuidesOptions}
* - pixelTolerance {number | undefined} distance (in px) to snap to a guideline, default 10 px
* - style {ol.style.Style | Array<ol.style.Style> | undefined} Style for the sektch features.
*/
ol.interaction.SnapGuides = function(options)
{ if (!options) options = {};
// Intersect 2 guides
function getIntersectionPoint (d1, d2)
{ var d1x = d1[1][0] - d1[0][0];
var d1y = d1[1][1] - d1[0][1];
var d2x = d2[1][0] - d2[0][0];
var d2y = d2[1][1] - d2[0][1];
var det = d1x * d2y - d1y * d2x;
if (det != 0)
{ var k = (d1x * d1[0][1] - d1x * d2[0][1] - d1y * d1[0][0] + d1y * d2[0][0]) / det;
return [d2[0][0] + k*d2x, d2[0][1] + k*d2y];
}
else return false;
}
function dist2D (p1,p2)
{ var dx = p1[0]-p2[0];
var dy = p1[1]-p2[1];
return Math.sqrt(dx*dx+dy*dy);
}
// Snap distance (in px)
this.snapDistance_ = options.pixelTolerance || 10;
// Default style
var sketchStyle =
[ new ol.style.Style({
stroke: new ol.style.Stroke(
{ color: '#ffcc33',
lineDash: [8,5],
width: 1.25
})
})
];
// Custom style
if (options.style) sketchStyle = options.style instanceof Array ? options.style : [options.style];
// Create a new overlay for the sketch
this.overlaySource_ = new ol.source.Vector(
{ features: new ol.Collection(),
useSpatialIndex: false
});
/* Speed up with a ImageVector layer (deprecated)
this.overlayLayer_ = new ol.layer.Image(
{ source: new ol.source.ImageVector(
{ source: this.overlaySource_,
style: function(f)
{ return sketchStyle;
}
}),
name:'Snap overlay',
displayInLayerSwitcher: false
});
*/
this.overlayLayer_ = new ol.layer.Vector(
{ source: this.overlaySource_,
style: function(f)
{ return sketchStyle;
},
name:'Snap overlay',
displayInLayerSwitcher: false
});
// Use snap interaction
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ if (this.getActive())
{ var features = this.overlaySource_.getFeatures();
var prev = null;
var p = null;
var res = e.frameState.viewState.resolution;
for (var i=0, f; f = features[i]; i++)
{ var c = f.getGeometry().getClosestPoint(e.coordinate);
if ( dist2D(c, e.coordinate) / res < this.snapDistance_)
{ // Intersection on 2 lines
if (prev)
{ var c2 = getIntersectionPoint(prev.getGeometry().getCoordinates(), f.getGeometry().getCoordinates());
if (c2)
{ if (dist2D(c2, e.coordinate) / res < this.snapDistance_)
{ p = c2;
}
}
}
else
{ p = c;
}
prev = f;
}
}
if (p) e.coordinate = p;
}
return true;
}
});
};
ol.inherits(ol.interaction.SnapGuides, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.SnapGuides.prototype.setMap = function(map)
{ if (this.getMap()) this.getMap().removeLayer(this.overlayLayer_);
ol.interaction.Interaction.prototype.setMap.call (this, map);
this.overlayLayer_.setMap(map);
if (map) this.projExtent_ = map.getView().getProjection().getExtent();
};
/** Activate or deactivate the interaction.
* @param {boolean} active
*/
ol.interaction.SnapGuides.prototype.setActive = function(active)
{ this.overlayLayer_.setVisible(active);
ol.interaction.Interaction.prototype.setActive.call (this, active);
}
/** Clear previous added guidelines
* @param {Array<ol.Feature> | undefined} features a list of feature to remove, default remove all feature
*/
ol.interaction.SnapGuides.prototype.clearGuides = function(features)
{ if (!features) this.overlaySource_.clear();
else
{ for (var i=0, f; f=features[i]; i++)
{ this.overlaySource_.removeFeature(f);
}
}
}
/** Get guidelines
* @return {ol.Collection} guidelines features
*/
ol.interaction.SnapGuides.prototype.getGuides = function(features)
{ return this.overlaySource_.getFeaturesCollection();
}
/** Add a new guide to snap to
* @param {Array<ol.coordinate>} v the direction vector
* @return {ol.Feature} feature guide
*/
ol.interaction.SnapGuides.prototype.addGuide = function(v, ortho)
{ if (v)
{ var map = this.getMap();
// Limit extent
var extent = map.getView().calculateExtent(map.getSize());
extent = ol.extent.buffer(extent, Math.max (1e5+1, (extent[2]-extent[0])*100));
extent = ol.extent.getIntersection(extent, this.projExtent_);
var dx = v[0][0] - v[1][0];
var dy = v[0][1] - v[1][1];
var d = 1 / Math.sqrt(dx*dx+dy*dy);
var p, g = [];
var p0, p1;
for (var i= 0; i<1e8; i+=1e5)
{ if (ortho) p = [ v[0][0] + dy*d*i, v[0][1] - dx*d*i];
else p = [ v[0][0] + dx*d*i, v[0][1] + dy*d*i];
if (ol.extent.containsCoordinate(extent, p)) g.push(p);
else break;
}
var f0 = new ol.Feature(new ol.geom.LineString(g));
var g=[];
for (var i= 0; i>-1e8; i-=1e5)
{ if (ortho) p = [ v[0][0] + dy*d*i, v[0][1] - dx*d*i];
else p = [ v[0][0] + dx*d*i, v[0][1] + dy*d*i];
if (ol.extent.containsCoordinate(extent, p)) g.push(p);
else break;
}
var f1 = new ol.Feature(new ol.geom.LineString(g));
this.overlaySource_.addFeature(f0);
this.overlaySource_.addFeature(f1);
return [f0, f1];
}
};
/** Add a new orthogonal guide to snap to
* @param {Array<ol.coordinate>} v the direction vector
* @return {ol.Feature} feature guide
*/
ol.interaction.SnapGuides.prototype.addOrthoGuide = function(v)
{ return this.addGuide(v, true);
};
/** Listen to draw event to add orthogonal guidelines on the first and last point.
* @param {_ol_interaction_Draw_} drawi a draw interaction to listen to
* @api
*/
ol.interaction.SnapGuides.prototype.setDrawInteraction = function(drawi)
{ var self = this;
// Number of points currently drawing
var nb = 0;
// Current guidelines
var features = [];
function setGuides(e)
{ var coord = [];
var s = 2;
switch (e.target.getType())
{ case 'LineString':
coord = e.target.getCoordinates();
s = 2;
break;
case 'Polygon':
coord = e.target.getCoordinates()[0];
s = 3;
break;
default: break;
}
var l = coord.length;
if (l != nb && l > s)
{ self.clearGuides(features);
features = self.addOrthoGuide([coord[l-s],coord[l-s-1]]);
features = features.concat(self.addGuide([coord[0],coord[1]]));
features = features.concat(self.addOrthoGuide([coord[0],coord[1]]));
nb = l;
}
};
// New drawing
drawi.on ("drawstart", function(e)
{ // When geom is changing add a new orthogonal direction
e.feature.getGeometry().on("change", setGuides);
});
// end drawing, clear directions
drawi.on ("drawend", function(e)
{ self.clearGuides(features);
e.feature.getGeometry().un("change", setGuides);
nb = 0;
features = [];
});
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction split interaction for splitting feature geometry
* @constructor
* @extends {ol.interaction.Interaction}
* @fires beforesplit, aftersplit
* @param {olx.interaction.SplitOptions}
* - source {ol.source.Vector|Array{ol.source.Vector}} a list of source to split (configured with useSpatialIndex set to true)
* - features {ol.Collection.<ol.Feature>} collection of feature to split
* - snapDistance {integer} distance (in px) to snap to an object, default 25px
* - cursor {string|undefined} cursor name to display when hovering an objet
* - filter {function|undefined} a filter that takes a feature and return true if it can be clipped, default always split.
* - featureStyle {ol.style.Style | Array<ol.style.Style> | false | undefined} Style for the selected features, choose false if you don't want feature selection. By default the default edit style is used.
* - sketchStyle {ol.style.Style | Array<ol.style.Style> | undefined} Style for the sektch features.
* - tolerance {function|undefined} Distance between the calculated intersection and a vertex on the source geometry below which the existing vertex will be used for the split. Default is 1e-10.
*/
ol.interaction.Split = function(options)
{ if (!options) options = {};
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ switch (e.type)
{ case "singleclick":
return this.handleDownEvent(e);
case "pointermove":
return this.handleMoveEvent(e);
default:
return true;
}
return true;
}
});
// Snap distance (in px)
this.snapDistance_ = options.snapDistance || 25;
// Split tolerance between the calculated intersection and the geometry
this.tolerance_ = options.tolerance || 1e-10;
// Cursor
this.cursor_ = options.cursor;
// List of source to split
this.sources_ = options.sources ? (options.sources instanceof Array) ? options.sources:[options.sources] : [];
if (options.features)
{ this.sources_.push (new ol.source.Vector({ features: features }));
}
// Get all features candidate
this.filterSplit_ = options.filter || function(){ return true; };
// Default style
var white = [255, 255, 255, 1];
var blue = [0, 153, 255, 1];
var width = 3;
var fill = new ol.style.Fill({ color: 'rgba(255,255,255,0.4)' });
var stroke = new ol.style.Stroke({
color: '#3399CC',
width: 1.25
});
var sketchStyle =
[ new ol.style.Style({
image: new ol.style.Circle({
fill: fill,
stroke: stroke,
radius: 5
}),
fill: fill,
stroke: stroke
})
];
var featureStyle =
[ new ol.style.Style({
stroke: new ol.style.Stroke({
color: white,
width: width + 2
})
}),
new ol.style.Style({
image: new ol.style.Circle({
radius: 2*width,
fill: new ol.style.Fill({
color: blue
}),
stroke: new ol.style.Stroke({
color: white,
width: width/2
})
}),
stroke: new ol.style.Stroke({
color: blue,
width: width
})
}),
];
// Custom style
if (options.sketchStyle) sketchStyle = options.sketchStyle instanceof Array ? options.sketchStyle : [options.sketchStyle];
if (options.featureStyle) featureStyle = options.featureStyle instanceof Array ? options.featureStyle : [options.featureStyle];
// Create a new overlay for the sketch
this.overlayLayer_ = new ol.layer.Vector(
{ source: new ol.source.Vector({
useSpatialIndex: false
}),
name:'Split overlay',
displayInLayerSwitcher: false,
style: function(f)
{ if (f._sketch_) return sketchStyle;
else return featureStyle;
}
});
};
ol.inherits(ol.interaction.Split, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.Split.prototype.setMap = function(map)
{ if (this.getMap()) this.getMap().removeLayer(this.overlayLayer_);
ol.interaction.Interaction.prototype.setMap.call (this, map);
this.overlayLayer_.setMap(map);
};
/** Get closest feature at pixel
* @param {ol.Pixel}
* @return {ol.feature}
* @private
*/
ol.interaction.Split.prototype.getClosestFeature = function(e)
{ var f, c, g, d = this.snapDistance_+1;
for (var i=0; i<this.sources_.length; i++)
{ var source = this.sources_[i];
f = source.getClosestFeatureToCoordinate(e.coordinate);
if (f.getGeometry().splitAt)
{ c = f.getGeometry().getClosestPoint(e.coordinate);
g = new ol.geom.LineString([e.coordinate,c]);
d = g.getLength() / e.frameState.viewState.resolution;
break;
}
}
if (d > this.snapDistance_) return false;
else
{ // Snap to node
var coord = this.getNearestCoord (c, f.getGeometry().getCoordinates());
var p = this.getMap().getPixelFromCoordinate(coord);
if (ol.coordinate.dist2d(e.pixel, p) < this.snapDistance_)
{ c = coord;
}
//
return { source:source, feature:f, coord: c, link: g };
}
}
/** Get nearest coordinate in a list
* @param {ol.coordinate} pt the point to find nearest
* @param {Array<ol.coordinate>} coords list of coordinates
* @return {ol.coordinate} the nearest coordinate in the list
*/
ol.interaction.Split.prototype.getNearestCoord = function(pt, coords)
{ var d, dm=Number.MAX_VALUE, p0;
for (var i=0; i < coords.length; i++)
{ d = ol.coordinate.dist2d (pt, coords[i]);
if (d < dm)
{ dm = d;
p0 = coords[i];
}
}
return p0;
};
/**
* @param {ol.MapBrowserEvent} evt Map browser event.
* @return {boolean} `true` to start the drag sequence.
*/
ol.interaction.Split.prototype.handleDownEvent = function(evt)
{ // Something to split ?
var current = this.getClosestFeature(evt);
if (current)
{ var self = this;
self.overlayLayer_.getSource().clear();
var split = current.feature.getGeometry().splitAt(current.coord, this.tolerance_);
if (split.length > 1)
{ var tosplit = [];
for (var i=0; i<split.length; i++)
{ var f = current.feature.clone();
f.setGeometry(split[i]);
tosplit.push(f);
}
self.dispatchEvent({ type:'beforesplit', original: current.feature, features: tosplit });
current.source.dispatchEvent({ type:'beforesplit', original: current.feature, features: tosplit });
current.source.removeFeature(current.feature);
for (var i=0; i<tosplit.length; i++)
{ current.source.addFeature(tosplit[i]);
}
self.dispatchEvent({ type:'aftersplit', original: current.feature, features: tosplit });
current.source.dispatchEvent({ type:'aftersplit', original: current.feature, features: tosplit });
}
}
return false;
};
/**
* @param {ol.MapBrowserEvent} evt Event.
*/
ol.interaction.Split.prototype.handleMoveEvent = function(e)
{ var map = e.map;
this.overlayLayer_.getSource().clear();
var current = this.getClosestFeature(e);
if (current && this.filterSplit_(current.feature))
{ var coord, p, l;
// Draw sketch
this.overlayLayer_.getSource().addFeature(current.feature);
p = new ol.Feature(new ol.geom.Point(current.coord));
p._sketch_ = true;
this.overlayLayer_.getSource().addFeature(p);
//
l = new ol.Feature(new ol.geom.LineString([e.coordinate,current.coord]));
l._sketch_ = true;
this.overlayLayer_.getSource().addFeature(l);
}
var element = map.getTargetElement();
if (this.cursor_)
{ if (current)
{ if (element.style.cursor != this.cursor_)
{ this.previousCursor_ = element.style.cursor;
element.style.cursor = this.cursor_;
}
}
else if (this.previousCursor_ !== undefined)
{ element.style.cursor = this.previousCursor_;
this.previousCursor_ = undefined;
}
}
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction splitter: acts as a split feature agent while editing vector features (LineString).
* @constructor
* @extends {ol.interaction.Interaction}
* @fires beforesplit, aftersplit
* @param {olx.interaction.SplitOptions}
* - source {ol.source.Vector|Array{ol.source.Vector}} The target source (or array of source) with features to be split (configured with useSpatialIndex set to true)
* - triggerSource {ol.source.Vector} Any newly created or modified features from this source will be used to split features on the target source. If none is provided the target source is used instead.
* - features {ol.Collection.<ol.Feature>} A collection of feature to be split (replace source target).
* - triggerFeatures {ol.Collection.<ol.Feature>} Any newly created or modified features from this collection will be used to split features on the target source (replace triggerSource).
* - filter {function|undefined} a filter that takes a feature and return true if the feature is eligible for splitting, default always split.
* - tolerance {function|undefined} Distance between the calculated intersection and a vertex on the source geometry below which the existing vertex will be used for the split. Default is 1e-10.
* @todo verify auto intersection on features that split.
*/
ol.interaction.Splitter = function(options)
{ if (!options) options = {};
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ // Hack to get only one changeFeature when draging with ol.interaction.Modify on.
if (e.type != "pointermove" && e.type != "pointerdrag")
{ if (this.lastEvent_)
{ this.splitSource(this.lastEvent_.feature);
this.lastEvent_ = null;
}
this.moving_ = false;
}
else this.moving_ = true;
return true;
},
});
// Features added / remove
this.added_ = [];
this.removed_ = [];
// Source to split
if (options.features)
{ this.source_ = new ol.source.Vector({ features: options.features });
}
else
{ this.source_ = options.source ? options.source : new ol.source.Vector({ features: new ol.Collection() });
}
var trigger = this.triggerSource;
if (options.triggerFeatures)
{ trigger = new ol.source.Vector({ features: options.triggerFeatures });
}
if (trigger)
{ trigger.on("addfeature", this.onAddFeature.bind(this));
trigger.on("changefeature", this.onChangeFeature.bind(this));
trigger.on("removefeature", this.onRemoveFeature.bind(this));
}
else
{ this.source_.on("addfeature", this.onAddFeature.bind(this));
this.source_.on("changefeature", this.onChangeFeature.bind(this));
this.source_.on("removefeature", this.onRemoveFeature.bind(this));
}
// Split tolerance between the calculated intersection and the geometry
this.tolerance_ = options.tolerance || 1e-10;
// Get all features candidate
this.filterSplit_ = options.filter || function(){ return true; };
};
ol.inherits(ol.interaction.Splitter, ol.interaction.Interaction);
/** Calculate intersection on 2 segs
* @param {Array<_ol_coordinate_>} s1 first seg to intersect (2 points)
* @param {Array<_ol_coordinate_>} s2 second seg to intersect (2 points)
* @return { boolean | _ol_coordinate_ } intersection point or false no intersection
*/
ol.interaction.Splitter.prototype.intersectSegs = function(s1,s2)
{ var tol = this.tolerance_;
// Solve
var x12 = s1[0][0] - s1[1][0];
var x34 = s2[0][0] - s2[1][0];
var y12 = s1[0][1] - s1[1][1];
var y34 = s2[0][1] - s2[1][1];
var det = x12 * y34 - y12 * x34;
// No intersection
if (Math.abs(det) < tol)
{ return false;
}
else
{ // Outside segement
var r1 = ((s1[0][0] - s2[1][0])*y34 - (s1[0][1] - s2[1][1])*x34) / det;
if (Math.abs(r1)<tol) return s1[0];
if (Math.abs(1-r1)<tol) return s1[1];
if (r1<0 || r1>1) return false;
var r2 = ((s1[0][1] - s2[1][1])*x12 - (s1[0][0] - s2[1][0])*y12) / det;
if (Math.abs(r2)<tol) return s2[1];
if (Math.abs(1-r2)<tol) return s2[0];
if (r2<0 || r2>1) return false;
// Intersection
var a = s1[0][0] * s1[1][1] - s1[0][1] * s1[1][0];
var b = s2[0][0] * s2[1][1] - s2[0][1] * s2[1][0];
var p = [(a * x34 - b * x12) / det, (a * y34 - b * y12) / det];
// Test start / end
/*
console.log("r1: "+r1)
console.log("r2: "+r2)
console.log ("s10: "+(_ol_coordinate_.dist2d(p,s1[0])<tol)) ;
console.log ("s11: "+(_ol_coordinate_.dist2d(p,s1[1])<tol)) ;
console.log ("s20: "+(_ol_coordinate_.dist2d(p,s2[0])<tol)) ;
console.log ("s21: "+(_ol_coordinate_.dist2d(p,s2[1])<tol)) ;
*/
return p;
}
};
/** Split the source using a feature
* @param {ol.Feature} feature The feature to use to split.
*/
ol.interaction.Splitter.prototype.splitSource = function(feature)
{ // Allready perform a split
if (this.splitting) return;
var self = this;
var i, k, f2;
// Start splitting
this.source_.dispatchEvent({ type:'beforesplit', feaure: feature, source: this.source_ });
this.splitting = true;
this.added_ = [];
this.removed_ = [];
var c = feature.getGeometry().getCoordinates();
var seg, split = [];
function intersect (f)
{ if (f !== feature)
{ var c2 = f.getGeometry().getCoordinates();
for (var j=0; j<c2.length-1; j++)
{ var p = this.intersectSegs (seg, [c2[j],c2[j+1]]);
if (p)
{ split.push(p);
g = f.getGeometry().splitAt(p, this.tolerance_);
if (g && g.length>1)
{ found = f;
return true;
}
}
}
}
return false;
}
// Split existing features
for (i=0; i<c.length-1; i++)
{ seg = [c[i],c[i+1]];
var extent = ol.extent.buffer(ol.extent.boundingExtent(seg), this.tolerance_ /*0.01*/ );
var g;
while (true)
{ var found = false;
this.source_.forEachFeatureIntersectingExtent(extent, intersect.bind(this));
// Split feature
if (found)
{ var f = found;
this.source_.removeFeature(f);
for (k=0; k<g.length; k++)
{ f2 = f.clone();
f2.setGeometry(g[k]);
this.source_.addFeature(f2);
}
}
else break;
}
}
// Auto intersect
for (i=0; i<c.length-2; i++)
{ for (var j=i+1; j<c.length-1; j++)
{ var p = this.intersectSegs ([c[i],c[i+1]], [c[j],c[j+1]]);
if (p && p!=c[i+1])
{ split.push(p);
}
}
}
// Split original
var splitOriginal = false;
if (split.length)
{ var result = feature.getGeometry().splitAt(split, this.tolerance_);
if (result.length>1)
{ for (k=0; k<result.length; k++)
{ f2 = feature.clone();
f2.setGeometry(result[k]);
this.source_.addFeature(f2);
}
splitOriginal = true;
}
}
// If the interaction is inserted after modify interaction, the objet is not consistant
// > wait end of other interactions
setTimeout (function()
{ if (splitOriginal) self.source_.removeFeature(feature);
self.source_.dispatchEvent({ type:'aftersplit', featureAdded: self.added_, featureRemoved: self.removed_, source: this.source_ });
// Finish
self.splitting = false;
},0);
};
/** New feature source is added
*/
ol.interaction.Splitter.prototype.onAddFeature = function(e)
{ this.splitSource(e.feature);
if (this.splitting)
{ this.added_.push(e.feature);
}
/*
if (this.splitting) return;
var self = this;
setTimeout (function() { self.splitSource(e.feature); }, 0);
*/
};
/** Feature source is removed > count features added/removed
*/
ol.interaction.Splitter.prototype.onRemoveFeature = function(e)
{ if (this.splitting)
{ var n = this.added_.indexOf(e.feature);
if (n==-1)
{ this.removed_.push(e.feature);
}
else
{ this.added_.splice(n,1);
}
}
};
/** Feature source is changing
*/
ol.interaction.Splitter.prototype.onChangeFeature = function(e)
{ if (this.moving_)
{ this.lastEvent_ = e;
}
else this.splitSource(e.feature);
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction synchronize
* @constructor
* @extends {ol.interaction.Interaction}
* @param {olx.interaction.SynchronizeOptions}
* - maps {Array<ol.Map>} An array of maps to synchronize with the map of the interaction
*/
ol.interaction.Synchronize = function(options)
{ if (!options) options={};
var self = this;
ol.interaction.Interaction.call(this,
{ handleEvent: function(e)
{ if (e.type=="pointermove") { self.handleMove_(e); }
return true;
}
});
this.maps = options.maps;
};
ol.inherits(ol.interaction.Synchronize, ol.interaction.Interaction);
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.Synchronize.prototype.setMap = function(map)
{
if (this._listener) {
ol.Observable.unByKey(this._listener.center);
ol.Observable.unByKey(this._listener.rotation);
ol.Observable.unByKey(this._listener.resolution);
$(this.getMap().getTargetElement()).off('mouseout', this._listener.mouseout);
}
this._listener = null;
ol.interaction.Interaction.prototype.setMap.call (this, map);
if (map) {
this._listener = {};
this._listener.center = this.getMap().getView().on('change:center', this.syncMaps.bind(this));
this._listener.rotation = this.getMap().getView().on('change:rotation', this.syncMaps.bind(this));
this._listener.resolution = this.getMap().getView().on('change:resolution', this.syncMaps.bind(this));
this._listener.mouseout = this.handleMouseOut_.bind(this);
$(this.getMap().getTargetElement()).on('mouseout', this._listener.mouseout);
this.syncMaps();
}
};
/** Synchronize the maps
*/
ol.interaction.Synchronize.prototype.syncMaps = function(e)
{ var map = this.getMap();
if (!e) e = { type:'all' };
if (map)
{ for (var i=0; i<this.maps.length; i++)
{ switch (e.type)
{ case 'change:rotation':
if (this.maps[i].getView().getRotation() != map.getView().getRotation())
this.maps[i].getView().setRotation(map.getView().getRotation());
break;
case 'change:center':
if (this.maps[i].getView().getCenter() != map.getView().getCenter())
this.maps[i].getView().setCenter(map.getView().getCenter());
break;
case 'change:resolution':
if (this.maps[i].getView().getResolution() != map.getView().getResolution())
{ /* old version prior to 1.19.1
this.maps[i].beforeRender ( ol.animation.zoom(
{ duration: 250,
resolution: this.maps[i].getView().getResolution()
}));
*/
this.maps[i].getView().setResolution(map.getView().getResolution());
}
break;
default:
this.maps[i].getView().setRotation(map.getView().getRotation());
this.maps[i].getView().setCenter(map.getView().getCenter());
this.maps[i].getView().setResolution(map.getView().getResolution());
break;
}
}
}
};
/** Cursor move > tells other maps to show the cursor
* @param {ol.event} e "move" event
*/
ol.interaction.Synchronize.prototype.handleMove_ = function(e)
{ for (var i=0; i<this.maps.length; i++)
{ this.maps[i].showTarget(e.coordinate);
}
this.getMap().showTarget();
};
/** Cursor out of map > tells other maps to hide the cursor
* @param {event} e "mouseOut" event
*/
ol.interaction.Synchronize.prototype.handleMouseOut_ = function(e) {
for (var i=0; i<this.maps.length; i++) {
this.maps[i].targetOverlay_.setPosition(undefined);
}
};
/** Show a target overlay at coord
* @param {ol.coordinate} coord
*/
ol.Map.prototype.showTarget = function(coord)
{ if (!this.targetOverlay_)
{ var elt = $("<div>").addClass("ol-target");
this.targetOverlay_ = new ol.Overlay({ element: elt.get(0) });
this.targetOverlay_.setPositioning('center-center');
this.addOverlay(this.targetOverlay_);
elt.parent().addClass("ol-target-overlay");
// hack to render targetOverlay before positioning it
this.targetOverlay_.setPosition([0,0]);
}
this.targetOverlay_.setPosition(coord);
};
/** Hide the target overlay
*/
ol.Map.prototype.hideTarget = function()
{
this.removeOverlay(this.targetOverlay_);
this.targetOverlay_ = undefined;
};
/*
Tinker Bell effect on maps.
Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
@link https://github.com/Viglino
*/
/**
* @constructor
* @extends {ol.interaction.Pointer}
* @param {ol.interaction.TinkerBell.options} options flashlight param
* - color {ol.color} color of the sparkles
*/
ol.interaction.TinkerBell = function(options)
{ options = options || {};
ol.interaction.Pointer.call(this,
{ handleDownEvent: this.onMove,
handleMoveEvent: this.onMove
});
this.set('color', options.color ? ol.color.asString(options.color) : "#fff");
this.sparkle = [0,0];
this.sparkles = [];
this.lastSparkle = this.time = new Date();
var self = this;
this.out_ = function() { self.isout_=true; };
this.isout_ = true;
};
ol.inherits(ol.interaction.TinkerBell, ol.interaction.Pointer);
/** Set the map > start postcompose
*/
ol.interaction.TinkerBell.prototype.setMap = function(map) {
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
if (this.getMap()) {
map.getViewport().removeEventListener('mouseout', this.out_, false);
this.getMap().render();
}
ol.interaction.Pointer.prototype.setMap.call(this, map);
if (map) {
this._listener = map.on('postcompose', this.postcompose_.bind(this));
map.getViewport().addEventListener('mouseout', this.out_, false);
}
};
ol.interaction.TinkerBell.prototype.onMove = function(e)
{ this.sparkle = e.pixel;
this.isout_ = false;
this.getMap().render();
};
/** Postcompose function
*/
ol.interaction.TinkerBell.prototype.postcompose_ = function(e)
{ var delta = 15;
var ctx = e.context;
var canvas = ctx.canvas;
var dt = e.frameState.time - this.time;
this.time = e.frameState.time;
if (e.frameState.time-this.lastSparkle > 30 && !this.isout_)
{ this.lastSparkle = e.frameState.time;
this.sparkles.push({ p:[this.sparkle[0]+Math.random()*delta-delta/2, this.sparkle[1]+Math.random()*delta], o:1 });
}
ctx.save();
ctx.scale(e.frameState.pixelRatio,e.frameState.pixelRatio);
ctx.fillStyle = this.get("color");
for (var i=this.sparkles.length-1, p; p=this.sparkles[i]; i--)
{ if (p.o < 0.2)
{ this.sparkles.splice(0,i+1);
break;
}
ctx.globalAlpha = p.o;
ctx.beginPath();
ctx.arc (p.p[0], p.p[1], 2.2, 0, 2 * Math.PI, false);
ctx.fill();
p.o *= 0.98;
p.p[0] += (Math.random()-0.5);
p.p[1] += dt*(1+Math.random())/30;
};
ctx.restore();
// tell OL3 to continue postcompose animation
if (this.sparkles.length) this.getMap().render();
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Interaction splitter: acts as a split feature agent while editing vector features (LineString).
* @constructor
* @extends {ol.interaction.Pointer}
* @param {olx.interaction.TouchCompass}
* - onDrag {function|undefined} Function handling "drag" events. It provides a dpixel and a traction (in projection) vector form the center of the compas
* - size {Number} size of the compass in px, default 80
* - alpha {Number} opacity of the compass, default 0.5
*/
ol.interaction.TouchCompass = function(options)
{ var options = options||{};
var self = this;
var opt = {};
// Click on the compass
opt.handleDownEvent = function(e)
{ var s = this.getCenter_();
var dx = e.pixel[0]-s[0];
var dy = e.pixel[1]-s[1];
this.start = e;
return (Math.sqrt(dx*dx+dy*dy) < this.size/2);
};
// Pn drag
opt.handleDragEvent = function(e)
{ if (!this.pos)
{ this.pos = this.start;
this.getMap().renderSync();
}
this.pos = e;
};
// Stop drag
opt.handleUpEvent = function(e)
{ this.pos = false;
return true;
};
ol.interaction.Pointer.call(this, opt);
this.ondrag_ = options.onDrag;
this.size = options.size || 80;
this.alpha = options.alpha || 0.5;
if (!ol.interaction.TouchCompass.prototype.compass)
{ var canvas = ol.interaction.TouchCompass.prototype.compass = document.createElement('canvas');
var ctx = canvas.getContext("2d");
var s = canvas.width = canvas.height = this.size;
var w = s/10;
var r = s/2;
var r2 = 0.22*r;
ctx.translate(r,r);
ctx.fillStyle = "#999";
ctx.strokeStyle = "#ccc";
ctx.lineWidth = w;
ctx.beginPath();
ctx.arc (0,0, s*0.42, 0, 2*Math.PI);
ctx.fill();
ctx.stroke();
ctx.fillStyle = "#99f";
ctx.beginPath();
ctx.moveTo (0,0);
ctx.lineTo (r,0); ctx.lineTo (r2,r2); ctx.moveTo (0,0);
ctx.lineTo (-r,0); ctx.lineTo (-r2,-r2); ctx.moveTo (0,0);
ctx.lineTo (0,r); ctx.lineTo (-r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,-r); ctx.lineTo (r2,-r2); ctx.moveTo (0,0);
ctx.fill();
ctx.fillStyle = "#eee";
ctx.beginPath();
ctx.moveTo (0,0);
ctx.lineTo (r,0); ctx.lineTo (r2,-r2); ctx.moveTo (0,0);
ctx.lineTo (-r,0); ctx.lineTo (-r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,r); ctx.lineTo (r2,r2); ctx.moveTo (0,0);
ctx.lineTo (0,-r); ctx.lineTo (-r2,-r2); ctx.moveTo (0,0);
ctx.fill();
}
};
ol.inherits(ol.interaction.TouchCompass, ol.interaction.Pointer);
/** Compass Image as a JS Image object
* @api
*/
ol.interaction.TouchCompass.prototype.compass = null;
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {_ol_Map_} map Map.
* @api stable
*/
ol.interaction.TouchCompass.prototype.setMap = function(map) {
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.interaction.Pointer.prototype.setMap.call (this, map);
if (map) {
this._listener = map.on('postcompose', this.drawCompass_.bind(this));
}
};
/**
* Activate or deactivate the interaction.
* @param {boolean} active Active.
* @observable
* @api
*/
ol.interaction.TouchCompass.prototype.setActive = function(b)
{ ol.interaction.Pointer.prototype.setActive.call (this, b);
if (this.getMap()) this.getMap().renderSync();
}
/**
* Get the center of the compass
* @param {_ol_coordinate_}
* @private
*/
ol.interaction.TouchCompass.prototype.getCenter_ = function()
{ var margin = 10;
var s = this.size;
var c = this.getMap().getSize();
return [c[0]/2, c[1]-margin-s/2];
}
/**
* Draw the compass on post compose
* @private
*/
ol.interaction.TouchCompass.prototype.drawCompass_ = function(e)
{ if (!this.getActive()) return;
var ctx = e.context;
var ratio = e.frameState.pixelRatio;
ctx.save();
ctx.scale(ratio,ratio);
ctx.globalAlpha = this.alpha;
ctx.strokeStyle = "#fff";
ctx.lineWidth = 5;
var s = this.size;
var c = this.getCenter_();
ctx.drawImage(this.compass, 0,0,this.compass.width,this.compass.height, c[0]-s/2, c[1]-s/2, s,s);
if (this.pos)
{ var dx = this.pos.pixel[0]-this.start.pixel[0];
var dy = this.pos.pixel[1]-this.start.pixel[1];
for (var i=1; i<=4; i++)
{ ctx.beginPath();
ctx.arc (c[0] +dx/4*i, c[1] +dy/4*i, s/2*(0.6+0.4*i/4), 0, 2*Math.PI);
ctx.stroke();
}
}
ctx.restore();
if (this.pos)
{ // Get delta
if (this.ondrag_)
{ var r = this.getMap().getView().getResolution();
var delta =
{ dpixel: [ this.pos.pixel[0] - this.start.pixel[0], this.pos.pixel[1] - this.start.pixel[1] ]
}
delta.traction = [ delta.dpixel[0]*r, -delta.dpixel[1]*r];
this.ondrag_(delta, this.pos);
}
// Continue animation
e.frameState.animate = true;
}
};
/** Interaction rotate
* @constructor
* @extends {ol.interaction.Pointer}
* @fires select | rotatestart | rotating | rotateend | translatestart | translating | translateend | scalestart | scaling | scaleend
* @param {any} options
* @param {Array<ol.Layer>} options.layers array of layers to transform,
* @param {ol.Collection<ol.Feature>} options.features collection of feature to transform,
* @param {ol.EventsConditionType|undefined} options.addCondition A function that takes an ol.MapBrowserEvent and returns a boolean to indicate whether that event should be handled. default: ol.events.condition.never.
* @param {number | undefined} options.hitTolerance Tolerance to select feature in pixel, default 0
* @param {bool} options.translateFeature Translate when click on feature
* @param {bool} options.translate Can translate the feature
* @param {bool} options.stretch can stretch the feature
* @param {bool} options.scale can scale the feature
* @param {bool} options.rotate can rotate the feature
* @param {ol.events.ConditionType | undefined} options.keepAspectRatio A function that takes an ol.MapBrowserEvent and returns a boolean to keep aspect ratio, default ol.events.condition.shiftKeyOnly.
* @param {ol.events.ConditionType | undefined} options.modifyCenter A function that takes an ol.MapBrowserEvent and returns a boolean to apply scale & strech from the center, default ol.events.condition.metaKey or ol.events.condition.ctrlKey.
* @param {} options.style list of ol.style for handles
*
*/
ol.interaction.Transform = function(options) {
if (!options) options = {};
var self = this;
// Create a new overlay layer for the sketch
this.handles_ = new ol.Collection();
this.overlayLayer_ = new ol.layer.Vector({
source: new ol.source.Vector({
features: this.handles_,
useSpatialIndex: false
}),
name:'Transform overlay',
displayInLayerSwitcher: false,
// Return the style according to the handle type
style: function (feature) {
return (self.style[(feature.get('handle')||'default')+(feature.get('constraint')||'')+(feature.get('option')||'')]);
}
});
// Extend pointer
ol.interaction.Pointer.call(this, {
handleDownEvent: this.handleDownEvent_,
handleDragEvent: this.handleDragEvent_,
handleMoveEvent: this.handleMoveEvent_,
handleUpEvent: this.handleUpEvent_
});
/** Collection of feature to transform */
this.features_ = options.features;
/** List of layers to transform */
this.layers_ = options.layers ? (options.layers instanceof Array) ? options.layers:[options.layers] : null;
this.addFn_ = options.addCondition || function() { return false; };
/* Translate when click on feature */
this.set('translateFeature', (options.translateFeature!==false));
/* Can translate the feature */
this.set('translate', (options.translate!==false));
/* Can stretch the feature */
this.set('stretch', (options.stretch!==false));
/* Can scale the feature */
this.set('scale', (options.scale!==false));
/* Can rotate the feature */
this.set('rotate', (options.rotate!==false));
/* Keep aspect ratio */
this.set('keepAspectRatio', (options.keepAspectRatio || function(e){ return e.originalEvent.shiftKey }));
/* Modify center */
this.set('modifyCenter', (options.modifyCenter || function(e){ return e.originalEvent.metaKey || e.originalEvent.ctrlKey }));
/* */
this.set('hitTolerance', (options.hitTolerance || 0));
this.selection_ = [];
// Force redraw when changed
this.on ('propertychange', function() {
this.drawSketch_();
});
// setstyle
this.setDefaultStyle();
};
ol.inherits(ol.interaction.Transform, ol.interaction.Pointer);
/** Cursors for transform
*/
ol.interaction.Transform.prototype.Cursors = {
'default': 'auto',
'select': 'pointer',
'translate': 'move',
'rotate': 'move',
'rotate0': 'move',
'scale': 'nesw-resize',
'scale1': 'nwse-resize',
'scale2': 'nesw-resize',
'scale3': 'nwse-resize',
'scalev': 'ew-resize',
'scaleh1': 'ns-resize',
'scalev2': 'ew-resize',
'scaleh3': 'ns-resize'
};
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.Transform.prototype.setMap = function(map) {
if (this.getMap()) this.getMap().removeLayer(this.overlayLayer_);
ol.interaction.Pointer.prototype.setMap.call (this, map);
this.overlayLayer_.setMap(map);
if (map !== null) {
this.isTouch = /touch/.test(map.getViewport().className);
this.setDefaultStyle();
}
};
/**
* Activate/deactivate interaction
* @param {bool}
* @api stable
*/
ol.interaction.Transform.prototype.setActive = function(b) {
this.select(null);
this.overlayLayer_.setVisible(b);
ol.interaction.Pointer.prototype.setActive.call (this, b);
};
/** Set efault sketch style
*/
ol.interaction.Transform.prototype.setDefaultStyle = function() {
// Style
var stroke = new ol.style.Stroke({ color: [255,0,0,1], width: 1 });
var strokedash = new ol.style.Stroke({ color: [255,0,0,1], width: 1, lineDash:[4,4] });
var fill0 = new ol.style.Fill({ color:[255,0,0,0.01] });
var fill = new ol.style.Fill({ color:[255,255,255,0.8] });
var circle = new ol.style.RegularShape({
fill: fill,
stroke: stroke,
radius: this.isTouch ? 12 : 6,
points: 15
});
circle.getAnchor()[0] = this.isTouch ? -10 : -5;
var bigpt = new ol.style.RegularShape({
fill: fill,
stroke: stroke,
radius: this.isTouch ? 16 : 8,
points: 4,
angle: Math.PI/4
});
var smallpt = new ol.style.RegularShape({
fill: fill,
stroke: stroke,
radius: this.isTouch ? 12 : 6,
points: 4,
angle: Math.PI/4
});
function createStyle (img, stroke, fill) {
return [ new ol.style.Style({image:img, stroke:stroke, fill:fill}) ];
}
/** Style for handles */
this.style = {
'default': createStyle (bigpt, strokedash, fill0),
'translate': createStyle (bigpt, stroke, fill),
'rotate': createStyle (circle, stroke, fill),
'rotate0': createStyle (bigpt, stroke, fill),
'scale': createStyle (bigpt, stroke, fill),
'scale1': createStyle (bigpt, stroke, fill),
'scale2': createStyle (bigpt, stroke, fill),
'scale3': createStyle (bigpt, stroke, fill),
'scalev': createStyle (smallpt, stroke, fill),
'scaleh1': createStyle (smallpt, stroke, fill),
'scalev2': createStyle (smallpt, stroke, fill),
'scaleh3': createStyle (smallpt, stroke, fill),
};
this.drawSketch_();
}
/**
* Set sketch style.
* @param {ol.Map} map Map.
* @api stable
*/
ol.interaction.Transform.prototype.setStyle = function(style, olstyle) {
if (!olstyle) return;
if (olstyle instanceof Array) this.style[style] = olstyle;
else this.style[style] = [ olstyle ];
for (var i=0; i<this.style[style].length; i++) {
var im = this.style[style][i].getImage();
if (im) {
if (style == 'rotate') im.getAnchor()[0] = -5;
if (this.isTouch) im.setScale(1.8);
}
var tx = this.style[style][i].getText();
if (tx) {
if (style == 'rotate') tx.setOffsetX(this.isTouch ? 14 : 7);
if (this.isTouch) tx.setScale(1.8);
}
}
this.drawSketch_();
};
/** Get Feature at pixel
* @param {ol.Pixel}
* @return {ol.feature}
* @private
*/
ol.interaction.Transform.prototype.getFeatureAtPixel_ = function(pixel) {
var self = this;
return this.getMap().forEachFeatureAtPixel(pixel,
function(feature, layer) {
var found = false;
// Overlay ?
if (!layer) {
if (feature===self.bbox_) return false;
self.handles_.forEach (function(f) { if (f===feature) found=true; });
if (found) return { feature: feature, handle:feature.get('handle'), constraint:feature.get('constraint'), option:feature.get('option') };
}
// feature belong to a layer
if (self.layers_) {
for (var i=0; i<self.layers_.length; i++) {
if (self.layers_[i]===layer) return { feature: feature };
}
return null;
}
// feature in the collection
else if (self.features_) {
self.features_.forEach (function(f) { if (f===feature) found=true; });
if (found) return { feature: feature };
else return null;
}
// Others
else return { feature: feature };
},
{ hitTolerance: this.get('hitTolerance') }
) || {};
}
/** Draw transform sketch
* @param {boolean} draw only the center
*/
ol.interaction.Transform.prototype.drawSketch_ = function(center) {
this.overlayLayer_.getSource().clear();
if (!this.selection_.length) return;
var ext = this.selection_[0].getGeometry().getExtent();
// Clone and extend
ext = ol.extent.buffer(ext, 0);
for (var i=1, f; f = this.selection_[i]; i++) {
ol.extent.extend(ext, f.getGeometry().getExtent());
}
if (center===true) {
if (!this.ispt_) {
this.overlayLayer_.getSource().addFeature(new ol.Feature( { geometry: new ol.geom.Point(this.center_), handle:'rotate0' }) );
var geom = ol.geom.Polygon.fromExtent(ext);
var f = this.bbox_ = new ol.Feature(geom);
this.overlayLayer_.getSource().addFeature (f);
}
}
else {
if (this.ispt_) {
var p = this.getMap().getPixelFromCoordinate([ext[0], ext[1]]);
ext = ol.extent.boundingExtent([
this.getMap().getCoordinateFromPixel([p[0]-10, p[1]-10]),
this.getMap().getCoordinateFromPixel([p[0]+10, p[1]+10])
]);
}
var geom = ol.geom.Polygon.fromExtent(ext);
var f = this.bbox_ = new ol.Feature(geom);
var features = [];
var g = geom.getCoordinates()[0];
if (!this.ispt_) {
features.push(f);
// Middle
if (this.get('stretch') && this.get('scale')) for (var i=0; i<g.length-1; i++) {
f = new ol.Feature( { geometry: new ol.geom.Point([(g[i][0]+g[i+1][0])/2,(g[i][1]+g[i+1][1])/2]), handle:'scale', constraint:i%2?"h":"v", option:i });
features.push(f);
}
// Handles
if (this.get('scale')) for (var i=0; i<g.length-1; i++) {
f = new ol.Feature( { geometry: new ol.geom.Point(g[i]), handle:'scale', option:i });
features.push(f);
}
// Center
if (this.get('translate') && !this.get('translateFeature')) {
f = new ol.Feature( { geometry: new ol.geom.Point([(g[0][0]+g[2][0])/2, (g[0][1]+g[2][1])/2]), handle:'translate' });
features.push(f);
}
}
// Rotate
if (this.get('rotate')) {
f = new ol.Feature( { geometry: new ol.geom.Point(g[3]), handle:'rotate' });
features.push(f);
}
// Add sketch
this.overlayLayer_.getSource().addFeatures(features);
}
};
/** Select a feature to transform
* @param {ol.Feature} feature the feature to transform
* @param {boolean} add true to add the feature to the selection, default false
*/
ol.interaction.Transform.prototype.select = function(feature, add) {
if (!feature) {
this.selection_ = [];
return;
}
if (!feature.getGeometry || !feature.getGeometry()) return;
// Add to selection
if (add) this.selection_.push(feature);
else this.selection_ = [feature];
this.ispt_ = (this.selection_.length===1 ? (this.selection_[0].getGeometry().getType() == "Point") : false);
this.drawSketch_();
this.dispatchEvent({ type:'select', feature: feature, features: this.selection_ });
}
/**
* @param {ol.MapBrowserEvent} evt Map browser event.
* @return {boolean} `true` to start the drag sequence.
*/
ol.interaction.Transform.prototype.handleDownEvent_ = function(evt) {
var sel = this.getFeatureAtPixel_(evt.pixel);
var feature = sel.feature;
if (this.selection_.length
&& this.selection_.indexOf(feature) >=0
&& ((this.ispt_ && this.get('translate')) || this.get('translateFeature'))
){
sel.handle = 'translate';
}
if (sel.handle) {
this.mode_ = sel.handle;
this.opt_ = sel.option;
this.constraint_ = sel.constraint;
// Save info
this.coordinate_ = evt.coordinate;
this.pixel_ = evt.pixel;
this.geoms_ = [];
var extent = ol.extent.createEmpty();
for (var i=0, f; f=this.selection_[i]; i++) {
this.geoms_.push(f.getGeometry().clone());
extent = ol.extent.extend(extent, f.getGeometry().getExtent());
}
this.extent_ = (ol.geom.Polygon.fromExtent(extent)).getCoordinates()[0];
if (this.mode_==='rotate') {
this.center_ = this.getCenter() || ol.extent.getCenter(extent);
// we are now rotating (cursor down on rotate mode), so apply the grabbing cursor
var element = evt.map.getTargetElement();
element.style.cursor = this.Cursors.rotate0;
this.previousCursor_ = element.style.cursor;
} else {
this.center_ = ol.extent.getCenter(extent);
}
this.angle_ = Math.atan2(this.center_[1]-evt.coordinate[1], this.center_[0]-evt.coordinate[0]);
this.dispatchEvent({
type: this.mode_+'start',
feature: this.selection_[0], // backward compatibility
features: this.selection_,
pixel: evt.pixel,
coordinate: evt.coordinate
});
return true;
}
else {
if (feature){
if (!this.addFn_(evt)) this.selection_ = [];
var index = this.selection_.indexOf(feature);
if (index < 0) this.selection_.push(feature);
else this.selection_.splice(index,1);
} else {
this.selection_ = [];
}
this.ispt_ = this.selection_.length===1 ? (this.selection_[0].getGeometry().getType() == "Point") : false;
this.drawSketch_();
this.dispatchEvent({ type:'select', feature: feature, features: this.selection_, pixel: evt.pixel, coordinate: evt.coordinate });
return false;
}
};
/**
* Get features to transform
* @return {Array<ol.Feature>}
*/
ol.interaction.Transform.prototype.getFeatures = function() {
return this.selection_;
};
/**
* Get the rotation center
* @return {ol.coordinates|undefined}
*/
ol.interaction.Transform.prototype.getCenter = function() {
return this.get('center');
};
/**
* Set the rotation center
* @param {ol.coordinates|undefined} c the center point, default center on the objet
*/
ol.interaction.Transform.prototype.setCenter = function(c) {
return this.set('center', c);
}
/**
* @param {ol.MapBrowserEvent} evt Map browser event.
*/
ol.interaction.Transform.prototype.handleDragEvent_ = function(evt) {
switch (this.mode_) {
case 'rotate': {
var a = Math.atan2(this.center_[1]-evt.coordinate[1], this.center_[0]-evt.coordinate[0]);
if (!this.ispt) {
// var geometry = this.geom_.clone();
// geometry.rotate(a-this.angle_, this.center_);
// this.feature_.setGeometry(geometry);
for (var i=0, f; f=this.selection_[i]; i++) {
var geometry = this.geoms_[i].clone();
geometry.rotate(a - this.angle_, this.center_);
f.setGeometry(geometry);
}
}
this.drawSketch_(true);
this.dispatchEvent({
type:'rotating',
feature: this.selection_[0],
features: this.selection_,
angle: a-this.angle_,
pixel: evt.pixel,
coordinate: evt.coordinate
});
break;
}
case 'translate': {
var deltaX = evt.coordinate[0] - this.coordinate_[0];
var deltaY = evt.coordinate[1] - this.coordinate_[1];
//this.feature_.getGeometry().translate(deltaX, deltaY);
for (var i=0, f; f=this.selection_[i]; i++) {
f.getGeometry().translate(deltaX, deltaY);
}
this.handles_.forEach(function(f) {
f.getGeometry().translate(deltaX, deltaY);
});
this.coordinate_ = evt.coordinate;
this.dispatchEvent({
type:'translating',
feature: this.selection_[0],
features: this.selection_,
delta:[deltaX,deltaY],
pixel: evt.pixel,
coordinate: evt.coordinate
});
break;
}
case 'scale': {
var center = this.center_;
if (this.get('modifyCenter')(evt)) {
center = this.extent_[(Number(this.opt_)+2)%4];
}
var scx = (evt.coordinate[0] - center[0]) / (this.coordinate_[0] - center[0]);
var scy = (evt.coordinate[1] - center[1]) / (this.coordinate_[1] - center[1]);
if (this.constraint_) {
if (this.constraint_=="h") scx=1;
else scy=1;
} else {
if (this.get('keepAspectRatio')(evt)) {
scx = scy = Math.min(scx,scy);
}
}
for (var i=0, f; f=this.selection_[i]; i++) {
var geometry = this.geoms_[i].clone();
geometry.applyTransform(function(g1, g2, dim) {
if (dim<2) return g2;
for (var i=0; i<g1.length; i+=dim) {
if (scx!=1) g2[i] = center[0] + (g1[i]-center[0])*scx;
if (scy!=1) g2[i+1] = center[1] + (g1[i+1]-center[1])*scy;
}
return g2;
});
f.setGeometry(geometry);
}
this.drawSketch_();
this.dispatchEvent({
type:'scaling',
feature: this.selection_[0],
features: this.selection_,
scale:[scx,scy],
pixel: evt.pixel,
coordinate: evt.coordinate
});
}
default: break;
}
};
/**
* @param {ol.MapBrowserEvent} evt Event.
*/
ol.interaction.Transform.prototype.handleMoveEvent_ = function(evt) {
// console.log("handleMoveEvent");
if (!this.mode_)
{ var map = evt.map;
var sel = this.getFeatureAtPixel_(evt.pixel);
var element = evt.map.getTargetElement();
if (sel.feature)
{ var c = sel.handle ? this.Cursors[(sel.handle||'default')+(sel.constraint||'')+(sel.option||'')] : this.Cursors.select;
if (this.previousCursor_===undefined)
{ this.previousCursor_ = element.style.cursor;
}
element.style.cursor = c;
}
else
{ if (this.previousCursor_!==undefined) element.style.cursor = this.previousCursor_;
this.previousCursor_ = undefined;
}
}
};
/**
* @param {ol.MapBrowserEvent} evt Map browser event.
* @return {boolean} `false` to stop the drag sequence.
*/
ol.interaction.Transform.prototype.handleUpEvent_ = function(evt) {
// remove rotate0 cursor on Up event, otherwise it's stuck on grab/grabbing
if (this.mode_ === 'rotate') {
var element = evt.map.getTargetElement();
element.style.cursor = this.Cursors.default;
this.previousCursor_ = undefined;
}
//dispatchEvent
this.dispatchEvent({
type:this.mode_+'end',
feature: this.selection_[0],
features: this.selection_,
oldgeom: this.geoms_[0],
oldgeoms: this.geoms_
});
this.drawSketch_();
this.mode_ = null;
return false;
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
@classdesc
ol.source.DBPedia is a DBPedia layer source that load DBPedia located content in a vector layer.
olx.source.DBPedia: olx.source.Vector
{ url: {string} Url for DBPedia SPARQL
}
@require jQuery
Inherits from:
<ol.source.Vector>
*/
/**
* @constructor ol.source.DBPedia
* @extends {ol.source.Vector}
* @param {olx.source.DBPedia=} opt_options
*/
ol.source.DBPedia = function(opt_options)
{ var options = opt_options || {};
var self = this;
options.loader = this._loaderFn;
/** Url for DBPedia SPARQL */
this._url = options.url || "http://fr.dbpedia.org/sparql";
/** Max resolution to load features */
this._maxResolution = options.maxResolution || 100;
/** Result language */
this._lang = options.lang || "fr";
/** Query limit */
this._limit = options.limit || 1000;
/** Default attribution */
if (!options.attributions) options.attributions = [ new ol.Attribution({ html:"&copy; <a href='http://dbpedia.org/'>DBpedia</a> CC-by-SA" }) ];
// Bbox strategy : reload at each move
if (!options.strategy) options.strategy = ol.loadingstrategy.bbox;
ol.source.Vector.call (this, options);
};
ol.inherits (ol.source.DBPedia, ol.source.Vector);
/** Decode RDF attributes and choose to add feature to the layer
* @param {feature} the feature
* @param {attributes} RDF attributes
* @param {lastfeature} last feature added (null if none)
* @return {boolean} true: add the feature to the layer
* @API stable
*/
ol.source.DBPedia.prototype.readFeature = function (feature, attributes, lastfeature)
{ // Copy RDF attributes values
for (var i in attributes) feature.set (i, attributes[i].value);
// Prevent same feature with different type duplication
if (lastfeature && lastfeature.get("subject") == attributes.subject.value)
{ // Kepp dbpedia.org type ?
// if (bindings[i].type.match ("dbpedia.org") lastfeature.get("type") = bindings[i].type.value;
// Concat types
lastfeature.set("type", lastfeature.get("type") +"\n"+ attributes.type.value);
return false;
}
else
{ return true;
}
};
/** Set RDF query subject, default: select label, thumbnail, abstract and type
* @API stable
*/
ol.source.DBPedia.prototype.querySubject = function ()
{ return "?subject rdfs:label ?label. "
+ "OPTIONAL {?subject dbpedia-owl:thumbnail ?thumbnail}."
+ "OPTIONAL {?subject dbpedia-owl:abstract ?abstract} . "
+ "OPTIONAL {?subject rdf:type ?type}";
}
/** Set RDF query filter, default: select language
* @API stable
*/
ol.source.DBPedia.prototype.queryFilter = function ()
{ return "lang(?label) = '"+this._lang+"' "
+ "&& lang(?abstract) = '"+this._lang+"'"
// Filter on type
//+ "&& regex (?type, 'Monument|Sculpture|Museum', 'i')"
}
/** Loader function used to load features.
* @private
*/
ol.source.DBPedia.prototype._loaderFn = function(extent, resolution, projection)
{ if (resolution > this._maxResolution) return;
var self = this;
var bbox = ol.proj.transformExtent(extent, projection, "EPSG:4326");
// SPARQL request: for more info @see http://fr.dbpedia.org/
query = "PREFIX geo: <http://www.w3.org/2003/01/geo/wgs84_pos#> "
+ "SELECT DISTINCT * WHERE { "
+ "?subject geo:lat ?lat . "
+ "?subject geo:long ?long . "
+ this.querySubject()+" . "
+ "FILTER("+this.queryFilter()+") . "
// Filter bbox
+ "FILTER(xsd:float(?lat) <= " + bbox[3] + " && " + bbox[1] + " <= xsd:float(?lat) "
+ "&& xsd:float(?long) <= " + bbox[2] + " && " + bbox[0] + " <= xsd:float(?long) "
+ ") . "
+ "} LIMIT "+this._limit;
// Ajax request to get the tile
$.ajax(
{ url: this._url,
dataType: 'jsonp',
data: { query: query, format:"json" },
success: function(data)
{ var bindings = data.results.bindings;
var features = [];
var att, pt, feature, lastfeature = null;
for ( var i in bindings )
{ att = bindings[i];
pt = [Number(bindings[i].long.value), Number(bindings[i].lat.value)];
feature = new ol.Feature(new ol.geom.Point(ol.proj.transform (pt,"EPSG:4326",projection)));
if (self.readFeature(feature, att, lastfeature))
{ features.push(feature);
lastfeature = feature;
}
}
self.addFeatures(features);
}});
};
(function(){
// Style cache
var styleCache = {};
/** Reset the cache (when fonts are loaded)
*/
ol.style.clearDBPediaStyleCache = function()
{ styleCache = {};
}
/** Get a default style function for dbpedia
* @param {} options
* @param {string|function|undefined} options.glyph a glyph name or a function that takes a feature and return a glyph
* @param {number} options.radius radius of the symbol, default 8
* @param {ol.style.Fill} options.fill style for fill, default navy
* @param {ol.style.stroke} options.stroke style for stroke, default 2px white
* @param {string} options.prefix a prefix if many style used for the same type
*
* @require ol.style.FontSymbol and FontAwesome defs are required for dbPediaStyleFunction()
*/
ol.style.dbPediaStyleFunction = function(options)
{ if (!options) options={};
// Get font function using dbPedia type
var getFont;
switch (typeof(options.glyph))
{ case "function": getFont = options.glyph; break;
case "string": getFont = function(){ return options.glyph; }; break;
default:
{ getFont = function (f)
{ var type = f.get("type");
if (type)
{ if (type.match("/Museum")) return "fa-camera";
else if (type.match("/Monument")) return "fa-building";
else if (type.match("/Sculpture")) return "fa-android";
else if (type.match("/Religious")) return "fa-institution";
else if (type.match("/Castle")) return "fa-key";
else if (type.match("Water")) return "fa-tint";
else if (type.match("Island")) return "fa-leaf";
else if (type.match("/Event")) return "fa-heart";
else if (type.match("/Artwork")) return "fa-asterisk";
else if (type.match("/Stadium")) return "fa-futbol-o";
else if (type.match("/Place")) return "fa-street-view";
}
return "fa-star";
}
break;
}
}
// Default values
var radius = options.radius || 8;
var fill = options.fill || new ol.style.Fill({ color:"navy"});
var stroke = options.stroke || new ol.style.Stroke({ color: "#fff", width: 2 });
var prefix = options.prefix ? options.prefix+"_" : "";
// Vector style function
return function (feature, resolution)
{ var glyph = getFont(feature);
var k = prefix + glyph;
var style = styleCache[k];
if (!style)
{ styleCache[k] = style = new ol.style.Style
({ image: new ol.style.FontSymbol(
{ glyph: glyph,
radius: radius,
fill: fill,
stroke: stroke
})
});
}
return [style];
}
};
})();
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
ol.source.GeoImage is a layer source with georeferencement to place it on a map.
olx.source.GeoImageOptions:
{ url: {string} url of the static image
image: {image} the static image, if not provided, use url to load an image
imageCenter: {ol.Coordinate} of the center of the image
imageScale: {ol.Size|Number} [scalex, scaley] of the image
imageRotate: {number} angle of the image in radian, default 0
imageCrop: {ol.Extent} of the image to be show (in the image) default: [0,0,imageWidth,imageHeight]
imageMask: {Array.<ol.Coordinate>} - linestring to mask the image on the map
}
*/
/** Layer source with georeferencement to place it on a map
* @constructor
* @extends {ol.source.ImageCanvas}
* @param {olx.source.GeoImageOptions=} options
*/
ol.source.GeoImage = function(opt_options)
{ var options = {
attributions: opt_options.attributions,
logo: opt_options.logo,
projection: opt_options.projection
};
// options.projection = opt_options.projection;
// Coordinate of the image center
this.center = opt_options.imageCenter;
// Scale of the image
this.scale = opt_options.imageScale;
// Rotation of the image
this.rotate = opt_options.imageRotate ? opt_options.imageRotate : 0;
// Crop of the image
this.crop = opt_options.imageCrop;
// Mask of the image
this.mask = opt_options.imageMask;
// Load Image
this._image = (opt_options.image ? opt_options.image : new Image );
this._image.crossOrigin = opt_options.crossOrigin; // 'anonymous';
// Show image on load
var self = this;
this._image.onload = function()
{ self.setCrop (self.crop);
self.changed();
}
if (!opt_options.image) this._image.src = opt_options.url;
// Draw image on canvas
options.canvasFunction = function(extent, resolution, pixelRatio, size, projection)
{ var canvas = document.createElement('canvas');
canvas.width = size[0];
canvas.height = size[1];
var ctx = canvas.getContext('2d');
if (!this._imageSize) return canvas;
// transform coords to pixel
function tr(xy)
{ return [(xy[0]-extent[0])/(extent[2]-extent[0]) * size[0],
(xy[1]-extent[3])/(extent[1]-extent[3]) * size[1]
];
}
// Clipping mask
if (this.mask)
{ ctx.beginPath();
var p = tr(this.mask[0]);
ctx.moveTo(p[0],p[1]);
for (var i=1; i<this.mask.length; i++)
{ p = tr(this.mask[i]);
ctx.lineTo(p[0],p[1]);
}
ctx.clip();
}
// Draw
var pixel = tr(this.center);
var dx = (this._image.naturalWidth/2 - this.crop[0]) *this.scale[0] /resolution *pixelRatio;
var dy = (this._image.naturalHeight/2 - this.crop[1]) *this.scale[1] /resolution *pixelRatio;
var sx = this._imageSize[0]*this.scale[0]/resolution *pixelRatio;
var sy = this._imageSize[1]*this.scale[1]/resolution *pixelRatio;
ctx.translate(pixel[0],pixel[1]);
if (this.rotate) ctx.rotate(this.rotate);
ctx.drawImage(this._image, this.crop[0], this.crop[1], this._imageSize[0], this._imageSize[1], -dx, -dy, sx,sy);
return canvas;
}
ol.source.ImageCanvas.call (this, options);
this.setCrop (this.crop);
};
ol.inherits (ol.source.GeoImage, ol.source.ImageCanvas);
/**
* Get coordinate of the image center.
* @return {ol.Coordinate} coordinate of the image center.
* @api stable
*/
ol.source.GeoImage.prototype.getCenter = function()
{ return this.center;
}
/**
* Set coordinate of the image center.
* @param {ol.Coordinate} coordinate of the image center.
* @api stable
*/
ol.source.GeoImage.prototype.setCenter = function(center)
{ this.center = center;
this.changed();
}
/**
* Get image scale.
* @return {ol.size} image scale (along x and y axis).
* @api stable
*/
ol.source.GeoImage.prototype.getScale = function()
{ return this.scale;
}
/**
* Set image scale.
* @param {ol.size|Number} image scale (along x and y axis or both).
* @api stable
*/
ol.source.GeoImage.prototype.setScale = function(scale)
{ switch (typeof(scale))
{ case 'number':
scale = [scale,scale];
break;
case 'object':
if (scale.length != 2) return;
break;
default: return;
}
this.scale = scale;
this.changed();
};
/**
* Get image rotation.
* @return {Number} rotation in degre.
* @api stable
*/
ol.source.GeoImage.prototype.getRotation = function()
{ return this.rotate;
};
/**
* Set image rotation.
* @param {Number} rotation in radian.
* @api stable
*/
ol.source.GeoImage.prototype.setRotation = function(angle)
{ this.rotate = angle;
this.changed();
};
/**
* Get the image.
* @api stable
*/
ol.source.GeoImage.prototype.getGeoImage = function()
{ return this._image;
};
/**
* Get image crop extent.
* @return {ol.extent} image crop extent.
* @api stable
*/
ol.source.GeoImage.prototype.getCrop = function()
{ return this.crop;
};
/**
* Set image mask.
* @param {ol.geom.LineString} coords of the mask
* @api stable
*/
ol.source.GeoImage.prototype.setMask = function(mask)
{ this.mask = mask;
this.changed();
};
/**
* Get image mask.
* @return {ol.geom.LineString} coords of the mask
* @api stable
*/
ol.source.GeoImage.prototype.getMask = function()
{ return this.mask;
};
/**
* Set image crop extent.
* @param {ol.extent|Number} image crop extent or a number to crop from original size.
* @api stable
*/
ol.source.GeoImage.prototype.setCrop = function(crop)
{ // Image not loaded => get it latter
if (!this._image.naturalWidth)
{ this.crop = crop;
return;
}
if (crop)
{ switch (typeof(crop))
{ case 'number':
crop = [crop,crop,this._image.naturalWidth-crop,this._image.naturalHeight-crop];
break;
case 'object':
if (crop.length != 4) return;
break;
default: return;
}
var crop = ol.extent.boundingExtent([ [crop[0],crop[1]], [crop[2],crop[3]] ]);
this.crop = [ Math.max(0,crop[0]), Math.max(0,crop[1]), Math.min(this._image.naturalWidth,crop[2]), Math.min(this._image.naturalHeight,crop[3]) ];
}
else this.crop = [0,0, this._image.naturalWidth,this._image.naturalHeight];
if (this.crop[2]<=this.crop[0]) this.crop[2] = this.crop[0]+1;
if (this.crop[3]<=this.crop[1]) this.crop[3] = this.crop[1]+1;
this._imageSize = [ this.crop[2]-this.crop[0], this.crop[3]-this.crop[1] ];
this.changed();
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** A source for hexagonal binning
* @constructor
* @extends {ol.source.Vector}
* @param {} options ol.source.VectorOptions + ol.HexGridOptions
* @param {ol.source.Vector} options.source Source
* @param {Number} options.size size of the exagon in map units, default 80000
* @param {ol.coordinate} options.origin orgin of the grid, default [0,0]
* @param {pointy|flat} options.layout grid layout, default pointy
* @param {function|undefined} options.geometryFunction Function that takes an ol.Feature as argument and returns an ol.geom.Point as feature's center.
*/
ol.source.HexBin = function(options) {
options = options || {} ;
// bind function for callback
this._bind = { modify: this._onModifyFeature.bind(this) };
ol.source.Vector.call (this, options);
// The HexGrid
this._hexgrid = new ol.HexGrid(options);
this._bin = {};
// Source and origin
this._origin = options.source;
// Geometry function to get a point
this._geomFn = options.geometryFunction || ol.coordinate.getFeatureCenter || function(f) { return f.getGeometry().getFirstCoordinate(); };
// Existing features
this.reset();
// Future features
this._origin.on("addfeature", this._onAddFeature.bind(this));
this._origin.on("removefeature", this._onRemoveFeature.bind(this));
};
ol.inherits (ol.source.HexBin, ol.source.Vector);
/**
* On add feature
* @param {ol.Event} e
* @private
*/
ol.source.HexBin.prototype._onAddFeature = function(e) {
var f = e.feature || e.target;
var h = this._hexgrid.coord2hex(this._geomFn(f));
var id = h.toString();
if (this._bin[id]) {
this._bin[id].get('features').push(f);
} else {
var ex = new ol.Feature(new ol.geom.Polygon([this._hexgrid.getHexagon(h)]));
ex.set('features',[f]);
ex.set('center', new ol.geom.Point(ol.extent.getCenter(ex.getGeometry().getExtent())));
this._bin[id] = ex;
this.addFeature(ex);
}
f.on("change", this._bind.modify);
};
/**
* Get the hexagon of a feature
* @param {ol.Feature} f
* @return {} the bin id, the index of the feature in the bin and a boolean if the feature has moved to an other bin
*/
ol.source.HexBin.prototype.getBin = function(f) {
// Test if feature exists in the current hex
var id = this._hexgrid.coord2hex(this._geomFn(f)).toString();
if (this._bin[id]) {
var index = this._bin[id].get('features').indexOf(f);
if (index > -1) return { id:id, index:index };
}
// The feature has moved > check all bins
for (id in this._bin) {
var index = this._bin[id].get('features').indexOf(f);
if (index > -1) return { id:id, index:index, moved:true };
}
return false;
};
/**
* On remove feature
* @param {ol.Event} e
* @param {*} bin
* @private
*/
ol.source.HexBin.prototype._onRemoveFeature = function(e, bin) {
var f = e.feature || e.target;
var b = bin || this.getBin(f);
if (b) {
var features = this._bin[b.id].get('features');
features.splice(b.index, 1);
if (!features.length) {
this.removeFeature(this._bin[b.id]);
delete this._bin[b.id];
}
} else {
console.log("[ERROR:HexBin] remove feature feature doesn't exists anymore.");
}
f.un("change", this._bind.modify);
};
/**
* A feature has been modified
* @param {ol.Event} e
* @private
*/
ol.source.HexBin.prototype._onModifyFeature = function(e) {
var bin = this.getBin(e.target);
if (bin && bin.moved) {
// remove from the bin
this._onRemoveFeature(e, bin);
// insert in the new bin
this._onAddFeature(e);
}
this.changed();
};
/** Clear all bins and generate a new one
*/
ol.source.HexBin.prototype.reset = function() {
this._bin = {};
this.clear();
var features = this._origin.getFeatures();
for (var i=0, f; f=features[i]; i++) {
this._onAddFeature({ feature:f });
};
};
/**
* Get the orginal source
* @return {ol.source.Vector}
*/
ol.source.HexBin.prototype.getSource = function() {
return this._origin;
};
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
@classdesc
ol.source.Mapillary is a source that load Mapillary's geotagged photos in a vector layer.
@require jQuery
Inherits from:
<ol.source.Vector>
*/
/**
* @constructor ol.source.Mapillary
* @extends {ol.source.Vector}
* @param {olx.source.Mapillary=} options
*/
ol.source.Mapillary = function(opt_options)
{ var options = opt_options || {};
var self = this;
options.loader = this._loaderFn;
/** Max resolution to load features */
this._maxResolution = options.maxResolution || 100;
/** Query limit */
this._limit = options.limit || 100;
/** Default attribution */
if (!options.attributions) options.attributions = [ new ol.Attribution({ html:"&copy; <a href='https://www.mapillary.com/'>Mapillary</a>" }) ];
// Bbox strategy : reload at each move
if (!options.strategy) options.strategy = ol.loadingstrategy.bbox;
// Init parent
ol.source.Vector.call (this, options);
// Client ID
// this.set("clientId", options.clientId);
};
ol.inherits (ol.source.Mapillary, ol.source.Vector);
/** Decode wiki attributes and choose to add feature to the layer
* @param {feature} the feature
* @param {attributes} wiki attributes
* @return {boolean} true: add the feature to the layer
* @API stable
*/
ol.source.Mapillary.prototype.readFeature = function (feature, attributes)
{ // Allways read feature (no filter)
return true;
};
/** Loader function used to load features.
* @private
*/
ol.source.Mapillary.prototype._loaderFn = function(extent, resolution, projection)
{ if (resolution > this._maxResolution) return;
var self = this;
var bbox = ol.proj.transformExtent(extent, projection, "EPSG:4326");
// Commons API: for more info @see https://www.mapillary.com/developer
var date = Date.now() - 6 * 30 * 24 * 60 * 60 * 1000;
var url = "https://a.mapillary.com/v2/search/im?client_id="
+ this.get('clientId')
+ "&max_lat=" + bbox[3]
+ "&max_lon=" + bbox[2]
+ "&min_lat=" + bbox[1]
+ "&min_lon=" + bbox[0]
+ "&limit="+(this._limit-1)
+ "&start_time=" + date;
// Ajax request to get the tile
$.ajax(
{ url: url,
dataType: 'jsonp',
success: function(data)
{ console.log(data);
/*
var features = [];
var att, pt, feature, lastfeature = null;
if (data.query && data.query.pages) return;
for ( var i in data.query.pages)
{ att = data.query.pages[i];
if (att.coordinates && att.coordinates.length )
{ pt = [att.coordinates[0].lon, att.coordinates[0].lat];
}
else
{ var meta = att.imageinfo[0].metadata;
if (!meta)
{ //console.log(att);
continue;
}
pt = [];
for (var k=0; k<meta.length; k++)
{ if (meta[k].name=="GPSLongitude") pt[0] = meta[k].value;
if (meta[k].name=="GPSLatitude") pt[1] = meta[k].value;
}
if (!pt.length)
{ //console.log(att);
continue;
}
}
feature = new ol.Feature(new ol.geom.Point(ol.proj.transform (pt,"EPSG:4326",projection)));
att.imageinfo[0].title = att.title;
if (self.readFeature(feature, att.imageinfo[0]))
{ features.push(feature);
}
}
self.addFeatures(features);
*/
}});
};
/* Copyright (c) 2018 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* OSM layer using the Ovepass API
* @constructor ol.source.Overpass
* @extends {ol.source.Vector}
* @param {any} options
* @param {string} options.url service url, default: https://overpass-api.de/api/interpreter
* @param {Array<string>} options.filter an array of tag filters, ie. ["key", "key=value", "key~value", ...]
* @param {boolean} options.node get nodes, default: true
* @param {boolean} options.way get ways, default: true
* @param {boolean} options.rel get relations, default: false
* @param {number} options.maxResolution maximum resolution to load features
* @param {string|ol.Attribution|Array<string>} options.attributions source attribution, default OSM attribution
* @param {ol.loadingstrategy} options.strategy loading strategy, default ol.loadingstrategy.bbox
*/
ol.source.Overpass = function(options) {
options = options || {};
var self = this;
options.loader = this._loaderFn;
/** Ovepass API Url */
this._url = options.url || 'https://overpass-api.de/api/interpreter';
/** Max resolution to load features */
this._maxResolution = options.maxResolution || 100;
/** Default attribution */
if (!options.attributions) {
options.attributions = ol.source.OSM.ATTRIBUTION;
}
// Bbox strategy : reload at each move
if (!options.strategy) options.strategy = ol.loadingstrategy.bbox;
ol.source.Vector.call (this, options);
this._types = {
node: options.node!==false,
way: options.way!==false,
rel: options.rel===true
};
this._filter = options.filter;
};
ol.inherits (ol.source.Overpass, ol.source.Vector);
/** Loader function used to load features.
* @private
*/
ol.source.Overpass.prototype._loaderFn = function(extent, resolution, projection) {
if (resolution > this._maxResolution) return;
var self = this;
var bbox = ol.proj.transformExtent(extent, projection, "EPSG:4326");
bbox = bbox[1] + ',' + bbox[0] + ',' + bbox[3] + ',' + bbox[2];
// Overpass QL
var query = '[bbox:'+bbox+'][out:xml][timeout:25];';
query += '(';
// Search attributes
for (var t in this._types) {
if (this._types[t]) {
query += t;
for (var n=0, filter; filter = this._filter[n]; n++) {
query += '['+filter+']';
}
query += ';'
}
}
query +=');out;>;out skel qt;'
var ajax = new XMLHttpRequest();
ajax.open('POST', this._url, true);
ajax.onload = function () {
var features = new ol.format.OSMXML().readFeatures(this.responseText,{featureProjection: projection});
var result = [];
// Remove duplicated features
for (var i=0, f; f=features[i]; i++) {
if (!self.hasFeature(f)) result.push(f);
}
vectorSource.addFeatures(result);
};
ajax.onerror = function () {
console.log(arguments);
};
ajax.send('data='+query);
};
/**
* Search if feature is allready loaded
* @param {ol.Feature} feature
* @return {boolean}
* @private
*/
ol.source.Overpass.prototype.hasFeature = function(feature) {
var p = feature.getGeometry().getFirstCoordinate();
var id = feature.getId();
var existing = this.getFeaturesInExtent([p[0]-0.1, p[1]-0.1, p[0]+0.1, p[1]+0.1]);
for (var i=0, f; f=existing[i]; i++) {
if (id===f.getId()) {
return true;
}
}
return false;
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
@classdesc
ol.source.WikiCommons is a source that load Wikimedia Commons content in a vector layer.
@require jQuery
Inherits from:
<ol.source.Vector>
*/
/**
* @constructor ol.source.WikiCommons
* @extends {ol.source.Vector}
* @param {olx.source.WikiCommons=} options
*/
ol.source.WikiCommons = function(opt_options)
{ var options = opt_options || {};
var self = this;
options.loader = this._loaderFn;
/** Max resolution to load features */
this._maxResolution = options.maxResolution || 100;
/** Result language */
this._lang = options.lang || "fr";
/** Query limit */
this._limit = options.limit || 100;
/** Default attribution */
if (!options.attributions) options.attributions = [ new ol.Attribution({ html:"&copy; <a href='https://commons.wikimedia.org/'>Wikimedia Commons</a>" }) ];
// Bbox strategy : reload at each move
if (!options.strategy) options.strategy = ol.loadingstrategy.bbox;
ol.source.Vector.call (this, options);
};
ol.inherits (ol.source.WikiCommons, ol.source.Vector);
/** Decode wiki attributes and choose to add feature to the layer
* @param {feature} the feature
* @param {attributes} wiki attributes
* @return {boolean} true: add the feature to the layer
* @API stable
*/
ol.source.WikiCommons.prototype.readFeature = function (feature, attributes)
{ feature.set("descriptionurl", attributes.descriptionurl);
feature.set("url", attributes.url);
feature.set("title", attributes.title.replace(/^file:|.jpg$/ig,""));
feature.set("thumbnail", attributes.url.replace(/^(.+wikipedia\/commons)\/([a-zA-Z0-9]\/[a-zA-Z0-9]{2})\/(.+)$/,"$1/thumb/$2/$3/200px-$3"));
feature.set("user", attributes.user);
if (attributes.extmetadata && attributes.extmetadata.LicenseShortName) feature.set("copy", attributes.extmetadata.LicenseShortName.value);
return true;
};
/** Loader function used to load features.
* @private
*/
ol.source.WikiCommons.prototype._loaderFn = function(extent, resolution, projection)
{ if (resolution > this._maxResolution) return;
var self = this;
var bbox = ol.proj.transformExtent(extent, projection, "EPSG:4326");
// Commons API: for more info @see https://commons.wikimedia.org/wiki/Commons:API/MediaWiki
var url = "https://commons.wikimedia.org/w/api.php?action=query&format=json&origin=*&prop=coordinates|imageinfo"
+ "&generator=geosearch&iiprop=timestamp|user|url|extmetadata|metadata|size&iiextmetadatafilter=LicenseShortName"
+ "&ggsbbox=" + bbox[3] + "|" + bbox[0] + "|" + bbox[1] + "|" + bbox[2]
+ "&ggslimit="+this._limit
+ "&iilimit="+(this._limit-1)
+ "&ggsnamespace=6";
// Ajax request to get the tile
$.ajax(
{ url: url,
dataType: 'jsonp',
success: function(data)
{ //console.log(data);
var features = [];
var att, pt, feature, lastfeature = null;
if (!data.query || !data.query.pages) return;
for ( var i in data.query.pages)
{ att = data.query.pages[i];
if (att.coordinates && att.coordinates.length )
{ pt = [att.coordinates[0].lon, att.coordinates[0].lat];
}
else
{ var meta = att.imageinfo[0].metadata;
if (!meta)
{ //console.log(att);
continue;
}
pt = [];
var found=0;
for (var k=0; k<meta.length; k++)
{ if (meta[k].name=="GPSLongitude")
{ pt[0] = meta[k].value;
found++;
}
if (meta[k].name=="GPSLatitude")
{ pt[1] = meta[k].value;
found++;
}
}
if (found!=2)
{ //console.log(att);
continue;
}
}
feature = new ol.Feature(new ol.geom.Point(ol.proj.transform (pt,"EPSG:4326",projection)));
att.imageinfo[0].title = att.title;
if (self.readFeature(feature, att.imageinfo[0]))
{ features.push(feature);
}
}
self.addFeatures(features);
}});
};
/*
Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (http://www.cecill.info/).
ol.layer.AnimatedCluster is a vector layer that animate cluster
*/
/**
* A vector layer for animated cluster
* @constructor
* @extends {ol.layer.Vector}
* @param {olx.layer.AnimatedClusterOptions=} options extend olx.layer.Options
* @param {Number} options.animationDuration animation duration in ms, default is 700ms
* @param {ol.easingFunction} animationMethod easing method to use, default ol.easing.easeOut
*/
ol.layer.AnimatedCluster = function(opt_options)
{ var options = opt_options || {};
ol.layer.Vector.call (this, options);
this.oldcluster = new ol.source.Vector();
this.clusters = [];
this.animation={start:false};
this.set('animationDuration', typeof(options.animationDuration)=='number' ? options.animationDuration : 700);
this.set('animationMethod', options.animationMethod || ol.easing.easeOut);
// Save cluster before change
this.getSource().on('change', this.saveCluster.bind(this));
// Animate the cluster
this.on('precompose', this.animate.bind(this));
this.on('postcompose', this.postanimate.bind(this));
};
ol.inherits (ol.layer.AnimatedCluster, ol.layer.Vector);
/** save cluster features before change
* @private
*/
ol.layer.AnimatedCluster.prototype.saveCluster = function() {
if (this.oldcluster) {
this.oldcluster.clear();
if (!this.get('animationDuration')) return;
var features = this.getSource().getFeatures();
if (features.length && features[0].get('features'))
{ this.oldcluster.addFeatures (this.clusters);
this.clusters = features.slice(0);
this.sourceChanged = true;
}
}
};
/**
* Get the cluster that contains a feature
* @private
*/
ol.layer.AnimatedCluster.prototype.getClusterForFeature = function(f, cluster)
{ for (var j=0, c; c=cluster[j]; j++)
{ var features = cluster[j].get('features');
if (features && features.length)
{ for (var k=0, f2; f2=features[k]; k++)
{ if (f===f2)
{ return cluster[j];
}
}
}
}
return false;
};
/**
* Stop animation
* @private
*/
ol.layer.AnimatedCluster.prototype.stopAnimation = function()
{ this.animation.start = false;
this.animation.cA = [];
this.animation.cB = [];
};
/**
* animate the cluster
* @private
*/
ol.layer.AnimatedCluster.prototype.animate = function(e)
{ var duration = this.get('animationDuration');
if (!duration) return;
var resolution = e.frameState.viewState.resolution;
var a = this.animation;
var time = e.frameState.time;
// Start a new animation, if change resolution and source has changed
if (a.resolution != resolution && this.sourceChanged)
{ var extent = e.frameState.extent;
if (a.resolution < resolution)
{ extent = ol.extent.buffer(extent, 100*resolution);
a.cA = this.oldcluster.getFeaturesInExtent(extent);
a.cB = this.getSource().getFeaturesInExtent(extent);
a.revers = false;
}
else
{ extent = ol.extent.buffer(extent, 100*resolution);
a.cA = this.getSource().getFeaturesInExtent(extent);
a.cB = this.oldcluster.getFeaturesInExtent(extent);
a.revers = true;
}
a.clusters = [];
for (var i=0, c0; c0=a.cA[i]; i++)
{ var f = c0.get('features');
if (f && f.length)
{ var c = this.getClusterForFeature (f[0], a.cB);
if (c) a.clusters.push({ f:c0, pt:c.getGeometry().getCoordinates() });
}
}
// Save state
a.resolution = resolution;
this.sourceChanged = false;
// No cluster or too much to animate
if (!a.clusters.length || a.clusters.length>1000)
{ this.stopAnimation();
return;
}
// Start animation from now
time = a.start = (new Date()).getTime();
}
// Run animation
if (a.start)
{ var vectorContext = e.vectorContext;
var d = (time - a.start) / duration;
// Animation ends
if (d > 1.0)
{ this.stopAnimation();
d = 1;
}
d = this.get('animationMethod')(d);
// Animate
var style = this.getStyle();
var stylefn = (typeof(style) == 'function') ? style : style.length ? function(){ return style; } : function(){ return [style]; } ;
// Layer opacity
e.context.save();
e.context.globalAlpha = this.getOpacity();
// Retina device
var ratio = e.frameState.pixelRatio;
for (var i=0, c; c=a.clusters[i]; i++)
{ var pt = c.f.getGeometry().getCoordinates();
var dx = pt[0]-c.pt[0];
var dy = pt[1]-c.pt[1];
if (a.revers)
{ pt[0] = c.pt[0] + d * dx;
pt[1] = c.pt[1] + d * dy;
}
else
{ pt[0] = pt[0] - d * dx;
pt[1] = pt[1] - d * dy;
}
// Draw feature
var st = stylefn(c.f, resolution, true);
// Preserve pixel ration on retina
var geo = new ol.geom.Point(pt);
for (var k=0, s; s=st[k]; k++)
{ var sc;
// OL < v4.3 : setImageStyle doesn't check retina
var imgs = ol.Map.prototype.getFeaturesAtPixel ? false : s.getImage();
if (imgs)
{ sc = imgs.getScale();
imgs.setScale(sc*ratio);
}
// OL3 > v3.14
if (vectorContext.setStyle)
{ // If one feature: draw the feature
if (c.f.get("features").length===1 && !dx && !dy) {
vectorContext.drawFeature(c.f.get("features")[0], s);
}
// else draw a point
else {
vectorContext.setStyle(s);
vectorContext.drawGeometry(geo);
}
}
// older version
else
{ vectorContext.setImageStyle(imgs);
vectorContext.setTextStyle(s.getText());
vectorContext.drawPointGeometry(geo);
}
if (imgs) imgs.setScale(sc);
}
}
e.context.restore();
// tell OL3 to continue postcompose animation
e.frameState.animate = true;
// Prevent layer drawing (clip with null rect)
e.context.save();
e.context.beginPath();
e.context.rect(0,0,0,0);
e.context.clip();
this.clip_ = true;
}
return;
};
/**
* remove clipping after the layer is drawn
* @private
*/
ol.layer.AnimatedCluster.prototype.postanimate = function(e)
{ if (this.clip_)
{ e.context.restore();
this.clip_ = false;
}
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Return a preview image of the source.
* @param {ol.Coordinate|undefined} lonlat The center of the preview.
* @param {number} resolution of the preview.
* @return {String} the preview url
* @api
*/
ol.source.Source.prototype.getPreview = function(lonlat, resolution)
{ return "";
};
/**
* Return the tile image of the source.
* @param {ol.Coordinate|undefined} lonlat The center of the preview.
* @param {number} resolution of the preview.
* @return {String} the preview url
* @api
*/
ol.source.Tile.prototype.getPreview = function(lonlat, resolution)
{ if (!lonlat) lonlat = [21020, 6355964];
if (!resolution) resolution = 150;
var coord = this.getTileGrid().getTileCoordForCoordAndResolution(lonlat, resolution);
var fn = this.getTileUrlFunction();
return fn.call(this, coord, this.getProjection());
};
/**
* Return the tile image of the source.
* @param {ol.Coordinate|undefined} lonlat The center of the preview.
* @param {number} resolution of the preview.
* @return {String} the preview url
* @api
*/
ol.source.TileWMS.prototype.getPreview = function(lonlat, resolution)
{ if (!lonlat) lonlat = [21020, 6355964];
if (!resolution) resolution = 150;
/* No way to acces tileUrlFunction...
var fn = this.getTileUrlFunction();
return fn.call(this, lonlat, this.getProjection());
*/
// Use getfeature info instead
var url = this.getGetFeatureInfoUrl(lonlat, resolution, this.getProjection() || 'EPSG:3857', {});
url = url.replace(/getfeatureinfo/i,"GetMap");
return url;
};
/**
* Return a preview for the layer.
* @param {ol.Coordinate|undefined} lonlat The center of the preview.
* @param {number} resolution of the preview.
* @return {Array<String>} list of preview url
* @api
*/
ol.layer.Layer.prototype.getPreview = function(lonlat, resolution)
{ if (this.get("preview")) return [ this.get("preview") ];
if (!resolution) resolution = 150;
// Get middle resolution
if (resolution < this.getMinResolution() || resolution > this.getMaxResolution())
{ var rmin = this.getMinResolution(),
rmax = this.getMaxResolution();
if (rmax>100000) rmax = 156543; // min zoom : world
if (rmin<0.15) rmin = 0.15; // max zoom
resolution = rmax;
while (rmax>rmin)
{ rmin *= 2;
rmax /= 2;
resolution = rmin;
}
}
var e = this.getExtent();
if (!lonlat) lonlat = [21020, 6355964]; // Default lonlat
if (e && !ol.extent.containsCoordinate(e,lonlat)) lonlat = [ (e[0]+e[2])/2, (e[1]+e[3])/2 ];
if (this.getSource) return [ this.getSource().getPreview(lonlat, resolution) ];
return [];
};
/**
* Return a preview for the layer.
* @param {_ol_coordinate_|undefined} lonlat The center of the preview.
* @param {number} resolution of the preview.
* @return {Array<String>} list of preview url
* @api
*/
ol.layer.Group.prototype.getPreview = function(lonlat, resolution)
{ if (this.get("preview")) return [ this.get("preview") ];
var t = [];
if (this.getLayers)
{ var l = this.getLayers().getArray();
for (var i=0; i<l.length; i++)
{ t = t.concat(l[i].getPreview(lonlat, resolution));
}
}
return t;
};
//NB: (Not confirmed)To use this module, you just have to :
// import('ol-ext/layer/getpreview')
/** ol.layer.Vector.prototype.setRender3D
* @extends {ol.layer.Vector}
* @param {ol.render3D}
*/
ol.layer.Vector.prototype.setRender3D = function (r)
{ r.setLayer(this);
}
/**
* @classdesc
* ol.render3D 3D vector layer rendering
* @constructor
* @param {olx.render3DOption}
* - masResolution {number} max resolution to render 3D
* - defaultHeight {number} default height if none is return by a propertie
* - height {function|string|Number} a height function (return height giving a feature) or a popertie name for the height or a fixed value
*/
ol.render3D = function (options)
{ var options = options || {};
this.maxResolution_ = options.maxResolution || 100
this.defaultHeight_ = options.defaultHeight || 0;
this.height_ = this.getHfn (options.height);
}
/** Calculate 3D at potcompose
*/
ol.render3D.prototype.onPostcompose_ = function(e)
{ var res = e.frameState.viewState.resolution;
if (res > this.maxResolution_) return;
this.res_ = res*400;
if (this.animate_)
{ var elapsed = e.frameState.time - this.animate_;
if (elapsed < this.animateDuration_)
{ this.elapsedRatio_ = this.easing_(elapsed / this.animateDuration_);
// tell OL3 to continue postcompose animation
e.frameState.animate = true;
}
else
{ this.animate_ = false;
this.height_ = this.toHeight_
}
}
var ratio = e.frameState.pixelRatio;
var ctx = e.context;
var m = this.matrix_ = e.frameState.coordinateToPixelTransform;
// Old version (matrix)
if (!m)
{ m = e.frameState.coordinateToPixelMatrix,
m[2] = m[4];
m[3] = m[5];
m[4] = m[12];
m[5] = m[13];
}
this.center_ = [ctx.canvas.width/2/ratio, ctx.canvas.height/ratio];
var f = this.layer_.getSource().getFeaturesInExtent(e.frameState.extent);
ctx.save();
ctx.scale(ratio,ratio);
ctx.lineWidth = 1;
ctx.strokeStyle = "red";
ctx.fillStyle = "rgba(0,0,255,0.5)";
var builds = [];
for (var i=0; i<f.length; i++)
{ builds.push (this.getFeature3D_ (f[i], this.getFeatureHeight(f[i])));
}
this.drawFeature3D_ (ctx, builds);
ctx.restore();
}
/** Set layer to render 3D
*/
ol.render3D.prototype.setLayer = function(l)
{ if (this.layer_) this.layer_.un ('postcompose', this.onPostcompose_, this);
this.layer_ = l;
l.on ('postcompose', this.onPostcompose_, this);
}
/** Create a function that return height of a feature
* @param {function|string|number} a height function or a popertie name or a fixed value
* @return {function} function(f) return height of the feature f
*/
ol.render3D.prototype.getHfn= function(h)
{ switch (typeof(h))
{ case 'function': return h;
case 'string':
{ var dh = this.defaultHeight_;
return (function(f)
{ return (Number(f.get(h)) || dh);
});
}
case 'number': return (function(f) { return h; });
default: return (function(f) { return 10; });
}
}
/** Animate rendering
* @param {olx.render3D.animateOptions}
* - height {string|function|number} an attribute name or a function returning height of a feature or a fixed value
* - durtion {number} the duration of the animatioin ms, default 1000
* - easing {ol.easing} an ol easing function
* @api
*/
ol.render3D.prototype.animate = function(options)
{ options = options || {};
this.toHeight_ = this.getHfn(options.height);
this.animate_ = new Date().getTime();
this.animateDuration_ = options.duration ||1000;
this.easing_ = options.easing || ol.easing.easeOut;
// Force redraw
this.layer_.changed();
}
/** Check if animation is on
* @return {bool}
*/
ol.render3D.prototype.animating = function()
{ if (this.animate_ && new Date().getTime() - this.animate_ > this.animateDuration_)
{ this.animate_ = false;
}
return !!this.animate_;
}
/**
*/
ol.render3D.prototype.getFeatureHeight = function (f)
{ if (this.animate_)
{ var h1 = this.height_(f);
var h2 = this.toHeight_(f);
return (h1*(1-this.elapsedRatio_)+this.elapsedRatio_*h2);
}
else return this.height_(f);
}
/**
*/
ol.render3D.prototype.hvector_ = function (pt, h)
{ p0 = [ pt[0]*this.matrix_[0] + pt[1]*this.matrix_[1] + this.matrix_[4],
pt[0]*this.matrix_[2] + pt[1]*this.matrix_[3] + this.matrix_[5]
];
p1 = [ p0[0] + h/this.res_*(p0[0]-this.center_[0]),
p0[1] + h/this.res_*(p0[1]-this.center_[1])
];
return {p0:p0, p1:p1};
}
/**
*/
ol.render3D.prototype.getFeature3D_ = function (f, h)
{ var c = f.getGeometry().getCoordinates();
switch (f.getGeometry().getType())
{ case "Polygon":
c = [c];
case "MultiPolygon":
var build = [];
for (var i=0; i<c.length; i++)
{ var p0, p1;
for (var j=0; j<c[i].length; j++)
{ var b = [];
for (var k=0; k<c[i][j].length; k++)
{ b.push( this.hvector_(c[i][j][k], h) );
}
build.push(b);
}
}
return { type:"MultiPolygon", feature:f, geom:build };
case "Point":
return { type:"Point", feature:f, geom:this.hvector_(c,h) };
default: return {};
}
}
/**
*/
ol.render3D.prototype.drawFeature3D_ = function(ctx, build)
{ // Construct
for (var i=0; i<build.length; i++)
{
switch (build[i].type)
{ case "MultiPolygon":
for (var j=0; j<build[i].geom.length; j++)
{ var b = build[i].geom[j];
for (var k=0; k < b.length; k++)
{ ctx.beginPath();
ctx.moveTo(b[k].p0[0], b[k].p0[1]);
ctx.lineTo(b[k].p1[0], b[k].p1[1]);
ctx.stroke();
}
}
break;
case "Point":
{ var g = build[i].geom;
ctx.beginPath();
ctx.moveTo(g.p0[0], g.p0[1]);
ctx.lineTo(g.p1[0], g.p1[1]);
ctx.stroke();
break;
}
default: break;
}
}
// Roof
for (var i=0; i<build.length; i++)
{ switch (build[i].type)
{ case "MultiPolygon":
{ ctx.beginPath();
for (var j=0; j<build[i].geom.length; j++)
{ var b = build[i].geom[j];
if (j==0)
{ ctx.moveTo(b[0].p1[0], b[0].p1[1]);
for (var k=1; k < b.length; k++)
{ ctx.lineTo(b[k].p1[0], b[k].p1[1]);
}
}
else
{ ctx.moveTo(b[0].p1[0], b[0].p1[1]);
for (var k=b.length-2; k>=0; k--)
{ ctx.lineTo(b[k].p1[0], b[k].p1[1]);
}
}
ctx.closePath();
}
ctx.fill("evenodd");
ctx.stroke();
break;
}
case "Point":
{ var b = build[i];
var t = b.feature.get('label');
var p = b.geom.p1;
var f = ctx.fillStyle;
ctx.fillStyle = ctx.strokeStyle;
ctx.textAlign = 'center';
ctx.textBaseline = 'bottom';
ctx.fillText ( t, p[0], p[1] );
var m = ctx.measureText(t);
var h = Number (ctx.font.match(/\d+(\.\d+)?/g).join([]));
ctx.fillStyle = "rgba(255,255,255,0.5)";
ctx.fillRect (p[0]-m.width/2 -5, p[1]-h -5, m.width +10, h +10)
ctx.strokeRect (p[0]-m.width/2 -5, p[1]-h -5, m.width +10, h +10)
ctx.fillStyle = f;
//console.log(build[i].feature.getProperties())
}
default: break;
}
}
}
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc
* The Magnify overlay add a "magnifying glass" effect to an OL3 map that displays
* a portion of the map in a different zoom (and actually display different content).
*
* @constructor
* @extends {ol.Overlay}
* @param {olx.OverlayOptions} options Overlay options
* @api stable
*/
ol.Overlay.Magnify = function (options)
{ var self = this;
var elt = $("<div>").addClass("ol-magnify");
this._elt = elt.get(0);
ol.Overlay.call(this,
{ positioning: options.positioning || "center-center",
element: this._elt,
stopEvent: false
});
// Create magnify map
this.mgmap_ = new ol.Map(
{ controls: new ol.Collection(),
interactions: new ol.Collection(),
target: options.target || this._elt,
view: new ol.View({ projection: options.projection }),
layers: options.layers
});
this.mgview_ = this.mgmap_.getView();
this.external_ = options.target?true:false;
this.set("zoomOffset", options.zoomOffset||1);
this.set("active", true);
this.on("propertychange", this.setView_.bind(this));
};
ol.inherits(ol.Overlay.Magnify, ol.Overlay);
/**
* Set the map instance the overlay is associated with.
* @param {ol.Map} map The map instance.
*/
ol.Overlay.Magnify.prototype.setMap = function(map) {
if (this.getMap()) {
$(this.getMap().getViewport()).off("mousemove", this.onMouseMove_);
}
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
ol.Overlay.prototype.setMap.call(this, map);
$(map.getViewport()).on("mousemove", {self:this}, this.onMouseMove_);
this._listener = map.getView().on('propertychange', this.setView_.bind(this));
this.setView_();
};
/** Get the magnifier map
* @return {_ol_Map_}
*/
ol.Overlay.Magnify.prototype.getMagMap = function()
{ return this.mgmap_;
};
/** Magnify is active
* @return {boolean}
*/
ol.Overlay.Magnify.prototype.getActive = function()
{ return this.get("active");
};
/** Activate or deactivate
* @param {boolean} active
*/
ol.Overlay.Magnify.prototype.setActive = function(active)
{ return this.set("active", active);
};
/** Mouse move
* @private
*/
ol.Overlay.Magnify.prototype.onMouseMove_ = function(e)
{ var self = e.data.self;
if (!self.get("active"))
{ self.setPosition();
}
else
{ var px = self.getMap().getEventCoordinate(e);
if (!self.external_) self.setPosition(px);
self.mgview_.setCenter(px);
if ($("canvas", self._elt).css("display")=="none") self.mgmap_.updateSize();
}
};
/** View has changed
* @private
*/
ol.Overlay.Magnify.prototype.setView_ = function(e)
{ if (!this.get("active"))
{ this.setPosition();
return;
}
if (!e)
{ // refresh all
this.setView_({key:'rotation'});
this.setView_({key:'resolution'});
return;
}
// Set the view params
switch (e.key)
{ case 'rotation':
this.mgview_.setRotation(this.getMap().getView().getRotation());
break;
case 'zoomOffset':
case 'resolution':
{ var z = Math.max(0,this.getMap().getView().getZoom()+Number(this.get("zoomOffset")));
this.mgview_.setZoom(z);
break;
}
default: break;
}
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc
* A popup element to be displayed over the map and attached to a single map
* location. The popup are customized using CSS.
*
* @example
var popup = new ol.Overlay.Popup();
map.addOverlay(popup);
popup.show(coordinate, "Hello!");
popup.hide();
*
* @constructor
* @extends {ol.Overlay}
* @param {} options Extend Overlay options
* @param {String} options.popupClass the a class of the overlay to style the popup.
* @param {bool} options.closeBox popup has a close box, default false.
* @param {function|undefined} options.onclose: callback function when popup is closed
* @param {function|undefined} options.onshow callback function when popup is shown
* @param {Number|Array<number>} options.offsetBox an offset box
* @param {ol.OverlayPositioning | string | undefined} options.positionning
* the 'auto' positioning var the popup choose its positioning to stay on the map.
* @api stable
*/
ol.Overlay.Popup = function (options)
{ var self = this;
if (typeof(options.offsetBox)==='number') this.offsetBox = [options.offsetBox,options.offsetBox,options.offsetBox,options.offsetBox];
else this.offsetBox = options.offsetBox;
// Popup div
var d = $("<div>").addClass('ol-overlaycontainer-stopevent');
options.element = d.get(0);
// Anchor div
$("<div>").addClass("anchor").appendTo(d);
// Content
this.content = $("<div>").addClass("content").appendTo(d).get(0);
// Closebox
this.closeBox = options.closeBox;
this.onclose = options.onclose;
this.onshow = options.onshow;
$("<button>").addClass("closeBox").addClass(options.closeBox?"hasclosebox":"")
.attr('type', 'button')
.prependTo(d)
.click(function()
{ self.hide();
});
// Stop event
options.stopEvent = true;
d.on("mousedown touchstart", function(e){ e.stopPropagation(); })
ol.Overlay.call(this, options);
this._elt = $(this.element);
// call setPositioning first in constructor so getClassPositioning is called only once
this.setPositioning(options.positioning);
this.setPopupClass(options.popupClass);
};
ol.inherits(ol.Overlay.Popup, ol.Overlay);
/**
* Get CSS class of the popup according to its positioning.
* @private
*/
ol.Overlay.Popup.prototype.getClassPositioning = function ()
{ var c = "";
var pos = this.getPositioning();
if (/bottom/.test(pos)) c += "ol-popup-bottom ";
if (/top/.test(pos)) c += "ol-popup-top ";
if (/left/.test(pos)) c += "ol-popup-left ";
if (/right/.test(pos)) c += "ol-popup-right ";
if (/^center/.test(pos)) c += "ol-popup-middle ";
if (/center$/.test(pos)) c += "ol-popup-center ";
return c;
};
/**
* Set a close box to the popup.
* @param {bool} b
* @api stable
*/
ol.Overlay.Popup.prototype.setClosebox = function (b)
{ this.closeBox = b;
if (b) this._elt.addClass("hasclosebox");
else this._elt.removeClass("hasclosebox");
};
/**
* Set the CSS class of the popup.
* @param {string} c class name.
* @api stable
*/
ol.Overlay.Popup.prototype.setPopupClass = function (c)
{ this._elt.removeClass()
.addClass("ol-popup "+(c||"default")+" "+this.getClassPositioning()+(this.closeBox?" hasclosebox":""));
};
/**
* Add a CSS class to the popup.
* @param {string} c class name.
* @api stable
*/
ol.Overlay.Popup.prototype.addPopupClass = function (c)
{ this._elt.addClass(c);
};
/**
* Remove a CSS class to the popup.
* @param {string} c class name.
* @api stable
*/
ol.Overlay.Popup.prototype.removePopupClass = function (c)
{ this._elt.removeClass(c);
};
/**
* Set positionning of the popup
* @param {ol.OverlayPositioning | string | undefined} pos an ol.OverlayPositioning
* or 'auto' to var the popup choose the best position
* @api stable
*/
ol.Overlay.Popup.prototype.setPositioning = function (pos)
{ if (pos === undefined)
return;
if (/auto/.test(pos))
{ this.autoPositioning = pos.split('-');
if (this.autoPositioning.length==1) this.autoPositioning[1]="auto";
}
else this.autoPositioning = false;
pos = pos.replace(/auto/g,"center");
if (pos=="center") pos = "bottom-center";
this.setPositioning_(pos);
};
/** @private
* @param {ol.OverlayPositioning | string | undefined} pos
*/
ol.Overlay.Popup.prototype.setPositioning_ = function (pos) {
if (this._elt) {
ol.Overlay.prototype.setPositioning.call(this, pos);
this._elt.removeClass("ol-popup-top ol-popup-bottom ol-popup-left ol-popup-right ol-popup-center ol-popup-middle");
this._elt.addClass(this.getClassPositioning());
}
};
/** Check if popup is visible
* @return {boolean}
*/
ol.Overlay.Popup.prototype.getVisible = function ()
{ return this._elt.hasClass("visible");
};
/**
* Set the position and the content of the popup.
* @param {ol.Coordinate|string} coordinate the coordinate of the popup or the HTML content.
* @param {string|undefined} html the HTML content (undefined = previous content).
* @example
var popup = new ol.Overlay.Popup();
// Show popup
popup.show([166000, 5992000], "Hello world!");
// Move popup at coord with the same info
popup.show([167000, 5990000]);
// set new info
popup.show("New informations");
* @api stable
*/
ol.Overlay.Popup.prototype.show = function (coordinate, html)
{ if (!html && typeof(coordinate)=='string')
{ html = coordinate;
coordinate = null;
}
var self = this;
var map = this.getMap();
if (!map) return;
if (html && html !== this.prevHTML)
{ // Prevent flickering effect
this.prevHTML = html;
$(this.content).html("").append(html);
// Refresh when loaded (img)
$("*", this.content).on('load',function()
{ map.renderSync();
})
}
if (coordinate)
{ // Auto positionning
if (this.autoPositioning)
{ var p = map.getPixelFromCoordinate(coordinate);
var s = map.getSize();
var pos=[];
if (this.autoPositioning[0]=='auto')
{ pos[0] = (p[1]<s[1]/3) ? "top" : "bottom";
}
else pos[0] = this.autoPositioning[0];
pos[1] = (p[0]<2*s[0]/3) ? "left" : "right";
this.setPositioning_(pos[0]+"-"+pos[1]);
if (this.offsetBox)
{ this.setOffset([this.offsetBox[pos[1]=="left"?2:0], this.offsetBox[pos[0]=="top"?3:1] ]);
}
} else {
if (this.offsetBox){
this.setOffset(this.offsetBox);
}
}
// Show
this.setPosition(coordinate);
// Set visible class (wait to compute the size/position first)
this._elt.parent().show();
if (typeof (this.onshow) == 'function') this.onshow();
this._tout = setTimeout (function()
{ self._elt.addClass("visible");
}, 0);
}
};
/**
* Hide the popup
* @api stable
*/
ol.Overlay.Popup.prototype.hide = function ()
{ if (this.getPosition() == undefined) return;
if (typeof (this.onclose) == 'function') this.onclose();
this.setPosition(undefined);
if (this._tout) clearTimeout(this._tout);
this._elt.removeClass("visible");
};
/*
Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (http://www.cecill.info/).
ol.coordinate.convexHull compute a convex hull using Andrew's Monotone Chain Algorithm.
@see https://en.wikipedia.org/wiki/Convex_hull_algorithms
*/
/** Tests if a point is left or right of line (a,b).
* @param {ol.coordinate} a point on the line
* @param {ol.coordinate} b point on the line
* @param {ol.coordinate} 0
* @return {bool} true if (a,b,o) turns clockwise
*/
ol.coordinate.clockwise = function (a, b, o) {
return ((a[0] - o[0]) * (b[1] - o[1]) - (a[1] - o[1]) * (b[0] - o[0]) <= 0);
};
/** Compute a convex hull using Andrew's Monotone Chain Algorithm
* @param {Array<ol.geom.Point>} points an array of 2D points
* @return {Array<ol.geom.Point>} the convex hull vertices
*/
ol.coordinate.convexHull = function (points) { // Sort by increasing x and then y coordinate
points.sort(function(a, b) {
return a[0] == b[0] ? a[1] - b[1] : a[0] - b[0];
});
// Compute the lower hull
var lower = [];
for (var i = 0; i < points.length; i++) {
while (lower.length >= 2 && ol.coordinate.clockwise(lower[lower.length - 2], lower[lower.length - 1], points[i])) {
lower.pop();
}
lower.push(points[i]);
}
// Compute the upper hull
var upper = [];
for (var i = points.length - 1; i >= 0; i--) {
while (upper.length >= 2 && ol.coordinate.clockwise(upper[upper.length - 2], upper[upper.length - 1], points[i])) {
upper.pop();
}
upper.push(points[i]);
}
upper.pop();
lower.pop();
return lower.concat(upper);
};
/* Get coordinates of a geometry */
var getCoordinates = function (geom) {
var h = [];
switch (geom.getType()) {
case "Point":h.push(geom.getCoordinates());
break;
case "LineString":
case "LinearRing":
case "MultiPoint":h = geom.getCoordinates();
break;
case "MultiLineString":
var p = geom.getLineStrings();
for (var i = 0; i < p.length; i++) h.concat(getCoordinates(p[i]));
break;
case "Polygon":
h = getCoordinates(geom.getLinearRing(0));
break;
case "MultiPolygon":
var p = geom.getPolygons();
for (var i = 0; i < p.length; i++) h.concat(getCoordinates(p[i]));
break;
case "GeometryCollection":
var p = geom.getGeometries();
for (var i = 0; i < p.length; i++) h.concat(getCoordinates(p[i]));
break;
default:break;
}
return h;
};
/** Compute a convex hull on a geometry using Andrew's Monotone Chain Algorithm
* @return {Array<ol.geom.Point>} the convex hull vertices
*/
ol.geom.Geometry.prototype.convexHull = function() {
return ol.coordinate.convexHull(getCoordinates(this));
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
Usefull function to handle geometric operations
*/
/** Distance beetween 2 points
* Usefull geometric functions
* @param {ol.coordinate} p1 first point
* @param {ol.coordinate} p2 second point
* @return {number} distance
*/
ol.coordinate.dist2d = function(p1, p2)
{ var dx = p1[0]-p2[0];
var dy = p1[1]-p2[1];
return Math.sqrt(dx*dx+dy*dy);
}
/** 2 points are equal
* Usefull geometric functions
* @param {ol.coordinate} p1 first point
* @param {ol.coordinate} p2 second point
* @return {boolean}
*/
ol.coordinate.equal = function(p1, p2)
{ return (p1[0]==p2[0] && p1[1]==p2[1]);
}
/** Get center coordinate of a feature
* @param {ol.Feature} f
* @return {ol.coordinate} the center
*/
ol.coordinate.getFeatureCenter = function(f)
{ return ol.coordinate.getGeomCenter (f.getGeometry());
};
/** Get center coordinate of a geometry
* @param {ol.Feature} geom
* @return {ol.coordinate} the center
*/
ol.coordinate.getGeomCenter = function(geom)
{ switch (geom.getType())
{ case 'Point':
return geom.getCoordinates();
case "MultiPolygon":
geom = geom.getPolygon(0);
case "Polygon":
return geom.getInteriorPoint().getCoordinates();
default:
return geom.getClosestPoint(ol.extent.getCenter(geom.getExtent()));
};
};
/** Split a lineString by a point or a list of points
* NB: points must be on the line, use getClosestPoint() to get one
* @param {ol.Coordinate | Array<ol.Coordinate>} pt points to split the line
* @param {Number} tol distance tolerance for 2 points to be equal
*/
ol.geom.LineString.prototype.splitAt = function(pt, tol)
{ if (!pt) return [this];
if (!tol) tol = 1e-10;
// Test if list of points
if (pt.length && pt[0].length)
{ var result = [this];
for (var i=0; i<pt.length; i++)
{ var r = [];
for (var k=0; k<result.length; k++)
{ var ri = result[k].splitAt(pt[i], tol);
r = r.concat(ri);
}
result = r;
}
return result;
}
// Nothing to do
if (ol.coordinate.equal(pt,this.getFirstCoordinate())
|| ol.coordinate.equal(pt,this.getLastCoordinate()))
{ return [this];
}
// Get
var c0 = this.getCoordinates();
var ci=[c0[0]], p0, p1;
var c = [];
for (var i=0; i<c0.length-1; i++)
{ // Filter equal points
if (ol.coordinate.equal(c0[i],c0[i+1])) continue;
// Extremity found
if (ol.coordinate.equal(pt,c0[i+1]))
{ ci.push(c0[i+1]);
c.push(new ol.geom.LineString(ci));
ci = [];
}
// Test alignement
else if (!ol.coordinate.equal(pt,c0[i]))
{ var d1, d2;
if (c0[i][0] == c0[i+1][0])
{ d1 = d2 = (c0[i][1]-pt[1]) / (c0[i][1]-c0[i+1][1]);
}
else if (c0[i][1] == c0[i+1][1])
{ d1 = d2 = (c0[i][0]-pt[0]) / (c0[i][0]-c0[i+1][0]);
}
else
{ d1 = (c0[i][0]-pt[0]) / (c0[i][0]-c0[i+1][0]);
d2 = (c0[i][1]-pt[1]) / (c0[i][1]-c0[i+1][1]);
}
if (Math.abs(d1-d2)<tol && 0<=d1 && d1<=1)
{ ci.push(pt);
c.push (new ol.geom.LineString(ci));
ci = [pt];
}
}
ci.push(c0[i+1]);
}
if (ci.length>1) c.push (new ol.geom.LineString(ci));
if (c.length) return c;
else return [this];
}
/** Offset a polyline
* @param {Array<ol.coordinate>} coords
* @param {Number} offset
* @return {Array<ol.coordinates>} resulting coord
* @see http://stackoverflow.com/a/11970006/796832
* @see https://drive.google.com/viewerng/viewer?a=v&pid=sites&srcid=ZGVmYXVsdGRvbWFpbnxqa2dhZGdldHN0b3JlfGd4OjQ4MzI5M2Y0MjNmNzI2MjY
*/
ol.coordinate.offsetCoords = function (coords, offset) {
var path = [];
var N = coords.length-1;
var max = N;
var mi, mi1, li, li1, ri, ri1, si, si1, Xi1, Yi1;
var p0, p1, p2;
var isClosed = ol.coordinate.equal(coords[0],coords[N]);
if (!isClosed) {
p0 = coords[0];
p1 = coords[1];
p2 = [
p0[0] + (p1[1] - p0[1]) / ol.coordinate.dist2d(p0,p1) *offset,
p0[1] - (p1[0] - p0[0]) / ol.coordinate.dist2d(p0,p1) *offset
];
path.push(p2);
coords.push(coords[N])
N++;
max--;
}
for (var i = 0; i < max; i++) {
p0 = coords[i];
p1 = coords[(i+1) % N];
p2 = coords[(i+2) % N];
mi = (p1[1] - p0[1])/(p1[0] - p0[0]);
mi1 = (p2[1] - p1[1])/(p2[0] - p1[0]);
// Prevent alignements
if (Math.abs(mi-mi1) > 1e-10) {
li = Math.sqrt((p1[0] - p0[0])*(p1[0] - p0[0])+(p1[1] - p0[1])*(p1[1] - p0[1]));
li1 = Math.sqrt((p2[0] - p1[0])*(p2[0] - p1[0])+(p2[1] - p1[1])*(p2[1] - p1[1]));
ri = p0[0] + offset*(p1[1] - p0[1])/li;
ri1 = p1[0] + offset*(p2[1] - p1[1])/li1;
si = p0[1] - offset*(p1[0] - p0[0])/li;
si1 = p1[1] - offset*(p2[0] - p1[0])/li1;
Xi1 = (mi1*ri1-mi*ri+si-si1) / (mi1-mi);
Yi1 = (mi*mi1*(ri1-ri)+mi1*si-mi*si1) / (mi1-mi);
// Correction for vertical lines
if(p1[0] - p0[0] == 0) {
Xi1 = p1[0] + offset*(p1[1] - p0[1])/Math.abs(p1[1] - p0[1]);
Yi1 = mi1*Xi1 - mi1*ri1 + si1;
}
if (p2[0] - p1[0] == 0 ) {
Xi1 = p2[0] + offset*(p2[1] - p1[1])/Math.abs(p2[1] - p1[1]);
Yi1 = mi*Xi1 - mi*ri + si;
}
path.push([Xi1, Yi1]);
}
}
if (isClosed) {
path.push(path[0]);
} else {
coords.pop();
p0 = coords[coords.length-1];
p1 = coords[coords.length-2];
p2 = [
p0[0] - (p1[1] - p0[1]) / ol.coordinate.dist2d(p0,p1) *offset,
p0[1] + (p1[0] - p0[0]) / ol.coordinate.dist2d(p0,p1) *offset
];
path.push(p2);
}
return path;
}
/** Find the segment a point belongs to
* @param {ol.coordinate} pt
* @param {Array<ol.coordinate>} coords
* @return {} the index (-1 if not found) and the segment
*/
ol.coordinate.findSegment = function (pt, coords) {
for (var i=0; i<coords.length-1; i++) {
var p0 = coords[i];
var p1 = coords[i+1];
if (ol.coordinate.equal(pt, p0) || ol.coordinate.equal(pt, p1)) {
return { index:1, segment: [p0,p1] };
} else {
var d0 = ol.coordinate.dist2d(p0,p1);
var v0 = [ (p1[0] - p0[0]) / d0, (p1[1] - p0[1]) / d0 ];
var d1 = ol.coordinate.dist2d(p0,pt);
var v1 = [ (pt[0] - p0[0]) / d1, (pt[1] - p0[1]) / d1 ];
if (Math.abs(v0[0]*v1[1] - v0[1]*v1[0]) < 1e-10) {
return { index:1, segment: [p0,p1] };
}
}
}
return { index: -1 };
};
/* Use
*/
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
Usefull function to handle geometric operations
*/
/**
* Split a Polygon geom with horizontal lines
* @param {Array<ol.coordinate>} geom
* @param {Number} y the y to split
* @param {Number} n contour index
* @return {Array<Array<ol.coordinate>>}
*/
ol.coordinate.splitH = function (geom, y, n) {
var x, abs;
var list = [];
for (var i=0; i<geom.length-1; i++) {
// Hole separator?
if (!geom[i].length || !geom[i+1].length) continue;
// Intersect
if (geom[i][1]<=y && geom[i+1][1]>y || geom[i][1]>=y && geom[i+1][1]<y) {
abs = (y-geom[i][1]) / (geom[i+1][1]-geom[i][1]);
x = abs * (geom[i+1][0]-geom[i][0]) + geom[i][0];
list.push ({ contour: n, index: i, pt: [x,y], abs: abs });
}
}
// Sort x
list.sort(function(a,b) { return a.pt[0] - b.pt[0] });
// Horizontal segement
var result = [];
for (var j=0; j<list.length-1; j += 2) {
result.push([list[j], list[j+1]])
}
return result;
};
/**
* Calculate a MultiPolyline to fill a Polygon with a scribble effect that appears hand-made
* @param {} options
* @param {Number} options.interval interval beetween lines
* @param {Number} options.angle hatch angle in radian, default PI/2
* @return {ol.geom.MultiLineString|null} the resulting MultiLineString geometry or null if none
*/
ol.geom.MultiPolygon.prototype.scribbleFill = function (options) {
var scribbles = [];
var poly = this.getPolygons();
for (var i=0, p; p=poly[i]; i++) {
var mls = p.scribbleFill(options);
if (mls) scribbles.push(mls);
}
if (!scribbles.length) return null;
// Merge scribbles
var scribble = scribbles[0];
for (var i=0, s; s=scribbles[i]; i++) {
ls = s.getLineStrings();
for (k=0; k<ls.length; k++) {
scribble.appendLineString(ls[k]);
}
}
return scribble;
};
/**
* Calculate a MultiPolyline to fill a Polygon with a scribble effect that appears hand-made
* @param {} options
* @param {Number} options.interval interval beetween lines
* @param {Number} options.angle hatch angle in radian, default PI/2
* @return {ol.geom.MultiLineString|null} the resulting MultiLineString geometry or null if none
*/
ol.geom.Polygon.prototype.scribbleFill = function (options) {
var step = options.interval;
var angle = options.angle || Math.PI/2;
// Geometry + rotate
var geom = this.clone();
geom.rotate(angle, [0,0]);
var coords = geom.getCoordinates();
// Merge holes
var coord = coords[0];
for (var i=1; i<coords.length; i++) {
// Add a separator
coord.push([]);
// Add the hole
coord = coord.concat(coords[i]);
}
// Extent
var ext = geom.getExtent();
// Split polygon with horizontal lines
var lines = [];
for (var y = (Math.floor(ext[1]/step)+1)*step; y<ext[3]; y += step) {
var l = ol.coordinate.splitH(coord, y, i);
lines = lines.concat(l);
}
if (!lines.length) return null;
// Order lines on segment index
var mod = coord.length-1;
var first = lines[0][0].index;
for (var k=0, l; l=lines[k]; k++) {
lines[k][0].index = (lines[k][0].index-first+mod) % mod;
lines[k][1].index = (lines[k][1].index-first+mod) % mod;
}
var scribble = [];
while(true) {
for (var k=0, l; l=lines[k]; k++) {
if (!l[0].done) break;
}
if (!l) break;
var scrib = [];
while (l) {
l[0].done = true;
scrib.push(l[0].pt);
scrib.push(l[1].pt);
var nexty = l[0].pt[1] + step;
var d0 = Infinity;
var l2 = null;
while (lines[k]) {
if (lines[k][0].pt[1] > nexty) break;
if (lines[k][0].pt[1] === nexty) {
var d = Math.min(
(lines[k][0].index - l[0].index + mod) % mod,
(l[0].index - lines[k][0].index + mod) % mod
);
var d2 = Math.min(
(l[1].index - l[0].index + mod) % mod,
(l[0].index - l[1].index + mod) % mod
);
if (d<d0 && d<d2) {
d0 = d;
if (!lines[k][0].done) l2 = lines[k];
else l2 = null;
}
}
k++;
}
l = l2;
}
if (scrib.length) {
scribble.push(scrib);
}
}
// Return the scribble as MultiLineString
if (!scribble.length) return null;
var mline = new ol.geom.MultiLineString(scribble);
mline.rotate(-angle,[0,0]);
return mline.cspline({ pointsPerSeg:8, tension:.9 });
};
// import('ol-ext/geom/Scribble')
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Pulse an extent on postcompose
* @param {ol.coordinates} point to pulse
* @param {ol.pulse.options} options pulse options param
* @param {ol.projectionLike|undefined} options.projection projection of coords, default no transform
* @param {Number} options.duration animation duration in ms, default 2000
* @param {ol.easing} options.easing easing function, default ol.easing.upAndDown
* @param {ol.style.Stroke} options.style stroke style, default 2px red
*/
ol.Map.prototype.animExtent = function(extent, options)
{ var listenerKey;
options = options || {};
// Change to map's projection
if (options.projection)
{ extent = ol.proj.transformExtent (extent, options.projection, this.getView().getProjection());
}
// options
var start = new Date().getTime();
var duration = options.duration || 1000;
var easing = options.easing || ol.easing.upAndDown;
var width = options.style ? options.style.getWidth() || 2 : 2;
var color = options.style ? options.style.getColr() || 'red' : 'red';
// Animate function
function animate(event)
{ var frameState = event.frameState;
var ratio = frameState.pixelRatio;
var elapsed = frameState.time - start;
if (elapsed > duration) ol.Observable.unByKey(listenerKey);
else
{ var elapsedRatio = elapsed / duration;
var p0 = this.getPixelFromCoordinate([extent[0],extent[1]]);
var p1 = this.getPixelFromCoordinate([extent[2],extent[3]]);
var context = event.context;
context.save();
context.scale(ratio,ratio);
context.beginPath();
var e = easing(elapsedRatio)
context.globalAlpha = easing(1 - elapsedRatio);
context.lineWidth = width;
context.strokeStyle = color;
context.rect(p0[0], p0[1], p1[0]-p0[0], p1[1]-p0[1]);
context.stroke();
context.restore();
// tell OL3 to continue postcompose animation
frameState.animate = true;
}
}
// Launch animation
listenerKey = this.on('postcompose', animate.bind(this));
this.renderSync();
}
/** Create a cardinal spline version of this geometry.
* Original https://github.com/epistemex/cardinal-spline-js
* @see https://en.wikipedia.org/wiki/Cubic_Hermite_spline#Cardinal_spline
*
* @param {} options
* @param {Number} options.tension a [0,1] number / can be interpreted as the "length" of the tangent, default 0.5
* @param {Number} options.resolution size of segment to split
* @param {Interger} options.pointsPerSeg number of points per segment to add if no resolution is provided, default add 10 points per segment
*/
/** Cache cspline calculation
*/
ol.geom.Geometry.prototype.cspline = function(options)
{ // Calculate cspline
if (this.calcCSpline_)
{ if (this.csplineGeometryRevision != this.getRevision()
|| this.csplineOption != JSON.stringify(options))
{ this.csplineGeometry_ = this.calcCSpline_(options)
this.csplineGeometryRevision = this.getRevision();
this.csplineOption = JSON.stringify(options);
}
return this.csplineGeometry_;
}
// Default do nothing
else
{ return this;
}
}
ol.geom.GeometryCollection.prototype.calcCSpline_ = function(options)
{ var g=[], g0=this.getGeometries();
for (var i=0; i<g0.length; i++)
{ g.push(g0[i].cspline(options));
}
return new ol.geom.GeometryCollection(g);
}
ol.geom.MultiLineString.prototype.calcCSpline_ = function(options)
{ var g=[], lines = this.getLineStrings();
for (var i=0; i<lines.length; i++)
{ g.push(lines[i].cspline(options).getCoordinates());
}
return new ol.geom.MultiLineString(g);
}
ol.geom.Polygon.prototype.calcCSpline_ = function(options)
{ var g=[], g0=this.getCoordinates();
for (var i=0; i<g0.length; i++)
{ g.push((new ol.geom.LineString(g0[i])).cspline(options).getCoordinates());
}
return new ol.geom.Polygon(g);
}
ol.geom.MultiPolygon.prototype.calcCSpline_ = function(options)
{ var g=[], g0=this.getPolygons();
for (var i=0; i<g0.length; i++)
{ g.push(g0[i].cspline(options).getCoordinates());
}
return new ol.geom.MultiPolygon(g);
}
/**
*/
ol.geom.LineString.prototype.calcCSpline_ = function(options)
{ if (!options) options={};
var line = this.getCoordinates();
var tension = typeof options.tension === "number" ? options.tension : 0.5;
var resolution = options.resolution || (this.getLength() / line.length / (options.pointsPerSeg || 10));
var pts, res = [], // clone array
x, y, // our x,y coords
t1x, t2x, t1y, t2y, // tension vectors
c1, c2, c3, c4, // cardinal points
st, t, i; // steps based on num. of segments
// clone array so we don't change the original
//
pts = line.slice(0);
// The algorithm require a previous and next point to the actual point array.
// Check if we will draw closed or open curve.
// If closed, copy end points to beginning and first points to end
// If open, duplicate first points to befinning, end points to end
if (line.length>2 && line[0][0]==line[line.length-1][0] && line[0][1]==line[line.length-1][1])
{ pts.unshift(line[line.length-2]);
pts.push(line[1]);
}
else
{ pts.unshift(line[0]);
pts.push(line[line.length-1]);
}
// ok, lets start..
function dist2d(x1, y1, x2, y2)
{ var dx = x2-x1;
var dy = y2-y1;
return Math.sqrt(dx*dx+dy*dy);
}
// 1. loop goes through point array
// 2. loop goes through each segment between the 2 pts + 1e point before and after
for (i=1; i < (pts.length - 2); i++)
{ var d1 = dist2d (pts[i][0], pts[i][1], pts[i+1][0], pts[i+1][1]);
var numOfSegments = Math.round(d1/resolution);
var d=1;
if (options.normalize)
{ var d1 = dist2d (pts[i+1][0], pts[i+1][1], pts[i-1][0], pts[i-1][1]);
var d2 = dist2d (pts[i+2][0], pts[i+2][1], pts[i][0], pts[i][1]);
if (d1<d2) d = d1/d2;
else d = d2/d1;
}
// calc tension vectors
t1x = (pts[i+1][0] - pts[i-1][0]) * tension *d;
t2x = (pts[i+2][0] - pts[i][0]) * tension *d;
t1y = (pts[i+1][1] - pts[i-1][1]) * tension *d;
t2y = (pts[i+2][1] - pts[i][1]) * tension *d;
for (t=0; t <= numOfSegments; t++)
{ // calc step
st = t / numOfSegments;
// calc cardinals
c1 = 2 * Math.pow(st, 3) - 3 * Math.pow(st, 2) + 1;
c2 = -(2 * Math.pow(st, 3)) + 3 * Math.pow(st, 2);
c3 = Math.pow(st, 3) - 2 * Math.pow(st, 2) + st;
c4 = Math.pow(st, 3) - Math.pow(st, 2);
// calc x and y cords with common control vectors
x = c1 * pts[i][0] + c2 * pts[i+1][0] + c3 * t1x + c4 * t2x;
y = c1 * pts[i][1] + c2 * pts[i+1][1] + c3 * t1y + c4 * t2y;
//store points in array
if (x && y) res.push([x,y]);
}
}
return new ol.geom.LineString(res);
}
//NB: (Not confirmed)To use this module, you just have to :
// import('ol-ext/utils/cspline')
/* Copyright (c) 2017 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* Hexagonal grids
* @classdesc ol.HexGrid is a class to compute hexagonal grids
* @see http://www.redblobgames.com/grids/hexagons
*
* @constructor ol.HexGrid
* @extends {ol.Object}
* @param {olx.HexGrid=} options
* @param {Number} options.size size of the exagon in map units, default 80000
* @param {_ol_coordinate_} options.origin orgin of the grid, default [0,0]
* @param {pointy|flat} options.layout grid layout, default pointy
*/
ol.HexGrid = function (options)
{ options = options || {};
ol.Object.call (this, options);
// Options
this.size_ = options.size||80000;
this.origin_ = options.origin || [0,0];
this.layout_ = this.layout[options.layout] || this.layout.pointy;
};
ol.inherits (ol.HexGrid, ol.Object);
/** Layout
*/
ol.HexGrid.prototype.layout =
{ pointy:
[ Math.sqrt(3), Math.sqrt(3)/2, 0, 3/2,
Math.sqrt(3)/3, -1/3, 0, 2/3,
// corners
Math.cos(Math.PI / 180 * (60 * 0 + 30)), Math.sin(Math.PI / 180 * (60 * 0 + 30)),
Math.cos(Math.PI / 180 * (60 * 1 + 30)), Math.sin(Math.PI / 180 * (60 * 1 + 30)),
Math.cos(Math.PI / 180 * (60 * 2 + 30)), Math.sin(Math.PI / 180 * (60 * 2 + 30)),
Math.cos(Math.PI / 180 * (60 * 3 + 30)), Math.sin(Math.PI / 180 * (60 * 3 + 30)),
Math.cos(Math.PI / 180 * (60 * 4 + 30)), Math.sin(Math.PI / 180 * (60 * 4 + 30)),
Math.cos(Math.PI / 180 * (60 * 5 + 30)), Math.sin(Math.PI / 180 * (60 * 5 + 30))
],
flat:
[ 3/2, 0, Math.sqrt(3)/2, Math.sqrt(3), 2/3,
0, -1/3, Math.sqrt(3) / 3,
// corners
Math.cos(Math.PI / 180 * (60 * 0)), Math.sin(Math.PI / 180 * (60 * 0)),
Math.cos(Math.PI / 180 * (60 * 1)), Math.sin(Math.PI / 180 * (60 * 1)),
Math.cos(Math.PI / 180 * (60 * 2)), Math.sin(Math.PI / 180 * (60 * 2)),
Math.cos(Math.PI / 180 * (60 * 3)), Math.sin(Math.PI / 180 * (60 * 3)),
Math.cos(Math.PI / 180 * (60 * 4)), Math.sin(Math.PI / 180 * (60 * 4)),
Math.cos(Math.PI / 180 * (60 * 5)), Math.sin(Math.PI / 180 * (60 * 5))
]
};
/** Set layout
* @param {pointy | flat | undefined} layout name, default pointy
*/
ol.HexGrid.prototype.setLayout = function (layout)
{ this.layout_ = this.layout[layout] || this.layout.pointy;
this.changed();
}
/** Get layout
* @return {pointy | flat} layout name
*/
ol.HexGrid.prototype.getLayout = function ()
{ return (this.layout_[9]!=0 ? 'pointy' : 'flat');
}
/** Set hexagon origin
* @param {ol.coordinate} coord origin
*/
ol.HexGrid.prototype.setOrigin = function (coord)
{ this.origin_ = coord;
this.changed();
}
/** Get hexagon origin
* @return {ol.coordinate} coord origin
*/
ol.HexGrid.prototype.getOrigin = function (coord)
{ return this.origin_;
}
/** Set hexagon size
* @param {Number} hexagon size
*/
ol.HexGrid.prototype.setSize = function (s)
{ this.size_ = s || 80000;
this.changed();
}
/** Get hexagon size
* @return {Number} hexagon size
*/
ol.HexGrid.prototype.getSize = function (s)
{ return this.size_;
}
/** Convert cube to axial coords
* @param {ol.coordinate} c cube coordinate
* @return {ol.coordinate} axial coordinate
*/
ol.HexGrid.prototype.cube2hex = function (c)
{ return [c[0], c[2]];
};
/** Convert axial to cube coords
* @param {ol.coordinate} h axial coordinate
* @return {ol.coordinate} cube coordinate
*/
ol.HexGrid.prototype.hex2cube = function(h)
{ return [h[0], -h[0]-h[1], h[1]];
};
/** Convert offset to axial coords
* @param {ol.coordinate} h axial coordinate
* @return {ol.coordinate} offset coordinate
*/
ol.HexGrid.prototype.hex2offset = function (h)
{ if (this.layout_[9]) return [ h[0] + (h[1] - (h[1]&1)) / 2, h[1] ];
else return [ h[0], h[1] + (h[0] + (h[0]&1)) / 2 ];
}
/** Convert axial to offset coords
* @param {ol.coordinate} o offset coordinate
* @return {ol.coordinate} axial coordinate
*/
ol.HexGrid.prototype.offset2hex = function(o)
{ if (this.layout_[9]) return [ q = o[0] - (o[1] - (o[1]&1)) / 2, r = o[1] ];
else return [ o[0], o[1] - (o[0] + (o[0]&1)) / 2 ];
}
/** Convert offset to cube coords
* @param {ol.coordinate} c cube coordinate
* @return {ol.coordinate} offset coordinate
* /
ol.HexGrid.prototype.cube2offset = function(c)
{ return hex2offset(cube2hex(c));
};
/** Convert cube to offset coords
* @param {ol.coordinate} o offset coordinate
* @return {ol.coordinate} cube coordinate
* /
ol.HexGrid.prototype.offset2cube = function (o)
{ return hex2cube(offset2Hex(o));
};
/** Round cube coords
* @param {ol.coordinate} h cube coordinate
* @return {ol.coordinate} rounded cube coordinate
*/
ol.HexGrid.prototype.cube_round = function(h)
{ var rx = Math.round(h[0])
var ry = Math.round(h[1])
var rz = Math.round(h[2])
var x_diff = Math.abs(rx - h[0])
var y_diff = Math.abs(ry - h[1])
var z_diff = Math.abs(rz - h[2])
if (x_diff > y_diff && x_diff > z_diff) rx = -ry-rz
else if (y_diff > z_diff) ry = -rx-rz
else rz = -rx-ry
return [rx, ry, rz];
};
/** Round axial coords
* @param {ol.coordinate} h axial coordinate
* @return {ol.coordinate} rounded axial coordinate
*/
ol.HexGrid.prototype.hex_round = function(h)
{ return this.cube2hex( this.cube_round( this.hex2cube(h )) );
};
/** Get hexagon corners
*/
ol.HexGrid.prototype.hex_corner = function(center, size, i)
{ return [ center[0] + size * this.layout_[8+(2*(i%6))], center[1] + size * this.layout_[9+(2*(i%6))]];
};
/** Get hexagon coordinates at a coordinate
* @param {ol.coord} coord
* @return {Arrary<ol.coord>}
*/
ol.HexGrid.prototype.getHexagonAtCoord = function (coord)
{ returhn (this.getHexagon(this.coord2hex(coord)));
};
/** Get hexagon coordinates at hex
* @param {ol.coord} hex
* @return {Arrary<ol.coord>}
*/
ol.HexGrid.prototype.getHexagon = function (hex)
{ var p = [];
var c = this.hex2coord(hex);
for (var i=0; i<=7; i++)
{ p.push(this.hex_corner(c, this.size_, i, this.layout_[8]));
}
return p;
};
/** Convert hex to coord
* @param {ol.hex} hex
* @return {ol.coord}
*/
ol.HexGrid.prototype.hex2coord = function (hex)
{ return [
this.origin_[0] + this.size_ * (this.layout_[0] * hex[0] + this.layout_[1] * hex[1]),
this.origin_[1] + this.size_ * (this.layout_[2] * hex[0] + this.layout_[3] * hex[1])
];
};
/** Convert coord to hex
* @param {ol.coord} coord
* @return {ol.hex}
*/
ol.HexGrid.prototype.coord2hex = function (coord)
{ var c = [ (coord[0]-this.origin_[0]) / this.size_, (coord[1]-this.origin_[1]) / this.size_ ];
var q = this.layout_[4] * c[0] + this.layout_[5] * c[1];
var r = this.layout_[6] * c[0] + this.layout_[7] * c[1];
return this.hex_round([q, r]);
};
/** Calculate distance between to hexagon (number of cube)
* @param {ol.coordinate} a first cube coord
* @param {ol.coordinate} a second cube coord
* @return {Number} distance
*/
ol.HexGrid.prototype.cube_distance = function (a, b)
{ //return ( (Math.abs(a[0] - b[0]) + Math.abs(a[1] - b[1]) + Math.abs(a[2] - b[2])) / 2 );
return ( Math.max (Math.abs(a[0] - b[0]), Math.abs(a[1] - b[1]), Math.abs(a[2] - b[2])) );
};
(function(){
/** Line interpolation
*/
function lerp(a, b, t)
{ // for floats
return a + (b - a) * t;
};
function cube_lerp(a, b, t)
{ // for hexes
return [
lerp (a[0]+1e-6, b[0], t),
lerp (a[1]+1e-6, b[1], t),
lerp (a[2]+1e-6, b[2], t)
];
};
/** Calculate line between to hexagon
* @param {ol.coordinate} a first cube coord
* @param {ol.coordinate} b second cube coord
* @return {Array<ol.coordinate>} array of cube coordinates
*/
ol.HexGrid.prototype.cube_line = function (a, b)
{ var d = this.cube_distance(a, b);
if (!d) return [a];
var results = []
for (var i=0; i<=d; i++)
{ results.push ( this.cube_round ( cube_lerp(a, b, i/d) ) );
}
return results;
};
})();
ol.HexGrid.prototype.neighbors =
{ 'cube': [ [+1, -1, 0], [+1, 0, -1], [0, +1, -1], [-1, +1, 0], [-1, 0, +1], [0, -1, +1] ],
'hex': [ [+1, 0], [+1, -1], [0, -1], [-1, 0], [-1, +1], [0, +1] ]
};
/** Get the neighbors for an hexagon
* @param {ol.coordinate} h axial coord
* @param {Number} direction
* @return { ol.coordinate | Array<ol.coordinates> } neighbor || array of neighbors
*/
ol.HexGrid.prototype.hex_neighbors = function (h, d)
{ if (d!==undefined)
{ return [ h[0] + this.neighbors.hex[d%6][0], h[1] + this.neighbors.hex[d%6][1] ];
}
else
{ var n = [];
for (d=0; d<6; d++)
{ n.push ([ h[0] + this.neighbors.hex[d][0], h[1] + this.neighbors.hex[d][1] ]);
}
return n;
}
};
/** Get the neighbors for an hexagon
* @param {ol.coordinate} c cube coord
* @param {Number} direction
* @return { ol.coordinate | Array<ol.coordinates> } neighbor || array of neighbors
*/
ol.HexGrid.prototype.cube_neighbors = function (c, d)
{ if (d!==undefined)
{ return [ c[0] + this.neighbors.cube[d%6][0], c[1] + this.neighbors.cube[d%6][1], c[2] + this.neighbors.cube[d%6][2] ];
}
else
{ var n = [];
for (d=0; d<6; d++)
{ n.push ([ c[0] + this.neighbors.cube[d][0], c[1] + this.neighbors.cube[d][1], c[2] + this.neighbors.cube[d][2] ]);
}
for (d=0; d<6; d++) n[d] = this.cube2hex(n[d])
return n;
}
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Show a markup a point on postcompose
* @deprecated use map.animateFeature instead
* @param {ol.coordinates} point to pulse
* @param {ol.markup.options} pulse options param
* - projection {ol.projection|String|undefined} projection of coords, default none
* - delay {Number} delay before mark fadeout
* - maxZoom {Number} zoom when mark fadeout
* - style {ol.style.Image|ol.style.Style|Array<ol.style.Style>} Image to draw as markup, default red circle
* @return Unique key for the listener with a stop function to stop animation
*/
ol.Map.prototype.markup = function(coords, options)
{ var listenerKey;
var self = this;
options = options || {};
// Change to map's projection
if (options.projection)
{ coords = ol.proj.transform(coords, options.projection, this.getView().getProjection());
}
// options
var start = new Date().getTime();
var delay = options.delay || 3000;
var duration = 1000;
var maxZoom = options.maxZoom || 100;
var easing = ol.easing.easeOut;
var style = options.style;
if (!style) style = new ol.style.Circle({ radius:10, stroke:new ol.style.Stroke({color:'red', width:2 }) });
if (style instanceof ol.style.Image) style = new ol.style.Style({ image: style });
if (!(style instanceof Array)) style = [style];
// Animate function
function animate(event)
{ var frameState = event.frameState;
var elapsed = frameState.time - start;
if (elapsed > delay+duration)
{ ol.Observable.unByKey(listenerKey);
listenerKey = null;
}
else
{ if (delay>elapsed && this.getView().getZoom()>maxZoom) delay = elapsed;
var ratio = frameState.pixelRatio;
var elapsedRatio = 0;
if (elapsed > delay) elapsedRatio = (elapsed-delay) / duration;
var context = event.context;
context.save();
context.beginPath();
context.globalAlpha = easing(1 - elapsedRatio);
for (var i=0; i<style.length; i++)
{ var imgs = style[i].getImage();
var sc = imgs.getScale();
imgs.setScale(sc*ratio);
event.vectorContext.setStyle(style[i]);
event.vectorContext.drawGeometry(new ol.geom.Point(coords));
imgs.setScale(sc);
}
context.restore();
// tell OL3 to continue postcompose animation
if (elapsed >= delay) frameState.animate = true;
}
}
setTimeout (function()
{ if (listenerKey) self.renderSync();
}, delay);
// Launch animation
listenerKey = this.on('postcompose', animate.bind(this));
this.renderSync();
listenerKey.stop = function()
{ delay = duration = 0;
this.target.renderSync();
};
return listenerKey;
}
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Ordering function for ol.layer.Vector renderOrder parameter
* ol.ordering.fn (options)
* It will return an ordering function (f0,f1)
* @namespace
*/
ol.ordering = {};
/** y-Ordering
* @return ordering function (f0,f1)
*/
ol.ordering.yOrdering = function(options)
{ return function(f0,f1)
{ return f1.getGeometry().getExtent()[1] - f0.getGeometry().getExtent()[1] ;
};
};
/** Order with a feature attribute
* @param options
* attribute: ordering attribute, default zIndex
* equalFn: ordering function for equal values
* @return ordering function (f0,f1)
*/
ol.ordering.zIndex = function(options)
{ if (!options) options = {};
var attr = options.attribute || 'zIndex';
if (option.equalFn)
{ return function(f0,f1)
{ if (f0.get(attr) == f1.get(attr)) return option.equalFn(f0,f1);
else return f0.get(attr) < f1.get(attr) ? 1:-1;
};
}
else
{ return function(f0,f1)
{ if (f0.get(attr) == f1.get(attr)) return 0;
else return f0.get(attr) < f1.get(attr) ? 1:-1;
};
}
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** Pulse a point on postcompose
* @deprecated use map.animateFeature instead
* @param {ol.coordinates} point to pulse
* @param {ol.pulse.options} pulse options param
* - projection {ol.projection||String} projection of coords
* - duration {Number} animation duration in ms, default 3000
* - amplitude {Number} movement amplitude 0: none - 0.5: start at 0.5*radius of the image - 1: max, default 1
* - easing {ol.easing} easing function, default ol.easing.easeOut
* - style {ol.style.Image|ol.style.Style|Array<ol.style.Style>} Image to draw as markup, default red circle
*/
ol.Map.prototype.pulse = function(coords, options)
{ var listenerKey;
options = options || {};
// Change to map's projection
if (options.projection)
{ coords = ol.proj.transform(coords, options.projection, this.getView().getProjection());
}
// options
var start = new Date().getTime();
var duration = options.duration || 3000;
var easing = options.easing || ol.easing.easeOut;
var style = options.style;
if (!style) style = new ol.style.Circle({ radius:30, stroke:new ol.style.Stroke({color:'red', width:2 }) });
if (style instanceof ol.style.Image) style = new ol.style.Style({ image: style });
if (!(style instanceof Array)) style = [style];
var amplitude = options.amplitude || 1;
if (amplitude<0) amplitude=0;
var maxRadius = options.radius || 15;
if (maxRadius<0) maxRadius = 5;
var minRadius = maxRadius - (options.amplitude || maxRadius); //options.minRadius || 0;
var width = options.lineWidth || 2;
var color = options.color || 'red';
console.log("pulse")
// Animate function
function animate(event)
{ var frameState = event.frameState;
var ratio = frameState.pixelRatio;
var elapsed = frameState.time - start;
if (elapsed > duration) ol.Observable.unByKey(listenerKey);
else
{ var elapsedRatio = elapsed / duration;
var context = event.context;
context.save();
context.beginPath();
var e = easing(elapsedRatio)
context.globalAlpha = easing(1 - elapsedRatio);
console.log("anim")
for (var i=0; i<style.length; i++)
{ var imgs = style[i].getImage();
var sc = imgs.getScale();
imgs.setScale(ratio*sc*(1+amplitude*(e-1)));
event.vectorContext.setStyle(style[i]);
event.vectorContext.drawGeometry(new ol.geom.Point(coords));
imgs.setScale(sc);
}
context.restore();
// tell OL3 to continue postcompose animation
frameState.animate = true;
}
}
// Launch animation
listenerKey = this.on('postcompose', animate.bind(this));
this.renderSync();
}
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*
* Add a chart style to display charts (pies or bars) on a map
*/
/**
* @requires ol.style.Circle
* @requires ol.structs.IHasChecksum
*/
/**
* @classdesc
* Set chart style for vector features.
*
* @constructor
* @param {} options
* @param {String} options.type Chart type: pie,pie3D, donut or bar
* @param {number} options.radius Chart radius/size, default 20
* @param {number} options.rotation Rotation in radians (positive rotation clockwise). Default is 0.
* @param {bool} options.snapToPixel use integral numbers of pixels, default true
* @param {_ol_style_Stroke_} options.stroke stroke style
* @param {String|Array<ol.color>} options.colors predefined color set "classic","dark","pale","pastel","neon" / array of color string, default classic
* @param {number} options.offsetX X offset in px
* @param {number} options.offsetY Y offset in px
* @param {number} options.animation step in an animation sequence [0,1]
* @see [Statistic charts example](../../examples/map.style.chart.html)
* @extends {ol.style.RegularShape}
* @implements {ol.structs.IHasChecksum}
* @api
*/
ol.style.Chart = function(opt_options)
{ options = opt_options || {};
var strokeWidth = 0;
if (opt_options.stroke) strokeWidth = opt_options.stroke.getWidth();
ol.style.RegularShape.call (this,
{ radius: options.radius + strokeWidth,
fill: new ol.style.Fill({color: [0,0,0]}),
rotation: options.rotation,
snapToPixel: options.snapToPixel
});
if (options.scale) this.setScale(options.scale);
this.stroke_ = options.stroke;
this.radius_ = options.radius || 20;
this.donutratio_ = options.donutRatio || 0.5;
this.type_ = options.type;
this.offset_ = [options.offsetX ? options.offsetX : 0, options.offsetY ? options.offsetY : 0];
this.animation_ = (typeof(options.animation) == 'number') ? { animate:true, step:options.animation } : this.animation_ = { animate:false, step:1 };
this.data_ = options.data;
if (options.colors instanceof Array)
{ this.colors_ = options.colors;
}
else
{ this.colors_ = ol.style.Chart.colors[options.colors];
if(!this.colors_) this.colors_ = ol.style.Chart.colors.classic;
}
this.renderChart_();
};
ol.inherits(ol.style.Chart, ol.style.RegularShape);
/** Default color set: classic, dark, pale, pastel, neon
*/
ol.style.Chart.colors =
{ "classic": ["#ffa500","blue","red","green","cyan","magenta","yellow","#0f0"],
"dark": ["#960","#003","#900","#060","#099","#909","#990","#090"],
"pale": ["#fd0","#369","#f64","#3b7","#880","#b5d","#666"],
"pastel": ["#fb4","#79c","#f66","#7d7","#acc","#fdd","#ff9","#b9b"],
"neon": ["#ff0","#0ff","#0f0","#f0f","#f00","#00f"]
}
/**
* Clones the style.
* @return {ol.style.Chart}
*/
ol.style.Chart.prototype.clone = function()
{ var s = new ol.style.Chart(
{ type: this.type_,
radius: this.radius_,
rotation: this.getRotation(),
scale: this.getScale(),
data: this.getData(),
snapToPixel: this.getSnapToPixel(),
stroke: this.stroke_,
colors: this.colors_,
offsetX: this.offset_[0],
offsetY: this.offset_[1],
animation: this.animation_
});
s.setScale(this.getScale());
s.setOpacity(this.getOpacity());
return s;
};
/** Get data associatied with the chart
*/
ol.style.Chart.prototype.getData = function()
{ return this.data_;
}
/** Set data associatied with the chart
* @param {Array<number>}
*/
ol.style.Chart.prototype.setData = function(data)
{ this.data_ = data;
this.renderChart_();
}
/** Get symbol radius
*/
ol.style.Chart.prototype.getRadius = function()
{ return this.radius_;
}
/** Set symbol radius
* @param {number} symbol radius
* @param {number} donut ratio
*/
ol.style.Chart.prototype.setRadius = function(radius, ratio)
{ this.radius_ = radius;
this.donuratio_ = ratio || this.donuratio_;
this.renderChart_();
}
/** Set animation step
* @param {false|number} false to stop animation or the step of the animation [0,1]
*/
ol.style.Chart.prototype.setAnimation = function(step)
{ if (step===false)
{ if (this.animation_.animate == false) return;
this.animation_.animate = false;
}
else
{ if (this.animation_.step == step) return;
this.animation_.animate = true;
this.animation_.step = step;
}
this.renderChart_();
}
/** @private
*/
ol.style.Chart.prototype.renderChart_ = function(atlasManager)
{ var strokeStyle;
var strokeWidth = 0;
if (this.stroke_)
{ strokeStyle = ol.color.asString(this.stroke_.getColor());
strokeWidth = this.stroke_.getWidth();
}
// no atlas manager is used, create a new canvas
var canvas = this.getImage();
// draw the circle on the canvas
var context = (canvas.getContext('2d'));
context.clearRect(0, 0, canvas.width, canvas.height);
context.lineJoin = 'round';
var sum=0;
for (var i=0; i<this.data_.length; i++)
sum += this.data_[i];
// reset transform
context.setTransform(1, 0, 0, 1, 0, 0);
// then move to (x, y)
context.translate(0,0);
var step = this.animation_.animate ? this.animation_.step : 1;
//console.log(this.animation_.step)
// Draw pie
switch (this.type_)
{ case "donut":
case "pie3D":
case "pie":
{ var a, a0 = Math.PI * (step-1.5);
var c = canvas.width/2;
context.strokeStyle = strokeStyle;
context.lineWidth = strokeWidth;
context.save();
if (this.type_=="pie3D")
{ context.translate(0, c*0.3);
context.scale(1, 0.7);
context.beginPath();
context.fillStyle = "#369";
context.arc ( c, c*1.4, this.radius_ *step, 0, 2*Math.PI);
context.fill();
context.stroke();
}
if (this.type_=="donut")
{ context.save();
context.beginPath();
context.rect ( 0,0,2*c,2*c );
context.arc ( c, c, this.radius_ *step *this.donutratio_, 0, 2*Math.PI);
context.clip("evenodd");
}
for (var i=0; i<this.data_.length; i++)
{ context.beginPath();
context.moveTo(c,c);
context.fillStyle = this.colors_[i%this.colors_.length];
a = a0 + 2*Math.PI*this.data_[i]/sum *step;
context.arc ( c, c, this.radius_ *step, a0, a);
context.closePath();
context.fill();
context.stroke();
a0 = a;
}
if (this.type_=="donut")
{ context.restore();
context.beginPath();
context.strokeStyle = strokeStyle;
context.lineWidth = strokeWidth;
context.arc ( c, c, this.radius_ *step *this.donutratio_, Math.PI * (step-1.5), a0);
context.stroke();
}
context.restore();
break;
}
case "bar":
default:
{ var max=0;
for (var i=0; i<this.data_.length; i++)
{ if (max < this.data_[i]) max = this.data_[i];
}
var s = Math.min(5,2*this.radius_/this.data_.length);
var c = canvas.width/2;
var b = canvas.width - strokeWidth;
var x, x0 = c - this.data_.length*s/2
context.strokeStyle = strokeStyle;
context.lineWidth = strokeWidth;
for (var i=0; i<this.data_.length; i++)
{ context.beginPath();
context.fillStyle = this.colors_[i%this.colors_.length];
x = x0 + s;
var h = this.data_[i]/max*2*this.radius_ *step;
context.rect ( x0, b-h, s, h);
//console.log ( x0+", "+(b-this.data_[i]/max*2*this.radius_)+", "+x+", "+b);
context.closePath();
context.fill();
context.stroke();
x0 = x;
}
}
}
// Set Anchor
var a = this.getAnchor();
a[0] = c - this.offset_[0];
a[1] = c - this.offset_[1];
};
/**
* @inheritDoc
*/
ol.style.Chart.prototype.getChecksum = function()
{
var strokeChecksum = (this.stroke_!==null) ?
this.stroke_.getChecksum() : '-';
var recalculate = (this.checksums_===null) ||
(strokeChecksum != this.checksums_[1] ||
fillChecksum != this.checksums_[2] ||
this.radius_ != this.checksums_[3] ||
this.data_.join('|') != this.checksums_[4]);
if (recalculate) {
var checksum = 'c' + strokeChecksum + fillChecksum
+ ((this.radius_ !== void 0) ? this.radius_.toString() : '-')
+ this.data_.join('|');
this.checksums_ = [checksum, strokeChecksum, fillChecksum, this.radius_, this.data_.join('|')];
}
return this.checksums_[0];
};
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc
* Fill style with named pattern
*
* @constructor
* @param {olx.style.FillPatternOption=} options
* @param {ol.style.Image|undefined} options.image an image pattern, image must be preloaded to draw on first call
* @param {number|undefined} options.opacity opacity with image pattern, default:1
* @param {olx.style.fillPattern} options.pattern pattern name (override by image option)
* @param {ol.color} options.color pattern color
* @param {ol.style.Fill} options.fill fill color (background)
* @param {number} options.offset pattern offset for hash/dot/circle/cross pattern
* @param {number} options.size line size for hash/dot/circle/cross pattern
* @param {number} options.spacing spacing for hash/dot/circle/cross pattern
* @param {number|bool} options.angle angle for hash pattern / true for 45deg dot/circle/cross
* @param {number} options.scale pattern scale
* @extends {ol.style.Fill}
* @implements {ol.structs.IHasChecksum}
* @api
*/
ol.style.FillPattern = function(options)
{ if (!options) options = {};
var pattern;
var canvas = this.canvas_ = document.createElement('canvas');
var scale = Number(options.scale)>0 ? Number(options.scale) : 1;
var ratio = scale*ol.has.DEVICE_PIXEL_RATIO || ol.has.DEVICE_PIXEL_RATIO;
var ctx = canvas.getContext('2d');
if (options.image)
{ options.image.load();
var img = options.image.getImage();
if (img.width)
{ canvas.width = Math.round(img.width *ratio);
canvas.height = Math.round(img.height *ratio);
ctx.globalAlpha = typeof(options.opacity) == 'number' ? options.opacity:1;
ctx.drawImage(img, 0,0, img.width, img.height, 0, 0, canvas.width, canvas.height);
pattern = ctx.createPattern(canvas, 'repeat');
}
else
{ var self = this;
pattern = [0,0,0,0];
img.onload = function ()
{ canvas.width = Math.round(img.width *ratio);
canvas.height = Math.round(img.height *ratio);
ctx.globalAlpha = typeof(options.opacity) == 'number' ? options.opacity:1;
ctx.drawImage(img, 0,0, img.width, img.height, 0, 0, canvas.width, canvas.height);
pattern = ctx.createPattern(canvas, 'repeat');
self.setColor(pattern);
}
}
}
else
{ var pat = this.getPattern_(options);
canvas.width = Math.round(pat.width *ratio);
canvas.height = Math.round(pat.height *ratio);
ctx.beginPath();
if (options.fill)
{ ctx.fillStyle = ol.color.asString(options.fill.getColor());
ctx.fillRect(0,0, canvas.width, canvas.height);
}
ctx.scale(ratio,ratio);
ctx.lineCap = "round";
ctx.lineWidth = pat.stroke || 1;
ctx.fillStyle = ol.color.asString(options.color||"#000");
ctx.strokeStyle = ol.color.asString(options.color||"#000");
if (pat.circles) for (var i=0; i<pat.circles.length; i++)
{ var ci = pat.circles[i];
ctx.beginPath();
ctx.arc(ci[0], ci[1], ci[2], 0,2*Math.PI);
if (pat.fill) ctx.fill();
if (pat.stroke) ctx.stroke();
}
if (!pat.repeat) pat.repeat=[[0,0]];
if (pat.char)
{ ctx.font = pat.font || (pat.width)+"px Arial";
ctx.textAlign = 'center';
ctx.textBaseline = 'middle';
if (pat.angle)
{ ctx.fillText(pat.char, pat.width/4, pat.height/4);
ctx.fillText(pat.char, 5*pat.width/4, 5*pat.height/4);
ctx.fillText(pat.char, pat.width/4, 5*pat.height/4);
ctx.fillText(pat.char, 5*pat.width/4, pat.height/4);
ctx.fillText(pat.char, 3*pat.width/4, 3*pat.height/4);
ctx.fillText(pat.char, -pat.width/4, -pat.height/4);
ctx.fillText(pat.char, 3*pat.width/4, -pat.height/4);
ctx.fillText(pat.char, -pat.width/4, 3*pat.height/4);
}
else ctx.fillText(pat.char, pat.width/2, pat.height/2);
}
if (pat.lines) for (var i=0; i<pat.lines.length; i++) for (var r=0; r<pat.repeat.length; r++)
{ var li = pat.lines[i];
ctx.beginPath();
ctx.moveTo(li[0]+pat.repeat[r][0],li[1]+pat.repeat[r][1]);
for (var k=2; k<li.length; k+=2)
{ ctx.lineTo(li[k]+pat.repeat[r][0],li[k+1]+pat.repeat[r][1]);
}
if (pat.fill) ctx.fill();
if (pat.stroke) ctx.stroke();
ctx.save()
ctx.strokeStyle = 'red';
ctx.strokeWidth = 0.1;
//ctx.strokeRect(0,0,canvas.width,canvas.height);
ctx.restore()
}
pattern = ctx.createPattern(canvas, 'repeat');
if (options.offset)
{ var offset = options.offset;
if (typeof(offset) == "number") offset = [offset,offset];
if (offset instanceof Array)
{ var dx = Math.round((offset[0]*ratio));
var dy = Math.round((offset[1]*ratio));
// New pattern
ctx.scale(1/ratio,1/ratio)
ctx.clearRect(0,0,canvas.width,canvas.height);
ctx.translate(dx,dy);
ctx.fillStyle = pattern;
ctx.fillRect(-dx, -dy, canvas.width,canvas.height);
pattern = ctx.createPattern(canvas, 'repeat');
}
}
}
ol.style.Fill.call (this, { color: pattern });
};
ol.inherits(ol.style.FillPattern, ol.style.Fill);
/**
* Clones the style.
* @return {ol.style.FillPattern}
*/
ol.style.FillPattern.prototype.clone = function()
{ var s = ol.style.Fill.prototype.clone.call(this);
s.canvas_ = this.canvas_;
return s;
};
/** Get canvas used as pattern
* @return {canvas}
*/
ol.style.FillPattern.prototype.getImage = function()
{ return this.canvas_;
}
/** Get pattern
* @param {olx.style.FillPatternOption}
*/
ol.style.FillPattern.prototype.getPattern_ = function(options)
{ var pat = ol.style.FillPattern.prototype.patterns[options.pattern]
|| ol.style.FillPattern.prototype.patterns.dot;
var d = Math.round(options.spacing)||10;
var d2 = Math.round(d/2)+0.5;
switch (options.pattern)
{ case 'dot':
case 'circle':
{ var size = options.size===0 ? 0 : options.size/2 || 2;
if (!options.angle)
{ pat.width = pat.height = d;
pat.circles = [[ d/2, d/2, size ]]
if (options.pattern=='circle')
{ pat.circles = pat.circles.concat([
[ d/2+d, d/2, size ],
[ d/2-d, d/2, size ],
[ d/2, d/2+d, size ],
[ d/2, d/2-d, size ],
[ d/2+d, d/2+d, size ],
[ d/2+d, d/2-d, size ],
[ d/2-d, d/2+d, size ],
[ d/2-d, d/2-d, size ] ])
};
}
else
{ d = pat.width = pat.height = Math.round(d*1.4);
pat.circles = [[ d/4, d/4, size ], [ 3*d/4, 3*d/4, size ]];
if (options.pattern=='circle')
{ pat.circles = pat.circles.concat([
[ d/4+d, d/4, size ],
[ d/4, d/4+d, size ],
[ 3*d/4-d, 3*d/4, size ],
[ 3*d/4, 3*d/4-d, size ],
[ d/4+d, d/4+d, size ],
[ 3*d/4-d, 3*d/4-d, size ] ]);
}
}
break;
}
case 'tile':
case 'square':
{ var size = options.size===0 ? 0 : options.size/2 || 2;
if (!options.angle)
{ pat.width = pat.height = d;
pat.lines = [[ d/2-size, d/2-size, d/2+size, d/2-size, d/2+size, d/2+size, d/2-size,d/2+size, d/2-size, d/2-size ]]
}
else
{ pat.width = pat.height = d;
//size *= Math.sqrt(2);
pat.lines = [[ d/2-size,d/2, d/2,d/2-size, d/2+size,d/2, d/2,d/2+size, d/2-size,d/2 ]]
}
if (options.pattern=='square') pat.repeat = [[0,0], [0,d], [d,0], [0,-d], [-d,0], [-d,-d], [d,d], [-d,d], [d,-d] ]
break;
}
case 'cross':
{ // Limit angle to 0 | 45
if (options.angle) options.angle = 45;
}
case 'hatch':
{ var a = Math.round(((options.angle||0)-90)%360);
if (a>180) a -= 360;
a *= Math.PI/180;
var cos = Math.cos(a);
var sin = Math.sin(a);
if (Math.abs(sin)<0.0001)
{ pat.width = pat.height = d;
pat.lines = [ [ 0,0.5, d, 0.5 ] ];
pat.repeat = [ [0,0], [0,d] ];
}
else if (Math.abs(cos)<0.0001)
{ pat.width = pat.height = d;
pat.lines = [ [ 0.5,0, 0.5, d] ];
pat.repeat = [ [0,0], [d,0] ];
if (options.pattern=='cross')
{ pat.lines.push ([ 0,0.5, d, 0.5 ]);
pat.repeat.push([0,d]);
}
}
else
{ var w = pat.width = Math.round(Math.abs(d/sin)) || 1;
var h = pat.height = Math.round(Math.abs(d/cos)) || 1;
if (options.pattern=='cross')
{ pat.lines = [ [-w,-h, 2*w,2*h], [2*w,-h, -w,2*h] ];
pat.repeat = [ [0,0] ];
}
else if (cos*sin>0)
{ pat.lines = [ [-w,-h, 2*w,2*h] ];
pat.repeat = [ [0,0], [w,0], [0,h] ];
}
else
{ pat.lines = [ [2*w,-h, -w,2*h] ];
pat.repeat = [ [0,0], [-w,0], [0,h] ];
}
}
pat.stroke = options.size===0 ? 0 : options.size||4;
}
default: break;
}
return pat
}
/** Static fuction to add char patterns
* @param {title}
* @param {olx.fillpattern.Option}
* - size {integer} default 10
* - width {integer} default 10
* - height {integer} default 10
* - circles {Array<circles>}
* - lines: {Array<pointlist>}
* - stroke {integer}
* - fill {bool}
* - char {char}
* - font {string} default "10px Arial"
*/
ol.style.FillPattern.addPattern = function (title, options)
{ if (!options) options={};
ol.style.FillPattern.prototype.patterns[title || options.char] =
{ width: options.width || options.size || 10,
height: options.height || options.size || 10,
font: options.font,
char: options.char,
circles: options.circles,
lines: options.lines,
repeat: options.repeat,
stroke: options.stroke,
angle: options.angle,
fill: options.fill
}
}
/** Patterns definitions
Examples : http://seig.ensg.ign.fr/fichchap.php?NOFICHE=FP31&NOCHEM=CHEMS009&NOLISTE=1&N=8
*/
ol.style.FillPattern.prototype.patterns =
{
"hatch":
{ width:5,
height:5,
lines:[[0,2.5,5,2.5]],
stroke:1
},
"cross":
{ width:7,
height:7,
lines:[[0,3,10,3],[3,0,3,10]],
stroke:1
},
"dot":
{ width:8,
height:8,
circles:[[5,5,2]],
stroke:false,
fill:true,
},
"circle":
{ width:10,
height:10,
circles:[[5,5,2]],
stroke:1,
fill:false,
},
"square":
{ width:10,
height:10,
lines:[[3,3, 3,8, 8,8, 8,3, 3,3]],
stroke:1,
fill:false,
},
"tile":
{ width:10,
height:10,
lines:[[3,3, 3,8, 8,8, 8,3, 3,3]],
fill:true,
},
"woven":
{ width: 12,
height: 12,
lines: [[ 3,3, 9,9 ],[0,12, 3,9], [9,3, 12,0], [-1,1,1,-1], [13,11,11,13]],
stroke: 1
},
"crosses":
{ width: 8,
height: 8,
lines: [[ 2,2, 6,6 ],[2,6,6,2]],
stroke: 1
},
"caps":
{ width: 8,
height: 8,
lines: [[ 2,6, 4,2, 6,6 ]],
stroke: 1
},
"nylon":
{ width: 20,
height: 20,
// lines: [[ 0,5, 0,0, 5,0 ],[ 5,10, 10,10, 10,5 ], [ 10,15, 10,20, 15,20 ],[ 15,10, 20,10, 20,15 ]],
// repeat: [[0,0], [20,0], [0,20], [-20,0], [0,-20], [-20,-20]],
lines: [[ 1,6, 1,1, 6,1 ],[ 6,11, 11,11, 11,6 ], [ 11,16, 11,21, 16,21 ],[ 16,11, 21,11, 21,16 ]],
repeat: [[0,0], [-20,0], [0,-20] ],
stroke: 1
},
"hexagon":
{ width: 20,
height: 12,
lines: [[ 0,10, 4,4, 10,4, 14,10, 10,16, 4,16, 0,10 ]],
stroke:1,
repeat:[[0,0],[10,6],[10,-6],[-10,-6]]
},
"cemetry":
{ width:15,
height:19,
lines:[[0,3.5,7,3.5],[3.5,0,3.5,10],
//[7,12.5,14,12.5],[10.5,9,10.5,19]
],
stroke:1,
repeat:[[0,0],[7,9]]
},
"sand":
{ width:20,
height:20,
circles:[[1,2,1],[9,3,1],[2,16,1],
[7,8,1],[6,14,1],[4,19,1],
[14,2,1],[12,10,1],[14,18,1],
[18,8,1],[18,14,1]],
fill:1
},
"conglomerate":
{ width:30,
height:20,
circles:[[2,4,1],[17,3,1],[26,18,1],[12,17,1],[5,17,2],[28,11,2]],
lines:[[7,5, 6,7, 9,9, 11,8, 11,6, 9,5, 7,5],
[16,10, 15,13, 16,14, 19,15, 21,13, 22,9, 20,8, 19,8, 16,10],
[24,6, 26,7, 27,5, 26,4, 24,4, 24,6]],
stroke:1
},
"gravel":
{ width:15,
height:10,
circles:[[4,2,1],[5,9,1],[1,7,1]],//[9,9,1],,[15,2,1]],
lines:[[7,5, 6,6, 7,7, 8,7, 9,7, 10,5, 9,4, 7,5], [11,2, 14,4, 14,1, 12,1, 11,2]],
stroke:1
},
"brick":
{ width:18,
height:16,
lines:[ [0,1,18,1],[0,10,18,10], [6,1,6,10],[12,10,12,18],[12,0,12,1]],
stroke:1
},
"dolomite":
{ width:20,
height:16,
lines:[[0,1,20,1],[0,9,20,9],[1,9,6,1],[11,9,14,16],[14,0,14.4,1]],
stroke:1
},
"coal":
{ width:20,
height:16,
lines:[[1,5, 7,1, 7,7], [11,10, 12,5, 18,9], [5,10, 2,15, 9,15,], [15,16, 15,13, 20,16], [15,0, 15,2, 20,0]],
fill:1
},
"breccia":
{ width:20,
height:16,
lines:[[1,5, 7,1, 7,7, 1,5], [11,10, 12,5, 18,9, 11,10], [5,10, 2,15, 9,15, 5,10], [15,16, 15,13, 22,18], [15,0, 15,2, 20,0] ],
stroke:1,
},
"clay":
{ width:20,
height:20,
lines:[[0,0, 3,11, 0,20], [11,0, 10,3, 13,13, 11,20], [0,0, 10,3, 20,0], [0,12, 3,11, 13,13, 20,12]],
stroke:1
},
"flooded":
{ width:15,
height:10,
lines:[ [0,1,10,1],[0,6,5,6], [10,6,15,6]],
stroke:1
},
"chaos":
{ width:40,
height:40,
lines:[[40,2, 40,0, 38,0, 40,2],
[4,0, 3,2, 2,5, 0,0, 0,3, 2,7, 5,6, 7,7, 8,10, 9,12, 9,13, 9,14, 8,14, 6,15, 2,15, 0,20, 0,22, 2,20, 5,19,
8,15, 10,14, 11,12.25, 10,12, 10,10, 12,9, 13,7, 12,6, 13,4, 16,7, 17,4, 20,0, 18,0, 15,3, 14,2, 14,0,
12,1, 11,0, 10,1, 11,4, 10,7, 9,8, 8,5, 6,4, 5,3, 5,1, 5,0, 4,0],
[7,1, 7,3, 8,3, 8,2, 7,1], [4,3, 5,5, 4,5, 4,3], [34,5, 33,7, 38,10, 38,8, 36,5, 34,5],
[ 27,0, 23,2, 21,8, 30,0, 27,0],
[25,8, 26,12, 26,16, 22.71875,15.375, 20,13, 18,15, 17,18, 13,22, 17,21, 19,22, 21,20, 19,18, 22,17, 30,25,
26,26, 24,28, 21.75,33.34375, 20,36, 18,40, 20,40, 24,37, 25,32, 27,31, 26,38, 27,37, 30,32, 32,35, 36,37,
38,40, 38,39, 40,40, 37,36, 34,32, 37,31, 36,29, 33,27, 34,24, 39,21, 40,21, 40,16, 37,20, 31,22, 32,25,
27,20, 29,15, 30,20, 32,20, 34,18, 33,12, 31,11, 29,14, 26,9, 25,8], [39,24, 37,26, 40,28, 39,24],
[13,15, 9,19, 14,18, 13,15], [18,23, 14,27, 16,27, 17,25, 20,26, 18,23],
[6,24, 2,26, 1,28, 2,30, 5,28, 12,30, 16,32, 18,30, 15,30, 12,28, 9,25, 7,27, 6,24],
[29,27, 32,28, 33,31, 30,29, 27,28, 29,27],
[5,35, 1,33, 3,36, 13,38, 15,35, 10,36, 5,35]],
fill:1,
},
"grass":
{ width:27,
height:22,
lines: [[0,10.5,13,10.5], [2.5,10,1.5,7], [4.5,10, 4.5,5, 3.5,4 ], [7,10, 7.5,6, 8.5,3], [10,10,11,6]],
repeat: [[0,0],[14,10]],
stroke:1
},
"swamp":
{ width:24,
height:23,
lines:[ [0,10.5,9.5,10.5], [2.5,10,2.5,7], [4.5,10,4.5,4], [6.5,10,6.5,6], [3,12.5,7,12.5] ],
repeat: [[0,0],[14,10]],
stroke:1
},
"wave":
{ width:10,
height:8,
lines:[ [0,0, 5,4, 10,0] ],
stroke:1
},
"vine":
{ width:13,
height:13,
lines:[[3,0,3,6],[9,7,9,13]],
stroke:1.0
},
"forest":
{ width:55,
height:30,
circles:[[7,7,3.5],[20,20,1.5],[42,22,3.5],[35,5,1.5]],
stroke:1
},
"scrub":
{ width:26,
height:20,
lines:[ [1,4, 4,8, 6,4] ],
circles:[[20,13,1.5]],
stroke:1,
},
"tree":
{ width:30,
height:30,
lines:[[7.78,10.61,4.95,10.61,4.95,7.78,3.54,7.78,2.12,6.36,0.71,6.36,0,4.24,0.71,2.12,4.24,0,7.78,0.71,9.19,3.54,7.78,4.95,7.07,7.07,4.95,7.78]],
repeat: [[3,1],[18,16]],
stroke:1
},
"pine":
{ width:30,
height:30,
lines:[[5.66,11.31,2.83,11.31,2.83,8.49,0,8.49,2.83,0,5.66,8.49,2.83,8.49]],
repeat:[[3,1],[18,16]],
stroke:1
},
"pines":
{ width:22,
height:20,
lines:[[1,4,3.5,1,6,4],[1,8,3.5,5,6,8],[3.5,1,3.5,11],[12,14.5,14.5,14,17,14.5],[12,18,17,18],[14.5,12,14.5,18]],
repeat: [[2,1]],
stroke:1
},
"rock":
{ width:20,
height:20,
lines:[ [1,0,1,9],[4,0,4,9],[7,0,7,9],
[10,1,19,1],[10,4,19,4],[10,7,19,7],
[0,11,9,11],[0,14,9,14],[0,17,9,17],
[12,10,12,19],[15,10,15,19],[18,10,18,19] ],
repeat:[[0.5,0.5]],
stroke:1
},
"rocks":
{ width:20,
height:20,
lines:[ [5,0, 3,0, 5,4, 4,6, 0,3, 0,5, 3,6, 5,9, 3.75,10, 2.5,10, 0,9, 0,10, 4,11, 5,14, 4,15, 0,13,
0,13, 0,13, 0,14, 0,14, 5,16, 5,18, 3,19, 0,19, -0.25,19.9375, 5,20, 10,19, 10,20, 11,20, 12,19,
14,20, 15,20, 17,19, 20,20, 20,19, 19,16, 20,15, 20,11, 20,10, 19,8, 20,5, 20,0, 19,0, 20,2, 19,4,
17,4, 16,3, 15,0, 14,0, 15,4, 11,5, 10,4, 11,0, 10,0, 9,4, 6,5, 5,0,],
[18,5, 19,6, 18,10, 16,10, 14,9, 16,5, 18,5],
[5,6, 9,5, 10,6, 10,9, 6,10, 5,6],
[14,5, 14,8, 13,9, 12,9, 11,7, 12,5, 14,5],
[ 5,11, 8,10, 9,11, 10,14, 6,15, 6,15, 5,11],
[13,10, 14,11, 15,14, 15,14, 15,14, 11,15, 10,11, 11,10, 13,10],
[15,12, 16,11, 19,11, 19,15, 16,14, 16,14, 15,12],
[6,16, 9,15, 10,18, 5,19, 6,16],
[10,16, 14,16, 14,18, 13,19, 11,18, 10,16],
[15,15, 18,16, 18,18, 16,19, 15,18, 15,15]],
stroke:1
}
}
/**
* /
ol.style.FillPattern.prototype.getChecksum = function()
{
var strokeChecksum = (this.stroke_!==null) ?
this.stroke_.getChecksum() : '-';
var fillChecksum = (this.fill_!==null) ?
this.fill_.getChecksum() : '-';
var recalculate = (this.checksums_===null) ||
(strokeChecksum != this.checksums_[1] ||
fillChecksum != this.checksums_[2] ||
this.radius_ != this.checksums_[3] ||
this.form_+"-"+this.glyphs_ != this.checksums_[4]);
if (recalculate) {
var checksum = 'c' + strokeChecksum + fillChecksum
+ ((this.radius_ !== void 0) ? this.radius_.toString() : '-')
+ this.form_+"-"+this.glyphs_;
this.checksums_ = [checksum, strokeChecksum, fillChecksum, this.radius_, this.form_+"-"+this.glyphs_];
}
return this.checksums_[0];
};
/**/
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @requires ol.style.Circle
* @requires ol.structs.IHasChecksum
*/
/**
* @classdesc
* A marker style to use with font symbols.
*
* @constructor
* @param {} options Options.
* @param {number} options.glyph the glyph name or a char to display as symbol.
* The name must be added using the {@link ol.style.FontSymbol.addDefs} function.
* @param {string} options.form
* none|circle|poi|bubble|marker|coma|shield|blazon|bookmark|hexagon|diamond|triangle|sign|ban|lozenge|square
* a form that will enclose the glyph, default none
* @param {number} options.radius
* @param {number} options.rotation
* @param {number} options.rotateWithView
* @param {number} options.opacity
* @param {number} options.fontSize, default 1
* @param {boolean} options.gradient true to display a gradient on the symbol
* @param {_ol_style_Fill_} options.fill
* @param {_ol_style_Stroke_} options.stroke
* @extends {ol.style.RegularShape}
* @implements {ol.structs.IHasChecksum}
* @api
*/
ol.style.FontSymbol = function(options)
{ options = options || {};
var strokeWidth = 0;
if (options.stroke) strokeWidth = options.stroke.getWidth();
ol.style.RegularShape.call (this,{ radius: options.radius, fill:options.fill,
rotation:options.rotation, rotateWithView: options.rotateWithView });
if (typeof(options.opacity)=="number") this.setOpacity(options.opacity);
this.color_ = options.color;
this.fontSize_ = options.fontSize || 1;
this.stroke_ = options.stroke;
this.fill_ = options.fill;
this.radius_ = options.radius -strokeWidth;
this.form_ = options.form || "none";
this.gradient_ = options.gradient;
this.offset_ = [options.offsetX ? options.offsetX :0, options.offsetY ? options.offsetY :0];
this.glyph_ = this.getGlyph(options.glyph) || "";
this.renderMarker_();
};
ol.inherits(ol.style.FontSymbol, ol.style.RegularShape);
/** Cool stuff to get the image symbol for a style
*/
ol.style.Image.prototype.getImagePNG = function()
{ var canvas = this.getImage();
if (canvas)
{ try { return canvas.toDataURL("image/png"); }
catch(e) { return false; }
}
else return false;
}
/**
* Font defs
*/
ol.style.FontSymbol.prototype.defs = { 'fonts':{}, 'glyphs':{} };
/** Static function : add new font defs
* @param {String|Object} font the font desciption
* @param {} glyphs a key / value list of glyph definitions.
* Each key is the name of the glyph,
* the value is an object that code the font, the caracter code,
* the name and a search string for the glyph.
*/
ol.style.FontSymbol.addDefs = function(font, glyphs)
{ var thefont = font;
if (typeof(font) == "string") thefont = {"font":font, "name":font, "copyright":"" };
if (!thefont.font || typeof(thefont.font) != "string")
{ console.log("bad font def");
return;
}
var fontname = thefont.font;
ol.style.FontSymbol.prototype.defs.fonts[fontname] = thefont;
for (var i in glyphs)
{ var g = glyphs[i];
if (typeof(g) == "string" && g.length==1) g = { char: g };
ol.style.FontSymbol.prototype.defs.glyphs[i] =
{ font: thefont.font,
char: g.char || ""+String.fromCharCode(g.code) || "",
theme: g.theme || thefont.name,
name: g.name || i,
search: g.search || ""
};
}
};
/**
* Clones the style.
* @return {ol.style.FontSymbol}
*/
ol.style.FontSymbol.prototype.clone = function()
{ var g = new ol.style.FontSymbol(
{ glyph: "",
color: this.color_,
fontSize: this.fontSize_,
stroke: this.stroke_,
fill: this.fill_,
radius: this.radius_ + (this.stroke_ ? this.stroke_.getWidth():0),
form: this.form_,
gradient: this.gradient_,
offsetX: this.offset_[0],
offsetY: this.offset_[1],
opacity: this.getOpacity(),
rotation: this.getRotation(),
rotateWithView: this.getRotateWithView()
});
g.setScale(this.getScale());
g.glyph_ = this.glyph_;
g.renderMarker_();
return g;
};
/**
* Get the fill style for the symbol.
* @return {ol.style.Fill} Fill style.
* @api
*/
ol.style.FontSymbol.prototype.getFill = function() {
return this.fill_;
};
/**
* Get the stroke style for the symbol.
* @return {_ol_style_Stroke_} Stroke style.
* @api
*/
ol.style.FontSymbol.prototype.getStroke = function() {
return this.stroke_;
};
/**
* Get the glyph definition for the symbol.
* @param {string|undefined} name a glyph name to get the definition, default return the glyph definition for the style.
* @return {_ol_style_Stroke_} Stroke style.
* @api
*/
ol.style.FontSymbol.prototype.getGlyph = function(name)
{ if (name) return ol.style.FontSymbol.prototype.defs.glyphs[name] || { "font":"none","char":name.charAt(0),"theme":"none","name":"none", "search":""};
else return this.glyph_;
};
/**
* Get the glyph name.
* @return {string} the name
* @api
*/
ol.style.FontSymbol.prototype.getGlyphName = function()
{ for (var i in ol.style.FontSymbol.prototype.defs.glyphs)
{ if (ol.style.FontSymbol.prototype.defs.glyphs[i] === this.glyph_) return i;
}
return "";
};
/**
* Get the stroke style for the symbol.
* @return {_ol_style_Stroke_} Stroke style.
* @api
*/
ol.style.FontSymbol.prototype.getFontInfo = function(glyph)
{ return ol.style.FontSymbol.prototype.defs.fonts[glyph.font];
}
/** @private
*/
ol.style.FontSymbol.prototype.renderMarker_ = function(atlasManager)
{
var strokeStyle;
var strokeWidth = 0;
if (this.stroke_)
{ strokeStyle = ol.color.asString(this.stroke_.getColor());
strokeWidth = this.stroke_.getWidth();
}
// no atlas manager is used, create a new canvas
var canvas = this.getImage();
//console.log(this.getImage().width+" / "+(2 * (this.radius_ + strokeWidth) + 1));
/** @type {ol.style.FontSymbol.RenderOptions} */
var renderOptions = {
strokeStyle: strokeStyle,
strokeWidth: strokeWidth,
size: canvas.width,
};
// draw the circle on the canvas
var context = (canvas.getContext('2d'));
context.clearRect(0, 0, canvas.width, canvas.height);
this.drawMarker_(renderOptions, context, 0, 0);
// Set Anchor
var a = this.getAnchor();
a[0] = canvas.width / 2 - this.offset_[0];
a[1] = canvas.width / 2 - this.offset_[1];
//this.createMarkerHitDetectionCanvas_(renderOptions);
};
/**
* @private
* @param {ol.style.FontSymbol.RenderOptions} renderOptions
* @param {CanvasRenderingContext2D} context
*/
ol.style.FontSymbol.prototype.drawPath_ = function(renderOptions, context)
{
var s = 2*this.radius_+renderOptions.strokeWidth+1;
var w = renderOptions.strokeWidth/2;
var c = renderOptions.size / 2;
// Transfo to place the glyph at the right place
var transfo = { fac:1, posX:renderOptions.size / 2, posY:renderOptions.size / 2 };
context.lineJoin = 'round';
context.beginPath();
// Draw the path with the form
switch (this.form_)
{ case "none": transfo.fac=1; break;
case "circle":
case "ban":
context.arc ( c, c, s/2, 0, 2 * Math.PI, true);
break;
case "poi":
context.arc ( c, c -0.4*this.radius_, 0.6*this.radius_, 0.15*Math.PI, 0.85*Math.PI, true);
context.lineTo ( c-0.89*0.05*s, (0.95+0.45*0.05)*s+w);
context.arc ( c, 0.95*s+w, 0.05*s, 0.85*Math.PI, 0.15*Math.PI, true);
transfo = { fac:0.45, posX:c, posY:c -0.35*this.radius_ };
break;
case "bubble":
context.arc ( c, c -0.2*this.radius_, 0.8*this.radius_, 0.4*Math.PI, 0.6*Math.PI, true);
context.lineTo ( 0.5*s+w, s+w);
transfo = { fac:0.7, posX:c, posY:c -0.2*this.radius_ };
break;
case "marker":
context.arc ( c, c -0.2*this.radius_, 0.8*this.radius_, 0.25*Math.PI, 0.75*Math.PI, true);
context.lineTo ( 0.5*s+w, s+w);
transfo = { fac:0.7, posX: c, posY: c -0.2*this.radius_ };
break;
case "coma":
/*
context.arc( renderOptions.size / 2, renderOptions.size / 2 +0.2*this.radius_, 0.8*this.radius_, 0.5*Math.PI, 0, true);
context.arc( renderOptions.size / 2, renderOptions.size / 2 -0.2*this.radius_, 0.8*this.radius_, 0, 0.5*Math.PI, true);
*/
context.moveTo ( c + 0.8*this.radius_, c -0.2*this.radius_);
context.quadraticCurveTo ( 0.95*s+w, 0.75*s+w, 0.5*s+w, s+w);
context.arc ( c, c -0.2*this.radius_, 0.8*this.radius_, 0.45*Math.PI, 0, false);
transfo = { fac:0.7, posX: c, posY: c -0.2*this.radius_ };
break;
default:
{ var pts;
switch (this.form_)
{ case "shield":
pts = [ 0.05,0, 0.95,0, 0.95,0.8, 0.5,1, 0.05,0.8, 0.05,0 ];
transfo.posY = 0.45*s+w ;
break;
case "blazon":
pts = [ 0.1,0, 0.9,0, 0.9,0.8, 0.6,0.8, 0.5,1, 0.4,0.8, 0.1,0.8, 0.1,0 ];
transfo.fac = 0.8;
transfo.posY = 0.4*s+w ;
break;
case "bookmark":
pts = [ 0.05,0, 0.95,0, 0.95,1, 0.5,0.8, 0.05,1, 0.05,0 ];
transfo.fac = 0.9;
transfo.posY = 0.4*s+w ;
break;
case "hexagon":
pts = [ 0.05,0.2, 0.5,0, 0.95,0.2, 0.95,0.8, 0.5,1, 0.05,0.8, 0.05,0.2 ];
transfo.fac = 0.9;
transfo.posY = 0.5*s+w ;
break;
case "diamond":
pts = [ 0.25,0, 0.75,0, 1,0.2, 1,0.4, 0.5,1, 0,0.4, 0,0.2, 0.25,0 ];
transfo.fac = 0.75 ;
transfo.posY = 0.35*s+w ;
break;
case "triangle":
pts = [ 0,0, 1,0, 0.5,1, 0,0 ];
transfo.fac = 0.6 ;
transfo.posY = 0.3*s+w ;
break;
case "sign":
pts = [ 0.5,0.05, 1,0.95, 0,0.95, 0.5,0.05 ];
transfo.fac = 0.7 ;
transfo.posY = 0.65*s+w ;
break;
case "lozenge":
pts = [ 0.5,0, 1,0.5, 0.5,1, 0,0.5, 0.5,0 ];
transfo.fac = 0.7;
break;
case "square":
default:
pts = [ 0,0, 1,0, 1,1, 0,1, 0,0 ]; break;
}
for (var i=0; i<pts.length; i+=2) context.lineTo ( pts[i]*s+w, pts[i+1]*s+w);
}
}
context.closePath();
return transfo;
}
/**
* @private
* @param {ol.style.FontSymbol.RenderOptions} renderOptions
* @param {CanvasRenderingContext2D} context
* @param {number} x The origin for the symbol (x).
* @param {number} y The origin for the symbol (y).
*/
ol.style.FontSymbol.prototype.drawMarker_ = function(renderOptions, context, x, y)
{ var fcolor = this.fill_ ? this.fill_.getColor() : "#000";
var scolor = this.stroke_ ? this.stroke_.getColor() : "#000";
if (this.form_ == "none" && this.stroke_ && this.fill_)
{ scolor = this.fill_.getColor();
fcolor = this.stroke_.getColor();
}
// reset transform
context.setTransform(1, 0, 0, 1, 0, 0);
// then move to (x, y)
context.translate(x, y);
var tr = this.drawPath_(renderOptions, context);
if (this.fill_)
{ if (this.gradient_ && this.form_!="none")
{ var grd = context.createLinearGradient(0,0,renderOptions.size/2,renderOptions.size);
grd.addColorStop (1, ol.color.asString(fcolor));
grd.addColorStop (0, ol.color.asString(scolor));
context.fillStyle = grd;
}
else context.fillStyle = ol.color.asString(fcolor);
context.fill();
}
if (this.stroke_ && renderOptions.strokeWidth) {
context.strokeStyle = renderOptions.strokeStyle;
context.lineWidth = renderOptions.strokeWidth;
context.stroke();
}
// Draw the symbol
if (this.glyph_.char)
{ context.font = (2*tr.fac*(this.radius_)*this.fontSize_)+"px "+this.glyph_.font;
context.strokeStyle = context.fillStyle;
context.lineWidth = renderOptions.strokeWidth * (this.form_ == "none" ? 2:1);
context.fillStyle = ol.color.asString(this.color_ || scolor);
context.textAlign = "center";
context.textBaseline = "middle";
var t = this.glyph_.char;
if (renderOptions.strokeWidth && scolor!="transparent") context.strokeText(t, tr.posX, tr.posY);
context.fillText(t, tr.posX, tr.posY);
}
if (this.form_=="ban" && this.stroke_ && renderOptions.strokeWidth)
{ context.strokeStyle = renderOptions.strokeStyle;
context.lineWidth = renderOptions.strokeWidth;
var r = this.radius_ + renderOptions.strokeWidth;
var d = this.radius_ * Math.cos(Math.PI/4);
context.moveTo(r + d, r - d);
context.lineTo(r - d, r + d);
context.stroke();
}
};
/**
* @inheritDoc
*/
ol.style.FontSymbol.prototype.getChecksum = function()
{
var strokeChecksum = (this.stroke_!==null) ?
this.stroke_.getChecksum() : '-';
var fillChecksum = (this.fill_!==null) ?
this.fill_.getChecksum() : '-';
var recalculate = (this.checksums_===null) ||
(strokeChecksum != this.checksums_[1] ||
fillChecksum != this.checksums_[2] ||
this.radius_ != this.checksums_[3] ||
this.form_+"-"+this.glyphs_ != this.checksums_[4]);
if (recalculate) {
var checksum = 'c' + strokeChecksum + fillChecksum
+ ((this.radius_ !== void 0) ? this.radius_.toString() : '-')
+ this.form_+"-"+this.glyphs_;
this.checksums_ = [checksum, strokeChecksum, fillChecksum, this.radius_, this.form_+"-"+this.glyphs_];
}
return this.checksums_[0];
};
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*
* Photo style for vector features
*/
/**
* @requires ol.style.RegularShape
* @requires ol.structs.IHasChecksum
*/
/**
* @classdesc
* Set Photo style for vector features.
*
* @constructor
* @param {} options
* @param { default | square | round | anchored | folio } options.kind
* @param {boolean} options.crop crop within square, default is false
* @param {Number} options.radius symbol size
* @param {boolean} options.shadow drop a shadow
* @param {ol.style.Stroke} options.stroke
* @param {String} options.src image src
* @param {String} options.crossOrigin The crossOrigin attribute for loaded images. Note that you must provide a crossOrigin value if you want to access pixel data with the Canvas renderer.
* @param {Number} options.offsetX Horizontal offset in pixels. Default is 0.
* @param {Number} options.offsetY Vertical offset in pixels. Default is 0.
* @param {function} options.onload callback when image is loaded (to redraw the layer)
* @extends {ol.style.RegularShape}
* @implements {ol.structs.IHasChecksum}
* @api
*/
ol.style.Photo = function(options)
{ options = options || {};
this.sanchor_ = options.kind=="anchored" ? 8:0;
this.shadow_ = Number(options.shadow) || 0;
if (!options.stroke)
{ options.stroke = new ol.style.Stroke({ width: 0, color: "#000"})
}
var strokeWidth = options.stroke.getWidth();
if (strokeWidth<0) strokeWidth = 0;
if (options.kind=='folio') strokeWidth += 6;
options.stroke.setWidth(strokeWidth);
ol.style.RegularShape.call (this,
{ radius: options.radius + strokeWidth + this.sanchor_/2 + this.shadow_/2,
points:0
// fill:new ol.style.Fill({color:"red"}) // No fill to create a hit detection Image
});
// Hack to get the hit detection Image (no API exported)
if (!this.hitDetectionCanvas_)
{ var img = this.getImage();
for (var i in this)
{ if (this[i] && this[i].getContext && this[i]!==img)
{ this.hitDetectionCanvas_ = this[i];
break;
}
}
}
this.stroke_ = options.stroke;
this.fill_ = options.fill;
this.crop_ = options.crop;
this.crossOrigin_ = options.crossOrigin;
this.kind_ = options.kind || "default";
this.radius_ = options.radius;
this.src_ = options.src;
this.offset_ = [options.offsetX ? options.offsetX :0, options.offsetY ? options.offsetY :0];
this.onload_ = options.onload;
if (typeof(options.opacity)=='number') this.setOpacity(options.opacity);
if (typeof(options.rotation)=='number') this.setRotation(options.rotation);
this.renderPhoto_();
};
ol.inherits(ol.style.Photo, ol.style.RegularShape);
/**
* Clones the style.
* @return {ol.style.Photo}
*/
ol.style.Photo.prototype.clone = function()
{ return new ol.style.Photo(
{ stroke: this.stroke_,
fill: this.fill_,
shadow: this.shadow_,
crop: this.crop_,
crossOrigin: this.crossOrigin_,
kind: this.kind_,
radius: this.radius_,
src: this.src_,
offsetX: this.offset_[0],
offsetY: this.offset_[1],
opacity: this.getOpacity(),
rotation: this.getRotation()
});
};
/**
* Draws a rounded rectangle using the current state of the canvas.
* Draw a rectangle if the radius is null.
* @param {Number} x The top left x coordinate
* @param {Number} y The top left y coordinate
* @param {Number} width The width of the rectangle
* @param {Number} height The height of the rectangle
* @param {Number} radius The corner radius.
*/
CanvasRenderingContext2D.prototype.roundRect = function (x, y, w, h, r)
{ if (!r) this.rect(x,y,w,h);
else
{ if (w < 2 * r) r = w / 2;
if (h < 2 * r) r = h / 2;
this.beginPath();
this.moveTo(x+r, y);
this.arcTo(x+w, y, x+w, y+h, r);
this.arcTo(x+w, y+h, x, y+h, r);
this.arcTo(x, y+h, x, y, r);
this.arcTo(x, y, x+w, y, r);
this.closePath();
}
return this;
}
/**
* Draw the form without the image
* @private
*/
ol.style.Photo.prototype.drawBack_ = function(context, color, strokeWidth)
{ var canvas = context.canvas;
context.beginPath();
context.fillStyle = color;
context.clearRect(0, 0, canvas.width, canvas.height);
switch (this.kind_)
{ case 'square':
context.rect(0,0,canvas.width-this.shadow_, canvas.height-this.shadow_);
break;
case 'circle':
context.arc(this.radius_+strokeWidth, this.radius_+strokeWidth, this.radius_+strokeWidth, 0, 2 * Math.PI, false);
break;
case 'folio':
offset = 6;
strokeWidth -= offset;
context.strokeStyle = 'rgba(0,0,0,0.5)';
var w = canvas.width-this.shadow_-2*offset;
var a = Math.atan(6/w);
context.save();
context.rotate(-a);
context.translate(-6,2);
context.beginPath();
context.rect(offset,offset,w,w);
context.stroke();
context.fill();
context.restore();
context.save();
context.translate(6,-1);
context.rotate(a);
context.beginPath();
context.rect(offset,offset,w,w);
context.stroke();
context.fill();
context.restore();
context.beginPath();
context.rect(offset,offset,w,w);
context.stroke();
break;
case 'anchored':
context.roundRect(this.sanchor_/2,0,canvas.width-this.sanchor_-this.shadow_, canvas.height-this.sanchor_-this.shadow_, strokeWidth);
context.moveTo(canvas.width/2-this.sanchor_-this.shadow_/2,canvas.height-this.sanchor_-this.shadow_);
context.lineTo(canvas.width/2+this.sanchor_-this.shadow_/2,canvas.height-this.sanchor_-this.shadow_);
context.lineTo(canvas.width/2-this.shadow_/2,canvas.height-this.shadow_);break;
default: /* roundrect */
context.roundRect(0,0,canvas.width-this.shadow_, canvas.height-this.shadow_, strokeWidth);
break;
}
context.closePath();
}
/**
* @private
*/
ol.style.Photo.prototype.renderPhoto_ = function()
{
var strokeStyle;
var strokeWidth = 0;
if (this.stroke_)
{ strokeStyle = ol.color.asString(this.stroke_.getColor());
strokeWidth = this.stroke_.getWidth();
}
var canvas = this.getImage();
// Draw hitdetection image
var context = this.hitDetectionCanvas_.getContext('2d');
this.drawBack_(context,"#000",strokeWidth);
context.fill();
// Draw the image
var context = (canvas.getContext('2d'));
this.drawBack_(context,strokeStyle,strokeWidth);
// Draw a shadow
if (this.shadow_)
{ context.shadowColor = 'rgba(0,0,0,0.5)';
context.shadowBlur = this.shadow_/2;
context.shadowOffsetX = this.shadow_/2;
context.shadowOffsetY = this.shadow_/2;
}
context.fill();
context.shadowColor = 'transparent';
var self = this;
var img = this.img_ = new Image();
if (this.crossOrigin_) img.crossOrigin = this.crossOrigin_;
img.src = this.src_;
// Draw image
if (img.width) self.drawImage_(img);
else img.onload = function()
{ self.drawImage_(img);
// Force change (?!)
// self.setScale(1);
if (self.onload_) self.onload_();
};
// Set anchor
var a = this.getAnchor();
a[0] = (canvas.width - this.shadow_)/2;
a[1] = (canvas.height - this.shadow_)/2;
if (this.sanchor_)
{ a[1] = canvas.height - this.shadow_;
}
}
/**
* Draw an timage when loaded
* @private
*/
ol.style.Photo.prototype.drawImage_ = function(img)
{ var canvas = this.getImage();
// Remove the circle on the canvas
var context = (canvas.getContext('2d'));
var strokeWidth = 0;
if (this.stroke_) strokeWidth = this.stroke_.getWidth();
var size = 2*this.radius_;
context.save();
if (this.kind_=='circle')
{ context.beginPath();
context.arc(this.radius_+strokeWidth, this.radius_+strokeWidth, this.radius_, 0, 2 * Math.PI, false);
context.clip();
}
var s, x, y, w, h, sx, sy, sw, sh;
// Crop the image to a square vignette
if (this.crop_)
{ s = Math.min (img.width/size, img.height/size);
sw = sh = s*size;
sx = (img.width-sw)/2;
sy = (img.height-sh)/2;
x = y = 0;
w = h = size+1;
}
// Fit the image to the size
else
{ s = Math.min (size/img.width, size/img.height);
sx = sy = 0;
sw = img.width;
sh = img.height;
w = s*sw;
h = s*sh;
x = (size-w)/2;
y = (size-h)/2;
}
x += strokeWidth + this.sanchor_/2;
y += strokeWidth;
context.drawImage(img, sx, sy, sw, sh, x, y, w, h);
context.restore();
// Draw a circle to avoid aliasing on clip
if (this.kind_=='circle' && strokeWidth)
{ context.beginPath();
context.strokeStyle = ol.color.asString(this.stroke_.getColor());
context.lineWidth = strokeWidth/4;
context.arc(this.radius_+strokeWidth, this.radius_+strokeWidth, this.radius_, 0, 2 * Math.PI, false);
context.stroke();
}
}
/**
* @inheritDoc
*/
ol.style.Photo.prototype.getChecksum = function()
{
var strokeChecksum = (this.stroke_!==null) ?
this.stroke_.getChecksum() : '-';
var fillChecksum = (this.fill_!==null) ?
this.fill_.getChecksum() : '-';
var recalculate = (this.checksums_===null) ||
(strokeChecksum != this.checksums_[1] ||
fillChecksum != this.checksums_[2] ||
this.radius_ != this.checksums_[3]);
if (recalculate) {
var checksum = 'c' + strokeChecksum + fillChecksum
+ ((this.radius_ !== void 0) ? this.radius_.toString() : '-');
this.checksums_ = [checksum, strokeChecksum, fillChecksum, this.radius_];
}
return this.checksums_[0];
};
/** Add a setTextPath style to draw text along linestrings
@toto letterpadding/spacing, wordpadding/spacing
*/
(function()
{
/** Internal drawing function called on postcompose
* @param {ol.eventPoscompose} e postcompose event
*/
function drawTextPath (e)
{ // Prent drawing at large resolution
if (e.frameState.viewState.resolution > this.textPathMaxResolution_) return;
var extent = e.frameState.extent;
var c2p = e.frameState.coordinateToPixelTransform;
// Get pixel path with coordinates
function getPath(c, readable)
{ var path1 = [];
for (var k=0; k<c.length; k++)
{ path1.push(c2p[0]*c[k][0]+c2p[1]*c[k][1]+c2p[4]);
path1.push(c2p[2]*c[k][0]+c2p[3]*c[k][1]+c2p[5]);
}
// Revert line ?
if (readable && path1[0]>path1[path1.length-2])
{ var path2 = [];
for (var k=path1.length-2; k>=0; k-=2)
{ path2.push(path1[k]);
path2.push(path1[k+1]);
}
return path2;
}
else return path1;
}
var ctx = e.context;
ctx.save();
ctx.scale(e.frameState.pixelRatio,e.frameState.pixelRatio);
var features = this.getSource().getFeaturesInExtent(extent);
for (var i=0, f; f=features[i]; i++)
{ { var style = this.textPathStyle_(f,e.frameState.viewState.resolution);
for (var s,j=0; s=style[j]; j++)
{
var g = s.getGeometry() || f.getGeometry();
var c;
switch (g.getType())
{ case "LineString": c = g.getCoordinates(); break;
case "MultiLineString": c = g.getLineString(0).getCoordinates(); break;
default: continue;
}
var st = s.getText();
var path = getPath(c, st.getRotateWithView() );
ctx.font = st.getFont();
ctx.textBaseline = st.getTextBaseline();
ctx.textAlign = st.getTextAlign();
ctx.lineWidth = st.getStroke() ? (st.getStroke().getWidth()||0) : 0;
ctx.strokeStyle = st.getStroke() ? (st.getStroke().getColor()||"#fff") : "#fff";
ctx.fillStyle = st.getFill() ? st.getFill().getColor()||"#000" : "#000";
// New params
ctx.textJustify = st.getTextAlign()=="justify";
ctx.textOverflow = st.getTextOverflow ? st.getTextOverflow():"";
ctx.minWidth = st.getMinWidth ? st.getMinWidth():0;
// Draw textpath
ctx.textPath(st.getText()||f.get("name"), path);
}
}
}
ctx.restore();
}
/** Set the style for features.
* This can be a single style object, an array of styles, or a function that takes a feature and resolution and
* returns an array of styles. If it is undefined the default style is used.
* If it is null the layer has no style (a null style).
* See ol.style for information on the default style.
* @param {ol.style.Style|Array.<ol.style.Style>|ol.StyleFunction} style
* @param {Number} maxResolution to display text, default: 0
*/
ol.layer.Vector.prototype.setTextPathStyle = function(style, maxResolution)
{
// Remove existing style
if (style===null)
{ if (this.textPath_) this.unByKey(this.textPath_);
this.textPath_ = null;
this.changed();
return;
}
// New postcompose
if (!this.textPath_)
{ this.textPath_ = this.on('postcompose', drawTextPath.bind(this));
}
// Set textPathStyle
if (style===undefined)
{ style = [ new ol.style.Style({ text: new ol.style.Text()}) ];
}
if (typeof(style) == "function") this.textPathStyle_ = style;
else this.textPathStyle_ = function() { return style; };
this.textPathMaxResolution_ = Number(maxResolution) || Number.MAX_VALUE;
// Force redraw
this.changed();
}
/** Add new properties to ol.style.Text
* to use with ol.layer.Vector.prototype.setTextPathStyle
* @constructor
* @param {} options
* @param {visible|ellipsis|string} textOverflow
* @param {number} minWidth minimum width (px) to draw text, default 0
*/
ol.style.TextPath = function(options)
{ if (!options) options={};
ol.style.Text.call (this, options);
this.textOverflow_ = typeof(options.textOverflow)!="undefined" ? options.textOverflow : "visible";
this.minWidth_ = options.minWidth || 0;
}
ol.inherits(ol.style.TextPath, ol.style.Text);
ol.style.TextPath.prototype.getTextOverflow = function()
{ return this.textOverflow_;
};
ol.style.TextPath.prototype.getMinWidth = function()
{ return this.minWidth_;
};
/**/
})();
/** CanvasRenderingContext2D: draw text along path
* @param {string} text
* @param {Array<Number>} path
*/
CanvasRenderingContext2D.prototype.textPath = function (text, path)
{
var ctx = this;
function dist2D(x1,y1,x2,y2)
{ var dx = x2-x1;
var dy = y2-y1;
return Math.sqrt(dx*dx+dy*dy);
}
var di, dpos=0;
var pos=2;
function getPoint(path, dl)
{ if (!di || dpos+di<dl)
{ for (; pos<path.length; )
{ di = dist2D(path[pos-2],path[pos-1],path[pos],path[pos+1]);
if (dpos+di>dl) break;
pos += 2;
if (pos>=path.length) break;
dpos += di;
}
}
var x, y, a, dt = dl-dpos;
if (pos>=path.length)
{ pos = path.length-2;
}
if (!dt)
{ x = path[pos-2];
y = path[pos-1];
a = Math.atan2(path[pos+1]-path[pos-1], path[pos]-path[pos-2]);
}
else
{ x = path[pos-2]+ (path[pos]-path[pos-2])*dt/di;
y = path[pos-1]+(path[pos+1]-path[pos-1])*dt/di;
a = Math.atan2(path[pos+1]-path[pos-1], path[pos]-path[pos-2]);
}
return [x,y,a];
}
var letterPadding = ctx.measureText(" ").width *0.25;
var start = 0;
var d = 0;
for (var i=2; i<path.length; i+=2)
{ d += dist2D(path[i-2],path[i-1],path[i],path[i+1])
}
if (d < ctx.minWidth) return;
var nbspace = text.split(" ").length -1;
// Remove char for overflow
if (ctx.textOverflow != "visible")
{ if (d < ctx.measureText(text).width + (text.length-1 + nbspace) * letterPadding)
{ var overflow = (ctx.textOverflow=="ellipsis") ? '\u2026' : ctx.textOverflow;
do
{ nbspace = text.split(" ").length -1;
text = text.slice(0,text.length-1);
} while (text && d < ctx.measureText(text+overflow).width + (text.length + overflow.length-1 + nbspace) * letterPadding)
text += overflow;
}
}
switch (ctx.textJustify || ctx.textAlign)
{ case true: // justify
case "center":
case "end":
case "right":
{ // Text align
if (ctx.textJustify)
{ start = 0;
letterPadding = (d - ctx.measureText(text).width) / (text.length-1 + nbspace);
}
else
{ start = d - ctx.measureText(text).width - (text.length + nbspace) * letterPadding;
if (ctx.textAlign == "center") start /= 2;
}
break;
}
default: break;
}
for (var t=0; t<text.length; t++)
{ var letter = text[t];
var wl = ctx.measureText(letter).width;
var p = getPoint(path, start+wl/2);
ctx.save();
ctx.textAlign = "center";
ctx.translate(p[0], p[1]);
ctx.rotate(p[2]);
if (ctx.lineWidth) ctx.strokeText(letter,0,0);
ctx.fillText(letter,0,0);
ctx.restore();
start += wl+letterPadding*(letter==" "?2:1);
}
};
//NB: (Not confirmed)To use this module, you just have to :
// import('ol-ext/layer/getpreview')
/* Copyright (c) 2015 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*
* Shadow image style for point vector features
*/
/**
* @requires ol.style.Circle
* @requires ol.structs.IHasChecksum
*/
/**
* @classdesc
* Set Shadow style for point vector features.
*
* @constructor
* @param {} options Options.
* @param {ol.style.Fill | undefined} options.fill fill style, default rgba(0,0,0,0.5)
* @param {number} options.radius point radius
* @param {number} options.blur lur radius, default radius/3
* @param {number} options.offsetX x offset, default 0
* @param {number} options.offsetY y offset, default 0
* @extends {ol.style.RegularShape}
* @implements {ol.structs.IHasChecksum}
* @api
*/
ol.style.Shadow = function(options)
{ options = options || {};
if (!options.fill) options.fill = new ol.style.Fill({ color: "rgba(0,0,0,0.5)" });
ol.style.RegularShape.call (this,{ radius: options.radius, fill: options.fill });
this.fill_ = options.fill;
this.radius_ = options.radius;
this.blur_ = options.blur===0 ? 0 : options.blur || options.radius/3;
this.offset_ = [options.offsetX ? options.offsetX : 0, options.offsetY ? options.offsetY : 0];
this.renderShadow_();
};
ol.inherits(ol.style.Shadow, ol.style.RegularShape);
/**
* Clones the style.
* @return {ol.style.Shadow}
*/
ol.style.Shadow.prototype.clone = function()
{ var s = new ol.style.Shadow(
{ fill: this.fill_,
radius: this.radius_,
blur: this.blur_,
offsetX: this.offset_[0],
offsetY: this.offset_[1]
});
s.setScale(this.getScale());
s.setOpacity(this.getOpacity());
return s;
};
/**
* @private
*/
ol.style.Shadow.prototype.renderShadow_ = function()
{
var radius = this.radius_;
var canvas = this.getImage();
var s = [canvas.width, canvas.height];
s[1] = radius;
// Remove the circle on the canvas
var context = (canvas.getContext('2d'));
context.beginPath();
context.clearRect(0, 0, canvas.width, canvas.height);
context.scale(1,0.5);
context.arc(radius, -radius, radius-this.blur_, 0, 2 * Math.PI, false);
context.fillStyle = '#000';
context.shadowColor = this.fill_.getColor();
context.shadowBlur = 0.7*this.blur_;
context.shadowOffsetX = 0;
context.shadowOffsetY = radius;
context.closePath();
context.fill();
context.shadowColor = 'transparent';
// Set anchor
var a = this.getAnchor();
a[0] = canvas.width /2 -this.offset_[0];
a[1] = canvas.height/5 -this.offset_[1];
}
/**
* @inheritDoc
*/
ol.style.Shadow.prototype.getChecksum = function()
{
var strokeChecksum = (this.stroke_!==null) ?
this.stroke_.getChecksum() : '-';
var fillChecksum = (this.fill_!==null) ?
this.fill_.getChecksum() : '-';
var recalculate = (this.checksums_===null) ||
(strokeChecksum != this.checksums_[1] ||
fillChecksum != this.checksums_[2] ||
this.radius_ != this.checksums_[3] ||
this.form_+"-"+this.glyphs_ != this.checksums_[4]);
if (recalculate) {
var checksum = 'c' + strokeChecksum + fillChecksum
+ ((this.radius_ !== void 0) ? this.radius_.toString() : '-')
+ this.form_+"-"+this.glyphs_;
this.checksums_ = [checksum, strokeChecksum, fillChecksum, this.radius_, this.form_+"-"+this.glyphs_];
}
return this.checksums_[0];
};
/* Copyright (c) 2018 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/**
* @classdesc
* Stroke style with named pattern
*
* @constructor
* @param {any} options
* @param {ol.style.Image|undefined} options.image an image pattern, image must be preloaded to draw on first call
* @param {number|undefined} options.opacity opacity with image pattern, default:1
* @param {olx.style.fillPattern} options.pattern pattern name (override by image option)
* @param {ol.colorLike} options.color pattern color
* @param {ol.style.Fill} options.fill fill color (background)
* @param {number} options.offset pattern offset for hash/dot/circle/cross pattern
* @param {number} options.size line size for hash/dot/circle/cross pattern
* @param {number} options.spacing spacing for hash/dot/circle/cross pattern
* @param {number|bool} options.angle angle for hash pattern / true for 45deg dot/circle/cross
* @param {number} options.scale pattern scale
* @extends {ol.style.Fill}
* @implements {ol.structs.IHasChecksum}
* @api
*/
ol.style.StrokePattern = function(options)
{ if (!options) options = {};
var pattern;
var canvas = this.canvas_ = document.createElement('canvas');
var scale = Number(options.scale)>0 ? Number(options.scale) : 1;
var ratio = scale*ol.has.DEVICE_PIXEL_RATIO || ol.has.DEVICE_PIXEL_RATIO;
var ctx = canvas.getContext('2d');
if (options.image)
{ options.image.load();
var img = options.image.getImage();
if (img.width)
{ canvas.width = Math.round(img.width *ratio);
canvas.height = Math.round(img.height *ratio);
ctx.globalAlpha = typeof(options.opacity) == 'number' ? options.opacity:1;
ctx.drawImage(img, 0,0, img.width, img.height, 0, 0, canvas.width, canvas.height);
pattern = ctx.createPattern(canvas, 'repeat');
}
else
{ var self = this;
pattern = [0,0,0,0];
img.onload = function ()
{ canvas.width = Math.round(img.width *ratio);
canvas.height = Math.round(img.height *ratio);
ctx.globalAlpha = typeof(options.opacity) == 'number' ? options.opacity:1;
ctx.drawImage(img, 0,0, img.width, img.height, 0, 0, canvas.width, canvas.height);
pattern = ctx.createPattern(canvas, 'repeat');
self.setColor(pattern);
}
}
}
else
{ var pat = this.getPattern_(options);
canvas.width = Math.round(pat.width *ratio);
canvas.height = Math.round(pat.height *ratio);
ctx.beginPath();
if (options.fill)
{ ctx.fillStyle = ol.color.asString(options.fill.getColor());
ctx.fillRect(0,0, canvas.width, canvas.height);
}
ctx.scale(ratio,ratio);
ctx.lineCap = "round";
ctx.lineWidth = pat.stroke || 1;
ctx.fillStyle = ol.color.asString(options.color||"#000");
ctx.strokeStyle = ol.color.asString(options.color||"#000");
if (pat.circles) for (var i=0; i<pat.circles.length; i++)
{ var ci = pat.circles[i];
ctx.beginPath();
ctx.arc(ci[0], ci[1], ci[2], 0,2*Math.PI);
if (pat.fill) ctx.fill();
if (pat.stroke) ctx.stroke();
}
if (!pat.repeat) pat.repeat=[[0,0]];
if (pat.char)
{ ctx.font = pat.font || (pat.width)+"px Arial";
ctx.textAlign = 'center';
ctx.textBaseline = 'middle';
if (pat.angle)
{ ctx.fillText(pat.char, pat.width/4, pat.height/4);
ctx.fillText(pat.char, 5*pat.width/4, 5*pat.height/4);
ctx.fillText(pat.char, pat.width/4, 5*pat.height/4);
ctx.fillText(pat.char, 5*pat.width/4, pat.height/4);
ctx.fillText(pat.char, 3*pat.width/4, 3*pat.height/4);
ctx.fillText(pat.char, -pat.width/4, -pat.height/4);
ctx.fillText(pat.char, 3*pat.width/4, -pat.height/4);
ctx.fillText(pat.char, -pat.width/4, 3*pat.height/4);
}
else ctx.fillText(pat.char, pat.width/2, pat.height/2);
}
if (pat.lines) for (var i=0; i<pat.lines.length; i++) for (var r=0; r<pat.repeat.length; r++)
{ var li = pat.lines[i];
ctx.beginPath();
ctx.moveTo(li[0]+pat.repeat[r][0],li[1]+pat.repeat[r][1]);
for (var k=2; k<li.length; k+=2)
{ ctx.lineTo(li[k]+pat.repeat[r][0],li[k+1]+pat.repeat[r][1]);
}
if (pat.fill) ctx.fill();
if (pat.stroke) ctx.stroke();
ctx.save()
ctx.strokeStyle = 'red';
ctx.strokeWidth = 0.1;
//ctx.strokeRect(0,0,canvas.width,canvas.height);
ctx.restore()
}
pattern = ctx.createPattern(canvas, 'repeat');
if (options.offset)
{ var offset = options.offset;
if (typeof(offset) == "number") offset = [offset,offset];
if (offset instanceof Array)
{ var dx = Math.round((offset[0]*ratio));
var dy = Math.round((offset[1]*ratio));
// New pattern
ctx.scale(1/ratio,1/ratio)
ctx.clearRect(0,0,canvas.width,canvas.height);
ctx.translate(dx,dy);
ctx.fillStyle = pattern;
ctx.fillRect(-dx, -dy, canvas.width,canvas.height);
pattern = ctx.createPattern(canvas, 'repeat');
}
}
}
options.color = pattern;
ol.style.Stroke.call (this, options);
};
ol.inherits(ol.style.StrokePattern, ol.style.Stroke);
/**
* Clones the style.
* @return {ol.style.StrokePattern}
*/
ol.style.StrokePattern.prototype.clone = function()
{ var s = ol.style.Fill.prototype.clone.call(this);
s.canvas_ = this.canvas_;
return s;
};
/** Get canvas used as pattern
* @return {canvas}
*/
ol.style.StrokePattern.prototype.getImage = function()
{ return this.canvas_;
}
/** Get pattern
* @param {olx.style.FillPatternOption}
*/
ol.style.StrokePattern.prototype.getPattern_ = function(options)
{ var pat = ol.style.FillPattern.prototype.patterns[options.pattern]
|| ol.style.FillPattern.prototype.patterns.dot;
var d = Math.round(options.spacing)||10;
var d2 = Math.round(d/2)+0.5;
switch (options.pattern)
{ case 'dot':
case 'circle':
{ var size = options.size===0 ? 0 : options.size/2 || 2;
if (!options.angle)
{ pat.width = pat.height = d;
pat.circles = [[ d/2, d/2, size ]]
if (options.pattern=='circle')
{ pat.circles = pat.circles.concat([
[ d/2+d, d/2, size ],
[ d/2-d, d/2, size ],
[ d/2, d/2+d, size ],
[ d/2, d/2-d, size ],
[ d/2+d, d/2+d, size ],
[ d/2+d, d/2-d, size ],
[ d/2-d, d/2+d, size ],
[ d/2-d, d/2-d, size ] ])
};
}
else
{ d = pat.width = pat.height = Math.round(d*1.4);
pat.circles = [[ d/4, d/4, size ], [ 3*d/4, 3*d/4, size ]];
if (options.pattern=='circle')
{ pat.circles = pat.circles.concat([
[ d/4+d, d/4, size ],
[ d/4, d/4+d, size ],
[ 3*d/4-d, 3*d/4, size ],
[ 3*d/4, 3*d/4-d, size ],
[ d/4+d, d/4+d, size ],
[ 3*d/4-d, 3*d/4-d, size ] ]);
}
}
break;
}
case 'tile':
case 'square':
{ var size = options.size===0 ? 0 : options.size/2 || 2;
if (!options.angle)
{ pat.width = pat.height = d;
pat.lines = [[ d/2-size, d/2-size, d/2+size, d/2-size, d/2+size, d/2+size, d/2-size,d/2+size, d/2-size, d/2-size ]]
}
else
{ pat.width = pat.height = d;
//size *= Math.sqrt(2);
pat.lines = [[ d/2-size,d/2, d/2,d/2-size, d/2+size,d/2, d/2,d/2+size, d/2-size,d/2 ]]
}
if (options.pattern=='square') pat.repeat = [[0,0], [0,d], [d,0], [0,-d], [-d,0], [-d,-d], [d,d], [-d,d], [d,-d] ]
break;
}
case 'cross':
{ // Limit angle to 0 | 45
if (options.angle) options.angle = 45;
}
case 'hatch':
{ var a = Math.round(((options.angle||0)-90)%360);
if (a>180) a -= 360;
a *= Math.PI/180;
var cos = Math.cos(a);
var sin = Math.sin(a);
if (Math.abs(sin)<0.0001)
{ pat.width = pat.height = d;
pat.lines = [ [ 0,0.5, d, 0.5 ] ];
pat.repeat = [ [0,0], [0,d] ];
}
else if (Math.abs(cos)<0.0001)
{ pat.width = pat.height = d;
pat.lines = [ [ 0.5,0, 0.5, d] ];
pat.repeat = [ [0,0], [d,0] ];
if (options.pattern=='cross')
{ pat.lines.push ([ 0,0.5, d, 0.5 ]);
pat.repeat.push([0,d]);
}
}
else
{ var w = pat.width = Math.round(Math.abs(d/sin)) || 1;
var h = pat.height = Math.round(Math.abs(d/cos)) || 1;
if (options.pattern=='cross')
{ pat.lines = [ [-w,-h, 2*w,2*h], [2*w,-h, -w,2*h] ];
pat.repeat = [ [0,0] ];
}
else if (cos*sin>0)
{ pat.lines = [ [-w,-h, 2*w,2*h] ];
pat.repeat = [ [0,0], [w,0], [0,h] ];
}
else
{ pat.lines = [ [2*w,-h, -w,2*h] ];
pat.repeat = [ [0,0], [-w,0], [0,h] ];
}
}
pat.stroke = options.size===0 ? 0 : options.size||4;
}
default: break;
}
return pat
}