/** * @module ol/source/Vector */ var __extends = (this && this.__extends) || (function () { var extendStatics = function (d, b) { extendStatics = Object.setPrototypeOf || ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; }; return extendStatics(d, b); }; return function (d, b) { if (typeof b !== "function" && b !== null) throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); extendStatics(d, b); function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; })(); import Collection from '../Collection.js'; import CollectionEventType from '../CollectionEventType.js'; import Event from '../events/Event.js'; import EventType from '../events/EventType.js'; import ObjectEventType from '../ObjectEventType.js'; import RBush from '../structs/RBush.js'; import Source from './Source.js'; import VectorEventType from './VectorEventType.js'; import { TRUE, VOID } from '../functions.js'; import { all as allStrategy } from '../loadingstrategy.js'; import { assert } from '../asserts.js'; import { containsExtent, equals, wrapAndSliceX } from '../extent.js'; import { extend } from '../array.js'; import { getUid } from '../util.js'; import { getValues, isEmpty } from '../obj.js'; import { listen, unlistenByKey } from '../events.js'; import { xhr } from '../featureloader.js'; /** * A function that takes an {@link module:ol/extent~Extent} and a resolution as arguments, and * returns an array of {@link module:ol/extent~Extent} with the extents to load. Usually this * is one of the standard {@link module:ol/loadingstrategy} strategies. * * @typedef {function(import("../extent.js").Extent, number, import("../proj/Projection.js").default): Array} LoadingStrategy * @api */ /** * @classdesc * Events emitted by {@link module:ol/source/Vector~VectorSource} instances are instances of this * type. * @template {import("../geom/Geometry.js").default} [Geometry=import("../geom/Geometry.js").default] */ var VectorSourceEvent = /** @class */ (function (_super) { __extends(VectorSourceEvent, _super); /** * @param {string} type Type. * @param {import("../Feature.js").default} [opt_feature] Feature. * @param {Array>} [opt_features] Features. */ function VectorSourceEvent(type, opt_feature, opt_features) { var _this = _super.call(this, type) || this; /** * The added or removed feature for the `ADDFEATURE` and `REMOVEFEATURE` events, `undefined` otherwise. * @type {import("../Feature.js").default|undefined} * @api */ _this.feature = opt_feature; /** * The loaded features for the `FEATURESLOADED` event, `undefined` otherwise. * @type {Array>|undefined} * @api */ _this.features = opt_features; return _this; } return VectorSourceEvent; }(Event)); export { VectorSourceEvent }; /*** * @template Return * @typedef {import("../Observable").OnSignature & * import("../Observable").OnSignature & * import("../Observable").OnSignature & * import("../Observable").CombinedOnSignature} VectorSourceOnSignature */ /** * @typedef {Object} Options * @property {import("./Source.js").AttributionLike} [attributions] Attributions. * @property {Array|Collection} [features] * Features. If provided as {@link module:ol/Collection~Collection}, the features in the source * and the collection will stay in sync. * @property {import("../format/Feature.js").default} [format] The feature format used by the XHR * feature loader when `url` is set. Required if `url` is set, otherwise ignored. * @property {import("../featureloader.js").FeatureLoader} [loader] * The loader function used to load features, from a remote source for example. * If this is not set and `url` is set, the source will create and use an XHR * feature loader. The `'featuresloadend'` and `'featuresloaderror'` events * will only fire if the `success` and `failure` callbacks are used. * * Example: * * ```js * import {Vector} from 'ol/source'; * import {GeoJSON} from 'ol/format'; * import {bbox} from 'ol/loadingstrategy'; * * var vectorSource = new Vector({ * format: new GeoJSON(), * loader: function(extent, resolution, projection, success, failure) { * var proj = projection.getCode(); * var url = 'https://ahocevar.com/geoserver/wfs?service=WFS&' + * 'version=1.1.0&request=GetFeature&typename=osm:water_areas&' + * 'outputFormat=application/json&srsname=' + proj + '&' + * 'bbox=' + extent.join(',') + ',' + proj; * var xhr = new XMLHttpRequest(); * xhr.open('GET', url); * var onError = function() { * vectorSource.removeLoadedExtent(extent); * failure(); * } * xhr.onerror = onError; * xhr.onload = function() { * if (xhr.status == 200) { * var features = vectorSource.getFormat().readFeatures(xhr.responseText); * vectorSource.addFeatures(features); * success(features); * } else { * onError(); * } * } * xhr.send(); * }, * strategy: bbox * }); * ``` * @property {boolean} [overlaps=true] This source may have overlapping geometries. * Setting this to `false` (e.g. for sources with polygons that represent administrative * boundaries or TopoJSON sources) allows the renderer to optimise fill and * stroke operations. * @property {LoadingStrategy} [strategy] The loading strategy to use. * By default an {@link module:ol/loadingstrategy.all} * strategy is used, a one-off strategy which loads all features at once. * @property {string|import("../featureloader.js").FeatureUrlFunction} [url] * Setting this option instructs the source to load features using an XHR loader * (see {@link module:ol/featureloader.xhr}). Use a `string` and an * {@link module:ol/loadingstrategy.all} for a one-off download of all features from * the given URL. Use a {@link module:ol/featureloader~FeatureUrlFunction} to generate the url with * other loading strategies. * Requires `format` to be set as well. * When default XHR feature loader is provided, the features will * be transformed from the data projection to the view projection * during parsing. If your remote data source does not advertise its projection * properly, this transformation will be incorrect. For some formats, the * default projection (usually EPSG:4326) can be overridden by setting the * dataProjection constructor option on the format. * Note that if a source contains non-feature data, such as a GeoJSON geometry * or a KML NetworkLink, these will be ignored. Use a custom loader to load these. * @property {boolean} [useSpatialIndex=true] * By default, an RTree is used as spatial index. When features are removed and * added frequently, and the total number of features is low, setting this to * `false` may improve performance. * * Note that * {@link module:ol/source/Vector~VectorSource#getFeaturesInExtent}, * {@link module:ol/source/Vector~VectorSource#getClosestFeatureToCoordinate} and * {@link module:ol/source/Vector~VectorSource#getExtent} cannot be used when `useSpatialIndex` is * set to `false`, and {@link module:ol/source/Vector~VectorSource#forEachFeatureInExtent} will loop * through all features. * * When set to `false`, the features will be maintained in an * {@link module:ol/Collection~Collection}, which can be retrieved through * {@link module:ol/source/Vector~VectorSource#getFeaturesCollection}. * @property {boolean} [wrapX=true] Wrap the world horizontally. For vector editing across the * -180° and 180° meridians to work properly, this should be set to `false`. The * resulting geometry coordinates will then exceed the world bounds. */ /** * @classdesc * Provides a source of features for vector layers. Vector features provided * by this source are suitable for editing. See {@link module:ol/source/VectorTile~VectorTile} for * vector data that is optimized for rendering. * * @fires VectorSourceEvent * @api * @template {import("../geom/Geometry.js").default} [Geometry=import("../geom/Geometry.js").default] */ var VectorSource = /** @class */ (function (_super) { __extends(VectorSource, _super); /** * @param {Options} [opt_options] Vector source options. */ function VectorSource(opt_options) { var _this = this; var options = opt_options || {}; _this = _super.call(this, { attributions: options.attributions, interpolate: true, projection: undefined, state: 'ready', wrapX: options.wrapX !== undefined ? options.wrapX : true, }) || this; /*** * @type {VectorSourceOnSignature} */ _this.on; /*** * @type {VectorSourceOnSignature} */ _this.once; /*** * @type {VectorSourceOnSignature} */ _this.un; /** * @private * @type {import("../featureloader.js").FeatureLoader} */ _this.loader_ = VOID; /** * @private * @type {import("../format/Feature.js").default|undefined} */ _this.format_ = options.format; /** * @private * @type {boolean} */ _this.overlaps_ = options.overlaps === undefined ? true : options.overlaps; /** * @private * @type {string|import("../featureloader.js").FeatureUrlFunction|undefined} */ _this.url_ = options.url; if (options.loader !== undefined) { _this.loader_ = options.loader; } else if (_this.url_ !== undefined) { assert(_this.format_, 7); // `format` must be set when `url` is set // create a XHR feature loader for "url" and "format" _this.loader_ = xhr(_this.url_, /** @type {import("../format/Feature.js").default} */ (_this.format_)); } /** * @private * @type {LoadingStrategy} */ _this.strategy_ = options.strategy !== undefined ? options.strategy : allStrategy; var useSpatialIndex = options.useSpatialIndex !== undefined ? options.useSpatialIndex : true; /** * @private * @type {RBush>} */ _this.featuresRtree_ = useSpatialIndex ? new RBush() : null; /** * @private * @type {RBush<{extent: import("../extent.js").Extent}>} */ _this.loadedExtentsRtree_ = new RBush(); /** * @type {number} * @private */ _this.loadingExtentsCount_ = 0; /** * @private * @type {!Object>} */ _this.nullGeometryFeatures_ = {}; /** * A lookup of features by id (the return from feature.getId()). * @private * @type {!Object>} */ _this.idIndex_ = {}; /** * A lookup of features by uid (using getUid(feature)). * @private * @type {!Object>} */ _this.uidIndex_ = {}; /** * @private * @type {Object>} */ _this.featureChangeKeys_ = {}; /** * @private * @type {Collection>|null} */ _this.featuresCollection_ = null; var collection, features; if (Array.isArray(options.features)) { features = /** @type {Array>} */ (options.features); } else if (options.features) { collection = /** @type {Collection>} */ (options.features); features = collection.getArray(); } if (!useSpatialIndex && collection === undefined) { collection = new Collection(features); } if (features !== undefined) { _this.addFeaturesInternal(features); } if (collection !== undefined) { _this.bindFeaturesCollection_(collection); } return _this; } /** * Add a single feature to the source. If you want to add a batch of features * at once, call {@link module:ol/source/Vector~VectorSource#addFeatures #addFeatures()} * instead. A feature will not be added to the source if feature with * the same id is already there. The reason for this behavior is to avoid * feature duplication when using bbox or tile loading strategies. * Note: this also applies if an {@link module:ol/Collection~Collection} is used for features, * meaning that if a feature with a duplicate id is added in the collection, it will * be removed from it right away. * @param {import("../Feature.js").default} feature Feature to add. * @api */ VectorSource.prototype.addFeature = function (feature) { this.addFeatureInternal(feature); this.changed(); }; /** * Add a feature without firing a `change` event. * @param {import("../Feature.js").default} feature Feature. * @protected */ VectorSource.prototype.addFeatureInternal = function (feature) { var featureKey = getUid(feature); if (!this.addToIndex_(featureKey, feature)) { if (this.featuresCollection_) { this.featuresCollection_.remove(feature); } return; } this.setupChangeEvents_(featureKey, feature); var geometry = feature.getGeometry(); if (geometry) { var extent = geometry.getExtent(); if (this.featuresRtree_) { this.featuresRtree_.insert(extent, feature); } } else { this.nullGeometryFeatures_[featureKey] = feature; } this.dispatchEvent(new VectorSourceEvent(VectorEventType.ADDFEATURE, feature)); }; /** * @param {string} featureKey Unique identifier for the feature. * @param {import("../Feature.js").default} feature The feature. * @private */ VectorSource.prototype.setupChangeEvents_ = function (featureKey, feature) { this.featureChangeKeys_[featureKey] = [ listen(feature, EventType.CHANGE, this.handleFeatureChange_, this), listen(feature, ObjectEventType.PROPERTYCHANGE, this.handleFeatureChange_, this), ]; }; /** * @param {string} featureKey Unique identifier for the feature. * @param {import("../Feature.js").default} feature The feature. * @return {boolean} The feature is "valid", in the sense that it is also a * candidate for insertion into the Rtree. * @private */ VectorSource.prototype.addToIndex_ = function (featureKey, feature) { var valid = true; var id = feature.getId(); if (id !== undefined) { if (!(id.toString() in this.idIndex_)) { this.idIndex_[id.toString()] = feature; } else { valid = false; } } if (valid) { assert(!(featureKey in this.uidIndex_), 30); // The passed `feature` was already added to the source this.uidIndex_[featureKey] = feature; } return valid; }; /** * Add a batch of features to the source. * @param {Array>} features Features to add. * @api */ VectorSource.prototype.addFeatures = function (features) { this.addFeaturesInternal(features); this.changed(); }; /** * Add features without firing a `change` event. * @param {Array>} features Features. * @protected */ VectorSource.prototype.addFeaturesInternal = function (features) { var extents = []; var newFeatures = []; var geometryFeatures = []; for (var i = 0, length_1 = features.length; i < length_1; i++) { var feature = features[i]; var featureKey = getUid(feature); if (this.addToIndex_(featureKey, feature)) { newFeatures.push(feature); } } for (var i = 0, length_2 = newFeatures.length; i < length_2; i++) { var feature = newFeatures[i]; var featureKey = getUid(feature); this.setupChangeEvents_(featureKey, feature); var geometry = feature.getGeometry(); if (geometry) { var extent = geometry.getExtent(); extents.push(extent); geometryFeatures.push(feature); } else { this.nullGeometryFeatures_[featureKey] = feature; } } if (this.featuresRtree_) { this.featuresRtree_.load(extents, geometryFeatures); } if (this.hasListener(VectorEventType.ADDFEATURE)) { for (var i = 0, length_3 = newFeatures.length; i < length_3; i++) { this.dispatchEvent(new VectorSourceEvent(VectorEventType.ADDFEATURE, newFeatures[i])); } } }; /** * @param {!Collection>} collection Collection. * @private */ VectorSource.prototype.bindFeaturesCollection_ = function (collection) { var modifyingCollection = false; this.addEventListener(VectorEventType.ADDFEATURE, /** * @param {VectorSourceEvent} evt The vector source event */ function (evt) { if (!modifyingCollection) { modifyingCollection = true; collection.push(evt.feature); modifyingCollection = false; } }); this.addEventListener(VectorEventType.REMOVEFEATURE, /** * @param {VectorSourceEvent} evt The vector source event */ function (evt) { if (!modifyingCollection) { modifyingCollection = true; collection.remove(evt.feature); modifyingCollection = false; } }); collection.addEventListener(CollectionEventType.ADD, /** * @param {import("../Collection.js").CollectionEvent} evt The collection event */ function (evt) { if (!modifyingCollection) { modifyingCollection = true; this.addFeature( /** @type {import("../Feature.js").default} */ (evt.element)); modifyingCollection = false; } }.bind(this)); collection.addEventListener(CollectionEventType.REMOVE, /** * @param {import("../Collection.js").CollectionEvent} evt The collection event */ function (evt) { if (!modifyingCollection) { modifyingCollection = true; this.removeFeature( /** @type {import("../Feature.js").default} */ (evt.element)); modifyingCollection = false; } }.bind(this)); this.featuresCollection_ = collection; }; /** * Remove all features from the source. * @param {boolean} [opt_fast] Skip dispatching of {@link module:ol/source/Vector.VectorSourceEvent#event:removefeature removefeature} events. * @api */ VectorSource.prototype.clear = function (opt_fast) { if (opt_fast) { for (var featureId in this.featureChangeKeys_) { var keys = this.featureChangeKeys_[featureId]; keys.forEach(unlistenByKey); } if (!this.featuresCollection_) { this.featureChangeKeys_ = {}; this.idIndex_ = {}; this.uidIndex_ = {}; } } else { if (this.featuresRtree_) { var removeAndIgnoreReturn = function (feature) { this.removeFeatureInternal(feature); }.bind(this); this.featuresRtree_.forEach(removeAndIgnoreReturn); for (var id in this.nullGeometryFeatures_) { this.removeFeatureInternal(this.nullGeometryFeatures_[id]); } } } if (this.featuresCollection_) { this.featuresCollection_.clear(); } if (this.featuresRtree_) { this.featuresRtree_.clear(); } this.nullGeometryFeatures_ = {}; var clearEvent = new VectorSourceEvent(VectorEventType.CLEAR); this.dispatchEvent(clearEvent); this.changed(); }; /** * Iterate through all features on the source, calling the provided callback * with each one. If the callback returns any "truthy" value, iteration will * stop and the function will return the same value. * Note: this function only iterate through the feature that have a defined geometry. * * @param {function(import("../Feature.js").default): T} callback Called with each feature * on the source. Return a truthy value to stop iteration. * @return {T|undefined} The return value from the last call to the callback. * @template T * @api */ VectorSource.prototype.forEachFeature = function (callback) { if (this.featuresRtree_) { return this.featuresRtree_.forEach(callback); } else if (this.featuresCollection_) { this.featuresCollection_.forEach(callback); } }; /** * Iterate through all features whose geometries contain the provided * coordinate, calling the callback with each feature. If the callback returns * a "truthy" value, iteration will stop and the function will return the same * value. * * @param {import("../coordinate.js").Coordinate} coordinate Coordinate. * @param {function(import("../Feature.js").default): T} callback Called with each feature * whose goemetry contains the provided coordinate. * @return {T|undefined} The return value from the last call to the callback. * @template T */ VectorSource.prototype.forEachFeatureAtCoordinateDirect = function (coordinate, callback) { var extent = [coordinate[0], coordinate[1], coordinate[0], coordinate[1]]; return this.forEachFeatureInExtent(extent, function (feature) { var geometry = feature.getGeometry(); if (geometry.intersectsCoordinate(coordinate)) { return callback(feature); } else { return undefined; } }); }; /** * Iterate through all features whose bounding box intersects the provided * extent (note that the feature's geometry may not intersect the extent), * calling the callback with each feature. If the callback returns a "truthy" * value, iteration will stop and the function will return the same value. * * If you are interested in features whose geometry intersects an extent, call * the {@link module:ol/source/Vector~VectorSource#forEachFeatureIntersectingExtent #forEachFeatureIntersectingExtent()} method instead. * * When `useSpatialIndex` is set to false, this method will loop through all * features, equivalent to {@link module:ol/source/Vector~VectorSource#forEachFeature #forEachFeature()}. * * @param {import("../extent.js").Extent} extent Extent. * @param {function(import("../Feature.js").default): T} callback Called with each feature * whose bounding box intersects the provided extent. * @return {T|undefined} The return value from the last call to the callback. * @template T * @api */ VectorSource.prototype.forEachFeatureInExtent = function (extent, callback) { if (this.featuresRtree_) { return this.featuresRtree_.forEachInExtent(extent, callback); } else if (this.featuresCollection_) { this.featuresCollection_.forEach(callback); } }; /** * Iterate through all features whose geometry intersects the provided extent, * calling the callback with each feature. If the callback returns a "truthy" * value, iteration will stop and the function will return the same value. * * If you only want to test for bounding box intersection, call the * {@link module:ol/source/Vector~VectorSource#forEachFeatureInExtent #forEachFeatureInExtent()} method instead. * * @param {import("../extent.js").Extent} extent Extent. * @param {function(import("../Feature.js").default): T} callback Called with each feature * whose geometry intersects the provided extent. * @return {T|undefined} The return value from the last call to the callback. * @template T * @api */ VectorSource.prototype.forEachFeatureIntersectingExtent = function (extent, callback) { return this.forEachFeatureInExtent(extent, /** * @param {import("../Feature.js").default} feature Feature. * @return {T|undefined} The return value from the last call to the callback. */ function (feature) { var geometry = feature.getGeometry(); if (geometry.intersectsExtent(extent)) { var result = callback(feature); if (result) { return result; } } }); }; /** * Get the features collection associated with this source. Will be `null` * unless the source was configured with `useSpatialIndex` set to `false`, or * with an {@link module:ol/Collection~Collection} as `features`. * @return {Collection>|null} The collection of features. * @api */ VectorSource.prototype.getFeaturesCollection = function () { return this.featuresCollection_; }; /** * Get a snapshot of the features currently on the source in random order. The returned array * is a copy, the features are references to the features in the source. * @return {Array>} Features. * @api */ VectorSource.prototype.getFeatures = function () { var features; if (this.featuresCollection_) { features = this.featuresCollection_.getArray().slice(0); } else if (this.featuresRtree_) { features = this.featuresRtree_.getAll(); if (!isEmpty(this.nullGeometryFeatures_)) { extend(features, getValues(this.nullGeometryFeatures_)); } } return /** @type {Array>} */ (features); }; /** * Get all features whose geometry intersects the provided coordinate. * @param {import("../coordinate.js").Coordinate} coordinate Coordinate. * @return {Array>} Features. * @api */ VectorSource.prototype.getFeaturesAtCoordinate = function (coordinate) { var features = []; this.forEachFeatureAtCoordinateDirect(coordinate, function (feature) { features.push(feature); }); return features; }; /** * Get all features whose bounding box intersects the provided extent. Note that this returns an array of * all features intersecting the given extent in random order (so it may include * features whose geometries do not intersect the extent). * * When `useSpatialIndex` is set to false, this method will return all * features. * * @param {import("../extent.js").Extent} extent Extent. * @param {import("../proj/Projection.js").default} [opt_projection] Include features * where `extent` exceeds the x-axis bounds of `projection` and wraps around the world. * @return {Array>} Features. * @api */ VectorSource.prototype.getFeaturesInExtent = function (extent, opt_projection) { var _this = this; if (this.featuresRtree_) { var multiWorld = opt_projection && opt_projection.canWrapX() && this.getWrapX(); if (!multiWorld) { return this.featuresRtree_.getInExtent(extent); } var extents = wrapAndSliceX(extent, opt_projection); return [].concat.apply([], extents.map(function (anExtent) { return _this.featuresRtree_.getInExtent(anExtent); })); } else if (this.featuresCollection_) { return this.featuresCollection_.getArray().slice(0); } else { return []; } }; /** * Get the closest feature to the provided coordinate. * * This method is not available when the source is configured with * `useSpatialIndex` set to `false`. * @param {import("../coordinate.js").Coordinate} coordinate Coordinate. * @param {function(import("../Feature.js").default):boolean} [opt_filter] Feature filter function. * The filter function will receive one argument, the {@link module:ol/Feature~Feature feature} * and it should return a boolean value. By default, no filtering is made. * @return {import("../Feature.js").default} Closest feature. * @api */ VectorSource.prototype.getClosestFeatureToCoordinate = function (coordinate, opt_filter) { // Find the closest feature using branch and bound. We start searching an // infinite extent, and find the distance from the first feature found. This // becomes the closest feature. We then compute a smaller extent which any // closer feature must intersect. We continue searching with this smaller // extent, trying to find a closer feature. Every time we find a closer // feature, we update the extent being searched so that any even closer // feature must intersect it. We continue until we run out of features. var x = coordinate[0]; var y = coordinate[1]; var closestFeature = null; var closestPoint = [NaN, NaN]; var minSquaredDistance = Infinity; var extent = [-Infinity, -Infinity, Infinity, Infinity]; var filter = opt_filter ? opt_filter : TRUE; this.featuresRtree_.forEachInExtent(extent, /** * @param {import("../Feature.js").default} feature Feature. */ function (feature) { if (filter(feature)) { var geometry = feature.getGeometry(); var previousMinSquaredDistance = minSquaredDistance; minSquaredDistance = geometry.closestPointXY(x, y, closestPoint, minSquaredDistance); if (minSquaredDistance < previousMinSquaredDistance) { closestFeature = feature; // This is sneaky. Reduce the extent that it is currently being // searched while the R-Tree traversal using this same extent object // is still in progress. This is safe because the new extent is // strictly contained by the old extent. var minDistance = Math.sqrt(minSquaredDistance); extent[0] = x - minDistance; extent[1] = y - minDistance; extent[2] = x + minDistance; extent[3] = y + minDistance; } } }); return closestFeature; }; /** * Get the extent of the features currently in the source. * * This method is not available when the source is configured with * `useSpatialIndex` set to `false`. * @param {import("../extent.js").Extent} [opt_extent] Destination extent. If provided, no new extent * will be created. Instead, that extent's coordinates will be overwritten. * @return {import("../extent.js").Extent} Extent. * @api */ VectorSource.prototype.getExtent = function (opt_extent) { return this.featuresRtree_.getExtent(opt_extent); }; /** * Get a feature by its identifier (the value returned by feature.getId()). * Note that the index treats string and numeric identifiers as the same. So * `source.getFeatureById(2)` will return a feature with id `'2'` or `2`. * * @param {string|number} id Feature identifier. * @return {import("../Feature.js").default|null} The feature (or `null` if not found). * @api */ VectorSource.prototype.getFeatureById = function (id) { var feature = this.idIndex_[id.toString()]; return feature !== undefined ? feature : null; }; /** * Get a feature by its internal unique identifier (using `getUid`). * * @param {string} uid Feature identifier. * @return {import("../Feature.js").default|null} The feature (or `null` if not found). */ VectorSource.prototype.getFeatureByUid = function (uid) { var feature = this.uidIndex_[uid]; return feature !== undefined ? feature : null; }; /** * Get the format associated with this source. * * @return {import("../format/Feature.js").default|undefined} The feature format. * @api */ VectorSource.prototype.getFormat = function () { return this.format_; }; /** * @return {boolean} The source can have overlapping geometries. */ VectorSource.prototype.getOverlaps = function () { return this.overlaps_; }; /** * Get the url associated with this source. * * @return {string|import("../featureloader.js").FeatureUrlFunction|undefined} The url. * @api */ VectorSource.prototype.getUrl = function () { return this.url_; }; /** * @param {Event} event Event. * @private */ VectorSource.prototype.handleFeatureChange_ = function (event) { var feature = /** @type {import("../Feature.js").default} */ (event.target); var featureKey = getUid(feature); var geometry = feature.getGeometry(); if (!geometry) { if (!(featureKey in this.nullGeometryFeatures_)) { if (this.featuresRtree_) { this.featuresRtree_.remove(feature); } this.nullGeometryFeatures_[featureKey] = feature; } } else { var extent = geometry.getExtent(); if (featureKey in this.nullGeometryFeatures_) { delete this.nullGeometryFeatures_[featureKey]; if (this.featuresRtree_) { this.featuresRtree_.insert(extent, feature); } } else { if (this.featuresRtree_) { this.featuresRtree_.update(extent, feature); } } } var id = feature.getId(); if (id !== undefined) { var sid = id.toString(); if (this.idIndex_[sid] !== feature) { this.removeFromIdIndex_(feature); this.idIndex_[sid] = feature; } } else { this.removeFromIdIndex_(feature); this.uidIndex_[featureKey] = feature; } this.changed(); this.dispatchEvent(new VectorSourceEvent(VectorEventType.CHANGEFEATURE, feature)); }; /** * Returns true if the feature is contained within the source. * @param {import("../Feature.js").default} feature Feature. * @return {boolean} Has feature. * @api */ VectorSource.prototype.hasFeature = function (feature) { var id = feature.getId(); if (id !== undefined) { return id in this.idIndex_; } else { return getUid(feature) in this.uidIndex_; } }; /** * @return {boolean} Is empty. */ VectorSource.prototype.isEmpty = function () { if (this.featuresRtree_) { return (this.featuresRtree_.isEmpty() && isEmpty(this.nullGeometryFeatures_)); } if (this.featuresCollection_) { return this.featuresCollection_.getLength() === 0; } return true; }; /** * @param {import("../extent.js").Extent} extent Extent. * @param {number} resolution Resolution. * @param {import("../proj/Projection.js").default} projection Projection. */ VectorSource.prototype.loadFeatures = function (extent, resolution, projection) { var loadedExtentsRtree = this.loadedExtentsRtree_; var extentsToLoad = this.strategy_(extent, resolution, projection); var _loop_1 = function (i, ii) { var extentToLoad = extentsToLoad[i]; var alreadyLoaded = loadedExtentsRtree.forEachInExtent(extentToLoad, /** * @param {{extent: import("../extent.js").Extent}} object Object. * @return {boolean} Contains. */ function (object) { return containsExtent(object.extent, extentToLoad); }); if (!alreadyLoaded) { ++this_1.loadingExtentsCount_; this_1.dispatchEvent(new VectorSourceEvent(VectorEventType.FEATURESLOADSTART)); this_1.loader_.call(this_1, extentToLoad, resolution, projection, function (features) { --this.loadingExtentsCount_; this.dispatchEvent(new VectorSourceEvent(VectorEventType.FEATURESLOADEND, undefined, features)); }.bind(this_1), function () { --this.loadingExtentsCount_; this.dispatchEvent(new VectorSourceEvent(VectorEventType.FEATURESLOADERROR)); }.bind(this_1)); loadedExtentsRtree.insert(extentToLoad, { extent: extentToLoad.slice() }); } }; var this_1 = this; for (var i = 0, ii = extentsToLoad.length; i < ii; ++i) { _loop_1(i, ii); } this.loading = this.loader_.length < 4 ? false : this.loadingExtentsCount_ > 0; }; VectorSource.prototype.refresh = function () { this.clear(true); this.loadedExtentsRtree_.clear(); _super.prototype.refresh.call(this); }; /** * Remove an extent from the list of loaded extents. * @param {import("../extent.js").Extent} extent Extent. * @api */ VectorSource.prototype.removeLoadedExtent = function (extent) { var loadedExtentsRtree = this.loadedExtentsRtree_; var obj; loadedExtentsRtree.forEachInExtent(extent, function (object) { if (equals(object.extent, extent)) { obj = object; return true; } }); if (obj) { loadedExtentsRtree.remove(obj); } }; /** * Remove a single feature from the source. If you want to remove all features * at once, use the {@link module:ol/source/Vector~VectorSource#clear #clear()} method * instead. * @param {import("../Feature.js").default} feature Feature to remove. * @api */ VectorSource.prototype.removeFeature = function (feature) { if (!feature) { return; } var featureKey = getUid(feature); if (featureKey in this.nullGeometryFeatures_) { delete this.nullGeometryFeatures_[featureKey]; } else { if (this.featuresRtree_) { this.featuresRtree_.remove(feature); } } var result = this.removeFeatureInternal(feature); if (result) { this.changed(); } }; /** * Remove feature without firing a `change` event. * @param {import("../Feature.js").default} feature Feature. * @return {import("../Feature.js").default|undefined} The removed feature * (or undefined if the feature was not found). * @protected */ VectorSource.prototype.removeFeatureInternal = function (feature) { var featureKey = getUid(feature); var featureChangeKeys = this.featureChangeKeys_[featureKey]; if (!featureChangeKeys) { return; } featureChangeKeys.forEach(unlistenByKey); delete this.featureChangeKeys_[featureKey]; var id = feature.getId(); if (id !== undefined) { delete this.idIndex_[id.toString()]; } delete this.uidIndex_[featureKey]; this.dispatchEvent(new VectorSourceEvent(VectorEventType.REMOVEFEATURE, feature)); return feature; }; /** * Remove a feature from the id index. Called internally when the feature id * may have changed. * @param {import("../Feature.js").default} feature The feature. * @return {boolean} Removed the feature from the index. * @private */ VectorSource.prototype.removeFromIdIndex_ = function (feature) { var removed = false; for (var id in this.idIndex_) { if (this.idIndex_[id] === feature) { delete this.idIndex_[id]; removed = true; break; } } return removed; }; /** * Set the new loader of the source. The next render cycle will use the * new loader. * @param {import("../featureloader.js").FeatureLoader} loader The loader to set. * @api */ VectorSource.prototype.setLoader = function (loader) { this.loader_ = loader; }; /** * Points the source to a new url. The next render cycle will use the new url. * @param {string|import("../featureloader.js").FeatureUrlFunction} url Url. * @api */ VectorSource.prototype.setUrl = function (url) { assert(this.format_, 7); // `format` must be set when `url` is set this.url_ = url; this.setLoader(xhr(url, this.format_)); }; return VectorSource; }(Source)); export default VectorSource; //# sourceMappingURL=Vector.js.map