Rendering vector layers

This commit is contained in:
Tim Schaub
2013-01-22 01:36:32 -07:00
parent 92c8c9c5df
commit 1dd17fc88f
4 changed files with 166 additions and 31 deletions

View File

@@ -9,6 +9,7 @@ goog.require('ol.Size');
goog.require('ol.layer.TileLayer');
goog.require('ol.renderer.Map');
goog.require('ol.renderer.canvas.TileLayer');
goog.require('ol.renderer.canvas.VectorLayer');
@@ -60,6 +61,8 @@ goog.inherits(ol.renderer.canvas.Map, ol.renderer.Map);
ol.renderer.canvas.Map.prototype.createLayerRenderer = function(layer) {
if (layer instanceof ol.layer.TileLayer) {
return new ol.renderer.canvas.TileLayer(this, layer);
} else if (layer instanceof ol.layer.Vector) {
return new ol.renderer.canvas.VectorLayer(this, layer);
} else {
goog.asserts.assert(false);
return null;
@@ -141,6 +144,8 @@ ol.renderer.canvas.Map.prototype.renderFrame = function(frameState) {
context.globalAlpha = 1;
context.fillRect(0, 0, size.width, size.height);
this.calculateMatrices2D(frameState);
goog.array.forEach(frameState.layersArray, function(layer) {
var layerState = frameState.layerStates[goog.getUid(layer)];
@@ -172,6 +177,4 @@ ol.renderer.canvas.Map.prototype.renderFrame = function(frameState) {
this.renderedVisible_ = true;
}
this.calculateMatrices2D(frameState);
};

View File

@@ -14,6 +14,7 @@ goog.require('ol.style.LiteralLine');
goog.require('ol.style.LiteralPoint');
goog.require('ol.style.LiteralPolygon');
goog.require('ol.style.LiteralShape');
goog.require('ol.style.LiteralSymbolizer');
goog.require('ol.style.ShapeType');
@@ -26,8 +27,8 @@ ol.renderer.canvas.isSupported = ol.canvas.isSupported;
/**
* @constructor
* @param {!HTMLCanvasElement} canvas Target canvas.
* @param {!goog.vec.Mat4.Number} transform Transform.
* @param {HTMLCanvasElement} canvas Target canvas.
* @param {goog.vec.Mat4.Number} transform Transform.
* @param {ol.Pixel=} opt_offset Pixel offset for top-left corner. This is
* provided as an optional argument as a convenience in cases where the
* transform applies to a separate canvas.
@@ -39,6 +40,11 @@ ol.renderer.canvas.Renderer = function(canvas, transform, opt_offset) {
dx = goog.isDef(opt_offset) ? opt_offset.x : 0,
dy = goog.isDef(opt_offset) ? opt_offset.y : 0;
/**
* @type {goog.vec.Mat4.Number}
* @private
*/
this.transform_ = transform;
context.setTransform(
goog.vec.Mat4.getElement(transform, 0, 0),
goog.vec.Mat4.getElement(transform, 1, 0),
@@ -47,6 +53,15 @@ ol.renderer.canvas.Renderer = function(canvas, transform, opt_offset) {
goog.vec.Mat4.getElement(transform, 0, 3) + dx,
goog.vec.Mat4.getElement(transform, 1, 3) + dy);
var vec = [1, 0, 0];
goog.vec.Mat4.multVec3NoTranslate(transform, vec, vec);
/**
* @type {number}
* @private
*/
this.inverseScale_ = 1 / Math.sqrt(vec[0] * vec[0] + vec[1] * vec[1]);
/**
* @type {CanvasRenderingContext2D}
* @private
@@ -57,22 +72,52 @@ ol.renderer.canvas.Renderer = function(canvas, transform, opt_offset) {
/**
* @param {Array.<ol.geom.LineString>} lines Line array.
* @param {ol.style.LiteralLine} symbolizer Line symbolizer.
* @param {ol.geom.GeometryType} type Geometry type.
* @param {Array.<ol.Feature>} features Array of features.
* @param {ol.style.LiteralSymbolizer} symbolizer Symbolizer.
*/
ol.renderer.canvas.Renderer.prototype.renderLineStrings =
function(lines, symbolizer) {
ol.renderer.canvas.Renderer.prototype.renderFeaturesByGeometryType =
function(type, features, symbolizer) {
switch (type) {
case ol.geom.GeometryType.POINT:
goog.asserts.assert(symbolizer instanceof ol.style.LiteralPoint);
this.renderPointFeatures_(
features, /** @type {ol.style.LiteralPoint} */ (symbolizer));
break;
case ol.geom.GeometryType.LINESTRING:
goog.asserts.assert(symbolizer instanceof ol.style.LiteralLine);
this.renderLineStringFeatures_(
features, /** @type {ol.style.LiteralLine} */ (symbolizer));
break;
case ol.geom.GeometryType.POLYGON:
goog.asserts.assert(symbolizer instanceof ol.style.LiteralPolygon);
this.renderPolygonFeatures_(
features, /** @type {ol.style.LiteralPolygon} */ (symbolizer));
break;
default:
throw new Error('Rendering not implemented for geometry type: ' + type);
}
};
/**
* @param {Array.<ol.Feature>} features Array of line features.
* @param {ol.style.LiteralLine} symbolizer Line symbolizer.
* @private
*/
ol.renderer.canvas.Renderer.prototype.renderLineStringFeatures_ =
function(features, symbolizer) {
var context = this.context_,
i, ii, line, coords, dim, j, jj, x, y;
context.globalAlpha = symbolizer.opacity;
context.strokeStyle = symbolizer.strokeStyle;
context.lineWidth = symbolizer.strokeWidth;
context.lineWidth = symbolizer.strokeWidth * this.inverseScale_;
context.beginPath();
for (i = 0, ii = lines.length; i < ii; ++i) {
line = lines[i];
for (i = 0, ii = features.length; i < ii; ++i) {
line = features[i].getGeometry();
dim = line.dimension;
coords = line.coordinates;
for (j = 0, jj = coords.length; j < jj; j += dim) {
@@ -91,14 +136,15 @@ ol.renderer.canvas.Renderer.prototype.renderLineStrings =
/**
* @param {Array.<ol.geom.Point>} points Point array.
* @param {Array.<ol.Feature>} features Array of point features.
* @param {ol.style.LiteralPoint} symbolizer Point symbolizer.
* @private
*/
ol.renderer.canvas.Renderer.prototype.renderPoints =
function(points, symbolizer) {
ol.renderer.canvas.Renderer.prototype.renderPointFeatures_ =
function(features, symbolizer) {
var context = this.context_,
canvas, i, ii, coords;
canvas, i, ii, coords, vec;
if (symbolizer instanceof ol.style.LiteralShape) {
canvas = ol.renderer.canvas.Renderer.renderShape(symbolizer);
@@ -108,22 +154,25 @@ ol.renderer.canvas.Renderer.prototype.renderPoints =
var mid = canvas.width / 2;
context.save();
context.translate(-mid, -mid);
context.setTransform(1, 0, 0, 1, -mid, -mid);
context.globalAlpha = 1;
for (i = 0, ii = points.length; i < ii; ++i) {
coords = points[i].coordinates;
context.drawImage(canvas, coords[0], coords[1]);
for (i = 0, ii = features.length; i < ii; ++i) {
coords = features[i].getGeometry().coordinates;
vec = goog.vec.Mat4.multVec3(
this.transform_, [coords[0], coords[1], 0], []);
context.drawImage(canvas, vec[0], vec[1]);
}
context.restore();
};
/**
* @param {Array.<ol.geom.Polygon>} polygons Array of polygons.
* @param {Array.<ol.Feature>} features Array of polygon features.
* @param {ol.style.LiteralPolygon} symbolizer Polygon symbolizer.
* @private
*/
ol.renderer.canvas.Renderer.prototype.renderPolygons =
function(polygons, symbolizer) {
ol.renderer.canvas.Renderer.prototype.renderPolygonFeatures_ =
function(features, symbolizer) {
var context = this.context_,
strokeStyle = symbolizer.strokeStyle,
@@ -133,7 +182,7 @@ ol.renderer.canvas.Renderer.prototype.renderPolygons =
context.globalAlpha = symbolizer.opacity;
if (strokeStyle) {
context.strokeStyle = symbolizer.strokeStyle;
context.lineWidth = symbolizer.strokeWidth;
context.lineWidth = symbolizer.strokeWidth * this.inverseScale_;
}
if (fillStyle) {
context.fillStyle = fillStyle;
@@ -147,8 +196,8 @@ ol.renderer.canvas.Renderer.prototype.renderPolygons =
* 4) holes - render polygon to sketch canvas first
*/
context.beginPath();
for (i = 0, ii = polygons.length; i < ii; ++i) {
poly = polygons[i];
for (i = 0, ii = features.length; i < ii; ++i) {
poly = features[i].getGeometry();
dim = poly.dimension;
rings = poly.rings;
numRings = rings.length;

View File

@@ -1,6 +1,8 @@
goog.provide('ol.renderer.canvas.VectorLayer');
goog.require('goog.vec.Mat4');
goog.require('ol.filter.Geometry');
goog.require('ol.geom.GeometryType');
goog.require('ol.layer.Vector');
@@ -38,6 +40,19 @@ ol.renderer.canvas.VectorLayer = function(mapRenderer, layer) {
* @type {!goog.vec.Mat4.Number}
*/
this.transform_ = goog.vec.Mat4.createNumber();
goog.vec.Mat4.makeIdentity(this.transform_);
/**
* Geometry filters in rendering order.
* @private
* @type {Array.<ol.filter.Geometry>}
* TODO: deal with multis
*/
this.geometryFilters_ = [
new ol.filter.Geometry(ol.geom.GeometryType.POLYGON),
new ol.filter.Geometry(ol.geom.GeometryType.LINESTRING),
new ol.filter.Geometry(ol.geom.GeometryType.POINT)
];
};
goog.inherits(ol.renderer.canvas.VectorLayer, ol.renderer.canvas.Layer);
@@ -72,4 +87,76 @@ ol.renderer.canvas.VectorLayer.prototype.getTransform = function() {
*/
ol.renderer.canvas.VectorLayer.prototype.renderFrame =
function(frameState, layerState) {
var view2DState = frameState.view2DState;
var layer = this.getVectorLayer();
var source = layer.getVectorSource();
var extent = frameState.extent;
var canvasSize = frameState.size;
var canvas, context;
if (goog.isNull(this.canvas_)) {
canvas = /** @type {HTMLCanvasElement} */
(goog.dom.createElement(goog.dom.TagName.CANVAS));
canvas.width = canvasSize.width;
canvas.height = canvasSize.height;
context = /** @type {CanvasRenderingContext2D} */ (canvas.getContext('2d'));
this.canvas_ = canvas;
this.canvasSize_ = canvasSize;
this.context_ = context;
} else {
canvas = this.canvas_;
context = this.context_;
// force clear the canvas
canvas.width = canvasSize.width;
canvas.height = canvasSize.height;
this.canvasSize_ = canvasSize;
}
/**
* For now, we create a canvas renderer and have it render all features with
* every call to renderFrame.
* TODO: only render newly visible/dirty areas
*/
var canvasRenderer = new ol.renderer.canvas.Renderer(
canvas, frameState.coordinateToPixelMatrix);
// TODO: get these elsewhere
var symbolizers = {
point: new ol.style.LiteralShape({
type: ol.style.ShapeType.CIRCLE,
size: 10,
fillStyle: '#ffcc99',
strokeStyle: '#ff9933',
strokeWidth: 2,
opacity: 0.75
}),
linestring: new ol.style.LiteralLine({
strokeStyle: '#ff9933',
strokeWidth: 2,
opacity: 1
}),
polygon: new ol.style.LiteralPolygon({
fillStyle: '#ffcc99',
strokeStyle: '#ff9933',
strokeWidth: 2,
opacity: 0.5
})
};
// render features by geometry type
var filters = this.geometryFilters_,
numFilters = filters.length,
i, filter, type, features, symbolizer;
for (i = 0; i < numFilters; ++i) {
filter = filters[i];
type = filter.getType();
features = source.getFeatures(filter);
symbolizer = symbolizers[type];
canvasRenderer.renderFeaturesByGeometryType(type, features, symbolizer);
}
};

View File

@@ -132,7 +132,6 @@ ol.source.VectorOptions;
/**
* @constructor
* @extends {ol.source.Source}
@@ -150,19 +149,16 @@ ol.source.Vector = function(options) {
* @type {ol.source.FeatureCache}
* @private
*/
this.featureCache_ = null;
this.featureCache_ = new ol.source.FeatureCache();
};
goog.inherits(ol.source.Vector, ol.source.Source);
/**
* @param {Array.<ol.Feature>} features Array of features.
*/
ol.source.Vector.prototype.setFeatures = function(features) {
this.featureCache_.clear();
ol.source.Vector.prototype.addFeatures = function(features) {
for (var i = 0, ii = features.length; i < ii; ++i) {
this.featureCache_.add(features[i]);
}