Write styles based on style objects appropriate for geometry. Write fill and outline in PolyStyle if false (i.e. non-default) Handle MultiLineString, MultiPoint and MultiPolygon within heterogenous MultiGeometry when writing features Add getGeometriesArrayRecursive method to ol/geom/GeometryCollection to allow for nested MultiGeometry Enhanced write GeometryCollection geometries test A more rigorous write GeometryCollection geometries test including nested collections (the output is simplified to a single MultiGeomtry) Add writeFeatures to outline and fill tests, setting geometry for geometry specific tests Add <fill>0</fill> and <outline>0</outline> to some existing tests
305 lines
7.0 KiB
JavaScript
305 lines
7.0 KiB
JavaScript
/**
|
|
* @module ol/geom/GeometryCollection
|
|
*/
|
|
import {listen, unlistenByKey} from '../events.js';
|
|
import EventType from '../events/EventType.js';
|
|
import {createOrUpdateEmpty, closestSquaredDistanceXY, extend, getCenter} from '../extent.js';
|
|
import Geometry from './Geometry.js';
|
|
import GeometryType from './GeometryType.js';
|
|
|
|
/**
|
|
* @classdesc
|
|
* An array of {@link module:ol/geom/Geometry} objects.
|
|
*
|
|
* @api
|
|
*/
|
|
class GeometryCollection extends Geometry {
|
|
|
|
/**
|
|
* @param {Array<Geometry>=} opt_geometries Geometries.
|
|
*/
|
|
constructor(opt_geometries) {
|
|
|
|
super();
|
|
|
|
/**
|
|
* @private
|
|
* @type {Array<Geometry>}
|
|
*/
|
|
this.geometries_ = opt_geometries ? opt_geometries : null;
|
|
|
|
/**
|
|
* @type {Array<import("../events.js").EventsKey>}
|
|
*/
|
|
this.changeEventsKeys_ = [];
|
|
|
|
this.listenGeometriesChange_();
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
unlistenGeometriesChange_() {
|
|
this.changeEventsKeys_.forEach(unlistenByKey);
|
|
this.changeEventsKeys_.length = 0;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
listenGeometriesChange_() {
|
|
if (!this.geometries_) {
|
|
return;
|
|
}
|
|
for (let i = 0, ii = this.geometries_.length; i < ii; ++i) {
|
|
this.changeEventsKeys_.push(listen(
|
|
this.geometries_[i], EventType.CHANGE,
|
|
this.changed, this));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Make a complete copy of the geometry.
|
|
* @return {!GeometryCollection} Clone.
|
|
* @override
|
|
* @api
|
|
*/
|
|
clone() {
|
|
const geometryCollection = new GeometryCollection(null);
|
|
geometryCollection.setGeometries(this.geometries_);
|
|
return geometryCollection;
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
closestPointXY(x, y, closestPoint, minSquaredDistance) {
|
|
if (minSquaredDistance < closestSquaredDistanceXY(this.getExtent(), x, y)) {
|
|
return minSquaredDistance;
|
|
}
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
minSquaredDistance = geometries[i].closestPointXY(
|
|
x, y, closestPoint, minSquaredDistance);
|
|
}
|
|
return minSquaredDistance;
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
containsXY(x, y) {
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
if (geometries[i].containsXY(x, y)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
computeExtent(extent) {
|
|
createOrUpdateEmpty(extent);
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
extend(extent, geometries[i].getExtent());
|
|
}
|
|
return extent;
|
|
}
|
|
|
|
/**
|
|
* Return the geometries that make up this geometry collection.
|
|
* @return {Array<Geometry>} Geometries.
|
|
* @api
|
|
*/
|
|
getGeometries() {
|
|
return cloneGeometries(this.geometries_);
|
|
}
|
|
|
|
/**
|
|
* @return {Array<Geometry>} Geometries.
|
|
*/
|
|
getGeometriesArray() {
|
|
return this.geometries_;
|
|
}
|
|
|
|
/**
|
|
* @return {Array<Geometry>} Geometries.
|
|
*/
|
|
getGeometriesArrayRecursive() {
|
|
/** @type {Array<Geometry>} */
|
|
let geometriesArray = [];
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
if (geometries[i].getType() === this.getType()) {
|
|
geometriesArray = geometriesArray.concat(/** @type {GeometryCollection} */ (geometries[i]).getGeometriesArrayRecursive());
|
|
} else {
|
|
geometriesArray.push(geometries[i]);
|
|
}
|
|
}
|
|
return geometriesArray;
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
getSimplifiedGeometry(squaredTolerance) {
|
|
if (this.simplifiedGeometryRevision !== this.getRevision()) {
|
|
this.simplifiedGeometryMaxMinSquaredTolerance = 0;
|
|
this.simplifiedGeometryRevision = this.getRevision();
|
|
}
|
|
if (squaredTolerance < 0 ||
|
|
(this.simplifiedGeometryMaxMinSquaredTolerance !== 0 &&
|
|
squaredTolerance < this.simplifiedGeometryMaxMinSquaredTolerance)) {
|
|
return this;
|
|
}
|
|
|
|
const simplifiedGeometries = [];
|
|
const geometries = this.geometries_;
|
|
let simplified = false;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
const geometry = geometries[i];
|
|
const simplifiedGeometry = geometry.getSimplifiedGeometry(squaredTolerance);
|
|
simplifiedGeometries.push(simplifiedGeometry);
|
|
if (simplifiedGeometry !== geometry) {
|
|
simplified = true;
|
|
}
|
|
}
|
|
if (simplified) {
|
|
const simplifiedGeometryCollection = new GeometryCollection(null);
|
|
simplifiedGeometryCollection.setGeometriesArray(simplifiedGeometries);
|
|
return simplifiedGeometryCollection;
|
|
} else {
|
|
this.simplifiedGeometryMaxMinSquaredTolerance = squaredTolerance;
|
|
return this;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
* @api
|
|
*/
|
|
getType() {
|
|
return GeometryType.GEOMETRY_COLLECTION;
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
* @api
|
|
*/
|
|
intersectsExtent(extent) {
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
if (geometries[i].intersectsExtent(extent)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @return {boolean} Is empty.
|
|
*/
|
|
isEmpty() {
|
|
return this.geometries_.length === 0;
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
* @api
|
|
*/
|
|
rotate(angle, anchor) {
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
geometries[i].rotate(angle, anchor);
|
|
}
|
|
this.changed();
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
* @api
|
|
*/
|
|
scale(sx, opt_sy, opt_anchor) {
|
|
let anchor = opt_anchor;
|
|
if (!anchor) {
|
|
anchor = getCenter(this.getExtent());
|
|
}
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
geometries[i].scale(sx, opt_sy, anchor);
|
|
}
|
|
this.changed();
|
|
}
|
|
|
|
/**
|
|
* Set the geometries that make up this geometry collection.
|
|
* @param {Array<Geometry>} geometries Geometries.
|
|
* @api
|
|
*/
|
|
setGeometries(geometries) {
|
|
this.setGeometriesArray(cloneGeometries(geometries));
|
|
}
|
|
|
|
/**
|
|
* @param {Array<Geometry>} geometries Geometries.
|
|
*/
|
|
setGeometriesArray(geometries) {
|
|
this.unlistenGeometriesChange_();
|
|
this.geometries_ = geometries;
|
|
this.listenGeometriesChange_();
|
|
this.changed();
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
* @api
|
|
*/
|
|
applyTransform(transformFn) {
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
geometries[i].applyTransform(transformFn);
|
|
}
|
|
this.changed();
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
* @api
|
|
*/
|
|
translate(deltaX, deltaY) {
|
|
const geometries = this.geometries_;
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
geometries[i].translate(deltaX, deltaY);
|
|
}
|
|
this.changed();
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
disposeInternal() {
|
|
this.unlistenGeometriesChange_();
|
|
super.disposeInternal();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @param {Array<Geometry>} geometries Geometries.
|
|
* @return {Array<Geometry>} Cloned geometries.
|
|
*/
|
|
function cloneGeometries(geometries) {
|
|
const clonedGeometries = [];
|
|
for (let i = 0, ii = geometries.length; i < ii; ++i) {
|
|
clonedGeometries.push(geometries[i].clone());
|
|
}
|
|
return clonedGeometries;
|
|
}
|
|
|
|
|
|
export default GeometryCollection;
|