UNPKG

geoportal-extensions-openlayers

Version:

![GitHub package.json version](https://img.shields.io/github/package-json/v/IGNF/geoportal-extensions?filename=build%2Fscripts%2Frelease%2Fpackage-openlayers.json)

1,228 lines (1,140 loc) 87.8 kB
// import CSS import "../CSS/Controls/Drawing/GPdrawingOpenLayers.css"; // import OpenLayers import Control from "ol/control/Control"; import { unByKey as olObservableUnByKey } from "ol/Observable"; import Collection from "ol/Collection"; import Overlay from "ol/Overlay"; import { transform as olTransformProj } from "ol/proj"; import VectorSource from "ol/source/Vector"; import VectorLayer from "ol/layer/Vector"; import { Fill, Icon, Stroke, Style, Text, Circle } from "ol/style"; import { LineString, Point, Polygon, MultiLineString, MultiPoint, MultiPolygon } from "ol/geom"; // FIXME not include into ol/geom !? import LinearRing from "ol/geom/LinearRing"; import { Select as SelectInteraction, Modify as ModifyInteraction, Draw as DrawInteraction } from "ol/interaction"; import { singleClick as eventSingleClick, pointerMove as eventPointerMove } from "ol/events/condition"; import { getArea as olGetAreaSphere, getDistance as olGetDistanceSphere } from "ol/sphere"; // import local import Logger from "../../Common/Utils/LoggerByDefault"; import Interactions from "./Utils/Interactions"; import MarkersOther from "./Utils/MarkersOther"; import Draggable from "../../Common/Utils/Draggable"; import SelectorID from "../../Common/Utils/SelectorID"; import Color from "../../Common/Utils/ColorUtils"; // DOM import DrawingDOM from "../../Common/Controls/DrawingDOM"; // import local with ol dependencies import KMLExtended from "../Formats/KML"; import GeoJSONExtended from "../Formats/GeoJSON"; import GPXExtended from "../Formats/GPX"; import LayerSwitcher from "./LayerSwitcher"; var logger = Logger.getLogger("Drawing"); /** * @classdesc * * Drawing Control. * * @constructor * @alias ol.control.Drawing * @type {ol.control.Drawing} * @extends {ol.control.Control} * @param {Object} options - options for function call. * @param {Boolean} [options.collapsed = true] - Specify if Drawing control should be collapsed at startup. Default is true. * @param {Boolean} [options.draggable = false] - Specify if widget is draggable * @param {Object} [options.layer = {}] - Openlayers layer that will hosts created features. If none, an empty vector layer will be created. * @param {Object} [options.popup = {}] - Popup informations * @param {Boolean} [options.popup.display = true] - Specify if popup is displayed when create a drawing * @param {Function} [options.popup.function] - Function to display popup informations if you want to cutomise it. You may also provide your own function with params : {geomType / feature / saveFunc(message) / closeFunc()}. This function must return the DOM object of the popup content. * @param {Object} [options.layerDescription = {}] - Layer informations to be displayed in LayerSwitcher widget (only if a LayerSwitcher is also added to the map) * @param {String} [options.layerDescription.title = "Croquis"] - Layer title to be displayed in LayerSwitcher * @param {String} [options.layerDescription.description = "Mon croquis"] - Layer description to be displayed in LayerSwitcher * @param {Object} options.tools - Tools to display in the drawing toolbox. All by default. * @param {Boolean} [options.tools.points = true] - Display points drawing tool * @param {Boolean} [options.tools.lines = true] - Display lines drawing tool * @param {Boolean} [options.tools.polygons = true] - Display polygons drawing tool * @param {Boolean} [options.tools.holes = false] - Display polygons with holes drawing tool * @param {Boolean} [options.tools.text = true] - Display text drawing tool * @param {Boolean} [options.tools.remove = true] - Display feature removing tool * @param {Boolean} [options.tools.display = true] - Display style editing tool * @param {Boolean} [options.tools.tooltip = true] - Display text editing tool * @param {Boolean} [options.tools.edit = true] - Display editing tool * @param {Boolean} [options.tools.export = true] - Display exporting tool * @param {Boolean} [options.tools.measure = false] - Display measure drawing into popup info * @param {String} [options.labels] - Labels for Control * @param {String} [options.labels.control] - Label for Control * @param {String} [options.labels.points] - Label for points drawing tool * @param {String} [options.labels.lines] - Label for lines drawing tool * @param {String} [options.labels.polygons] - Label for polygons drawing tool * @param {String} [options.labels.holes] - Label for polygons with holes drawing tool * @param {String} [options.labels.text] - Label for text drawing tool * @param {String} [options.labels.edit] - Label for editing tool * @param {String} [options.labels.display] - Label for style editing tool * @param {String} [options.labels.tooltip] - Label for text editing tool * @param {String} [options.labels.remove] - Label for feature removing tool * @param {String} [options.labels.export] - Label for exporting tool. * @param {String} [options.labels.exportTitle] - Title for exporting tool. * @param {String} [options.labels.applyToObject] - Label for apply to object button. * @param {String} [options.labels.saveDescription] - Label for save description button. * @param {String} [options.labels.setAsDefault] - Label for set as default style button. * @param {String} [options.labels.strokeColor] - Label for stroke color. * @param {String} [options.labels.strokeWidth] - Label for stroke width. * @param {String} [options.labels.fillColor] - Label for fill color. * @param {String} [options.labels.fillOpacity] - Label for fillOpacity. * @param {String} [options.labels.markerSize] - Label for markerSize. * @param {Array.<Object>} [options.markersList = [{"src" : "", "anchor" : [0.5,1]}]] - List of markers src to be used for points with their anchor offsets See {@link http://openlayers.org/en/latest/apidoc/ol.style.Icon.html OpenLayers params} for anchor offset options. * @param {Object} options.defaultStyles - Default styles applying to geometries (labels, lines and polygons). * @param {String} [options.defaultStyles.textFillColor = "#000000"] - Text fill color for labels (RGB hex value). * @param {String} [options.defaultStyles.textStrokeColor = "#FFFFFF"] - Text surrounding color for labels (RGB hex value). * @param {String} [options.defaultStyles.strokeColor = "#ffcc33"] - Stroke color (RGB hex value). * @param {Number} [options.defaultStyles.strokeWidth = 2] - Stroke width in pixels. * @param {String} [options.defaultStyles.polyStrokeColor = "#ffcc33"] - Stroke color (RGB hex value) for polygons. * @param {Number} [options.defaultStyles.polyStrokeWidth = 2] - Stroke width in pixels for polygons. * @param {String} [options.defaultStyles.polyFillColor = "#ffffff"] - Polygons fill color (RGB hex value). * @param {Number} [options.defaultStyles.polyFillOpacity = 0.2] - Polygon fill opacity (alpha value between 0:transparent and 1:opaque). * @param {Object} options.cursorStyle - cursor (circle) style when drawing or editing. * @param {String} [options.cursorStyle.fillColor = "rgba(0, 153, 255, 1)"] - Cursor fill color. * @param {String} [options.cursorStyle.strokeColor = "#FFF"] - Cursor stroke color. * @param {String} [options.cursorStyle.strokeWidth = 1] - Cursor surrounding stroke width. * @param {String} [options.cursorStyle.radius = 6] - Cursor radius. * @example * var drawing = new ol.control.Drawing({ * collapsed : false, * draggable : true, * layerswitcher : { * title : "Dessins", * description : "Mes dessins..." * }, * markersList : [{ * src : "http://api.ign.fr/api/images/api/markers/marker_01.png", * anchor : [0.5, 1] * }], * defaultStyles : {}, * cursorStyle : {}, * tools : { * points : true, * lines : true, * polygons :true, * holes : true, * text : false, * remove : true, * display : true, * tooltip : true, * export : true, * measure : true * }, * popup : { * display : true, * function : function (params) { * var container = document.createElement("div"); * // - params.geomType; * // - params.feature; * // Les 2 fonctions ferment la popup avec ou sans sauvegarde des informations * // dans les properties de la feature (key : description) * // - params.saveFunc(message); * // - params.closeFunc(); * return container; * } * }); */ var Drawing = (function (Control) { /** * See {@link ol.control.Drawing} * @module Drawing * @alias module:~Controls/Drawing * @param {*} options - options * @example * import Drawing from "src/OpenLayers/Controls/Drawing" */ function Drawing (options) { options = options || {}; if (!(this instanceof Drawing)) { throw new TypeError("ERROR CLASS_CONSTRUCTOR"); } this._initialize(options); // init control DOM container this._container = this._initContainer(); // call ol.control.Control constructor Control.call(this, { element : this._container, target : options.target, render : options.render }); }; // Inherits from ol.control.Control // olInherits(Drawing, Control); if (Control) Drawing.__proto__ = Control; /** * Default tools to display for widget * * @private */ Drawing.DefaultTools = { points : true, lines : true, polygons : true, holes : false, text : true, remove : true, display : true, tooltip : true, edit : true, export : true, measure : false }; /** * Default labels for widget * * @private */ Drawing.DefaultLabels = { control : "Annoter la carte", creatingTools : "Outils de création", points : "Placer des points", lines : "Dessiner des lignes", polygons : "Dessiner des polygones", holes : "Créer des trous sur un polygone", text : "Ecrire sur la carte", editingTools : "Outils d'édition", edit : "Editer les tracés", display : "Modifier l'apparence des objets", tooltip : "Modifier les textes / infos-bulles", remove : "Supprimer des objets", export : "Exporter", exportTitle : "Exporter en KML", applyToObject : "Appliquer à l'objet", saveDescription : "Enregistrer", setAsDefault : "Définir par défaut", strokeColor : "Couleur du trait : ", strokeWidth : "Epaisseur du trait : ", fillColor : "Couleur de remplissage : ", fillOpacity : "Opacité du remplissage : ", markerSize : "Taille du pictogramme : ", markerColor : "Couleur du pictogramme : ", labelDisplay : "Afficher le label : " }; /** * Default styles applyied to drawn features. * * @private */ Drawing.DefaultStyles = { textFillColor : "#000000", textStrokeColor : "#FFFFFF", textStrokeWidth : 6, // INFO : cette option n'est pas surchargeable via les options du constructeur ! textIcon1x1 : { src : "", anchor : [0, 0] }, polyFillColor : "#ffffff", polyFillOpacity : 0.4, polyStrokeColor : "#ffcc33", polyStrokeWidth : 4, strokeColor : "#ffcc33", strokeWidth : 4, markerSize : 1, markerColor : "#ffcc33", // INFO : cette option n'est pas surchargeable via les options du constructeur ! labelDisplay : true }; /** * Default styles when drawing * * @private */ Drawing.DefaultCursorStyle = { radius : 6, strokeColor : "#FFF", strokeWidth : 1, fillColor : "rgba(0, 153, 255, 1)" }; /** * @lends module:Drawing */ Drawing.prototype = Object.create(Control.prototype, {}); /** * Copies all source object members to Drawing prototype * * @param {Object} source - source object whose properties will be copied. * @private */ Drawing.prototype.assign = function (source) { for (var prop in source) { if (source.hasOwnProperty(prop)) { this[prop] = source[prop]; } } }; Drawing.prototype.assign(DrawingDOM); /** * Constructor (alias) * * @private */ Drawing.prototype.constructor = Drawing; /** * Overload of {@link http://openlayers.org/en/latest/apidoc/ol.control.Control.html#setMap ol.control.Control.setMap()} method, called when control is added to or removed from map. * * @param {Object} map - {@link http://openlayers.org/en/latest/apidoc/ol.Map.html ol.Map} object. */ Drawing.prototype.setMap = function (map) { // call original setMap method Control.prototype.setMap.call(this, map); if (this.getMap() && this.eventKey) { olObservableUnByKey(this.eventKey); } // nothing else to do if map == null if (map == null) { return; } // mode "draggable" if (this.draggable) { Draggable.dragElement( this._drawingPanel, this._drawingPanelHeader, map.getTargetElement() ); } // mode "collapsed" if (!this.collapsed) { var inputShow = document.getElementById(this._addUID("GPshowDrawing")); inputShow.checked = "checked"; } if (this.layer) { // ajout du layer de dessin à la carte s'il n'y est pas déjà this.setLayer(this.layer); } // gestion des suppressions "externes" de la couche de dessin. this.eventKey = this.getMap().getLayers().on("remove", (evtRm) => { if (evtRm.element === this.layer) { // FIXME object comparison // found layer removed. this.layer = null; // on supprime l'interaction en cours si besoin if (this.interactionCurrent) { this.getMap().removeInteraction(this.interactionCurrent); this.interactionCurrent = null; } } }); }; /** * Export features of current drawing layer (KML by default). * * @returns {String} a representation of drawn features (KML, GPX or GeoJSON) or null if not possible. */ Drawing.prototype.exportFeatures = function () { var result = null; if (Control.prototype.getMap.call(this) == null) { logger.log("Impossible to export : control isn't attached to any map."); return result; } if (!this.layer) { logger.log("Impossible to export : no layer is hosting features."); return result; } if (!this.layer.getSource() || !this.layer.getSource().getFeatures() || !this.layer.getSource().getFeatures().length) { logger.log("Impossible to export : no features found."); return result; } // on invalide les features... if (this.featuresCollectionSelected) { this.featuresCollectionSelected.clear(); } var ClassName = null; switch (this.getExportFormat()) { case "KML": ClassName = new KMLExtended({ writeStyles : true }); break; case "GPX": ClassName = new GPXExtended({ // readExtensions : function (ext) {/* only extensions nodes from wpt, rte and trk can be processed */ } }); break; case "GEOJSON": ClassName = new GeoJSONExtended({}); break; default: break; } if (!ClassName) { logger.log("Impossible to export : format unknown !?"); return result; } var featProj = this.layer.getSource().getProjection(); featProj = featProj || this.getMap().getView().getProjection(); result = ClassName.writeFeatures(this.layer.getSource().getFeatures(), { dataProjection : "EPSG:4326", featureProjection : featProj }); return result; }; // ################################################################### // // #################### user interface methods ####################### // // ################################################################### // /** * Collapse or display control main container * * @param {Boolean} collapsed - True to collapse control, False to display it */ Drawing.prototype.setCollapsed = function (collapsed) { if (collapsed === undefined) { logger.error("[ERROR] Drawing:setCollapsed - missing collapsed parameter"); return; } if ((collapsed && this.collapsed) || (!collapsed && !this.collapsed)) { return; } // on simule l'ouverture du panneau après un click this.onShowDrawingClick(); this._showDrawingContainer.checked = !collapsed; }; /** * Setter for Export Name. * * @param {String} name - Export Name. By default, "Croquis". */ Drawing.prototype.setExportName = function (name) { this._exportName = name; }; /** * getter for Export Name. * * @returns {String} export name */ Drawing.prototype.getExportName = function () { return this._exportName; }; /** * Setter for Export format (KML, GPX or GeoJSON). * * @param {String} format - Export format. By default, "KML". */ Drawing.prototype.setExportFormat = function (format) { this._exportFormat = (format) ? format.toUpperCase() : "KML"; switch (format.toUpperCase()) { case "KML": this._exportExt = ".kml"; this._exportMimeType = "application/vnd.google-earth.kml+xml"; break; case "GPX": this._exportExt = ".gpx"; this._exportMimeType = "application/gpx+xml"; break; case "GEOJSON": this._exportExt = ".geojson"; this._exportMimeType = "application/geo+json"; break; default: // redefine format by default ! this._exportFormat = "KML"; break; } }; /** * getter for Export format. * * @returns {String} export format */ Drawing.prototype.getExportFormat = function () { return this._exportFormat; }; /** * Sets vector layer to hosts feature. * * @param {ol.layer.Vector} vlayer - vector layer */ Drawing.prototype.setLayer = function (vlayer) { if (!vlayer) { this.layer = null; return; } if (!(vlayer instanceof VectorLayer)) { logger.log("no valid layer given for hosting drawn features."); return; } // ajout du layer de dessin à la carte s'il n'y est pas déjà var layers = this.getMap().getLayers(); if (layers) { var found = false; layers.forEach((mapLayer) => { if (mapLayer === vlayer) { logger.trace("layer already in map. Not adding."); found = true; } }); // si le layer n'est pas sur la carte, on l'ajoute. if (!found) { this.getMap().addLayer(vlayer); } // style par defaut ! // application des styles ainsi que ceux par defaut de ol sur le layer vlayer.getSource().getFeatures().forEach((feature) => { var style = feature.getStyle(); if (typeof style !== "function") { return; } var featureStyleFunction = feature.getStyleFunction(); if (featureStyleFunction) { var styles = featureStyleFunction.call(this, feature, 0); if (styles && styles.length !== 0) { feature.setStyle((Array.isArray(styles)) ? styles[0] : styles); } } }); this.layer = vlayer; // Si un layer switcher est présent dans la carte, on lui affecte des informations pour cette couche this.getMap().getControls().forEach( (control) => { if (control instanceof LayerSwitcher) { // un layer switcher est présent dans la carte var layerId = this.layer.gpLayerId; // on n'ajoute des informations que s'il n'y en a pas déjà (si le titre est le numéro par défaut) if (control._layers[layerId].title === layerId) { control.addLayer( this.layer, { title : this.options.layerDescription.title, description : this.options.layerDescription.description } ); } } } ); } }; /** * Get vector layer * * @returns {Object} layer - isocurve layer */ Drawing.prototype.getLayer = function () { return this.layer; }; /** * Get container * * @returns {DOMElement} container */ Drawing.prototype.getContainer = function () { return this._container; }; // ################################################################### // // ######################## initialize control ####################### // // ################################################################### // /** * Gets marker options in options.markersList given its src. * * @param {String} src - marker image URI, * @returns {Object} markers options * @private */ Drawing.prototype._getsMarkersOptionsFromSrc = function (src) { var markerOptions = null; for (var i = 0; i < this.options.markersList.length; i++) { if (src && this.options.markersList[i].src === src) { markerOptions = this.options.markersList[i]; return markerOptions; } } return markerOptions; }; /** * Converts markerElement options into Openlayers IconStyles options. * * @param {Object} markerElement - marker element * @returns {Object} ol.Style.Icon constructor options. * @private */ Drawing.prototype._getIconStyleOptions = function (markerElement) { var iconOptions = {}; Object.keys(markerElement).forEach((key) => { switch (key) { case "src": case "size": case "scale": case "color": case "anchor": case "anchorOrigin": case "anchorXUnits": case "anchorYUnits": iconOptions[key] = markerElement[key]; break; } }); return iconOptions; }; /** * Initialize control (called by Drawing constructor) * * @method _initialize * @param {Object} options - control options (set by user) * @private */ Drawing.prototype._initialize = function (options) { // determination d'un uid this._uid = SelectorID.generate(); // export name / format / ... this._exportName = "Croquis"; this._exportFormat = "KML"; this._exportMimeType = "application/vnd.google-earth.kml+xml"; this._exportExt = ".kml"; options = options || {}; // Set default options this.options = options; if (!this.options.layerDescription) { this.options.layerDescription = { title : "Croquis", description : "Mon croquis" }; } // applying default tools if (!this.options.tools) { this.options.tools = {}; } Object.keys(Drawing.DefaultTools).forEach((key) => { if (!this.options.tools.hasOwnProperty(key)) { this.options.tools[key] = Drawing.DefaultTools[key]; } }); // styles par defaut lors du dessin if (!this.options.cursorStyle) { this.options.cursorStyle = {}; } Object.keys(Drawing.DefaultCursorStyle).forEach((key) => { if (!this.options.cursorStyle.hasOwnProperty(key)) { this.options.cursorStyle[key] = Drawing.DefaultCursorStyle[key]; } }); this.options.collapsed = (options.collapsed !== undefined) ? options.collapsed : true; /** {Boolean} specify if Drawing control is collapsed (true) or not (false) */ this.collapsed = this.options.collapsed; this.options.draggable = (options.draggable !== undefined) ? options.draggable : false; /** {Boolean} specify if Drawing control is draggable (true) or not (false) */ this.draggable = this.options.draggable; this.options.markersList = options.markersList || MarkersOther["drawing_api"]; // applying default labels if (!this.options.labels) { this.options.labels = {}; } Object.keys(Drawing.DefaultLabels).forEach((key) => { if (!this.options.labels.hasOwnProperty(key)) { this.options.labels[key] = Drawing.DefaultLabels[key]; } }); // applying default styles if (!this.options.defaultStyles) { this.options.defaultStyles = {}; } Object.keys(Drawing.DefaultStyles).forEach((key) => { if (!options.defaultStyles.hasOwnProperty(key)) { this.options.defaultStyles[key] = Drawing.DefaultStyles[key]; return; } if (key === "polyFillOpacity" && (options.defaultStyles[key] < 0 || options.defaultStyles[key] > 1)) { logger.log("Wrong value (" + options.defaultStyles[key] + ") for defaultStyles.polyFillOpactity. Must be between 0 and 1"); this.options.defaultStyles[key] = Drawing.DefaultStyles[key]; return; } if (key === "strokeWidth" || key === "polyStrokeWidth") { var intValue = parseInt(options.defaultStyles[key], 10); if (isNaN(intValue) || intValue < 0) { logger.log("Wrong value (" + options.defaultStyles[key] + ") for defaultStyles.strokeWidth. Must be a positive interger value."); this.options.defaultStyles[key] = Drawing.DefaultStyles[key]; return; } this.options.defaultStyles[key] = intValue; } if (key === "markerSize") { var floatValue = parseFloat(options.defaultStyles[key]); if (isNaN(floatValue) || floatValue < 0) { logger.log("Wrong value (" + options.defaultStyles[key] + ") for defaultStyles.markerSize. Must be a positive value."); this.options.defaultStyles[key] = Drawing.DefaultStyles[key]; return; } this.options.defaultStyles[key] = floatValue; } }); this.interactionCurrent = null; this.interactionSelectEdit = null; this.featuresCollectionSelected = null; this.stylingOvl = null; this.popupOvl = null; this.layer = null; if (this.options.layer && this.options.layer instanceof VectorLayer) { this.layer = this.options.layer; } // detection du support : desktop ou tactile // FIXME : utile ? this._isDesktop = this._detectSupport(); // applying default popup if (!this.options.popup) { this.options.popup = { display : true, apply : null }; } }; /** * Creates empty layer to host features * * @private */ Drawing.prototype._createEmptyLayer = function () { var features = new Collection(); var layer = new VectorLayer({ source : new VectorSource({ features : features }) }); // on rajoute le champ gpResultLayerId permettant d'identifier une couche crée par le composant. layer.gpResultLayerId = "drawing"; // on le rajoute au controle (et à la carte) this.setLayer(layer); }; /** * this method is called by the constructor. * this information is useful to switch to touch mode. * Detection : test for desktop or tactile * * @method _detectSupport * * @returns {Boolean} is desktop * @private */ Drawing.prototype._detectSupport = function () { // TODO // Choix de gérer la détection dans le code du composant au lieu du DOM car : // Utilisation de l'implémentation Leaflet // http://leafletjs.com/reference.html#browser var isDesktop = true; var userAgent = window.navigator.userAgent.toLowerCase(); if (userAgent.indexOf("iphone") !== -1 || userAgent.indexOf("ipod") !== -1 || userAgent.indexOf("ipad") !== -1 || userAgent.indexOf("android") !== -1 || userAgent.indexOf("mobile") !== -1 || userAgent.indexOf("blackberry") !== -1 || userAgent.indexOf("tablet") !== -1 || userAgent.indexOf("phone") !== -1 || userAgent.indexOf("touch") !== -1) { isDesktop = false; } if (userAgent.indexOf("msie") !== -1 || userAgent.indexOf("trident") !== -1) { isDesktop = true; } return isDesktop; }; // ################################################################### // // ######################## methods handle dom ####################### // // ################################################################### // /** * Create control main container (called by Drawing constructor) * * @method _initContainer * * @returns {DOMElement} DOM element * @private */ Drawing.prototype._initContainer = function () { // creation du container principal var container = this._createMainContainerElement(); var inputShow = this._showDrawingContainer = this._createShowDrawingElement(); container.appendChild(inputShow); var picto = this._createShowDrawingPictoElement(); container.appendChild(picto); var panel = this._drawingPanel = this._createDrawingPanelElement(); var header = this._drawingPanelHeader = this._createDrawingPanelHeaderElement(); panel.appendChild(header); var tools = this._createDrawingToolsSections(); for (var i = 0; i < tools.length; i++) { panel.appendChild(tools[i]); } container.appendChild(panel); return container; }; // ################################################################### // // ##################### handlers events to control ################## // // ################################################################### // /** * Callback de fin de dessin de geometrie * @param {Object} feature - ol feature * @param {String} geomType - geometry type * @param {Boolean} clean - clean last feature * * @private */ Drawing.prototype._drawEndFeature = function (feature, geomType) { // application des styles par defaut. var style = null; switch (geomType) { case "Point": style = new Style({ image : new Icon(this._getIconStyleOptions(this.options.markersList[0])) }); break; case "LineString": style = new Style({ stroke : new Stroke({ color : this.options.defaultStyles.strokeColor, width : this.options.defaultStyles.strokeWidth }) }); break; case "Polygon": style = new Style({ fill : new Fill({ color : Color.hexToRgba( this.options.defaultStyles.polyFillColor, this.options.defaultStyles.polyFillOpacity ) }), stroke : new Stroke({ color : this.options.defaultStyles.polyStrokeColor, width : this.options.defaultStyles.polyStrokeWidth }) }); break; } feature.setStyle(style); // gestion des mesures this._updateMeasure(feature, geomType); if (this.options.popup.display) { // creation overlay pour saisie du label // contexte var context = this; /** * Enregistrement de la valeur saisie dans l'input. * * @param {String} key - clef de l'attribut. * @param {String} value - valeur de l'attribut. * @param {Boolean} save - true si on garde le label. */ var setAttValue = function (key, value, save) { context.getMap().removeOverlay(context.popupOvl); context.popupOvl = null; if (save && value && value.trim().length > 0) { var obj = {}; obj[key] = value.replace(/\n/g, "<br>"); feature.setProperties(obj); } }; var popup = null; var popupByDefault = true; var displayFunction = this.options.popup.function; if (displayFunction && typeof displayFunction === "function") { // la sauvegarde et la fermeture sont des actions à implementer par l'utilisateur // par contre, la destruction est à gerer en interne popup = displayFunction.call(context, { feature : feature, geomType : geomType, closeFunc : function () { setAttValue(null, false); }, saveFunc : function (message) { setAttValue(message, true); } }); if (popup) { // on est sûr que la popup customisée existe, // donc on n'utilise pas celle par defaut... popupByDefault = false; // FIXME comment forcer le focus sur une div ? popup.tabIndex = -1; // hack sur le focus sur une div ? popup.onblur = function () { context.getMap().removeOverlay(context.popupOvl); context.popupOvl = null; }; } } // use popup by default if (popupByDefault) { // function by default popup = this._createLabelDiv({ applyFunc : setAttValue, inputId : this._addUID("att-input"), placeholder : "Saisir une description...", measure : (this.options.tools.measure) ? feature.getProperties().measure : null, geomType : geomType, key : "description" }); } // un peu de menage... if (this.popupOvl) { this.getMap().removeOverlay(this.popupOvl); this.popupOvl = null; } // creation de l'overlay this.popupOvl = new Overlay({ element : popup, // FIXME : autres valeurs. positioning : "top-center" // stopEvent : false }); this.getMap().addOverlay(this.popupOvl); var geomExtent = feature.getGeometry().getExtent(); this.popupOvl.setPosition([ (geomExtent[0] + geomExtent[2]) / 2, (geomExtent[1] + geomExtent[3]) / 2 ]); if (document.getElementById(this._addUID("att-input"))) { document.getElementById(this._addUID("att-input")).focus(); } } }; /** * Creates Interaction for features removal. * * @returns {SelectInteraction} created interaction. * @private */ Drawing.prototype._createRemoveInteraction = function () { var interaction = new SelectInteraction({ // features : this.layer.getSource().getFeaturesCollection(), layers : [this.layer], style : false }); interaction.on("select", (seEv) => { if (!seEv || !seEv.selected || seEv.selected.length === 0) { return; } this.layer.getSource().removeFeature(seEv.selected[0]); // suppression puis rajout de l'interaction pour appliquer le changement tout de suite... this.getMap().removeInteraction(this.interactionCurrent); this.interactionCurrent = this._createRemoveInteraction(); this.getMap().addInteraction(this.interactionCurrent); }); return interaction; }; /** * Creates Interaction for features style definition. * * @returns {ol.interaction.Select} created interaction. * @private */ Drawing.prototype._createStylingInteraction = function () { var interaction = new SelectInteraction({ layers : [this.layer], style : false }); interaction.on("select", (seEv) => { // suppression de toute popup existante if (this.stylingOvl) { this.getMap().removeOverlay(this.stylingOvl); } if (!seEv || !seEv.selected || seEv.selected.length === 0) { return; } var valuesColor = null; var hexColor = null; var popupOvl = null; var geomType = null; var initValues = {}; // FIXME // l'appel feature.getStyle() est parfois nul pour des geometries Point // avec un style par defaut ! var geom = seEv.selected[0].getGeometry(); var style = seEv.selected[0].getStyle(); if (geom instanceof Point || geom instanceof MultiPoint) { // INFO // on determine si c'est un marker (ou cercle), un label ou les 2. // un label a un pixel transparent comme icone if (style && style.getImage() && typeof style.getImage().getSrc === "function" && style.getImage().getSrc() !== this.options.defaultStyles.textIcon1x1.src) { geomType = "Point"; // on traite un marker // mais si c'est un cercle !? if (typeof style.getImage().getSrc === "function") { initValues.markerSrc = style.getImage().getSrc(); initValues.markerSize = style.getImage().getScale() || 1; initValues.markerAnchor = style.getImage().getAnchor(); if (style.getImage().getColor()) { valuesColor = style.getImage().getColor(); if (Array.isArray(valuesColor)) { // FIXME Array !? valuesColor = "rgba(" + valuesColor.join() + ")"; } else { initValues.markerColor = valuesColor; } hexColor = Color.isRGB(valuesColor) ? Color.rgbaToHex(valuesColor) : { hex : valuesColor, opacity : 1 }; initValues.markerColor = hexColor.hex; initValues.markerOpacity = hexColor.opacity; } else { initValues.markerColor = this.options.markersList[0].color || "#ffffff"; } } else { initValues.markerSrc = this.options.markersList[0].src; initValues.markerSize = this.options.markersList[0].scale || 1; initValues.markerColor = this.options.markersList[0].color || "#ffffff"; initValues.markerAnchor = this.options.markersList[0].anchor; } initValues.markerCustom = !(this._getsMarkersOptionsFromSrc(initValues.markerSrc)); } if (style && style.getText()) { var labelName = seEv.selected[0].getProperties().name; if (labelName) { // test si on a un marker avec un label geomType = (geomType === "Point") ? "Point&Text" : "Text"; if (style.getText().getStroke() && style.getText().getStroke().getColor()) { valuesColor = style.getText().getStroke().getColor(); if (Array.isArray(valuesColor)) { // FIXME Array !? valuesColor = "rgba(" + valuesColor.join() + ")"; } else { initValues.strokeColor = valuesColor; } hexColor = Color.isRGB(valuesColor) ? Color.rgbaToHex(valuesColor) : { hex : valuesColor, opacity : 1 }; initValues.strokeColor = hexColor.hex; initValues.strokeOpacity = hexColor.opacity; } if (style.getText().getStroke() && style.getText().getStroke().getWidth()) { initValues.strokeWidth = style.getText().getStroke().getWidth(); } if (style.getText().getFill() && style.getText().getFill().getColor()) { valuesColor = style.getText().getFill().getColor(); if (Array.isArray(valuesColor)) { valuesColor = "rgba(" + valuesColor.join() + ")"; } else { initValues.fillColor = valuesColor; } hexColor = Color.isRGB(valuesColor) ? Color.rgbaToHex(valuesColor) : { hex : valuesColor, opacity : 1 }; initValues.fillColor = hexColor.hex; initValues.fillOpacity = hexColor.opacity; } initValues.strokeColor = initValues.hasOwnProperty("strokeColor") ? initValues.strokeColor : this.options.defaultStyles.textStrokeColor; initValues.strokeWidth = initValues.hasOwnProperty("strokeWidth") ? initValues.strokeWidth : this.options.defaultStyles.textStrokeWidth; initValues.fillColor = initValues.hasOwnProperty("fillColor") ? initValues.fillColor : this.options.defaultStyles.textFillColor; // Par defaut, pour un marker avec un label, on affiche le label si le tag "name" est renseigné. if (geomType === "Point&Text") { var value = style.getText().getText(); if (!value) { style.getText().setText(labelName); } var checked = seEv.selected[0].get("checked"); initValues.labelDisplay = (checked === undefined) ? this.options.defaultStyles.labelDisplay : checked; } } } } else if (geom instanceof LineString || geom instanceof MultiLineString) { geomType = "Line"; if (style && style.getStroke()) { if (style.getStroke().getWidth()) { initValues.strokeWidth = style.getStroke().getWidth(); } if (style.getStroke().getColor()) { valuesColor = style.getStroke().getColor(); if (Array.isArray(valuesColor)) { valuesColor = "rgba(" + valuesColor.join() + ")"; } else { initValues.strokeColor = valuesColor; } hexColor = Color.isRGB(valuesColor) ? Color.rgbaToHex(valuesColor) : { hex : valuesColor, opacity : 1 }; initValues.strokeColor = hexColor.hex; initValues.strokeOpacity = hexColor.opacity; } } initValues.strokeWidth = initValues.hasOwnProperty("strokeWidth") ? initValues.strokeWidth : this.options.defaultStyles.strokeWidth; initValues.strokeColor = initValues.hasOwnProperty("strokeColor") ? initValues.strokeColor : this.options.defaultStyles.strokeColor; } else if (geom instanceof Polygon || geom instanceof MultiPolygon) { geomType = "Polygon"; if (style && style.getStroke()) { if (style.getStroke().getWidth()) { initValues.strokeWidth = style.getStroke().getWidth(); } if (style.getStroke().getColor()) { valuesColor = style.getStroke().getColor(); if (Array.isArray(valuesColor)) { valuesColor = "rgba(" + valuesColor.join() + ")"; } else { initValues.strokeColor = valuesColor; } hexColor = Color.isRGB(valuesColor) ? Color.rgbaToHex(valuesColor) : { hex : valuesColor, opacity : 1 }; initValues.strokeColor = hexColor.hex; initValues.strokeOpacity = hexColor.opacity; } } if (style && style.getFill()) { if (style.getFill().getColor()) { valuesColor = style.getFill().getColor(); if (Array.isArray(valuesColor)) { valuesColor = "rgba(" + valuesColor.join() + ")"; } else { initValues.fillColor = valuesColor; } hexColor = Color.isRGB(valuesColor) ? Color.rgbaToHex(valuesColor) : { hex : valuesColor, opacity : 1 }; initValues.fillColor = hexColor.hex; initValues.fillOpacity = hexColor.opacity; } } initValues.strokeWidth = initValues.hasOwnProperty("strokeWidth") ? initValues.strokeWidth : this.options.defaultStyles.polyStrokeWidth; initValues.strokeColor = initValues.hasOwnProperty("strokeColor") ? initValues.strokeColor : this.options.defaultStyles.polyStrokeColor; initValues.fillColor = initValues.hasOwnProperty("fillColor") ? initValues.fillColor : this.options.defaultStyles.polyFillColor; initValues.fillOpacity = initValues.hasOwnProperty("fillOpacity") ? initValues.fillOpacity : this.options.defaultStyles.polyFillOpacity; } if (!geomType) { logger.log("Unhandled geometry type for styling."); return; } var dtObj = this; /** * function called when apply button is pressed. * * @param {String} action - "apply" (to selected object), "default" (set as default), "cancel" (do nothing). */ var applyStyle = function (action) { if (action === "cancel") { dtObj.getMap().removeOverlay(popupOvl); return; } var setDefault = action !== "apply"; var fillColorElem = document.getElementById(dtObj._addUID("fillColor")); var fillOpacityElem = document.getElementById(dtObj._addUID("fillOpacity")); var strokeColorElem = document.getElementByI