paper.js/src/svg/SvgExport.js

517 lines
15 KiB
JavaScript
Raw Normal View History

2012-11-02 20:47:14 -04:00
/*
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
2012-09-30 17:51:50 -04:00
* http://paperjs.org/
*
* Copyright (c) 2011 - 2013, Juerg Lehni & Jonathan Puckey
2012-09-30 17:51:50 -04:00
* http://lehni.org/ & http://jonathanpuckey.com/
*
* Distributed under the MIT license. See LICENSE file for details.
*
* All rights reserved.
*/
2012-11-04 02:04:15 -05:00
/**
* A function scope holding all the functionality needed to convert a
* Paper.js DOM to a SVG DOM.
2012-11-04 02:04:15 -05:00
*/
new function() {
// Shortcut to Format.number
var format = Format.number,
namespaces = {
href: 'http://www.w3.org/1999/xlink'
};
2013-02-10 22:02:53 -05:00
function setAttributes(node, attrs) {
for (var key in attrs) {
var val = attrs[key],
namespace = namespaces[key];
if (typeof val === 'number')
val = format(val);
if (namespace) {
node.setAttributeNS(namespace, key, val);
} else {
2013-02-10 22:02:53 -05:00
node.setAttribute(key, val);
}
}
2013-02-10 22:02:53 -05:00
return node;
2012-11-05 23:31:45 -05:00
}
function createElement(tag, attrs) {
return setAttributes(
document.createElementNS('http://www.w3.org/2000/svg', tag), attrs);
}
function getDistance(segments, index1, index2) {
return segments[index1]._point.getDistance(segments[index2]._point);
}
function getTransform(item, coordinates) {
var matrix = item._matrix,
trans = matrix.getTranslation(),
attrs = {};
if (coordinates) {
// If the item suppports x- and y- coordinates, we're taking out the
// translation part of the matrix and move it to x, y attributes, to
// produce more readable markup, and not have to use center points
// in rotate(). To do so, SVG requries us to inverse transform the
// translation point by the matrix itself, since they are provided
// in local coordinates.
matrix = matrix.shiftless();
var point = matrix._inverseTransform(trans);
attrs.x = point.x;
attrs.y = point.y;
trans = null;
}
if (matrix.isIdentity())
return attrs;
// See if we can decompose the matrix and can formulate it as a simple
// translate/scale/rotate command sequence.
var decomposed = matrix.decompose();
if (decomposed && !decomposed.shearing) {
var parts = [],
angle = decomposed.rotation,
scale = decomposed.scaling;
if (trans && !trans.isZero())
parts.push('translate(' + Format.point(trans) + ')');
if (!Numerical.isZero(scale.x - 1) || !Numerical.isZero(scale.y - 1))
parts.push('scale(' + Format.point(scale) +')');
if (angle)
parts.push('rotate(' + format(angle) + ')');
attrs.transform = parts.join(' ');
} else {
attrs.transform = 'matrix(' + matrix.getValues().join(',') + ')';
}
return attrs;
}
2012-11-06 14:00:58 -05:00
function determineAngle(path, segments, type, center) {
// If the object is a circle, ellipse, rectangle, or rounded rectangle,
// see if it is placed at an angle, by figuring out its topCenter point
// and measuring the angle to its center.
2012-11-06 14:00:58 -05:00
var 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;
var angle = topCenter && topCenter.subtract(center).getAngle() + 90;
return Numerical.isZero(angle || 0) ? 0 : angle;
2012-11-06 14:00:58 -05:00
}
function determineType(path, segments) {
// Returns true if the the two segment indices are the beggining of two
// lines and if the wto lines are parallel.
function isColinear(i, j) {
var seg1 = segments[i],
seg2 = seg1.getNext(),
seg3 = segments[j],
seg4 = seg3.getNext();
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
&& seg3._handleOut.isZero() && seg4._handleIn.isZero()
&& seg2._point.subtract(seg1._point).isColinear(
seg4._point.subtract(seg3._point));
}
// Kappa, see: http://www.whizkidtech.redprince.net/bezier/circle/kappa/
var kappa = 4 * (Math.sqrt(2) - 1) / 3;
// Returns true if the segment at the given index is the beginning of
// a orthogonal arc segment. The code is looking at the length of the
// handles and their relation to the distance to the imaginary corner
// point. If the relation is kappa (see above), then it's an arc.
function isArc(i) {
var segment = segments[i],
next = segment.getNext(),
handle1 = segment._handleOut,
handle2 = next._handleIn;
if (handle1.isOrthogonal(handle2)) {
var from = segment._point,
to = next._point,
// Find hte corner point by intersecting the lines described
// by both handles:
corner = new Line(from, handle1).intersect(
new Line(to, handle2));
return corner && Numerical.isZero(handle1.getLength() /
corner.subtract(from).getLength() - kappa)
&& Numerical.isZero(handle2.getLength() /
corner.subtract(to).getLength() - kappa);
}
}
// See if actually have any curves in the path. Differentiate
// between straight objects (line, polyline, rect, and polygon) and
// objects with curves(circle, ellipse, roundedRectangle).
if (path.isPolygon()) {
return segments.length === 4 && path._closed
&& isColinear(0, 2) && isColinear(1, 3)
? 'rect'
2012-12-09 21:04:56 -05:00
: segments.length === 0
? 'empty'
: segments.length >= 3
? path._closed ? 'polygon' : 'polyline'
: 'line';
2012-11-06 14:00:58 -05:00
} else if (path._closed) {
if (segments.length === 8
&& isArc(0) && isArc(2) && isArc(4) && isArc(6)
&& isColinear(1, 5) && isColinear(3, 7)) {
return 'roundrect';
} else if (segments.length === 4
&& isArc(0) && isArc(1) && isArc(2) && isArc(3)) {
// If the distance between (point0 and point2) and (point1
// and point3) are equal, then it is a circle
return Numerical.isZero(getDistance(segments, 0, 2)
- getDistance(segments, 1, 3))
? 'circle'
: 'ellipse';
}
}
return 'path';
}
2013-02-10 22:02:53 -05:00
function exportGroup(item) {
var attrs = getTransform(item),
children = item._children;
// Override default SVG style on groups, then apply style.
attrs.fill = 'none';
2013-02-10 22:02:53 -05:00
var node = createElement('g', attrs);
for (var i = 0, l = children.length; i < l; i++) {
2013-02-10 13:23:49 -05:00
var child = exportSvg(children[i]);
if (child)
2013-02-10 22:02:53 -05:00
node.appendChild(child);
}
2013-02-10 22:02:53 -05:00
return node;
}
2013-02-09 12:44:25 -05:00
function exportRaster(item) {
var attrs = getTransform(item, true),
size = item.getSize();
attrs.x -= size.width / 2;
attrs.y -= size.height / 2;
2013-02-10 13:23:49 -05:00
attrs.width = size.width;
attrs.height = size.height;
attrs.href = item.toDataURL();
return createElement('image', attrs);
2013-02-09 12:44:25 -05:00
}
function exportText(item) {
var attrs = getTransform(item, true),
style = item._style;
if (style._font != null)
attrs['font-family'] = style._font;
if (style._fontSize != null)
attrs['font-size'] = style._fontSize;
var node = createElement('text', attrs);
node.textContent = item._content;
return node;
}
function exportPath(item) {
var segments = item._segments,
center = item.getPosition(true),
type = determineType(item, segments),
angle = determineAngle(item, segments, type, center),
attrs;
2012-09-30 17:51:50 -04:00
switch (type) {
2012-12-09 21:04:56 -05:00
case 'empty':
return null;
2012-11-06 02:45:23 -05:00
case 'path':
var data = item.getPathData();
attrs = data && { d: data };
2012-11-06 02:45:23 -05:00
break;
case 'polyline':
case 'polygon':
var parts = [];
2012-11-06 14:28:50 -05:00
for(i = 0, l = segments.length; i < l; i++)
parts.push(Format.point(segments[i]._point));
2012-11-06 02:45:23 -05:00
attrs = {
points: parts.join(' ')
};
break;
case 'rect':
2012-11-05 23:31:45 -05:00
var width = getDistance(segments, 0, 3),
height = getDistance(segments, 0, 1),
2012-11-06 11:28:54 -05:00
// Counter-compensate the determined rotation angle
point = segments[1]._point.rotate(-angle, center);
attrs = {
x: point.x,
y: point.y,
2012-11-05 23:31:45 -05:00
width: width,
height: height
};
break;
case 'roundrect':
type = 'rect';
// d-variables and point are used to determine the rounded corners
// for the rounded rectangle
var width = getDistance(segments, 1, 6),
height = getDistance(segments, 0, 3),
// Subtract side lengths from total width and divide by 2 to get
// corner radius size
rx = (width - getDistance(segments, 0, 7)) / 2,
ry = (height - getDistance(segments, 1, 2)) / 2,
// Calculate topLeft corner point, by using sides vectors and
// subtracting normalized rx vector to calculate arc corner.
left = segments[3]._point, // top-left side point
right = segments[4]._point, // top-right side point
point = left.subtract(right.subtract(left).normalize(rx))
2012-11-06 11:28:54 -05:00
// Counter-compensate the determined rotation angle
.rotate(-angle, center);
attrs = {
x: point.x,
y: point.y,
2012-11-05 23:31:45 -05:00
width: width,
height: height,
rx: rx,
ry: ry
};
break;
case'line':
2012-11-05 23:31:45 -05:00
var first = segments[0]._point,
last = segments[segments.length - 1]._point;
attrs = {
2012-11-05 23:31:45 -05:00
x1: first._x,
y1: first._y,
x2: last._x,
y2: last._y
};
break;
case 'circle':
var radius = getDistance(segments, 0, 2) / 2;
attrs = {
cx: center.x,
cy: center.y,
2012-11-05 23:31:45 -05:00
r: radius
};
break;
case 'ellipse':
2012-11-06 00:19:53 -05:00
var rx = getDistance(segments, 2, 0) / 2,
ry = getDistance(segments, 3, 1) / 2;
attrs = {
cx: center.x,
cy: center.y,
2012-11-06 00:19:53 -05:00
rx: rx,
ry: ry
};
break;
}
2012-11-06 02:45:23 -05:00
if (angle) {
attrs.transform = 'rotate(' + format(angle) + ','
+ Format.point(center) + ')';
// Tell applyStyle() that to transform the gradient the other way
item._gradientMatrix = new Matrix().rotate(-angle, center);
2012-09-30 17:51:50 -04:00
}
2013-02-09 16:38:22 -05:00
return createElement(type, attrs);
}
2013-02-10 13:23:49 -05:00
function exportCompoundPath(item) {
var attrs = getTransform(item, true);
var data = item.getPathData();
if (data)
attrs.d = data;
2013-02-10 13:23:49 -05:00
return createElement('path', attrs);
}
function exportPlacedSymbol(item) {
var attrs = getTransform(item, true),
symbol = item.getSymbol(),
symbolNode = getDefinition(symbol);
2013-02-10 13:23:49 -05:00
definition = symbol.getDefinition(),
bounds = definition.getBounds();
if (!symbolNode) {
symbolNode = createElement('symbol', {
viewBox: Format.rectangle(bounds)
2013-02-10 13:23:49 -05:00
});
symbolNode.appendChild(exportSvg(definition));
setDefinition(symbol, symbolNode);
2013-02-10 13:23:49 -05:00
}
attrs.href = '#' + symbolNode.id;
2013-02-10 13:23:49 -05:00
attrs.x += bounds.x;
attrs.y += bounds.y;
attrs.width = format(bounds.width);
attrs.height = format(bounds.height);
2013-02-10 13:23:49 -05:00
return createElement('use', attrs);
}
function exportGradient(color, item) {
// NOTE: As long as the fillTransform attribute is not implemented,
// we need to create a separate gradient object for each gradient,
// even when they share the same gradient defintion.
// http://www.svgopen.org/2011/papers/20-Separating_gradients_from_geometry/
2013-02-10 22:40:15 -05:00
// TODO: Implement gradient merging in SvgImport
2013-02-12 20:23:56 -05:00
var gradientNode = getDefinition(color);
if (!gradientNode) {
2013-02-12 20:23:56 -05:00
var gradient = color.gradient,
type = gradient._type,
2013-02-12 20:23:56 -05:00
matrix = item._gradientMatrix,
origin = color._origin.transform(matrix),
destination = color._destination.transform(matrix),
highlight = color._hilite && color._hilite.transform(matrix),
attrs;
if (type == 'RadialGradient') {
attrs = {
cx: origin.x,
cy: origin.y,
r: origin.getDistance(destination)
};
if (highlight) {
attrs.fx = highlight.x;
attrs.fy = highlight.y;
}
} else {
attrs = {
x1: origin.x,
y1: origin.y,
x2: destination.x,
y2: destination.y
};
}
attrs.gradientUnits = 'userSpaceOnUse';
gradientNode = createElement(type[0].toLowerCase() + type.slice(1),
attrs);
var stops = gradient._stops;
for (var i = 0, l = stops.length; i < l; i++) {
var stop = stops[i],
2013-02-12 20:05:54 -05:00
stopColor = stop._color;
2013-02-10 22:40:44 -05:00
attrs = {
offset: stop._rampPoint,
2013-02-12 20:05:54 -05:00
'stop-color': stopColor.toCss(true)
2013-02-10 22:40:44 -05:00
};
// See applyStyle for an explanation of why there are separated
// opacity / color attributes.
2013-02-12 20:05:54 -05:00
if (stopColor.getAlpha() < 1)
attrs['stop-opacity'] = stopColor._alpha;
gradientNode.appendChild(createElement('stop', attrs));
}
2013-02-12 20:05:54 -05:00
setDefinition(color, gradientNode);
2013-02-10 22:02:53 -05:00
}
return 'url(#' + gradientNode.id + ')';
2013-02-10 22:02:53 -05:00
}
2012-11-06 14:00:58 -05:00
var exporters = {
Group: exportGroup,
Layer: exportGroup,
Raster: exportRaster,
PointText: exportText,
PlacedSymbol: exportPlacedSymbol,
Path: exportPath,
CompoundPath: exportCompoundPath
2012-11-06 14:00:58 -05:00
};
2013-02-10 22:02:53 -05:00
function applyStyle(item, node) {
var attrs = {},
style = item._style,
2012-11-05 12:05:32 -05:00
parent = item.getParent(),
parentStyle = parent && parent._style;
if (item._name != null)
2012-11-05 22:26:54 -05:00
attrs.id = item._name;
Base.each(SvgStyles, function(entry) {
// Get a given style only if it differs from the value on the parent
// (A layer or group which can have style values in SVG).
var value = style[entry.get]();
if (!parentStyle || !Base.equals(parentStyle[entry.get](), value)) {
// Support for css-style rgba() values is not in SVG 1.1, so
// separate the alpha value of colors with alpha into the
// separate fill- / stroke-opacity attribute:
if (entry.type === 'color' && value != null && value.getAlpha() < 1)
attrs[entry.attribute + '-opacity'] = value._alpha;
attrs[entry.attribute] = value == null
? 'none'
: entry.type === 'color'
? value.gradient
? exportGradient(value, item)
: value.toCss(true) // false for noAlpha, see above
: entry.type === 'array'
? value.join(',')
: entry.type === 'number'
? format(value)
: value;
}
});
if (item._opacity != null && item._opacity < 1)
attrs.opacity = item._opacity;
if (item._visibility != null && !item._visibility)
attrs.visibility = 'hidden';
delete item._gradientMatrix; // see exportPath()
2013-02-10 22:02:53 -05:00
return setAttributes(node, attrs);
}
2013-02-10 13:23:49 -05:00
var definitions;
function getDefinition(item) {
if (!definitions)
definitions = { ids: {}, svgs: {} };
return definitions.svgs[item._id];
}
function setDefinition(item, node) {
2013-04-06 03:16:08 -04:00
var type = item._class.toLowerCase(),
id = definitions.ids[type] = (definitions.ids[type] || 0) + 1;
2013-02-10 22:02:53 -05:00
node.id = type + '-' + id;
definitions.svgs[item._id] = node;
2013-02-10 13:23:49 -05:00
}
2013-02-10 22:02:53 -05:00
function exportDefinitions(node) {
2013-02-10 13:23:49 -05:00
if (!definitions)
2013-02-10 22:02:53 -05:00
return node;
// We can only use node nodes as defintion containers. Have the loop
2013-02-10 13:23:49 -05:00
// produce one if it's a single item of another type (when calling
// #exportSvg() on an item rather than a whole project)
2013-03-01 15:59:43 -05:00
var container = node.nodeName == 'svg' && node,
2013-02-10 22:02:53 -05:00
firstChild = container ? container.firstChild : node;
2013-02-10 13:23:49 -05:00
for (var i in definitions.svgs) {
if (!container) {
container = createElement('svg');
2013-02-10 22:02:53 -05:00
container.appendChild(node);
2013-02-10 13:23:49 -05:00
}
container.insertBefore(definitions.svgs[i], firstChild);
}
// Clear definitions at the end of export
definitions = null;
return container;
}
function exportSvg(item) {
2013-04-06 03:16:08 -04:00
var exporter = exporters[item._class],
node = exporter && exporter(item, item._class);
2013-02-10 22:02:53 -05:00
return node && applyStyle(item, node);
2013-02-10 13:23:49 -05:00
}
2012-11-06 16:34:46 -05:00
Item.inject(/** @lends Item# */{
/**
2012-11-06 16:34:46 -05:00
* {@grouptitle SVG Conversion}
*
* Exports the item and all its child items as an SVG DOM, all contained
* in one top level SVG group node.
*
* @return {SVGSVGElement} the item converted to an SVG node
*/
exportSvg: function() {
2013-02-10 22:02:53 -05:00
var node = exportSvg(this);
return exportDefinitions(node);
}
});
2012-11-06 16:34:46 -05:00
Project.inject(/** @lends Project# */{
/**
2012-11-06 16:34:46 -05:00
* {@grouptitle SVG Conversion}
*
* Exports the project and all its layers and child items as an SVG DOM,
* all contained in one top level SVG group node.
*
* @return {SVGSVGElement} the project converted to an SVG node
*/
exportSvg: function() {
2013-02-10 22:02:53 -05:00
var node = createElement('svg'),
layers = this.layers;
for (var i = 0, l = layers.length; i < l; i++)
2013-02-10 22:02:53 -05:00
node.appendChild(exportSvg(layers[i]));
return exportDefinitions(node);
}
});
};