1023 lines
31 KiB
JavaScript
1023 lines
31 KiB
JavaScript
goog.provide('ol.View');
|
|
|
|
goog.require('ol');
|
|
goog.require('ol.CenterConstraint');
|
|
goog.require('ol.Constraints');
|
|
goog.require('ol.Object');
|
|
goog.require('ol.ResolutionConstraint');
|
|
goog.require('ol.RotationConstraint');
|
|
goog.require('ol.array');
|
|
goog.require('ol.asserts');
|
|
goog.require('ol.coordinate');
|
|
goog.require('ol.easing');
|
|
goog.require('ol.extent');
|
|
goog.require('ol.geom.Polygon');
|
|
goog.require('ol.geom.SimpleGeometry');
|
|
goog.require('ol.proj');
|
|
goog.require('ol.proj.METERS_PER_UNIT');
|
|
goog.require('ol.proj.Units');
|
|
|
|
|
|
/**
|
|
* @classdesc
|
|
* An ol.View object represents a simple 2D view of the map.
|
|
*
|
|
* This is the object to act upon to change the center, resolution,
|
|
* and rotation of the map.
|
|
*
|
|
* ### The view states
|
|
*
|
|
* An `ol.View` is determined by three states: `center`, `resolution`,
|
|
* and `rotation`. Each state has a corresponding getter and setter, e.g.
|
|
* `getCenter` and `setCenter` for the `center` state.
|
|
*
|
|
* An `ol.View` has a `projection`. The projection determines the
|
|
* coordinate system of the center, and its units determine the units of the
|
|
* resolution (projection units per pixel). The default projection is
|
|
* Spherical Mercator (EPSG:3857).
|
|
*
|
|
* ### The constraints
|
|
*
|
|
* `setCenter`, `setResolution` and `setRotation` can be used to change the
|
|
* states of the view. Any value can be passed to the setters. And the value
|
|
* that is passed to a setter will effectively be the value set in the view,
|
|
* and returned by the corresponding getter.
|
|
*
|
|
* But an `ol.View` object also has a *resolution constraint*, a
|
|
* *rotation constraint* and a *center constraint*.
|
|
*
|
|
* As said above, no constraints are applied when the setters are used to set
|
|
* new states for the view. Applying constraints is done explicitly through
|
|
* the use of the `constrain*` functions (`constrainResolution` and
|
|
* `constrainRotation` and `constrainCenter`).
|
|
*
|
|
* The main users of the constraints are the interactions and the
|
|
* controls. For example, double-clicking on the map changes the view to
|
|
* the "next" resolution. And releasing the fingers after pinch-zooming
|
|
* snaps to the closest resolution (with an animation).
|
|
*
|
|
* The *resolution constraint* snaps to specific resolutions. It is
|
|
* determined by the following options: `resolutions`, `maxResolution`,
|
|
* `maxZoom`, and `zoomFactor`. If `resolutions` is set, the other three
|
|
* options are ignored. See documentation for each option for more
|
|
* information.
|
|
*
|
|
* The *rotation constraint* snaps to specific angles. It is determined
|
|
* by the following options: `enableRotation` and `constrainRotation`.
|
|
* By default the rotation value is snapped to zero when approaching the
|
|
* horizontal.
|
|
*
|
|
* The *center constraint* is determined by the `extent` option. By
|
|
* default the center is not constrained at all.
|
|
*
|
|
* @constructor
|
|
* @extends {ol.Object}
|
|
* @param {olx.ViewOptions=} opt_options View options.
|
|
* @api stable
|
|
*/
|
|
ol.View = function(opt_options) {
|
|
ol.Object.call(this);
|
|
var options = opt_options || {};
|
|
|
|
/**
|
|
* @private
|
|
* @type {Array.<number>}
|
|
*/
|
|
this.hints_ = [0, 0];
|
|
|
|
/**
|
|
* @private
|
|
* @type {Array.<Array.<ol.ViewAnimation>>}
|
|
*/
|
|
this.animations_ = [];
|
|
|
|
/**
|
|
* @private
|
|
* @type {number|undefined}
|
|
*/
|
|
this.updateAnimationKey_;
|
|
|
|
this.updateAnimations_ = this.updateAnimations_.bind(this);
|
|
|
|
/**
|
|
* @type {Object.<string, *>}
|
|
*/
|
|
var properties = {};
|
|
properties[ol.View.Property.CENTER] = options.center !== undefined ?
|
|
options.center : null;
|
|
|
|
/**
|
|
* @private
|
|
* @const
|
|
* @type {ol.proj.Projection}
|
|
*/
|
|
this.projection_ = ol.proj.createProjection(options.projection, 'EPSG:3857');
|
|
|
|
var resolutionConstraintInfo = ol.View.createResolutionConstraint_(
|
|
options);
|
|
|
|
/**
|
|
* @private
|
|
* @type {number}
|
|
*/
|
|
this.maxResolution_ = resolutionConstraintInfo.maxResolution;
|
|
|
|
/**
|
|
* @private
|
|
* @type {number}
|
|
*/
|
|
this.minResolution_ = resolutionConstraintInfo.minResolution;
|
|
|
|
/**
|
|
* @private
|
|
* @type {number}
|
|
*/
|
|
this.zoomFactor_ = resolutionConstraintInfo.zoomFactor;
|
|
|
|
/**
|
|
* @private
|
|
* @type {Array.<number>|undefined}
|
|
*/
|
|
this.resolutions_ = options.resolutions;
|
|
|
|
/**
|
|
* @private
|
|
* @type {number}
|
|
*/
|
|
this.minZoom_ = resolutionConstraintInfo.minZoom;
|
|
|
|
var centerConstraint = ol.View.createCenterConstraint_(options);
|
|
var resolutionConstraint = resolutionConstraintInfo.constraint;
|
|
var rotationConstraint = ol.View.createRotationConstraint_(options);
|
|
|
|
/**
|
|
* @private
|
|
* @type {ol.Constraints}
|
|
*/
|
|
this.constraints_ = new ol.Constraints(
|
|
centerConstraint, resolutionConstraint, rotationConstraint);
|
|
|
|
if (options.resolution !== undefined) {
|
|
properties[ol.View.Property.RESOLUTION] = options.resolution;
|
|
} else if (options.zoom !== undefined) {
|
|
properties[ol.View.Property.RESOLUTION] = this.constrainResolution(
|
|
this.maxResolution_, options.zoom - this.minZoom_);
|
|
}
|
|
properties[ol.View.Property.ROTATION] =
|
|
options.rotation !== undefined ? options.rotation : 0;
|
|
this.setProperties(properties);
|
|
};
|
|
ol.inherits(ol.View, ol.Object);
|
|
|
|
|
|
/**
|
|
* Animate the view. The view's center, zoom (or resolution), and rotation
|
|
* can be animated for smooth transitions between view states. For example,
|
|
* to animate the view to a new zoom level:
|
|
*
|
|
* view.animate({zoom: view.getZoom() + 1});
|
|
*
|
|
* By default, the animation lasts one second and uses in-and-out easing. You
|
|
* can customize this behavior by including `duration` (in milliseconds) and
|
|
* `easing` options (see {@link ol.easing}).
|
|
*
|
|
* To chain together multiple animations, call the method with multiple
|
|
* animation objects. For example, to first zoom and then pan:
|
|
*
|
|
* view.animate({zoom: 10}, {center: [0, 0]});
|
|
*
|
|
* If you provide a function as the last argument to the animate method, it
|
|
* will get called at the end of an animation series. The callback will be
|
|
* called with `true` if the animation series completed on its own or `false`
|
|
* if it was cancelled.
|
|
*
|
|
* Animations are cancelled by user interactions (e.g. dragging the map) or by
|
|
* calling `view.setCenter()`, `view.setResolution()`, or `view.setRotation()`
|
|
* (or another method that calls one of these).
|
|
*
|
|
* @param {...(olx.AnimationOptions|function(boolean))} var_args Animation
|
|
* options. Multiple animations can be run in series by passing multiple
|
|
* options objects. To run multiple animations in parallel, call the method
|
|
* multiple times. An optional callback can be provided as a final
|
|
* argument. The callback will be called with a boolean indicating whether
|
|
* the animation completed without being cancelled.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.animate = function(var_args) {
|
|
var start = Date.now();
|
|
var center = this.getCenter().slice();
|
|
var resolution = this.getResolution();
|
|
var rotation = this.getRotation();
|
|
var animationCount = arguments.length;
|
|
var callback;
|
|
if (animationCount > 1 && typeof arguments[animationCount - 1] === 'function') {
|
|
callback = arguments[animationCount - 1];
|
|
--animationCount;
|
|
}
|
|
var series = [];
|
|
for (var i = 0; i < animationCount; ++i) {
|
|
var options = /** @type olx.AnimationOptions */ (arguments[i]);
|
|
|
|
var animation = /** @type {ol.ViewAnimation} */ ({
|
|
start: start,
|
|
complete: false,
|
|
anchor: options.anchor,
|
|
duration: options.duration || 1000,
|
|
easing: options.easing || ol.easing.inAndOut
|
|
});
|
|
|
|
if (options.center) {
|
|
animation.sourceCenter = center;
|
|
animation.targetCenter = options.center;
|
|
center = animation.targetCenter;
|
|
}
|
|
|
|
if (options.zoom !== undefined) {
|
|
animation.sourceResolution = resolution;
|
|
animation.targetResolution = this.constrainResolution(
|
|
this.maxResolution_, options.zoom - this.minZoom_, 0);
|
|
resolution = animation.targetResolution;
|
|
} else if (options.resolution) {
|
|
animation.sourceResolution = this.getResolution();
|
|
animation.targetResolution = options.resolution;
|
|
resolution = animation.targetResolution;
|
|
}
|
|
|
|
if (options.rotation !== undefined) {
|
|
animation.sourceRotation = rotation;
|
|
animation.targetRotation = options.rotation;
|
|
rotation = animation.targetRotation;
|
|
}
|
|
|
|
animation.callback = callback;
|
|
start += animation.duration;
|
|
series.push(animation);
|
|
}
|
|
this.animations_.push(series);
|
|
this.setHint(ol.View.Hint.ANIMATING, 1);
|
|
this.updateAnimations_();
|
|
};
|
|
|
|
|
|
/**
|
|
* Determine if the view is being animated.
|
|
* @return {boolean} The view is being animated.
|
|
*/
|
|
ol.View.prototype.getAnimating = function() {
|
|
return this.getHints()[ol.View.Hint.ANIMATING] > 0;
|
|
};
|
|
|
|
|
|
/**
|
|
* Cancel any ongoing animations.
|
|
*/
|
|
ol.View.prototype.cancelAnimations = function() {
|
|
this.setHint(ol.View.Hint.ANIMATING, -this.getHints()[ol.View.Hint.ANIMATING]);
|
|
for (var i = 0, ii = this.animations_.length; i < ii; ++i) {
|
|
var series = this.animations_[i];
|
|
if (series[0].callback) {
|
|
series[0].callback(false);
|
|
}
|
|
}
|
|
this.animations_.length = 0;
|
|
};
|
|
|
|
/**
|
|
* Update all animations.
|
|
*/
|
|
ol.View.prototype.updateAnimations_ = function() {
|
|
if (this.updateAnimationKey_ !== undefined) {
|
|
cancelAnimationFrame(this.updateAnimationKey_);
|
|
this.updateAnimationKey_ = undefined;
|
|
}
|
|
if (!this.getAnimating()) {
|
|
return;
|
|
}
|
|
var now = Date.now();
|
|
var more = false;
|
|
for (var i = this.animations_.length - 1; i >= 0; --i) {
|
|
var series = this.animations_[i];
|
|
var seriesComplete = true;
|
|
for (var j = 0, jj = series.length; j < jj; ++j) {
|
|
var animation = series[j];
|
|
if (animation.complete) {
|
|
continue;
|
|
}
|
|
var elapsed = now - animation.start;
|
|
var fraction = elapsed / animation.duration;
|
|
if (fraction >= 1) {
|
|
animation.complete = true;
|
|
fraction = 1;
|
|
} else {
|
|
seriesComplete = false;
|
|
}
|
|
var progress = animation.easing(fraction);
|
|
if (animation.sourceCenter) {
|
|
var x0 = animation.sourceCenter[0];
|
|
var y0 = animation.sourceCenter[1];
|
|
var x1 = animation.targetCenter[0];
|
|
var y1 = animation.targetCenter[1];
|
|
var x = x0 + progress * (x1 - x0);
|
|
var y = y0 + progress * (y1 - y0);
|
|
this.set(ol.View.Property.CENTER, [x, y]);
|
|
}
|
|
if (animation.sourceResolution) {
|
|
var resolution = animation.sourceResolution +
|
|
progress * (animation.targetResolution - animation.sourceResolution);
|
|
if (animation.anchor) {
|
|
this.set(ol.View.Property.CENTER,
|
|
this.calculateCenterZoom(resolution, animation.anchor));
|
|
}
|
|
this.set(ol.View.Property.RESOLUTION, resolution);
|
|
}
|
|
if (animation.sourceRotation !== undefined) {
|
|
var rotation = animation.sourceRotation +
|
|
progress * (animation.targetRotation - animation.sourceRotation);
|
|
if (animation.anchor) {
|
|
this.set(ol.View.Property.CENTER,
|
|
this.calculateCenterRotate(rotation, animation.anchor));
|
|
}
|
|
this.set(ol.View.Property.ROTATION, rotation);
|
|
}
|
|
more = true;
|
|
if (!animation.complete) {
|
|
break;
|
|
}
|
|
}
|
|
if (seriesComplete) {
|
|
this.animations_[i] = null;
|
|
this.setHint(ol.View.Hint.ANIMATING, -1);
|
|
var callback = series[0].callback;
|
|
if (callback) {
|
|
callback(true);
|
|
}
|
|
}
|
|
}
|
|
// prune completed series
|
|
this.animations_ = this.animations_.filter(Boolean);
|
|
if (more && this.updateAnimationKey_ === undefined) {
|
|
this.updateAnimationKey_ = requestAnimationFrame(this.updateAnimations_);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @param {number} rotation Target rotation.
|
|
* @param {ol.Coordinate} anchor Rotation anchor.
|
|
* @return {ol.Coordinate|undefined} Center for rotation and anchor.
|
|
*/
|
|
ol.View.prototype.calculateCenterRotate = function(rotation, anchor) {
|
|
var center;
|
|
var currentCenter = this.getCenter();
|
|
if (currentCenter !== undefined) {
|
|
center = [currentCenter[0] - anchor[0], currentCenter[1] - anchor[1]];
|
|
ol.coordinate.rotate(center, rotation - this.getRotation());
|
|
ol.coordinate.add(center, anchor);
|
|
}
|
|
return center;
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {number} resolution Target resolution.
|
|
* @param {ol.Coordinate} anchor Zoom anchor.
|
|
* @return {ol.Coordinate|undefined} Center for resolution and anchor.
|
|
*/
|
|
ol.View.prototype.calculateCenterZoom = function(resolution, anchor) {
|
|
var center;
|
|
var currentCenter = this.getCenter();
|
|
var currentResolution = this.getResolution();
|
|
if (currentCenter !== undefined && currentResolution !== undefined) {
|
|
var x = anchor[0] -
|
|
resolution * (anchor[0] - currentCenter[0]) / currentResolution;
|
|
var y = anchor[1] -
|
|
resolution * (anchor[1] - currentCenter[1]) / currentResolution;
|
|
center = [x, y];
|
|
}
|
|
return center;
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the constrained center of this view.
|
|
* @param {ol.Coordinate|undefined} center Center.
|
|
* @return {ol.Coordinate|undefined} Constrained center.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.constrainCenter = function(center) {
|
|
return this.constraints_.center(center);
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the constrained resolution of this view.
|
|
* @param {number|undefined} resolution Resolution.
|
|
* @param {number=} opt_delta Delta. Default is `0`.
|
|
* @param {number=} opt_direction Direction. Default is `0`.
|
|
* @return {number|undefined} Constrained resolution.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.constrainResolution = function(
|
|
resolution, opt_delta, opt_direction) {
|
|
var delta = opt_delta || 0;
|
|
var direction = opt_direction || 0;
|
|
return this.constraints_.resolution(resolution, delta, direction);
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the constrained rotation of this view.
|
|
* @param {number|undefined} rotation Rotation.
|
|
* @param {number=} opt_delta Delta. Default is `0`.
|
|
* @return {number|undefined} Constrained rotation.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.constrainRotation = function(rotation, opt_delta) {
|
|
var delta = opt_delta || 0;
|
|
return this.constraints_.rotation(rotation, delta);
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the view center.
|
|
* @return {ol.Coordinate|undefined} The center of the view.
|
|
* @observable
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.getCenter = function() {
|
|
return /** @type {ol.Coordinate|undefined} */ (
|
|
this.get(ol.View.Property.CENTER));
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {Array.<number>=} opt_hints Destination array.
|
|
* @return {Array.<number>} Hint.
|
|
*/
|
|
ol.View.prototype.getHints = function(opt_hints) {
|
|
if (opt_hints !== undefined) {
|
|
opt_hints[0] = this.hints_[0];
|
|
opt_hints[1] = this.hints_[1];
|
|
return opt_hints;
|
|
} else {
|
|
return this.hints_.slice();
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Calculate the extent for the current view state and the passed size.
|
|
* The size is the pixel dimensions of the box into which the calculated extent
|
|
* should fit. In most cases you want to get the extent of the entire map,
|
|
* that is `map.getSize()`.
|
|
* @param {ol.Size} size Box pixel size.
|
|
* @return {ol.Extent} Extent.
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.calculateExtent = function(size) {
|
|
var center = /** @type {!ol.Coordinate} */ (this.getCenter());
|
|
ol.asserts.assert(center, 1); // The view center is not defined
|
|
var resolution = /** @type {!number} */ (this.getResolution());
|
|
ol.asserts.assert(resolution !== undefined, 2); // The view resolution is not defined
|
|
var rotation = /** @type {!number} */ (this.getRotation());
|
|
ol.asserts.assert(rotation !== undefined, 3); // The view rotation is not defined
|
|
|
|
return ol.extent.getForViewAndSize(center, resolution, rotation, size);
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the maximum resolution of the view.
|
|
* @return {number} The maximum resolution of the view.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.getMaxResolution = function() {
|
|
return this.maxResolution_;
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the minimum resolution of the view.
|
|
* @return {number} The minimum resolution of the view.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.getMinResolution = function() {
|
|
return this.minResolution_;
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the view projection.
|
|
* @return {ol.proj.Projection} The projection of the view.
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.getProjection = function() {
|
|
return this.projection_;
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the view resolution.
|
|
* @return {number|undefined} The resolution of the view.
|
|
* @observable
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.getResolution = function() {
|
|
return /** @type {number|undefined} */ (
|
|
this.get(ol.View.Property.RESOLUTION));
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the resolutions for the view. This returns the array of resolutions
|
|
* passed to the constructor of the {ol.View}, or undefined if none were given.
|
|
* @return {Array.<number>|undefined} The resolutions of the view.
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.getResolutions = function() {
|
|
return this.resolutions_;
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the resolution for a provided extent (in map units) and size (in pixels).
|
|
* @param {ol.Extent} extent Extent.
|
|
* @param {ol.Size} size Box pixel size.
|
|
* @return {number} The resolution at which the provided extent will render at
|
|
* the given size.
|
|
*/
|
|
ol.View.prototype.getResolutionForExtent = function(extent, size) {
|
|
var xResolution = ol.extent.getWidth(extent) / size[0];
|
|
var yResolution = ol.extent.getHeight(extent) / size[1];
|
|
return Math.max(xResolution, yResolution);
|
|
};
|
|
|
|
|
|
/**
|
|
* Return a function that returns a value between 0 and 1 for a
|
|
* resolution. Exponential scaling is assumed.
|
|
* @param {number=} opt_power Power.
|
|
* @return {function(number): number} Resolution for value function.
|
|
*/
|
|
ol.View.prototype.getResolutionForValueFunction = function(opt_power) {
|
|
var power = opt_power || 2;
|
|
var maxResolution = this.maxResolution_;
|
|
var minResolution = this.minResolution_;
|
|
var max = Math.log(maxResolution / minResolution) / Math.log(power);
|
|
return (
|
|
/**
|
|
* @param {number} value Value.
|
|
* @return {number} Resolution.
|
|
*/
|
|
function(value) {
|
|
var resolution = maxResolution / Math.pow(power, value * max);
|
|
ol.DEBUG && console.assert(resolution >= minResolution &&
|
|
resolution <= maxResolution,
|
|
'calculated resolution outside allowed bounds (%s <= %s <= %s)',
|
|
minResolution, resolution, maxResolution);
|
|
return resolution;
|
|
});
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the view rotation.
|
|
* @return {number} The rotation of the view in radians.
|
|
* @observable
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.getRotation = function() {
|
|
return /** @type {number} */ (this.get(ol.View.Property.ROTATION));
|
|
};
|
|
|
|
|
|
/**
|
|
* Return a function that returns a resolution for a value between
|
|
* 0 and 1. Exponential scaling is assumed.
|
|
* @param {number=} opt_power Power.
|
|
* @return {function(number): number} Value for resolution function.
|
|
*/
|
|
ol.View.prototype.getValueForResolutionFunction = function(opt_power) {
|
|
var power = opt_power || 2;
|
|
var maxResolution = this.maxResolution_;
|
|
var minResolution = this.minResolution_;
|
|
var max = Math.log(maxResolution / minResolution) / Math.log(power);
|
|
return (
|
|
/**
|
|
* @param {number} resolution Resolution.
|
|
* @return {number} Value.
|
|
*/
|
|
function(resolution) {
|
|
var value =
|
|
(Math.log(maxResolution / resolution) / Math.log(power)) / max;
|
|
ol.DEBUG && console.assert(value >= 0 && value <= 1,
|
|
'calculated value (%s) ouside allowed range (0-1)', value);
|
|
return value;
|
|
});
|
|
};
|
|
|
|
|
|
/**
|
|
* @return {olx.ViewState} View state.
|
|
*/
|
|
ol.View.prototype.getState = function() {
|
|
ol.DEBUG && console.assert(this.isDef(),
|
|
'the view was not defined (had no center and/or resolution)');
|
|
var center = /** @type {ol.Coordinate} */ (this.getCenter());
|
|
var projection = this.getProjection();
|
|
var resolution = /** @type {number} */ (this.getResolution());
|
|
var rotation = this.getRotation();
|
|
return /** @type {olx.ViewState} */ ({
|
|
center: center.slice(),
|
|
projection: projection !== undefined ? projection : null,
|
|
resolution: resolution,
|
|
rotation: rotation
|
|
});
|
|
};
|
|
|
|
|
|
/**
|
|
* Get the current zoom level. Return undefined if the current
|
|
* resolution is undefined or not within the "resolution constraints".
|
|
* @return {number|undefined} Zoom.
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.getZoom = function() {
|
|
var zoom;
|
|
var resolution = this.getResolution();
|
|
if (resolution !== undefined &&
|
|
resolution >= this.minResolution_ && resolution <= this.maxResolution_) {
|
|
var offset = this.minZoom_ || 0;
|
|
var max, zoomFactor;
|
|
if (this.resolutions_) {
|
|
var nearest = ol.array.linearFindNearest(this.resolutions_, resolution, 1);
|
|
offset += nearest;
|
|
if (nearest == this.resolutions_.length - 1) {
|
|
return offset;
|
|
}
|
|
max = this.resolutions_[nearest];
|
|
zoomFactor = max / this.resolutions_[nearest + 1];
|
|
} else {
|
|
max = this.maxResolution_;
|
|
zoomFactor = this.zoomFactor_;
|
|
}
|
|
zoom = offset + Math.log(max / resolution) / Math.log(zoomFactor);
|
|
}
|
|
return zoom;
|
|
};
|
|
|
|
|
|
/**
|
|
* Fit the given geometry or extent based on the given map size and border.
|
|
* The size is pixel dimensions of the box to fit the extent into.
|
|
* In most cases you will want to use the map size, that is `map.getSize()`.
|
|
* Takes care of the map angle.
|
|
* @param {ol.geom.SimpleGeometry|ol.Extent} geometry Geometry.
|
|
* @param {ol.Size} size Box pixel size.
|
|
* @param {olx.view.FitOptions=} opt_options Options.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.fit = function(geometry, size, opt_options) {
|
|
if (!(geometry instanceof ol.geom.SimpleGeometry)) {
|
|
ol.asserts.assert(Array.isArray(geometry),
|
|
24); // Invalid extent or geometry provided as `geometry`
|
|
ol.asserts.assert(!ol.extent.isEmpty(geometry),
|
|
25); // Cannot fit empty extent provided as `geometry`
|
|
geometry = ol.geom.Polygon.fromExtent(geometry);
|
|
}
|
|
|
|
var options = opt_options || {};
|
|
|
|
var padding = options.padding !== undefined ? options.padding : [0, 0, 0, 0];
|
|
var constrainResolution = options.constrainResolution !== undefined ?
|
|
options.constrainResolution : true;
|
|
var nearest = options.nearest !== undefined ? options.nearest : false;
|
|
var minResolution;
|
|
if (options.minResolution !== undefined) {
|
|
minResolution = options.minResolution;
|
|
} else if (options.maxZoom !== undefined) {
|
|
minResolution = this.constrainResolution(
|
|
this.maxResolution_, options.maxZoom - this.minZoom_, 0);
|
|
} else {
|
|
minResolution = 0;
|
|
}
|
|
var coords = geometry.getFlatCoordinates();
|
|
|
|
// calculate rotated extent
|
|
var rotation = this.getRotation();
|
|
ol.DEBUG && console.assert(rotation !== undefined, 'rotation was not defined');
|
|
var cosAngle = Math.cos(-rotation);
|
|
var sinAngle = Math.sin(-rotation);
|
|
var minRotX = +Infinity;
|
|
var minRotY = +Infinity;
|
|
var maxRotX = -Infinity;
|
|
var maxRotY = -Infinity;
|
|
var stride = geometry.getStride();
|
|
for (var i = 0, ii = coords.length; i < ii; i += stride) {
|
|
var rotX = coords[i] * cosAngle - coords[i + 1] * sinAngle;
|
|
var rotY = coords[i] * sinAngle + coords[i + 1] * cosAngle;
|
|
minRotX = Math.min(minRotX, rotX);
|
|
minRotY = Math.min(minRotY, rotY);
|
|
maxRotX = Math.max(maxRotX, rotX);
|
|
maxRotY = Math.max(maxRotY, rotY);
|
|
}
|
|
|
|
// calculate resolution
|
|
var resolution = this.getResolutionForExtent(
|
|
[minRotX, minRotY, maxRotX, maxRotY],
|
|
[size[0] - padding[1] - padding[3], size[1] - padding[0] - padding[2]]);
|
|
resolution = isNaN(resolution) ? minResolution :
|
|
Math.max(resolution, minResolution);
|
|
if (constrainResolution) {
|
|
var constrainedResolution = this.constrainResolution(resolution, 0, 0);
|
|
if (!nearest && constrainedResolution < resolution) {
|
|
constrainedResolution = this.constrainResolution(
|
|
constrainedResolution, -1, 0);
|
|
}
|
|
resolution = constrainedResolution;
|
|
}
|
|
|
|
// calculate center
|
|
sinAngle = -sinAngle; // go back to original rotation
|
|
var centerRotX = (minRotX + maxRotX) / 2;
|
|
var centerRotY = (minRotY + maxRotY) / 2;
|
|
centerRotX += (padding[1] - padding[3]) / 2 * resolution;
|
|
centerRotY += (padding[0] - padding[2]) / 2 * resolution;
|
|
var centerX = centerRotX * cosAngle - centerRotY * sinAngle;
|
|
var centerY = centerRotY * cosAngle + centerRotX * sinAngle;
|
|
var center = [centerX, centerY];
|
|
|
|
if (options.duration !== undefined) {
|
|
this.animate({
|
|
resolution: resolution,
|
|
center: center,
|
|
duration: options.duration,
|
|
easing: options.easing
|
|
});
|
|
} else {
|
|
this.setResolution(resolution);
|
|
this.setCenter(center);
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Center on coordinate and view position.
|
|
* @param {ol.Coordinate} coordinate Coordinate.
|
|
* @param {ol.Size} size Box pixel size.
|
|
* @param {ol.Pixel} position Position on the view to center on.
|
|
* @api
|
|
*/
|
|
ol.View.prototype.centerOn = function(coordinate, size, position) {
|
|
// calculate rotated position
|
|
var rotation = this.getRotation();
|
|
var cosAngle = Math.cos(-rotation);
|
|
var sinAngle = Math.sin(-rotation);
|
|
var rotX = coordinate[0] * cosAngle - coordinate[1] * sinAngle;
|
|
var rotY = coordinate[1] * cosAngle + coordinate[0] * sinAngle;
|
|
var resolution = this.getResolution();
|
|
rotX += (size[0] / 2 - position[0]) * resolution;
|
|
rotY += (position[1] - size[1] / 2) * resolution;
|
|
|
|
// go back to original angle
|
|
sinAngle = -sinAngle; // go back to original rotation
|
|
var centerX = rotX * cosAngle - rotY * sinAngle;
|
|
var centerY = rotY * cosAngle + rotX * sinAngle;
|
|
|
|
this.setCenter([centerX, centerY]);
|
|
};
|
|
|
|
|
|
/**
|
|
* @return {boolean} Is defined.
|
|
*/
|
|
ol.View.prototype.isDef = function() {
|
|
return !!this.getCenter() && this.getResolution() !== undefined;
|
|
};
|
|
|
|
|
|
/**
|
|
* Rotate the view around a given coordinate.
|
|
* @param {number} rotation New rotation value for the view.
|
|
* @param {ol.Coordinate=} opt_anchor The rotation center.
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.rotate = function(rotation, opt_anchor) {
|
|
if (opt_anchor !== undefined) {
|
|
var center = this.calculateCenterRotate(rotation, opt_anchor);
|
|
this.setCenter(center);
|
|
}
|
|
this.setRotation(rotation);
|
|
};
|
|
|
|
|
|
/**
|
|
* Set the center of the current view.
|
|
* @param {ol.Coordinate|undefined} center The center of the view.
|
|
* @observable
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.setCenter = function(center) {
|
|
this.set(ol.View.Property.CENTER, center);
|
|
if (this.getAnimating()) {
|
|
this.cancelAnimations();
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {ol.View.Hint} hint Hint.
|
|
* @param {number} delta Delta.
|
|
* @return {number} New value.
|
|
*/
|
|
ol.View.prototype.setHint = function(hint, delta) {
|
|
ol.DEBUG && console.assert(0 <= hint && hint < this.hints_.length,
|
|
'illegal hint (%s), must be between 0 and %s', hint, this.hints_.length);
|
|
this.hints_[hint] += delta;
|
|
ol.DEBUG && console.assert(this.hints_[hint] >= 0,
|
|
'Hint at %s must be positive, was %s', hint, this.hints_[hint]);
|
|
this.changed();
|
|
return this.hints_[hint];
|
|
};
|
|
|
|
|
|
/**
|
|
* Set the resolution for this view.
|
|
* @param {number|undefined} resolution The resolution of the view.
|
|
* @observable
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.setResolution = function(resolution) {
|
|
this.set(ol.View.Property.RESOLUTION, resolution);
|
|
if (this.getAnimating()) {
|
|
this.cancelAnimations();
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Set the rotation for this view.
|
|
* @param {number} rotation The rotation of the view in radians.
|
|
* @observable
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.setRotation = function(rotation) {
|
|
this.set(ol.View.Property.ROTATION, rotation);
|
|
if (this.getAnimating()) {
|
|
this.cancelAnimations();
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Zoom to a specific zoom level.
|
|
* @param {number} zoom Zoom level.
|
|
* @api stable
|
|
*/
|
|
ol.View.prototype.setZoom = function(zoom) {
|
|
var resolution = this.constrainResolution(
|
|
this.maxResolution_, zoom - this.minZoom_, 0);
|
|
this.setResolution(resolution);
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {olx.ViewOptions} options View options.
|
|
* @private
|
|
* @return {ol.CenterConstraintType} The constraint.
|
|
*/
|
|
ol.View.createCenterConstraint_ = function(options) {
|
|
if (options.extent !== undefined) {
|
|
return ol.CenterConstraint.createExtent(options.extent);
|
|
} else {
|
|
return ol.CenterConstraint.none;
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @private
|
|
* @param {olx.ViewOptions} options View options.
|
|
* @return {{constraint: ol.ResolutionConstraintType, maxResolution: number,
|
|
* minResolution: number, zoomFactor: number}} The constraint.
|
|
*/
|
|
ol.View.createResolutionConstraint_ = function(options) {
|
|
var resolutionConstraint;
|
|
var maxResolution;
|
|
var minResolution;
|
|
|
|
// TODO: move these to be ol constants
|
|
// see https://github.com/openlayers/ol3/issues/2076
|
|
var defaultMaxZoom = 28;
|
|
var defaultZoomFactor = 2;
|
|
|
|
var minZoom = options.minZoom !== undefined ?
|
|
options.minZoom : ol.DEFAULT_MIN_ZOOM;
|
|
|
|
var maxZoom = options.maxZoom !== undefined ?
|
|
options.maxZoom : defaultMaxZoom;
|
|
|
|
var zoomFactor = options.zoomFactor !== undefined ?
|
|
options.zoomFactor : defaultZoomFactor;
|
|
|
|
if (options.resolutions !== undefined) {
|
|
var resolutions = options.resolutions;
|
|
maxResolution = resolutions[0];
|
|
minResolution = resolutions[resolutions.length - 1];
|
|
resolutionConstraint = ol.ResolutionConstraint.createSnapToResolutions(
|
|
resolutions);
|
|
} else {
|
|
// calculate the default min and max resolution
|
|
var projection = ol.proj.createProjection(options.projection, 'EPSG:3857');
|
|
var extent = projection.getExtent();
|
|
var size = !extent ?
|
|
// use an extent that can fit the whole world if need be
|
|
360 * ol.proj.METERS_PER_UNIT[ol.proj.Units.DEGREES] /
|
|
projection.getMetersPerUnit() :
|
|
Math.max(ol.extent.getWidth(extent), ol.extent.getHeight(extent));
|
|
|
|
var defaultMaxResolution = size / ol.DEFAULT_TILE_SIZE / Math.pow(
|
|
defaultZoomFactor, ol.DEFAULT_MIN_ZOOM);
|
|
|
|
var defaultMinResolution = defaultMaxResolution / Math.pow(
|
|
defaultZoomFactor, defaultMaxZoom - ol.DEFAULT_MIN_ZOOM);
|
|
|
|
// user provided maxResolution takes precedence
|
|
maxResolution = options.maxResolution;
|
|
if (maxResolution !== undefined) {
|
|
minZoom = 0;
|
|
} else {
|
|
maxResolution = defaultMaxResolution / Math.pow(zoomFactor, minZoom);
|
|
}
|
|
|
|
// user provided minResolution takes precedence
|
|
minResolution = options.minResolution;
|
|
if (minResolution === undefined) {
|
|
if (options.maxZoom !== undefined) {
|
|
if (options.maxResolution !== undefined) {
|
|
minResolution = maxResolution / Math.pow(zoomFactor, maxZoom);
|
|
} else {
|
|
minResolution = defaultMaxResolution / Math.pow(zoomFactor, maxZoom);
|
|
}
|
|
} else {
|
|
minResolution = defaultMinResolution;
|
|
}
|
|
}
|
|
|
|
// given discrete zoom levels, minResolution may be different than provided
|
|
maxZoom = minZoom + Math.floor(
|
|
Math.log(maxResolution / minResolution) / Math.log(zoomFactor));
|
|
minResolution = maxResolution / Math.pow(zoomFactor, maxZoom - minZoom);
|
|
|
|
resolutionConstraint = ol.ResolutionConstraint.createSnapToPower(
|
|
zoomFactor, maxResolution, maxZoom - minZoom);
|
|
}
|
|
return {constraint: resolutionConstraint, maxResolution: maxResolution,
|
|
minResolution: minResolution, minZoom: minZoom, zoomFactor: zoomFactor};
|
|
};
|
|
|
|
|
|
/**
|
|
* @private
|
|
* @param {olx.ViewOptions} options View options.
|
|
* @return {ol.RotationConstraintType} Rotation constraint.
|
|
*/
|
|
ol.View.createRotationConstraint_ = function(options) {
|
|
var enableRotation = options.enableRotation !== undefined ?
|
|
options.enableRotation : true;
|
|
if (enableRotation) {
|
|
var constrainRotation = options.constrainRotation;
|
|
if (constrainRotation === undefined || constrainRotation === true) {
|
|
return ol.RotationConstraint.createSnapToZero();
|
|
} else if (constrainRotation === false) {
|
|
return ol.RotationConstraint.none;
|
|
} else if (typeof constrainRotation === 'number') {
|
|
return ol.RotationConstraint.createSnapToN(constrainRotation);
|
|
} else {
|
|
ol.DEBUG && console.assert(false,
|
|
'illegal option for constrainRotation (%s)', constrainRotation);
|
|
return ol.RotationConstraint.none;
|
|
}
|
|
} else {
|
|
return ol.RotationConstraint.disable;
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @enum {string}
|
|
*/
|
|
ol.View.Property = {
|
|
CENTER: 'center',
|
|
RESOLUTION: 'resolution',
|
|
ROTATION: 'rotation'
|
|
};
|
|
|
|
|
|
/**
|
|
* @enum {number}
|
|
*/
|
|
ol.View.Hint = {
|
|
ANIMATING: 0,
|
|
INTERACTING: 1
|
|
};
|