More work on SvgExporter code.

This commit is contained in:
Jürg Lehni 2012-11-05 23:45:23 -08:00
parent c544083a4c
commit a6d19ac681

View file

@ -71,23 +71,25 @@ var SvgExporter = this.SvgExporter = new function() {
function exportPath(path) { function exportPath(path) {
var segments = path._segments, var segments = path._segments,
type = determineType(path, segments), type = determineType(path, segments),
angle = determineAngle(path, segments, type),
attrs; attrs;
// If the object is a circle, ellipse, rectangle, or rounded rectangle,
// see if they are placed at an angle.
var angle = 0,
topCenter = type === 'rect'
? segments[1]._point.add(segments[2]._point).divide(2)
: type === 'roundrect'
? segments[3]._point.add(segments[4]._point).divide(2)
: type === 'circle' || type === 'ellipse'
? segments[1]._point
: null;
if (topCenter) {
angle = topCenter.subtract(path.getPosition()).getAngle() + 90;
if (Numerical.isZero(angle))
angle = 0;
}
switch (type) { switch (type) {
case 'path':
attrs = {
d: drawPath(path, segments)
};
break;
case 'polyline':
case 'polygon':
var parts = [];
for(i = 0; i < segments.length; i++) {
var point = segments[i]._point;
parts.push(point._x + ',' + point._y);
}
attrs = {
points: parts.join(' ')
};
break;
case 'rect': case 'rect':
var width = getDistance(segments, 0, 3), var width = getDistance(segments, 0, 3),
height = getDistance(segments, 0, 1), height = getDistance(segments, 0, 1),
@ -152,35 +154,33 @@ var SvgExporter = this.SvgExporter = new function() {
ry: ry ry: ry
}; };
break; break;
case 'polyline':
case 'polygon':
var parts = [];
for(i = 0; i < segments.length; i++) {
var point = segments[i]._point;
parts.push(point._x + ',' + point._y);
} }
attrs = { var svg = createElement(type, attrs);
points: parts.join(' ')
};
break;
}
if (attrs) {
var svg = createElement(type, attrs),
center = path.getPosition();
if (angle) { if (angle) {
var center = path.getPosition();
svg.setAttribute('transform', 'rotate(' + angle + ',' svg.setAttribute('transform', 'rotate(' + angle + ','
+ center._x + ',' + center._y + ')'); + center._x + ',' + center._y + ')');
} }
return svg; return svg;
} }
return pathSetup(path, segments);
function drawPath(path, segments) {
var parts = [],
style = path._style,
first = segments[0]._point;
parts.push('M' + first._x + ',' + first._y);
for (i = 0; i < segments.length - 1; i++)
drawCurve(parts, segments[i], segments[i + 1], false);
// We only need to draw the connecting curve if it is not a line, and if
// the path is cosed and has a stroke color, or if it is filled.
if (path._closed && style._strokeColor || style._fillColor)
drawCurve(parts, segments[segments.length - 1], segments[0], true);
if (path._closed)
parts.push('z');
return parts.join(' ');
} }
function pathSetup(path, segments) { function drawCurve(parts, seg1, seg2, skipLine) {
var svg = createElement('path');
var parts = [];
parts.push('M' + segments[0]._point._x + ',' + segments[0]._point._y);
function drawCurve(seg1, seg2, skipLine) {
var point1 = seg1._point, var point1 = seg1._point,
point2 = seg2._point, point2 = seg2._point,
x1 = point1._x, x1 = point1._x,
@ -191,11 +191,11 @@ var SvgExporter = this.SvgExporter = new function() {
handle2 = seg2._handleIn; handle2 = seg2._handleIn;
if (handle1.isZero() && handle2.isZero()) { if (handle1.isZero() && handle2.isZero()) {
if (!skipLine) { if (!skipLine) {
// L is lineto, moving to a point with drawing // L = lineto: moving to a point with drawing
parts.push('L' + x2 + ',' + y2 + ' '); parts.push('L' + x2 + ',' + y2 + ' ');
} }
} else { } else {
// c is curveto, relative: handle1, handle2 + end - start, end - start // c = relative curveto: handle1, handle2 + end - start, end - start
x2 -= x1; x2 -= x1;
y2 -= y1; y2 -= y1;
parts.push( parts.push(
@ -205,21 +205,24 @@ var SvgExporter = this.SvgExporter = new function() {
); );
} }
} }
for (i = 0; i < segments.length - 1; i++)
drawCurve(segments[i], segments[i + 1], false); function determineAngle(path, segments, type) {
// We only need to draw the connecting curve if the path is cosed and // If the object is a circle, ellipse, rectangle, or rounded rectangle,
// has a stroke color, or if it's filled. // see if they are placed at an angle.
if (path._closed && path._style._strokeColor || path._style._fillColor) var topCenter = type === 'rect'
drawCurve(segments[segments.length - 1], segments[0], true); ? segments[1]._point.add(segments[2]._point).divide(2)
if (path._closed) : type === 'roundrect'
parts.push('z'); ? segments[3]._point.add(segments[4]._point).divide(2)
svg.setAttribute('d', parts.join(' ')); : type === 'circle' || type === 'ellipse'
return svg; ? segments[1]._point
: null;
if (topCenter) {
var angle = topCenter.subtract(path.getPosition()).getAngle() + 90;
return Numerical.isZero(angle) ? 0 : angle;
}
return 0;
} }
/**
* Checks the type SVG object created by converting from Paper.js
*/
function determineType(path, segments) { function determineType(path, segments) {
// See if actually have any curves in the path. Differentiate // See if actually have any curves in the path. Differentiate
// between straight objects (line, polyline, rect, and polygon) and // between straight objects (line, polyline, rect, and polygon) and
@ -261,6 +264,7 @@ var SvgExporter = this.SvgExporter = new function() {
} }
} }
} }
return 'path';
} }
function applyStyle(item, svg) { function applyStyle(item, svg) {