// FIXME http://earth.google.com/kml/1.0 namespace? // FIXME why does node.getAttribute return an unknown type? // FIXME text // FIXME serialize arbitrary feature properties // FIXME don't parse style if extractStyles is false goog.provide('ol.format.KML'); goog.require('goog.Uri'); goog.require('goog.array'); goog.require('goog.asserts'); goog.require('goog.dom.NodeType'); goog.require('goog.math'); goog.require('goog.object'); goog.require('goog.string'); goog.require('ol'); goog.require('ol.Feature'); goog.require('ol.FeatureStyleFunction'); goog.require('ol.array'); goog.require('ol.color'); goog.require('ol.format.Feature'); goog.require('ol.format.XMLFeature'); goog.require('ol.format.XSD'); goog.require('ol.geom.Geometry'); goog.require('ol.geom.GeometryCollection'); goog.require('ol.geom.GeometryLayout'); goog.require('ol.geom.GeometryType'); goog.require('ol.geom.LineString'); goog.require('ol.geom.LinearRing'); goog.require('ol.geom.MultiLineString'); goog.require('ol.geom.MultiPoint'); goog.require('ol.geom.MultiPolygon'); goog.require('ol.geom.Point'); goog.require('ol.geom.Polygon'); goog.require('ol.proj'); goog.require('ol.style.Fill'); goog.require('ol.style.Icon'); goog.require('ol.style.IconAnchorUnits'); goog.require('ol.style.IconOrigin'); goog.require('ol.style.Image'); goog.require('ol.style.Stroke'); goog.require('ol.style.Style'); goog.require('ol.style.Text'); goog.require('ol.xml'); /** * @typedef {{x: number, xunits: (ol.style.IconAnchorUnits|undefined), * y: number, yunits: (ol.style.IconAnchorUnits|undefined)}} */ ol.format.KMLVec2_; /** * @typedef {{flatCoordinates: Array., * whens: Array.}} */ ol.format.KMLGxTrackObject_; /** * @classdesc * Feature format for reading and writing data in the KML format. * * @constructor * @extends {ol.format.XMLFeature} * @param {olx.format.KMLOptions=} opt_options Options. * @api stable */ ol.format.KML = function(opt_options) { var options = opt_options ? opt_options : {}; goog.base(this); /** * @inheritDoc */ this.defaultDataProjection = ol.proj.get('EPSG:4326'); /** * @private * @type {Array.} */ this.defaultStyle_ = options.defaultStyle ? options.defaultStyle : ol.format.KML.DEFAULT_STYLE_ARRAY_; /** * @private * @type {boolean} */ this.extractStyles_ = options.extractStyles !== undefined ? options.extractStyles : true; /** * @private * @type {Object.|string)>} */ this.sharedStyles_ = {}; }; goog.inherits(ol.format.KML, ol.format.XMLFeature); /** * @const * @type {Array.} * @private */ ol.format.KML.EXTENSIONS_ = ['.kml']; /** * @const * @type {Array.} * @private */ ol.format.KML.GX_NAMESPACE_URIS_ = [ 'http://www.google.com/kml/ext/2.2' ]; /** * @const * @type {Array.} * @private */ ol.format.KML.NAMESPACE_URIS_ = [ null, 'http://earth.google.com/kml/2.0', 'http://earth.google.com/kml/2.1', 'http://earth.google.com/kml/2.2', 'http://www.opengis.net/kml/2.2' ]; /** * @const * @type {string} * @private */ ol.format.KML.SCHEMA_LOCATION_ = 'http://www.opengis.net/kml/2.2 ' + 'https://developers.google.com/kml/schema/kml22gx.xsd'; /** * @const * @type {ol.Color} * @private */ ol.format.KML.DEFAULT_COLOR_ = [255, 255, 255, 1]; /** * @const * @type {ol.style.Fill} * @private */ ol.format.KML.DEFAULT_FILL_STYLE_ = new ol.style.Fill({ color: ol.format.KML.DEFAULT_COLOR_ }); /** * @const * @type {ol.Size} * @private */ ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_ = [20, 2]; // FIXME maybe [8, 32] ? /** * @const * @type {ol.style.IconAnchorUnits} * @private */ ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_X_UNITS_ = ol.style.IconAnchorUnits.PIXELS; /** * @const * @type {ol.style.IconAnchorUnits} * @private */ ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_Y_UNITS_ = ol.style.IconAnchorUnits.PIXELS; /** * @const * @type {ol.Size} * @private */ ol.format.KML.DEFAULT_IMAGE_STYLE_SIZE_ = [64, 64]; /** * @const * @type {string} * @private */ ol.format.KML.DEFAULT_IMAGE_STYLE_SRC_ = 'https://maps.google.com/mapfiles/kml/pushpin/ylw-pushpin.png'; /** * @const * @type {ol.style.Image} * @private */ ol.format.KML.DEFAULT_IMAGE_STYLE_ = new ol.style.Icon({ anchor: ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_, anchorOrigin: ol.style.IconOrigin.BOTTOM_LEFT, anchorXUnits: ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_X_UNITS_, anchorYUnits: ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_Y_UNITS_, crossOrigin: 'anonymous', rotation: 0, scale: 0.5, size: ol.format.KML.DEFAULT_IMAGE_STYLE_SIZE_, src: ol.format.KML.DEFAULT_IMAGE_STYLE_SRC_ }); /** * @const * @type {ol.style.Stroke} * @private */ ol.format.KML.DEFAULT_STROKE_STYLE_ = new ol.style.Stroke({ color: ol.format.KML.DEFAULT_COLOR_, width: 1 }); /** * @const * @type {ol.style.Text} * @private */ ol.format.KML.DEFAULT_TEXT_STYLE_ = new ol.style.Text({ font: 'normal 16px Helvetica', fill: ol.format.KML.DEFAULT_FILL_STYLE_, stroke: ol.format.KML.DEFAULT_STROKE_STYLE_, scale: 1 }); /** * @const * @type {ol.style.Style} * @private */ ol.format.KML.DEFAULT_STYLE_ = new ol.style.Style({ fill: ol.format.KML.DEFAULT_FILL_STYLE_, image: ol.format.KML.DEFAULT_IMAGE_STYLE_, text: ol.format.KML.DEFAULT_TEXT_STYLE_, stroke: ol.format.KML.DEFAULT_STROKE_STYLE_, zIndex: 0 }); /** * @const * @type {Array.} * @private */ ol.format.KML.DEFAULT_STYLE_ARRAY_ = [ol.format.KML.DEFAULT_STYLE_]; /** * @const * @type {Object.} * @private */ ol.format.KML.ICON_ANCHOR_UNITS_MAP_ = { 'fraction': ol.style.IconAnchorUnits.FRACTION, 'pixels': ol.style.IconAnchorUnits.PIXELS }; /** * @param {Array.|undefined} style Style. * @param {string} styleUrl Style URL. * @param {Array.} defaultStyle Default style. * @param {Object.|string)>} sharedStyles * Shared styles. * @return {ol.FeatureStyleFunction} Feature style function. * @private */ ol.format.KML.createFeatureStyleFunction_ = function( style, styleUrl, defaultStyle, sharedStyles) { return ( /** * @param {number} resolution Resolution. * @return {Array.} Style. * @this {ol.Feature} */ function(resolution) { if (style) { return style; } if (styleUrl) { return ol.format.KML.findStyle_(styleUrl, defaultStyle, sharedStyles); } return defaultStyle; }); }; /** * @param {Array.|string|undefined} styleValue Style value. * @param {Array.} defaultStyle Default style. * @param {Object.|string)>} sharedStyles * Shared styles. * @return {Array.} Style. * @private */ ol.format.KML.findStyle_ = function(styleValue, defaultStyle, sharedStyles) { if (goog.isArray(styleValue)) { return styleValue; } else if (goog.isString(styleValue)) { // KML files in the wild occasionally forget the leading `#` on styleUrls // defined in the same document. Add a leading `#` if it enables to find // a style. if (!(styleValue in sharedStyles) && ('#' + styleValue in sharedStyles)) { styleValue = '#' + styleValue; } return ol.format.KML.findStyle_( sharedStyles[styleValue], defaultStyle, sharedStyles); } else { return defaultStyle; } }; /** * @param {Node} node Node. * @private * @return {ol.Color|undefined} Color. */ ol.format.KML.readColor_ = function(node) { var s = ol.xml.getAllTextContent(node, false); // The KML specification states that colors should not include a leading `#` // but we tolerate them. var m = /^\s*#?\s*([0-9A-Fa-f]{8})\s*$/.exec(s); if (m) { var hexColor = m[1]; return [ parseInt(hexColor.substr(6, 2), 16), parseInt(hexColor.substr(4, 2), 16), parseInt(hexColor.substr(2, 2), 16), parseInt(hexColor.substr(0, 2), 16) / 255 ]; } else { return undefined; } }; /** * @param {Node} node Node. * @private * @return {Array.|undefined} Flat coordinates. */ ol.format.KML.readFlatCoordinates_ = function(node) { var s = ol.xml.getAllTextContent(node, false); var flatCoordinates = []; // The KML specification states that coordinate tuples should not include // spaces, but we tolerate them. var re = /^\s*([+\-]?\d*\.?\d+(?:e[+\-]?\d+)?)\s*,\s*([+\-]?\d*\.?\d+(?:e[+\-]?\d+)?)(?:\s*,\s*([+\-]?\d*\.?\d+(?:e[+\-]?\d+)?))?\s*/i; var m; while ((m = re.exec(s))) { var x = parseFloat(m[1]); var y = parseFloat(m[2]); var z = m[3] ? parseFloat(m[3]) : 0; flatCoordinates.push(x, y, z); s = s.substr(m[0].length); } if (s !== '') { return undefined; } return flatCoordinates; }; /** * @param {Node} node Node. * @private * @return {string|undefined} Style URL. */ ol.format.KML.readStyleUrl_ = function(node) { var s = goog.string.trim(ol.xml.getAllTextContent(node, false)); if (goog.isDefAndNotNull(node.baseURI)) { return goog.Uri.resolve(node.baseURI, s).toString(); } else { return s; } }; /** * @param {Node} node Node. * @private * @return {string} URI. */ ol.format.KML.readURI_ = function(node) { var s = ol.xml.getAllTextContent(node, false); if (goog.isDefAndNotNull(node.baseURI)) { return goog.Uri.resolve(node.baseURI, goog.string.trim(s)).toString(); } else { return goog.string.trim(s); } }; /** * @param {Node} node Node. * @private * @return {ol.format.KMLVec2_} Vec2. */ ol.format.KML.readVec2_ = function(node) { var xunits = node.getAttribute('xunits'); var yunits = node.getAttribute('yunits'); return { x: parseFloat(node.getAttribute('x')), xunits: ol.format.KML.ICON_ANCHOR_UNITS_MAP_[xunits], y: parseFloat(node.getAttribute('y')), yunits: ol.format.KML.ICON_ANCHOR_UNITS_MAP_[yunits] }; }; /** * @param {Node} node Node. * @private * @return {number|undefined} Scale. */ ol.format.KML.readScale_ = function(node) { var number = ol.format.XSD.readDecimal(node); if (number !== undefined) { return Math.sqrt(number); } else { return undefined; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {Array.|string|undefined} StyleMap. */ ol.format.KML.readStyleMapValue_ = function(node, objectStack) { return ol.xml.pushParseAndPop( /** @type {Array.|string|undefined} */ (undefined), ol.format.KML.STYLE_MAP_PARSERS_, node, objectStack); }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.IconStyleParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be an ELEMENT'); goog.asserts.assert(node.localName == 'IconStyle', 'localName should be IconStyle'); // FIXME refreshMode // FIXME refreshInterval // FIXME viewRefreshTime // FIXME viewBoundScale // FIXME viewFormat // FIXME httpQuery var object = ol.xml.pushParseAndPop( {}, ol.format.KML.ICON_STYLE_PARSERS_, node, objectStack); if (!object) { return; } var styleObject = /** @type {Object} */ (objectStack[objectStack.length - 1]); goog.asserts.assert(goog.isObject(styleObject), 'styleObject should be an Object'); var IconObject = /** @type {Object} */ (goog.object.get(object, 'Icon', {})); var src; var href = /** @type {string|undefined} */ (IconObject['href']); if (href) { src = href; } else { src = ol.format.KML.DEFAULT_IMAGE_STYLE_SRC_; } var anchor, anchorXUnits, anchorYUnits; var hotSpot = /** @type {ol.format.KMLVec2_|undefined} */ (object['hotSpot']); if (hotSpot) { anchor = [hotSpot.x, hotSpot.y]; anchorXUnits = hotSpot.xunits; anchorYUnits = hotSpot.yunits; } else if (src === ol.format.KML.DEFAULT_IMAGE_STYLE_SRC_) { anchor = ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_; anchorXUnits = ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_X_UNITS_; anchorYUnits = ol.format.KML.DEFAULT_IMAGE_STYLE_ANCHOR_Y_UNITS_; } else if (/^http:\/\/maps\.(?:google|gstatic)\.com\//.test(src)) { anchor = [0.5, 0]; anchorXUnits = ol.style.IconAnchorUnits.FRACTION; anchorYUnits = ol.style.IconAnchorUnits.FRACTION; } var offset; var x = /** @type {number|undefined} */ (IconObject['x']); var y = /** @type {number|undefined} */ (IconObject['y']); if (x !== undefined && y !== undefined) { offset = [x, y]; } var size; var w = /** @type {number|undefined} */ (IconObject['w']); var h = /** @type {number|undefined} */ (IconObject['h']); if (w !== undefined && h !== undefined) { size = [w, h]; } var rotation; var heading = /** @type {number} */ (object['heading']); if (heading !== undefined) { rotation = goog.math.toRadians(heading); } var scale = /** @type {number|undefined} */ (object['scale']); if (src == ol.format.KML.DEFAULT_IMAGE_STYLE_SRC_) { size = ol.format.KML.DEFAULT_IMAGE_STYLE_SIZE_; } var imageStyle = new ol.style.Icon({ anchor: anchor, anchorOrigin: ol.style.IconOrigin.BOTTOM_LEFT, anchorXUnits: anchorXUnits, anchorYUnits: anchorYUnits, crossOrigin: 'anonymous', // FIXME should this be configurable? offset: offset, offsetOrigin: ol.style.IconOrigin.BOTTOM_LEFT, rotation: rotation, scale: scale, size: size, src: src }); styleObject['imageStyle'] = imageStyle; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.LabelStyleParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'LabelStyle', 'localName should be LabelStyle'); // FIXME colorMode var object = ol.xml.pushParseAndPop( {}, ol.format.KML.LABEL_STYLE_PARSERS_, node, objectStack); if (!object) { return; } var styleObject = objectStack[objectStack.length - 1]; goog.asserts.assert(goog.isObject(styleObject), 'styleObject should be an Object'); var textStyle = new ol.style.Text({ fill: new ol.style.Fill({ color: /** @type {ol.Color} */ (goog.object.get(object, 'color', ol.format.KML.DEFAULT_COLOR_)) }), scale: /** @type {number|undefined} */ (object['scale']) }); styleObject['textStyle'] = textStyle; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.LineStyleParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'LineStyle', 'localName should be LineStyle'); // FIXME colorMode // FIXME gx:outerColor // FIXME gx:outerWidth // FIXME gx:physicalWidth // FIXME gx:labelVisibility var object = ol.xml.pushParseAndPop( {}, ol.format.KML.LINE_STYLE_PARSERS_, node, objectStack); if (!object) { return; } var styleObject = objectStack[objectStack.length - 1]; goog.asserts.assert(goog.isObject(styleObject), 'styleObject should be an Object'); var strokeStyle = new ol.style.Stroke({ color: /** @type {ol.Color} */ (goog.object.get(object, 'color', ol.format.KML.DEFAULT_COLOR_)), width: /** @type {number} */ (goog.object.get(object, 'width', 1)) }); styleObject['strokeStyle'] = strokeStyle; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.PolyStyleParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'PolyStyle', 'localName should be PolyStyle'); // FIXME colorMode var object = ol.xml.pushParseAndPop( {}, ol.format.KML.POLY_STYLE_PARSERS_, node, objectStack); if (!object) { return; } var styleObject = objectStack[objectStack.length - 1]; goog.asserts.assert(goog.isObject(styleObject), 'styleObject should be an Object'); var fillStyle = new ol.style.Fill({ color: /** @type {ol.Color} */ (goog.object.get(object, 'color', ol.format.KML.DEFAULT_COLOR_)) }); styleObject['fillStyle'] = fillStyle; var fill = /** @type {boolean|undefined} */ (object['fill']); if (fill !== undefined) { styleObject['fill'] = fill; } var outline = /** @type {boolean|undefined} */ (object['outline']); if (outline !== undefined) { styleObject['outline'] = outline; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {Array.} LinearRing flat coordinates. */ ol.format.KML.readFlatLinearRing_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'LinearRing', 'localName should be LinearRing'); return /** @type {Array.} */ (ol.xml.pushParseAndPop( null, ol.format.KML.FLAT_LINEAR_RING_PARSERS_, node, objectStack)); }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.gxCoordParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(ol.array.includes( ol.format.KML.GX_NAMESPACE_URIS_, node.namespaceURI), 'namespaceURI of the node should be known to the KML parser'); goog.asserts.assert(node.localName == 'coord', 'localName should be coord'); var gxTrackObject = /** @type {ol.format.KMLGxTrackObject_} */ (objectStack[objectStack.length - 1]); goog.asserts.assert(goog.isObject(gxTrackObject), 'gxTrackObject should be an Object'); var flatCoordinates = gxTrackObject.flatCoordinates; var s = ol.xml.getAllTextContent(node, false); var re = /^\s*([+\-]?\d+(?:\.\d*)?(?:e[+\-]?\d*)?)\s+([+\-]?\d+(?:\.\d*)?(?:e[+\-]?\d*)?)\s+([+\-]?\d+(?:\.\d*)?(?:e[+\-]?\d*)?)\s*$/i; var m = re.exec(s); if (m) { var x = parseFloat(m[1]); var y = parseFloat(m[2]); var z = parseFloat(m[3]); flatCoordinates.push(x, y, z, 0); } else { flatCoordinates.push(0, 0, 0, 0); } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.MultiLineString|undefined} MultiLineString. */ ol.format.KML.readGxMultiTrack_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(ol.array.includes( ol.format.KML.GX_NAMESPACE_URIS_, node.namespaceURI), 'namespaceURI of the node should be known to the KML parser'); goog.asserts.assert(node.localName == 'MultiTrack', 'localName should be MultiTrack'); var lineStrings = ol.xml.pushParseAndPop( /** @type {Array.} */ ([]), ol.format.KML.GX_MULTITRACK_GEOMETRY_PARSERS_, node, objectStack); if (!lineStrings) { return undefined; } var multiLineString = new ol.geom.MultiLineString(null); multiLineString.setLineStrings(lineStrings); return multiLineString; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.LineString|undefined} LineString. */ ol.format.KML.readGxTrack_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(ol.array.includes( ol.format.KML.GX_NAMESPACE_URIS_, node.namespaceURI), 'namespaceURI of the node should be known to the KML parser'); goog.asserts.assert(node.localName == 'Track', 'localName should be Track'); var gxTrackObject = ol.xml.pushParseAndPop( /** @type {ol.format.KMLGxTrackObject_} */ ({ flatCoordinates: [], whens: [] }), ol.format.KML.GX_TRACK_PARSERS_, node, objectStack); if (!gxTrackObject) { return undefined; } var flatCoordinates = gxTrackObject.flatCoordinates; var whens = gxTrackObject.whens; goog.asserts.assert(flatCoordinates.length / 4 == whens.length, 'the length of the flatCoordinates array divided by 4 should be the ' + 'length of the whens array'); var i, ii; for (i = 0, ii = Math.min(flatCoordinates.length, whens.length); i < ii; ++i) { flatCoordinates[4 * i + 3] = whens[i]; } var lineString = new ol.geom.LineString(null); lineString.setFlatCoordinates(ol.geom.GeometryLayout.XYZM, flatCoordinates); return lineString; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {Object} Icon object. */ ol.format.KML.readIcon_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Icon', 'localName should be Icon'); var iconObject = ol.xml.pushParseAndPop( {}, ol.format.KML.ICON_PARSERS_, node, objectStack); if (iconObject) { return iconObject; } else { return null; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {Array.} Flat coordinates. */ ol.format.KML.readFlatCoordinatesFromNode_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); return /** @type {Array.} */ (ol.xml.pushParseAndPop(null, ol.format.KML.GEOMETRY_FLAT_COORDINATES_PARSERS_, node, objectStack)); }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.LineString|undefined} LineString. */ ol.format.KML.readLineString_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'LineString', 'localName should be LineString'); var properties = ol.xml.pushParseAndPop(/** @type {Object} */ ({}), ol.format.KML.EXTRUDE_AND_ALTITUDE_MODE_PARSERS_, node, objectStack); var flatCoordinates = ol.format.KML.readFlatCoordinatesFromNode_(node, objectStack); if (flatCoordinates) { var lineString = new ol.geom.LineString(null); lineString.setFlatCoordinates(ol.geom.GeometryLayout.XYZ, flatCoordinates); lineString.setProperties(properties); return lineString; } else { return undefined; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.Polygon|undefined} Polygon. */ ol.format.KML.readLinearRing_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'LinearRing', 'localName should be LinearRing'); var properties = ol.xml.pushParseAndPop(/** @type {Object} */ ({}), ol.format.KML.EXTRUDE_AND_ALTITUDE_MODE_PARSERS_, node, objectStack); var flatCoordinates = ol.format.KML.readFlatCoordinatesFromNode_(node, objectStack); if (flatCoordinates) { var polygon = new ol.geom.Polygon(null); polygon.setFlatCoordinates(ol.geom.GeometryLayout.XYZ, flatCoordinates, [flatCoordinates.length]); polygon.setProperties(properties); return polygon; } else { return undefined; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.Geometry} Geometry. */ ol.format.KML.readMultiGeometry_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'MultiGeometry', 'localName should be MultiGeometry'); var geometries = ol.xml.pushParseAndPop( /** @type {Array.} */ ([]), ol.format.KML.MULTI_GEOMETRY_PARSERS_, node, objectStack); if (!geometries) { return null; } if (geometries.length === 0) { return new ol.geom.GeometryCollection(geometries); } var homogeneous = true; var type = geometries[0].getType(); var geometry, i, ii; for (i = 1, ii = geometries.length; i < ii; ++i) { geometry = geometries[i]; if (geometry.getType() != type) { homogeneous = false; break; } } if (homogeneous) { /** @type {ol.geom.GeometryLayout} */ var layout; /** @type {Array.} */ var flatCoordinates; if (type == ol.geom.GeometryType.POINT) { var point = geometries[0]; goog.asserts.assertInstanceof(point, ol.geom.Point, 'point should be an ol.geom.Point'); layout = point.getLayout(); flatCoordinates = point.getFlatCoordinates(); for (i = 1, ii = geometries.length; i < ii; ++i) { geometry = geometries[i]; goog.asserts.assertInstanceof(geometry, ol.geom.Point, 'geometry should be an ol.geom.Point'); goog.asserts.assert(geometry.getLayout() == layout, 'geometry layout should be consistent'); goog.array.extend(flatCoordinates, geometry.getFlatCoordinates()); } var multiPoint = new ol.geom.MultiPoint(null); multiPoint.setFlatCoordinates(layout, flatCoordinates); ol.format.KML.setCommonGeometryProperties_(multiPoint, geometries); return multiPoint; } else if (type == ol.geom.GeometryType.LINE_STRING) { var multiLineString = new ol.geom.MultiLineString(null); multiLineString.setLineStrings(geometries); ol.format.KML.setCommonGeometryProperties_(multiLineString, geometries); return multiLineString; } else if (type == ol.geom.GeometryType.POLYGON) { var multiPolygon = new ol.geom.MultiPolygon(null); multiPolygon.setPolygons(geometries); ol.format.KML.setCommonGeometryProperties_(multiPolygon, geometries); return multiPolygon; } else if (type == ol.geom.GeometryType.GEOMETRY_COLLECTION) { return new ol.geom.GeometryCollection(geometries); } else { goog.asserts.fail('Unexpected type: ' + type); return null; } } else { return new ol.geom.GeometryCollection(geometries); } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.Point|undefined} Point. */ ol.format.KML.readPoint_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Point', 'localName should be Point'); var properties = ol.xml.pushParseAndPop(/** @type {Object} */ ({}), ol.format.KML.EXTRUDE_AND_ALTITUDE_MODE_PARSERS_, node, objectStack); var flatCoordinates = ol.format.KML.readFlatCoordinatesFromNode_(node, objectStack); if (goog.isDefAndNotNull(flatCoordinates)) { var point = new ol.geom.Point(null); goog.asserts.assert(flatCoordinates.length == 3, 'flatCoordinates should have a length of 3'); point.setFlatCoordinates(ol.geom.GeometryLayout.XYZ, flatCoordinates); point.setProperties(properties); return point; } else { return undefined; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.geom.Polygon|undefined} Polygon. */ ol.format.KML.readPolygon_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Polygon', 'localName should be Polygon'); var properties = ol.xml.pushParseAndPop(/** @type {Object} */ ({}), ol.format.KML.EXTRUDE_AND_ALTITUDE_MODE_PARSERS_, node, objectStack); var flatLinearRings = ol.xml.pushParseAndPop( /** @type {Array.>} */ ([null]), ol.format.KML.FLAT_LINEAR_RINGS_PARSERS_, node, objectStack); if (goog.isDefAndNotNull(flatLinearRings) && !goog.isNull(flatLinearRings[0])) { var polygon = new ol.geom.Polygon(null); var flatCoordinates = flatLinearRings[0]; var ends = [flatCoordinates.length]; var i, ii; for (i = 1, ii = flatLinearRings.length; i < ii; ++i) { goog.array.extend(flatCoordinates, flatLinearRings[i]); ends.push(flatCoordinates.length); } polygon.setFlatCoordinates( ol.geom.GeometryLayout.XYZ, flatCoordinates, ends); polygon.setProperties(properties); return polygon; } else { return undefined; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {Array.} Style. */ ol.format.KML.readStyle_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Style', 'localName should be Style'); var styleObject = ol.xml.pushParseAndPop( {}, ol.format.KML.STYLE_PARSERS_, node, objectStack); if (!styleObject) { return null; } var fillStyle = /** @type {ol.style.Fill} */ (goog.object.get( styleObject, 'fillStyle', ol.format.KML.DEFAULT_FILL_STYLE_)); var fill = /** @type {boolean|undefined} */ (styleObject['fill']); if (fill !== undefined && !fill) { fillStyle = null; } var imageStyle = /** @type {ol.style.Image} */ (goog.object.get( styleObject, 'imageStyle', ol.format.KML.DEFAULT_IMAGE_STYLE_)); var textStyle = /** @type {ol.style.Text} */ (goog.object.get( styleObject, 'textStyle', ol.format.KML.DEFAULT_TEXT_STYLE_)); var strokeStyle = /** @type {ol.style.Stroke} */ (goog.object.get( styleObject, 'strokeStyle', ol.format.KML.DEFAULT_STROKE_STYLE_)); var outline = /** @type {boolean|undefined} */ (styleObject['outline']); if (outline !== undefined && !outline) { strokeStyle = null; } return [new ol.style.Style({ fill: fillStyle, image: imageStyle, stroke: strokeStyle, text: textStyle, zIndex: undefined // FIXME })]; }; /** * Reads an array of geometries and creates arrays for common geometry * properties. Then sets them to the multi geometry. * @param {ol.geom.MultiPoint|ol.geom.MultiLineString|ol.geom.MultiPolygon} * multiGeometry * @param {Array.} geometries * @private */ ol.format.KML.setCommonGeometryProperties_ = function(multiGeometry, geometries) { var ii = geometries.length; var extrudes = new Array(geometries.length); var altitudeModes = new Array(geometries.length); var geometry, i, hasExtrude, hasAltitudeMode; hasExtrude = hasAltitudeMode = false; for (i = 0; i < ii; ++i) { geometry = geometries[i]; extrudes[i] = geometry.get('extrude'); altitudeModes[i] = geometry.get('altitudeMode'); hasExtrude = hasExtrude || extrudes[i] !== undefined; hasAltitudeMode = hasAltitudeMode || altitudeModes[i]; } if (hasExtrude) { multiGeometry.set('extrude', extrudes); } if (hasAltitudeMode) { multiGeometry.set('altitudeMode', altitudeModes); } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.DataParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Data', 'localName should be Data'); var name = node.getAttribute('name'); if (!goog.isNull(name)) { var data = ol.xml.pushParseAndPop( undefined, ol.format.KML.DATA_PARSERS_, node, objectStack); if (data) { var featureObject = /** @type {Object} */ (objectStack[objectStack.length - 1]); goog.asserts.assert(goog.isObject(featureObject), 'featureObject should be an Object'); featureObject[name] = data; } } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.ExtendedDataParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'ExtendedData', 'localName should be ExtendedData'); ol.xml.parseNode(ol.format.KML.EXTENDED_DATA_PARSERS_, node, objectStack); }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.PairDataParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Pair', 'localName should be Pair'); var pairObject = ol.xml.pushParseAndPop( {}, ol.format.KML.PAIR_PARSERS_, node, objectStack); if (!pairObject) { return; } var key = /** @type {string|undefined} */ (pairObject['key']); if (key && key == 'normal') { var styleUrl = /** @type {string|undefined} */ (pairObject['styleUrl']); if (styleUrl) { objectStack[objectStack.length - 1] = styleUrl; } var Style = /** @type {ol.style.Style} */ (pairObject['Style']); if (Style) { objectStack[objectStack.length - 1] = Style; } } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.PlacemarkStyleMapParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'StyleMap', 'localName should be StyleMap'); var styleMapValue = ol.format.KML.readStyleMapValue_(node, objectStack); if (!styleMapValue) { return; } var placemarkObject = objectStack[objectStack.length - 1]; goog.asserts.assert(goog.isObject(placemarkObject), 'placemarkObject should be an Object'); if (goog.isArray(styleMapValue)) { placemarkObject['Style'] = styleMapValue; } else if (goog.isString(styleMapValue)) { placemarkObject['styleUrl'] = styleMapValue; } else { goog.asserts.fail('styleMapValue has an unknown type'); } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.SchemaDataParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'SchemaData', 'localName should be SchemaData'); ol.xml.parseNode(ol.format.KML.SCHEMA_DATA_PARSERS_, node, objectStack); }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.SimpleDataParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'SimpleData', 'localName should be SimpleData'); var name = node.getAttribute('name'); if (!goog.isNull(name)) { var data = ol.format.XSD.readString(node); var featureObject = /** @type {Object} */ (objectStack[objectStack.length - 1]); featureObject[name] = data; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.innerBoundaryIsParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'innerBoundaryIs', 'localName should be innerBoundaryIs'); var flatLinearRing = ol.xml.pushParseAndPop( /** @type {Array.|undefined} */ (undefined), ol.format.KML.INNER_BOUNDARY_IS_PARSERS_, node, objectStack); if (flatLinearRing) { var flatLinearRings = /** @type {Array.>} */ (objectStack[objectStack.length - 1]); goog.asserts.assert(goog.isArray(flatLinearRings), 'flatLinearRings should be an array'); goog.asserts.assert(flatLinearRings.length > 0, 'flatLinearRings array should not be empty'); flatLinearRings.push(flatLinearRing); } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.outerBoundaryIsParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'outerBoundaryIs', 'localName should be outerBoundaryIs'); var flatLinearRing = ol.xml.pushParseAndPop( /** @type {Array.|undefined} */ (undefined), ol.format.KML.OUTER_BOUNDARY_IS_PARSERS_, node, objectStack); if (flatLinearRing) { var flatLinearRings = /** @type {Array.>} */ (objectStack[objectStack.length - 1]); goog.asserts.assert(goog.isArray(flatLinearRings), 'flatLinearRings should be an array'); goog.asserts.assert(flatLinearRings.length > 0, 'flatLinearRings array should not be empty'); flatLinearRings[0] = flatLinearRing; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.LinkParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Link', 'localName should be Link'); ol.xml.parseNode(ol.format.KML.LINK_PARSERS_, node, objectStack); }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.whenParser_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'when', 'localName should be when'); var gxTrackObject = /** @type {ol.format.KMLGxTrackObject_} */ (objectStack[objectStack.length - 1]); goog.asserts.assert(goog.isObject(gxTrackObject), 'gxTrackObject should be an Object'); var whens = gxTrackObject.whens; var s = ol.xml.getAllTextContent(node, false); var re = /^\s*(\d{4})($|-(\d{2})($|-(\d{2})($|T(\d{2}):(\d{2}):(\d{2})(Z|(?:([+\-])(\d{2})(?::(\d{2}))?)))))\s*$/; var m = re.exec(s); if (m) { var year = parseInt(m[1], 10); var month = m[3] ? parseInt(m[3], 10) - 1 : 0; var day = m[5] ? parseInt(m[5], 10) : 1; var hour = m[7] ? parseInt(m[7], 10) : 0; var minute = m[8] ? parseInt(m[8], 10) : 0; var second = m[9] ? parseInt(m[9], 10) : 0; var when = Date.UTC(year, month, day, hour, minute, second); if (m[10] && m[10] != 'Z') { var sign = m[11] == '-' ? -1 : 1; when += sign * 60 * parseInt(m[12], 10); if (m[13]) { when += sign * 60 * 60 * parseInt(m[13], 10); } } whens.push(when); } else { whens.push(0); } }; /** * @const * @type {Object.>} * @private */ ol.format.KML.DATA_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'value': ol.xml.makeReplacer(ol.format.XSD.readString) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.EXTENDED_DATA_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Data': ol.format.KML.DataParser_, 'SchemaData': ol.format.KML.SchemaDataParser_ }); /** * @const * @type {Object.>} * @private */ ol.format.KML.EXTRUDE_AND_ALTITUDE_MODE_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'extrude': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean), 'altitudeMode': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.FLAT_LINEAR_RING_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'coordinates': ol.xml.makeReplacer(ol.format.KML.readFlatCoordinates_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.FLAT_LINEAR_RINGS_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'innerBoundaryIs': ol.format.KML.innerBoundaryIsParser_, 'outerBoundaryIs': ol.format.KML.outerBoundaryIsParser_ }); /** * @const * @type {Object.>} * @private */ ol.format.KML.GX_TRACK_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'when': ol.format.KML.whenParser_ }, ol.xml.makeStructureNS( ol.format.KML.GX_NAMESPACE_URIS_, { 'coord': ol.format.KML.gxCoordParser_ })); /** * @const * @type {Object.>} * @private */ ol.format.KML.GEOMETRY_FLAT_COORDINATES_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'coordinates': ol.xml.makeReplacer(ol.format.KML.readFlatCoordinates_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.ICON_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'href': ol.xml.makeObjectPropertySetter(ol.format.KML.readURI_) }, ol.xml.makeStructureNS( ol.format.KML.GX_NAMESPACE_URIS_, { 'x': ol.xml.makeObjectPropertySetter(ol.format.XSD.readDecimal), 'y': ol.xml.makeObjectPropertySetter(ol.format.XSD.readDecimal), 'w': ol.xml.makeObjectPropertySetter(ol.format.XSD.readDecimal), 'h': ol.xml.makeObjectPropertySetter(ol.format.XSD.readDecimal) })); /** * @const * @type {Object.>} * @private */ ol.format.KML.ICON_STYLE_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Icon': ol.xml.makeObjectPropertySetter(ol.format.KML.readIcon_), 'heading': ol.xml.makeObjectPropertySetter(ol.format.XSD.readDecimal), 'hotSpot': ol.xml.makeObjectPropertySetter(ol.format.KML.readVec2_), 'scale': ol.xml.makeObjectPropertySetter(ol.format.KML.readScale_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.INNER_BOUNDARY_IS_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'LinearRing': ol.xml.makeReplacer(ol.format.KML.readFlatLinearRing_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.LABEL_STYLE_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'color': ol.xml.makeObjectPropertySetter(ol.format.KML.readColor_), 'scale': ol.xml.makeObjectPropertySetter(ol.format.KML.readScale_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.LINE_STYLE_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'color': ol.xml.makeObjectPropertySetter(ol.format.KML.readColor_), 'width': ol.xml.makeObjectPropertySetter(ol.format.XSD.readDecimal) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.MULTI_GEOMETRY_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'LineString': ol.xml.makeArrayPusher(ol.format.KML.readLineString_), 'LinearRing': ol.xml.makeArrayPusher(ol.format.KML.readLinearRing_), 'MultiGeometry': ol.xml.makeArrayPusher(ol.format.KML.readMultiGeometry_), 'Point': ol.xml.makeArrayPusher(ol.format.KML.readPoint_), 'Polygon': ol.xml.makeArrayPusher(ol.format.KML.readPolygon_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.GX_MULTITRACK_GEOMETRY_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.GX_NAMESPACE_URIS_, { 'Track': ol.xml.makeArrayPusher(ol.format.KML.readGxTrack_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.NETWORK_LINK_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'ExtendedData': ol.format.KML.ExtendedDataParser_, 'Link': ol.format.KML.LinkParser_, 'address': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'description': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'name': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'open': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean), 'phoneNumber': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'visibility': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.LINK_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'href': ol.xml.makeObjectPropertySetter(ol.format.KML.readURI_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.OUTER_BOUNDARY_IS_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'LinearRing': ol.xml.makeReplacer(ol.format.KML.readFlatLinearRing_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.PAIR_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Style': ol.xml.makeObjectPropertySetter(ol.format.KML.readStyle_), 'key': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'styleUrl': ol.xml.makeObjectPropertySetter(ol.format.KML.readStyleUrl_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.PLACEMARK_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'ExtendedData': ol.format.KML.ExtendedDataParser_, 'MultiGeometry': ol.xml.makeObjectPropertySetter( ol.format.KML.readMultiGeometry_, 'geometry'), 'LineString': ol.xml.makeObjectPropertySetter( ol.format.KML.readLineString_, 'geometry'), 'LinearRing': ol.xml.makeObjectPropertySetter( ol.format.KML.readLinearRing_, 'geometry'), 'Point': ol.xml.makeObjectPropertySetter( ol.format.KML.readPoint_, 'geometry'), 'Polygon': ol.xml.makeObjectPropertySetter( ol.format.KML.readPolygon_, 'geometry'), 'Style': ol.xml.makeObjectPropertySetter(ol.format.KML.readStyle_), 'StyleMap': ol.format.KML.PlacemarkStyleMapParser_, 'address': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'description': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'name': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'open': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean), 'phoneNumber': ol.xml.makeObjectPropertySetter(ol.format.XSD.readString), 'styleUrl': ol.xml.makeObjectPropertySetter(ol.format.KML.readURI_), 'visibility': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean) }, ol.xml.makeStructureNS( ol.format.KML.GX_NAMESPACE_URIS_, { 'MultiTrack': ol.xml.makeObjectPropertySetter( ol.format.KML.readGxMultiTrack_, 'geometry'), 'Track': ol.xml.makeObjectPropertySetter( ol.format.KML.readGxTrack_, 'geometry') } )); /** * @const * @type {Object.>} * @private */ ol.format.KML.POLY_STYLE_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'color': ol.xml.makeObjectPropertySetter(ol.format.KML.readColor_), 'fill': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean), 'outline': ol.xml.makeObjectPropertySetter(ol.format.XSD.readBoolean) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.SCHEMA_DATA_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'SimpleData': ol.format.KML.SimpleDataParser_ }); /** * @const * @type {Object.>} * @private */ ol.format.KML.STYLE_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'IconStyle': ol.format.KML.IconStyleParser_, 'LabelStyle': ol.format.KML.LabelStyleParser_, 'LineStyle': ol.format.KML.LineStyleParser_, 'PolyStyle': ol.format.KML.PolyStyleParser_ }); /** * @const * @type {Object.>} * @private */ ol.format.KML.STYLE_MAP_PARSERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Pair': ol.format.KML.PairDataParser_ }); /** * @inheritDoc */ ol.format.KML.prototype.getExtensions = function() { return ol.format.KML.EXTENSIONS_; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {Array.|undefined} Features. */ ol.format.KML.prototype.readDocumentOrFolder_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); var localName = ol.xml.getLocalName(node); goog.asserts.assert(localName == 'Document' || localName == 'Folder', 'localName should be Document or Folder'); // FIXME use scope somehow var parsersNS = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Document': ol.xml.makeArrayExtender(this.readDocumentOrFolder_, this), 'Folder': ol.xml.makeArrayExtender(this.readDocumentOrFolder_, this), 'Placemark': ol.xml.makeArrayPusher(this.readPlacemark_, this), 'Style': goog.bind(this.readSharedStyle_, this), 'StyleMap': goog.bind(this.readSharedStyleMap_, this) }); var features = ol.xml.pushParseAndPop(/** @type {Array.} */ ([]), parsersNS, node, objectStack, this); if (features) { return features; } else { return undefined; } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private * @return {ol.Feature|undefined} Feature. */ ol.format.KML.prototype.readPlacemark_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Placemark', 'localName should be Placemark'); var object = ol.xml.pushParseAndPop({'geometry': null}, ol.format.KML.PLACEMARK_PARSERS_, node, objectStack); if (!object) { return undefined; } var feature = new ol.Feature(); var id = node.getAttribute('id'); if (!goog.isNull(id)) { feature.setId(id); } var options = /** @type {olx.format.ReadOptions} */ (objectStack[0]); var geometry = object['geometry']; if (goog.isDefAndNotNull(geometry)) { ol.format.Feature.transformWithOptions(geometry, false, options); } feature.setGeometry(geometry); delete object['geometry']; if (this.extractStyles_) { var style = object['Style']; var styleUrl = object['styleUrl']; var styleFunction = ol.format.KML.createFeatureStyleFunction_( style, styleUrl, this.defaultStyle_, this.sharedStyles_); feature.setStyle(styleFunction); } delete object['Style']; // we do not remove the styleUrl property from the object, so it // gets stored on feature when setProperties is called feature.setProperties(object); return feature; }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.prototype.readSharedStyle_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'Style', 'localName should be Style'); var id = node.getAttribute('id'); if (!goog.isNull(id)) { var style = ol.format.KML.readStyle_(node, objectStack); if (style) { var styleUri; if (goog.isDefAndNotNull(node.baseURI)) { styleUri = goog.Uri.resolve(node.baseURI, '#' + id).toString(); } else { styleUri = '#' + id; } this.sharedStyles_[styleUri] = style; } } }; /** * @param {Node} node Node. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.prototype.readSharedStyleMap_ = function(node, objectStack) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); goog.asserts.assert(node.localName == 'StyleMap', 'localName should be StyleMap'); var id = node.getAttribute('id'); if (goog.isNull(id)) { return; } var styleMapValue = ol.format.KML.readStyleMapValue_(node, objectStack); if (!styleMapValue) { return; } var styleUri; if (goog.isDefAndNotNull(node.baseURI)) { styleUri = goog.Uri.resolve(node.baseURI, '#' + id).toString(); } else { styleUri = '#' + id; } this.sharedStyles_[styleUri] = styleMapValue; }; /** * Read the first feature from a KML source. * * @function * @param {Document|Node|Object|string} source Source. * @param {olx.format.ReadOptions=} opt_options Read options. * @return {ol.Feature} Feature. * @api stable */ ol.format.KML.prototype.readFeature; /** * @inheritDoc */ ol.format.KML.prototype.readFeatureFromNode = function(node, opt_options) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); if (!ol.array.includes(ol.format.KML.NAMESPACE_URIS_, node.namespaceURI)) { return null; } goog.asserts.assert(node.localName == 'Placemark', 'localName should be Placemark'); var feature = this.readPlacemark_( node, [this.getReadOptions(node, opt_options)]); if (feature) { return feature; } else { return null; } }; /** * Read all features from a KML source. * * @function * @param {Document|Node|Object|string} source Source. * @param {olx.format.ReadOptions=} opt_options Read options. * @return {Array.} Features. * @api stable */ ol.format.KML.prototype.readFeatures; /** * @inheritDoc */ ol.format.KML.prototype.readFeaturesFromNode = function(node, opt_options) { goog.asserts.assert(node.nodeType == goog.dom.NodeType.ELEMENT, 'node.nodeType should be ELEMENT'); if (!ol.array.includes(ol.format.KML.NAMESPACE_URIS_, node.namespaceURI)) { return []; } var features; var localName = ol.xml.getLocalName(node); if (localName == 'Document' || localName == 'Folder') { features = this.readDocumentOrFolder_( node, [this.getReadOptions(node, opt_options)]); if (features) { return features; } else { return []; } } else if (localName == 'Placemark') { var feature = this.readPlacemark_( node, [this.getReadOptions(node, opt_options)]); if (feature) { return [feature]; } else { return []; } } else if (localName == 'kml') { features = []; var n; for (n = node.firstElementChild; !goog.isNull(n); n = n.nextElementSibling) { var fs = this.readFeaturesFromNode(n, opt_options); if (fs) { goog.array.extend(features, fs); } } return features; } else { return []; } }; /** * Read the name of the KML. * * @param {Document|Node|string} source Souce. * @return {string|undefined} Name. * @api stable */ ol.format.KML.prototype.readName = function(source) { if (ol.xml.isDocument(source)) { return this.readNameFromDocument(/** @type {Document} */ (source)); } else if (ol.xml.isNode(source)) { return this.readNameFromNode(/** @type {Node} */ (source)); } else if (goog.isString(source)) { var doc = ol.xml.parse(source); return this.readNameFromDocument(doc); } else { goog.asserts.fail('Unknown type for source'); return undefined; } }; /** * @param {Document} doc Document. * @return {string|undefined} Name. */ ol.format.KML.prototype.readNameFromDocument = function(doc) { var n; for (n = doc.firstChild; !goog.isNull(n); n = n.nextSibling) { if (n.nodeType == goog.dom.NodeType.ELEMENT) { var name = this.readNameFromNode(n); if (name) { return name; } } } return undefined; }; /** * @param {Node} node Node. * @return {string|undefined} Name. */ ol.format.KML.prototype.readNameFromNode = function(node) { var n; for (n = node.firstElementChild; !goog.isNull(n); n = n.nextElementSibling) { if (ol.array.includes(ol.format.KML.NAMESPACE_URIS_, n.namespaceURI) && n.localName == 'name') { return ol.format.XSD.readString(n); } } for (n = node.firstElementChild; !goog.isNull(n); n = n.nextElementSibling) { var localName = ol.xml.getLocalName(n); if (ol.array.includes(ol.format.KML.NAMESPACE_URIS_, n.namespaceURI) && (localName == 'Document' || localName == 'Folder' || localName == 'Placemark' || localName == 'kml')) { var name = this.readNameFromNode(n); if (name) { return name; } } } return undefined; }; /** * Read the network links of the KML. * * @param {Document|Node|string} source Source. * @return {Array.} Network links. * @api */ ol.format.KML.prototype.readNetworkLinks = function(source) { var networkLinks = []; if (ol.xml.isDocument(source)) { goog.array.extend(networkLinks, this.readNetworkLinksFromDocument( /** @type {Document} */ (source))); } else if (ol.xml.isNode(source)) { goog.array.extend(networkLinks, this.readNetworkLinksFromNode( /** @type {Node} */ (source))); } else if (goog.isString(source)) { var doc = ol.xml.parse(source); goog.array.extend(networkLinks, this.readNetworkLinksFromDocument(doc)); } else { goog.asserts.fail('unknown type for source'); } return networkLinks; }; /** * @param {Document} doc Document. * @return {Array.} Network links. */ ol.format.KML.prototype.readNetworkLinksFromDocument = function(doc) { var n, networkLinks = []; for (n = doc.firstChild; !goog.isNull(n); n = n.nextSibling) { if (n.nodeType == goog.dom.NodeType.ELEMENT) { goog.array.extend(networkLinks, this.readNetworkLinksFromNode(n)); } } return networkLinks; }; /** * @param {Node} node Node. * @return {Array.} Network links. */ ol.format.KML.prototype.readNetworkLinksFromNode = function(node) { var n, networkLinks = []; for (n = node.firstElementChild; !goog.isNull(n); n = n.nextElementSibling) { if (ol.array.includes(ol.format.KML.NAMESPACE_URIS_, n.namespaceURI) && n.localName == 'NetworkLink') { var obj = ol.xml.pushParseAndPop({}, ol.format.KML.NETWORK_LINK_PARSERS_, n, []); networkLinks.push(obj); } } for (n = node.firstElementChild; !goog.isNull(n); n = n.nextElementSibling) { var localName = ol.xml.getLocalName(n); if (ol.array.includes(ol.format.KML.NAMESPACE_URIS_, n.namespaceURI) && (localName == 'Document' || localName == 'Folder' || localName == 'kml')) { goog.array.extend(networkLinks, this.readNetworkLinksFromNode(n)); } } return networkLinks; }; /** * Read the projection from a KML source. * * @function * @param {Document|Node|Object|string} source Source. * @return {ol.proj.Projection} Projection. * @api stable */ ol.format.KML.prototype.readProjection; /** * @param {Node} node Node to append a TextNode with the color to. * @param {ol.Color|string} color Color. * @private */ ol.format.KML.writeColorTextNode_ = function(node, color) { var rgba = ol.color.asArray(color); var opacity = (rgba.length == 4) ? rgba[3] : 1; var abgr = [opacity * 255, rgba[2], rgba[1], rgba[0]]; var i; for (i = 0; i < 4; ++i) { var hex = parseInt(abgr[i], 10).toString(16); abgr[i] = (hex.length == 1) ? '0' + hex : hex; } ol.format.XSD.writeStringTextNode(node, abgr.join('')); }; /** * @param {Node} node Node to append a TextNode with the coordinates to. * @param {Array.} coordinates Coordinates. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeCoordinatesTextNode_ = function(node, coordinates, objectStack) { var context = objectStack[objectStack.length - 1]; goog.asserts.assert(goog.isObject(context), 'context should be an Object'); var layout = context['layout']; var stride = context['stride']; var dimension; if (layout == ol.geom.GeometryLayout.XY || layout == ol.geom.GeometryLayout.XYM) { dimension = 2; } else if (layout == ol.geom.GeometryLayout.XYZ || layout == ol.geom.GeometryLayout.XYZM) { dimension = 3; } else { goog.asserts.fail('Unknown geometry layout'); } var d, i; var ii = coordinates.length; var text = ''; if (ii > 0) { text += coordinates[0]; for (d = 1; d < dimension; ++d) { text += ',' + coordinates[d]; } for (i = stride; i < ii; i += stride) { text += ' ' + coordinates[i]; for (d = 1; d < dimension; ++d) { text += ',' + coordinates[i + d]; } } } ol.format.XSD.writeStringTextNode(node, text); }; /** * @param {Node} node Node. * @param {Array.} features Features. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeDocument_ = function(node, features, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; ol.xml.pushSerializeAndPop(context, ol.format.KML.DOCUMENT_SERIALIZERS_, ol.format.KML.DOCUMENT_NODE_FACTORY_, features, objectStack); }; /** * @param {Node} node Node. * @param {Object} icon Icon object. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeIcon_ = function(node, icon, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; var parentNode = objectStack[objectStack.length - 1].node; var orderedKeys = ol.format.KML.ICON_SEQUENCE_[parentNode.namespaceURI]; var values = ol.xml.makeSequence(icon, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.ICON_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, objectStack, orderedKeys); orderedKeys = ol.format.KML.ICON_SEQUENCE_[ol.format.KML.GX_NAMESPACE_URIS_[0]]; values = ol.xml.makeSequence(icon, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.ICON_SERIALIZERS_, ol.format.KML.GX_NODE_FACTORY_, values, objectStack, orderedKeys); }; /** * @param {Node} node Node. * @param {ol.style.Icon} style Icon style. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeIconStyle_ = function(node, style, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; var properties = {}; var src = style.getSrc(); var size = style.getSize(); var iconImageSize = style.getImageSize(); var iconProperties = { 'href': src }; if (!goog.isNull(size)) { iconProperties['w'] = size[0]; iconProperties['h'] = size[1]; var anchor = style.getAnchor(); // top-left var origin = style.getOrigin(); // top-left if (!goog.isNull(origin) && !goog.isNull(iconImageSize) && origin[0] !== 0 && origin[1] !== size[1]) { iconProperties['x'] = origin[0]; iconProperties['y'] = iconImageSize[1] - (origin[1] + size[1]); } if (!goog.isNull(anchor) && anchor[0] !== 0 && anchor[1] !== size[1]) { var /** @type {ol.format.KMLVec2_} */ hotSpot = { x: anchor[0], xunits: ol.style.IconAnchorUnits.PIXELS, y: size[1] - anchor[1], yunits: ol.style.IconAnchorUnits.PIXELS }; properties['hotSpot'] = hotSpot; } } properties['Icon'] = iconProperties; var scale = style.getScale(); if (scale !== 1) { properties['scale'] = scale; } var rotation = style.getRotation(); if (rotation !== 0) { properties['heading'] = rotation; // 0-360 } var parentNode = objectStack[objectStack.length - 1].node; var orderedKeys = ol.format.KML.ICON_STYLE_SEQUENCE_[parentNode.namespaceURI]; var values = ol.xml.makeSequence(properties, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.ICON_STYLE_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, objectStack, orderedKeys); }; /** * @param {Node} node Node. * @param {ol.style.Text} style style. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeLabelStyle_ = function(node, style, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; var properties = {}; var fill = style.getFill(); if (!goog.isNull(fill)) { properties['color'] = fill.getColor(); } var scale = style.getScale(); if (scale && scale !== 1) { properties['scale'] = scale; } var parentNode = objectStack[objectStack.length - 1].node; var orderedKeys = ol.format.KML.LABEL_STYLE_SEQUENCE_[parentNode.namespaceURI]; var values = ol.xml.makeSequence(properties, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.LABEL_STYLE_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, objectStack, orderedKeys); }; /** * @param {Node} node Node. * @param {ol.style.Stroke} style style. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeLineStyle_ = function(node, style, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; var properties = { 'color': style.getColor(), 'width': style.getWidth() }; var parentNode = objectStack[objectStack.length - 1].node; var orderedKeys = ol.format.KML.LINE_STYLE_SEQUENCE_[parentNode.namespaceURI]; var values = ol.xml.makeSequence(properties, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.LINE_STYLE_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, objectStack, orderedKeys); }; /** * @param {Node} node Node. * @param {ol.geom.Geometry} geometry Geometry. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeMultiGeometry_ = function(node, geometry, objectStack) { goog.asserts.assert( (geometry instanceof ol.geom.MultiPoint) || (geometry instanceof ol.geom.MultiLineString) || (geometry instanceof ol.geom.MultiPolygon), 'geometry should be one of: ol.geom.MultiPoint, ' + 'ol.geom.MultiLineString or ol.geom.MultiPolygon'); /** @type {ol.xml.NodeStackItem} */ var context = {node: node}; var type = geometry.getType(); /** @type {Array.} */ var geometries; /** @type {function(*, Array.<*>, string=): (Node|undefined)} */ var factory; if (type == ol.geom.GeometryType.MULTI_POINT) { geometries = (/** @type {ol.geom.MultiPoint} */ (geometry)).getPoints(); factory = ol.format.KML.POINT_NODE_FACTORY_; } else if (type == ol.geom.GeometryType.MULTI_LINE_STRING) { geometries = (/** @type {ol.geom.MultiLineString} */ (geometry)).getLineStrings(); factory = ol.format.KML.LINE_STRING_NODE_FACTORY_; } else if (type == ol.geom.GeometryType.MULTI_POLYGON) { geometries = (/** @type {ol.geom.MultiPolygon} */ (geometry)).getPolygons(); factory = ol.format.KML.POLYGON_NODE_FACTORY_; } else { goog.asserts.fail('Unknown geometry type: ' + type); } ol.xml.pushSerializeAndPop(context, ol.format.KML.MULTI_GEOMETRY_SERIALIZERS_, factory, geometries, objectStack); }; /** * @param {Node} node Node. * @param {ol.geom.LinearRing} linearRing Linear ring. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeBoundaryIs_ = function(node, linearRing, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; ol.xml.pushSerializeAndPop(context, ol.format.KML.BOUNDARY_IS_SERIALIZERS_, ol.format.KML.LINEAR_RING_NODE_FACTORY_, [linearRing], objectStack); }; /** * FIXME currently we do serialize arbitrary/custom feature properties * (ExtendedData). * @param {Node} node Node. * @param {ol.Feature} feature Feature. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writePlacemark_ = function(node, feature, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; // set id if (goog.isDefAndNotNull(feature.getId())) { node.setAttribute('id', feature.getId()); } // serialize properties (properties unknown to KML are not serialized) var properties = feature.getProperties(); var styleFunction = feature.getStyleFunction(); if (styleFunction) { // FIXME the styles returned by the style function are supposed to be // resolution-independent here var styles = styleFunction.call(feature, 0); if (!goog.isNull(styles) && styles.length > 0) { properties['Style'] = styles[0]; var textStyle = styles[0].getText(); if (!goog.isNull(textStyle)) { properties['name'] = textStyle.getText(); } } } var parentNode = objectStack[objectStack.length - 1].node; var orderedKeys = ol.format.KML.PLACEMARK_SEQUENCE_[parentNode.namespaceURI]; var values = ol.xml.makeSequence(properties, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.PLACEMARK_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, objectStack, orderedKeys); // serialize geometry var options = /** @type {olx.format.WriteOptions} */ (objectStack[0]); var geometry = feature.getGeometry(); if (goog.isDefAndNotNull(geometry)) { geometry = ol.format.Feature.transformWithOptions(geometry, true, options); } ol.xml.pushSerializeAndPop(context, ol.format.KML.PLACEMARK_SERIALIZERS_, ol.format.KML.GEOMETRY_NODE_FACTORY_, [geometry], objectStack); }; /** * @param {Node} node Node. * @param {ol.geom.SimpleGeometry} geometry Geometry. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writePrimitiveGeometry_ = function(node, geometry, objectStack) { goog.asserts.assert( (geometry instanceof ol.geom.Point) || (geometry instanceof ol.geom.LineString) || (geometry instanceof ol.geom.LinearRing), 'geometry should be one of ol.geom.Point, ol.geom.LineString ' + 'or ol.geom.LinearRing'); var flatCoordinates = geometry.getFlatCoordinates(); var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; context['layout'] = geometry.getLayout(); context['stride'] = geometry.getStride(); ol.xml.pushSerializeAndPop(context, ol.format.KML.PRIMITIVE_GEOMETRY_SERIALIZERS_, ol.format.KML.COORDINATES_NODE_FACTORY_, [flatCoordinates], objectStack); }; /** * @param {Node} node Node. * @param {ol.geom.Polygon} polygon Polygon. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writePolygon_ = function(node, polygon, objectStack) { goog.asserts.assertInstanceof(polygon, ol.geom.Polygon, 'polygon should be an ol.geom.Polygon'); var linearRings = polygon.getLinearRings(); goog.asserts.assert(linearRings.length > 0, 'linearRings should not be empty'); var outerRing = linearRings.shift(); var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; // inner rings ol.xml.pushSerializeAndPop(context, ol.format.KML.POLYGON_SERIALIZERS_, ol.format.KML.INNER_BOUNDARY_NODE_FACTORY_, linearRings, objectStack); // outer ring ol.xml.pushSerializeAndPop(context, ol.format.KML.POLYGON_SERIALIZERS_, ol.format.KML.OUTER_BOUNDARY_NODE_FACTORY_, [outerRing], objectStack); }; /** * @param {Node} node Node. * @param {ol.style.Fill} style Style. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writePolyStyle_ = function(node, style, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; ol.xml.pushSerializeAndPop(context, ol.format.KML.POLY_STYLE_SERIALIZERS_, ol.format.KML.COLOR_NODE_FACTORY_, [style.getColor()], objectStack); }; /** * @param {Node} node Node to append a TextNode with the scale to. * @param {number|undefined} scale Scale. * @private */ ol.format.KML.writeScaleTextNode_ = function(node, scale) { ol.format.XSD.writeDecimalTextNode(node, scale * scale); }; /** * @param {Node} node Node. * @param {ol.style.Style} style Style. * @param {Array.<*>} objectStack Object stack. * @private */ ol.format.KML.writeStyle_ = function(node, style, objectStack) { var /** @type {ol.xml.NodeStackItem} */ context = {node: node}; var properties = {}; var fillStyle = style.getFill(); var strokeStyle = style.getStroke(); var imageStyle = style.getImage(); var textStyle = style.getText(); if (!goog.isNull(imageStyle)) { properties['IconStyle'] = imageStyle; } if (!goog.isNull(textStyle)) { properties['LabelStyle'] = textStyle; } if (!goog.isNull(strokeStyle)) { properties['LineStyle'] = strokeStyle; } if (!goog.isNull(fillStyle)) { properties['PolyStyle'] = fillStyle; } var parentNode = objectStack[objectStack.length - 1].node; var orderedKeys = ol.format.KML.STYLE_SEQUENCE_[parentNode.namespaceURI]; var values = ol.xml.makeSequence(properties, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.STYLE_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, objectStack, orderedKeys); }; /** * @param {Node} node Node to append a TextNode with the Vec2 to. * @param {ol.format.KMLVec2_} vec2 Vec2. * @private */ ol.format.KML.writeVec2_ = function(node, vec2) { node.setAttribute('x', vec2.x); node.setAttribute('y', vec2.y); node.setAttribute('xunits', vec2.xunits); node.setAttribute('yunits', vec2.yunits); }; /** * @const * @type {Object.>} * @private */ ol.format.KML.KML_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'Document', 'Placemark' ]); /** * @const * @type {Object.>} * @private */ ol.format.KML.KML_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Document': ol.xml.makeChildAppender(ol.format.KML.writeDocument_), 'Placemark': ol.xml.makeChildAppender(ol.format.KML.writePlacemark_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.DOCUMENT_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Placemark': ol.xml.makeChildAppender(ol.format.KML.writePlacemark_) }); /** * @const * @type {Object.} * @private */ ol.format.KML.GEOMETRY_TYPE_TO_NODENAME_ = { 'Point': 'Point', 'LineString': 'LineString', 'LinearRing': 'LinearRing', 'Polygon': 'Polygon', 'MultiPoint': 'MultiGeometry', 'MultiLineString': 'MultiGeometry', 'MultiPolygon': 'MultiGeometry' }; /** * @const * @type {Object.>} * @private */ ol.format.KML.ICON_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'href' ], ol.xml.makeStructureNS( ol.format.KML.GX_NAMESPACE_URIS_, [ 'x', 'y', 'w', 'h' ])); /** * @const * @type {Object.>} * @private */ ol.format.KML.ICON_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'href': ol.xml.makeChildAppender(ol.format.XSD.writeStringTextNode) }, ol.xml.makeStructureNS( ol.format.KML.GX_NAMESPACE_URIS_, { 'x': ol.xml.makeChildAppender(ol.format.XSD.writeDecimalTextNode), 'y': ol.xml.makeChildAppender(ol.format.XSD.writeDecimalTextNode), 'w': ol.xml.makeChildAppender(ol.format.XSD.writeDecimalTextNode), 'h': ol.xml.makeChildAppender(ol.format.XSD.writeDecimalTextNode) })); /** * @const * @type {Object.>} * @private */ ol.format.KML.ICON_STYLE_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'scale', 'heading', 'Icon', 'hotSpot' ]); /** * @const * @type {Object.>} * @private */ ol.format.KML.ICON_STYLE_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'Icon': ol.xml.makeChildAppender(ol.format.KML.writeIcon_), 'heading': ol.xml.makeChildAppender(ol.format.XSD.writeDecimalTextNode), 'hotSpot': ol.xml.makeChildAppender(ol.format.KML.writeVec2_), 'scale': ol.xml.makeChildAppender(ol.format.KML.writeScaleTextNode_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.LABEL_STYLE_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'color', 'scale' ]); /** * @const * @type {Object.>} * @private */ ol.format.KML.LABEL_STYLE_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'color': ol.xml.makeChildAppender(ol.format.KML.writeColorTextNode_), 'scale': ol.xml.makeChildAppender(ol.format.KML.writeScaleTextNode_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.LINE_STYLE_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'color', 'width' ]); /** * @const * @type {Object.>} * @private */ ol.format.KML.LINE_STYLE_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'color': ol.xml.makeChildAppender(ol.format.KML.writeColorTextNode_), 'width': ol.xml.makeChildAppender(ol.format.XSD.writeDecimalTextNode) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.BOUNDARY_IS_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'LinearRing': ol.xml.makeChildAppender( ol.format.KML.writePrimitiveGeometry_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.MULTI_GEOMETRY_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'LineString': ol.xml.makeChildAppender( ol.format.KML.writePrimitiveGeometry_), 'Point': ol.xml.makeChildAppender( ol.format.KML.writePrimitiveGeometry_), 'Polygon': ol.xml.makeChildAppender(ol.format.KML.writePolygon_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.PLACEMARK_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'name', 'open', 'visibility', 'address', 'phoneNumber', 'description', 'styleUrl', 'Style' ]); /** * @const * @type {Object.>} * @private */ ol.format.KML.PLACEMARK_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'MultiGeometry': ol.xml.makeChildAppender( ol.format.KML.writeMultiGeometry_), 'LineString': ol.xml.makeChildAppender( ol.format.KML.writePrimitiveGeometry_), 'LinearRing': ol.xml.makeChildAppender( ol.format.KML.writePrimitiveGeometry_), 'Point': ol.xml.makeChildAppender( ol.format.KML.writePrimitiveGeometry_), 'Polygon': ol.xml.makeChildAppender(ol.format.KML.writePolygon_), 'Style': ol.xml.makeChildAppender(ol.format.KML.writeStyle_), 'address': ol.xml.makeChildAppender(ol.format.XSD.writeStringTextNode), 'description': ol.xml.makeChildAppender( ol.format.XSD.writeStringTextNode), 'name': ol.xml.makeChildAppender(ol.format.XSD.writeStringTextNode), 'open': ol.xml.makeChildAppender(ol.format.XSD.writeBooleanTextNode), 'phoneNumber': ol.xml.makeChildAppender( ol.format.XSD.writeStringTextNode), 'styleUrl': ol.xml.makeChildAppender(ol.format.XSD.writeStringTextNode), 'visibility': ol.xml.makeChildAppender( ol.format.XSD.writeBooleanTextNode) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.PRIMITIVE_GEOMETRY_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'coordinates': ol.xml.makeChildAppender( ol.format.KML.writeCoordinatesTextNode_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.POLYGON_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'outerBoundaryIs': ol.xml.makeChildAppender( ol.format.KML.writeBoundaryIs_), 'innerBoundaryIs': ol.xml.makeChildAppender( ol.format.KML.writeBoundaryIs_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.POLY_STYLE_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'color': ol.xml.makeChildAppender(ol.format.KML.writeColorTextNode_) }); /** * @const * @type {Object.>} * @private */ ol.format.KML.STYLE_SEQUENCE_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, [ 'IconStyle', 'LabelStyle', 'LineStyle', 'PolyStyle' ]); /** * @const * @type {Object.>} * @private */ ol.format.KML.STYLE_SERIALIZERS_ = ol.xml.makeStructureNS( ol.format.KML.NAMESPACE_URIS_, { 'IconStyle': ol.xml.makeChildAppender(ol.format.KML.writeIconStyle_), 'LabelStyle': ol.xml.makeChildAppender(ol.format.KML.writeLabelStyle_), 'LineStyle': ol.xml.makeChildAppender(ol.format.KML.writeLineStyle_), 'PolyStyle': ol.xml.makeChildAppender(ol.format.KML.writePolyStyle_) }); /** * @const * @param {*} value Value. * @param {Array.<*>} objectStack Object stack. * @param {string=} opt_nodeName Node name. * @return {Node|undefined} Node. * @private */ ol.format.KML.GX_NODE_FACTORY_ = function(value, objectStack, opt_nodeName) { return ol.xml.createElementNS(ol.format.KML.GX_NAMESPACE_URIS_[0], 'gx:' + opt_nodeName); }; /** * @const * @param {*} value Value. * @param {Array.<*>} objectStack Object stack. * @param {string=} opt_nodeName Node name. * @return {Node|undefined} Node. * @private */ ol.format.KML.DOCUMENT_NODE_FACTORY_ = function(value, objectStack, opt_nodeName) { goog.asserts.assertInstanceof(value, ol.Feature, 'value should be an ol.Feature'); var parentNode = objectStack[objectStack.length - 1].node; goog.asserts.assert(ol.xml.isNode(parentNode), 'parentNode should be an XML node'); return ol.xml.createElementNS(parentNode.namespaceURI, 'Placemark'); }; /** * @const * @param {*} value Value. * @param {Array.<*>} objectStack Object stack. * @param {string=} opt_nodeName Node name. * @return {Node|undefined} Node. * @private */ ol.format.KML.GEOMETRY_NODE_FACTORY_ = function(value, objectStack, opt_nodeName) { if (goog.isDefAndNotNull(value)) { goog.asserts.assertInstanceof(value, ol.geom.Geometry, 'value should be an ol.geom.Geometry'); var parentNode = objectStack[objectStack.length - 1].node; goog.asserts.assert(ol.xml.isNode(parentNode), 'parentNode should be an XML node'); return ol.xml.createElementNS(parentNode.namespaceURI, ol.format.KML.GEOMETRY_TYPE_TO_NODENAME_[value.getType()]); } }; /** * A factory for creating coordinates nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.COLOR_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('color'); /** * A factory for creating coordinates nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.COORDINATES_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('coordinates'); /** * A factory for creating innerBoundaryIs nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.INNER_BOUNDARY_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('innerBoundaryIs'); /** * A factory for creating Point nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.POINT_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('Point'); /** * A factory for creating LineString nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.LINE_STRING_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('LineString'); /** * A factory for creating LinearRing nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.LINEAR_RING_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('LinearRing'); /** * A factory for creating Polygon nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.POLYGON_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('Polygon'); /** * A factory for creating outerBoundaryIs nodes. * @const * @type {function(*, Array.<*>, string=): (Node|undefined)} * @private */ ol.format.KML.OUTER_BOUNDARY_NODE_FACTORY_ = ol.xml.makeSimpleNodeFactory('outerBoundaryIs'); /** * Encode an array of features in the KML format. * * @function * @param {Array.} features Features. * @param {olx.format.WriteOptions=} opt_options Options. * @return {string} Result. * @api stable */ ol.format.KML.prototype.writeFeatures; /** * Encode an array of features in the KML format as an XML node. * * @param {Array.} features Features. * @param {olx.format.WriteOptions=} opt_options Options. * @return {Node} Node. * @api */ ol.format.KML.prototype.writeFeaturesNode = function(features, opt_options) { opt_options = this.adaptOptions(opt_options); var kml = ol.xml.createElementNS(ol.format.KML.NAMESPACE_URIS_[4], 'kml'); var xmlnsUri = 'http://www.w3.org/2000/xmlns/'; var xmlSchemaInstanceUri = 'http://www.w3.org/2001/XMLSchema-instance'; ol.xml.setAttributeNS(kml, xmlnsUri, 'xmlns:gx', ol.format.KML.GX_NAMESPACE_URIS_[0]); ol.xml.setAttributeNS(kml, xmlnsUri, 'xmlns:xsi', xmlSchemaInstanceUri); ol.xml.setAttributeNS(kml, xmlSchemaInstanceUri, 'xsi:schemaLocation', ol.format.KML.SCHEMA_LOCATION_); var /** @type {ol.xml.NodeStackItem} */ context = {node: kml}; var properties = {}; if (features.length > 1) { properties['Document'] = features; } else if (features.length == 1) { properties['Placemark'] = features[0]; } var orderedKeys = ol.format.KML.KML_SEQUENCE_[kml.namespaceURI]; var values = ol.xml.makeSequence(properties, orderedKeys); ol.xml.pushSerializeAndPop(context, ol.format.KML.KML_SERIALIZERS_, ol.xml.OBJECT_PROPERTY_NODE_FACTORY, values, [opt_options], orderedKeys); return kml; };