'use strict'; var Pg = require('./../pg'); var svgWidth = 1024.0; var svgHeight = 768.0; var svgRatio = svgWidth / svgHeight; var radius = 5; // in pixels (based on svgWidth and svgHeight) var strokeWidth = 1; // in pixels (based on svgWidth and svgHeight) var strokeColor = 'black'; // fill settings affect polygons and points (circles) var fillOpacity = 0.5; // 0.0 is fully transparent, 1.0 is fully opaque // unused if fillColor='none' var fillColor = 'none'; // affects polygons and circles function SvgFormat () { this.totalRows = 0; this.bbox = null; // will be computed during the results scan this.buffer = ''; this._streamingStarted = false; } SvgFormat.prototype = new Pg('svg'); SvgFormat.prototype._contentType = 'image/svg+xml; charset=utf-8'; SvgFormat.prototype.getQuery = function (sql, options) { var gn = options.gn; var dp = options.dp; return 'WITH source AS ( ' + sql + '), extent AS ( ' + ' SELECT ST_Extent(' + gn + ') AS e FROM source ' + '), extent_info AS ( SELECT e, ' + 'st_xmin(e) as ex0, st_ymax(e) as ey0, ' + 'st_xmax(e)-st_xmin(e) as ew, ' + 'st_ymax(e)-st_ymin(e) as eh FROM extent )' + ', trans AS ( SELECT CASE WHEN ' + 'eh = 0 THEN ' + svgWidth + '/ COALESCE(NULLIF(ew,0),' + svgWidth + ') WHEN ' + svgRatio + ' <= (ew / eh) THEN (' + svgWidth + '/ew ) ELSE (' + svgHeight + '/eh ) END as s ' + ', ex0 as x0, ey0 as y0 FROM extent_info ) ' + 'SELECT st_TransScale(e, -x0, -y0, s, s)::box2d as ' + gn + '_box, ST_Dimension(' + gn + ') as ' + gn + '_dimension, ST_AsSVG(ST_TransScale(' + gn + ', ' + '-x0, -y0, s, s), 0, ' + dp + ') as ' + gn + // + ', ex0, ey0, ew, eh, s ' // DEBUG ONLY + ' FROM trans, extent_info, source' + ' ORDER BY the_geom_dimension ASC'; }; SvgFormat.prototype.startStreaming = function () { if (this.opts.beforeSink) { this.opts.beforeSink(); } var header = [ '', '' ]; var rootTag = '\n'; } else if (gdims === 1) { // Avoid filling closed linestrings this.buffer += '\n'; } else if (gdims === 2) { this.buffer += '\n'; } if (!this.bbox) { // Parse layer extent: "BOX(x y, X Y)" // NOTE: the name of the extent field is // determined by the same code adding the // ST_AsSVG call (in queryResult) // var bbox = row[this.opts.gn + '_box']; bbox = bbox.match(/BOX\(([^ ]*) ([^ ,]*),([^ ]*) ([^)]*)\)/); this.bbox = { xmin: parseFloat(bbox[1]), ymin: parseFloat(bbox[2]), xmax: parseFloat(bbox[3]), ymax: parseFloat(bbox[4]) }; } if (!this._streamingStarted && this.bbox) { this.startStreaming(); } if (this._streamingStarted && (this.totalRows % (this.opts.bufferedRows || 1000))) { this.opts.sink.write(this.buffer); this.buffer = ''; } }; SvgFormat.prototype.handleQueryEnd = function () { if (this.error && !this._streamingStarted) { this.callback(this.error); return; } if (this.opts.profiler) { this.opts.profiler.done('gotRows'); } if (!this._streamingStarted) { this.startStreaming(); } // rootTag close this.buffer += '\n'; this.opts.sink.write(this.buffer); this.opts.sink.end(); this.callback(); }; module.exports = SvgFormat;