/* * Paper.js - The Swiss Army Knife of Vector Graphics Scripting. * http://paperjs.org/ * * Copyright (c) 2011 - 2013, Juerg Lehni & Jonathan Puckey * http://lehni.org/ & http://jonathanpuckey.com/ * * Distributed under the MIT license. See LICENSE file for details. * * All rights reserved. */ /** * A function scope holding all the functionality needed to convert a * Paper.js DOM to a SVG DOM. */ new function() { var formatter = Formatter.instance; function setAttributes(node, attrs) { for (var key in attrs) { var val = attrs[key], namespace = SVGNamespaces[key]; if (typeof val === 'number') val = formatter.number(val); if (namespace) { node.setAttributeNS(namespace, key, val); } else { node.setAttribute(key, val); } } return node; } 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(' + formatter.point(trans) + ')'); if (!Numerical.isZero(scale.x - 1) || !Numerical.isZero(scale.y - 1)) parts.push('scale(' + formatter.point(scale) +')'); if (angle) parts.push('rotate(' + formatter.number(angle) + ')'); attrs.transform = parts.join(' '); } else { attrs.transform = 'matrix(' + matrix.getValues().join(',') + ')'; } return attrs; } 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. 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; } 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)); } // 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, then it's an arc. function isArc(i) { var segment = segments[i], next = segment.getNext(), handle1 = segment._handleOut, handle2 = next._handleIn, kappa = Numerical.KAPPA; if (handle1.isOrthogonal(handle2)) { var from = segment._point, to = next._point, // Find the corner point by intersecting the lines described // by both handles: corner = new Line(from, handle1, true).intersect( new Line(to, handle2, true), true); 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' : segments.length === 0 ? 'empty' : segments.length >= 3 ? path._closed ? 'polygon' : 'polyline' : 'line'; } 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'; } function exportGroup(item) { var attrs = getTransform(item), children = item._children; var node = createElement('g', attrs); for (var i = 0, l = children.length; i < l; i++) { var child = children[i]; var childNode = exportSVG(child); if (childNode) { if (child.isClipMask()) { var clip = createElement('clipPath'); clip.appendChild(childNode); setDefinition(child, clip, 'clip'); setAttributes(node, { 'clip-path': 'url(#' + clip.id + ')' }); } else { node.appendChild(childNode); } } } return node; } function exportRaster(item) { var attrs = getTransform(item, true), size = item.getSize(); attrs.x -= size.width / 2; attrs.y -= size.height / 2; attrs.width = size.width; attrs.height = size.height; attrs.href = item.toDataURL(); return createElement('image', attrs); } function exportText(item) { var attrs = getTransform(item, true), style = item._style, font = style.getFont(), fontSize = style.getFontSize(); if (font) attrs['font-family'] = font; if (fontSize) attrs['font-size'] = 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; switch (type) { case 'empty': return null; case 'path': var data = item.getPathData(); attrs = data && { d: data }; break; case 'polyline': case 'polygon': var parts = []; for(i = 0, l = segments.length; i < l; i++) parts.push(formatter.point(segments[i]._point)); attrs = { points: parts.join(' ') }; break; case 'rect': var width = getDistance(segments, 0, 3), height = getDistance(segments, 0, 1), // Counter-compensate the determined rotation angle point = segments[1]._point.rotate(-angle, center); attrs = { x: point.x, y: point.y, 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)) // Counter-compensate the determined rotation angle .rotate(-angle, center); attrs = { x: point.x, y: point.y, width: width, height: height, rx: rx, ry: ry }; break; case'line': var first = segments[0]._point, last = segments[segments.length - 1]._point; attrs = { 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, r: radius }; break; case 'ellipse': var rx = getDistance(segments, 2, 0) / 2, ry = getDistance(segments, 3, 1) / 2; attrs = { cx: center.x, cy: center.y, rx: rx, ry: ry }; break; } if (angle) { attrs.transform = 'rotate(' + formatter.number(angle) + ',' + formatter.point(center) + ')'; // Tell applyStyle() that to transform the gradient the other way item._gradientMatrix = new Matrix().rotate(-angle, center); } return createElement(type, attrs); } function exportCompoundPath(item) { var attrs = getTransform(item, true); var data = item.getPathData(); if (data) attrs.d = data; return createElement('path', attrs); } function exportPlacedSymbol(item) { var attrs = getTransform(item, true), symbol = item.getSymbol(), symbolNode = getDefinition(symbol); definition = symbol.getDefinition(), bounds = definition.getBounds(); if (!symbolNode) { symbolNode = createElement('symbol', { viewBox: formatter.rectangle(bounds) }); symbolNode.appendChild(exportSVG(definition)); setDefinition(symbol, symbolNode); } attrs.href = '#' + symbolNode.id; attrs.x += bounds.x; attrs.y += bounds.y; attrs.width = formatter.number(bounds.width); attrs.height = formatter.number(bounds.height); 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/ // TODO: Implement gradient merging in SVGImport var gradientNode = getDefinition(color); if (!gradientNode) { var gradient = color.getGradient(), radial = gradient._radial, matrix = item._gradientMatrix, origin = color.getOrigin().transform(matrix), destination = color.getDestination().transform(matrix), attrs; if (radial) { attrs = { cx: origin.x, cy: origin.y, r: origin.getDistance(destination) }; var highlight = color.getHighlight(); if (highlight) { highlight = highlight.transform(matrix); 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( (radial ? 'radial' : 'linear') + 'Gradient', attrs); var stops = gradient._stops; for (var i = 0, l = stops.length; i < l; i++) { var stop = stops[i], stopColor = stop._color, alpha = stopColor.getAlpha(); attrs = { offset: stop._rampPoint, 'stop-color': stopColor.toCSS(true) }; // See applyStyle for an explanation of why there are separated // opacity / color attributes. if (alpha < 1) attrs['stop-opacity'] = alpha; gradientNode.appendChild(createElement('stop', attrs)); } setDefinition(color, gradientNode); } return 'url(#' + gradientNode.id + ')'; } var exporters = { group: exportGroup, layer: exportGroup, raster: exportRaster, path: exportPath, 'point-text': exportText, 'placed-symbol': exportPlacedSymbol, 'compound-path': exportCompoundPath }; function applyStyle(item, node) { var attrs = {}, style = item._style, parent = item.getParent(), parentStyle = parent && parent._style; if (item._name != null) 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)) { if (entry.type === 'color' && value != null) { // 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: var alpha = value.getAlpha(); if (alpha < 1) attrs[entry.attribute + '-opacity'] = 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' ? formatter.number(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() return setAttributes(node, attrs); } var definitions; function getDefinition(item) { if (!definitions) definitions = { ids: {}, svgs: {} }; return item && definitions.svgs[item._id]; } function setDefinition(item, node, type) { if (!definitions) getDefinition(); // Have different id ranges per type type = type || item._type; var id = definitions.ids[type] = (definitions.ids[type] || 0) + 1; // Give the svg node an ide, and link to it from the item id. node.id = type + '-' + id; definitions.svgs[item._id] = node; } function exportDefinitions(node) { if (!definitions) return node; // We can only use svg nodes as defintion containers. Have the loop // produce one if it's a single item of another type (when calling // #exportSVG() on an item rather than a whole project) // jsdom in Node.js uses uppercase values for nodeName... var svg = node.nodeName.toLowerCase() === 'svg' && node, firstChild = svg ? svg.firstChild : node, defs = null; for (var i in definitions.svgs) { // This code is inside the loop so we only create a container if we // actually have svgs. if (!defs) { if (!svg) { svg = createElement('svg'); svg.appendChild(node); } defs = svg.insertBefore(createElement('defs'), svg.firstChild); } defs.appendChild(definitions.svgs[i]); } // Clear definitions at the end of export definitions = null; return svg; } function exportSVG(item) { var exporter = exporters[item._type], node = exporter && exporter(item, item._type); if (node && item._data) node.setAttribute('data-paper-data', JSON.stringify(item._data)); return node && applyStyle(item, node); } Item.inject(/** @lends Item# */{ /** * {@grouptitle SVG Conversion} * * Exports the item and all its children as an SVG DOM, all contained * in one top level SVG node. * * @return {SVGSVGElement} the item converted to an SVG node */ exportSVG: function() { return exportDefinitions(exportSVG(this)); } }); Project.inject(/** @lends Project# */{ /** * {@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() { var layers = this.layers, size = this.view.getSize(), node = createElement('svg', { x: 0, y: 0, width: size.width, height: size.height, version: '1.1', xmlns: 'http://www.w3.org/2000/svg', xlink: 'http://www.w3.org/1999/xlink' }); for (var i = 0, l = layers.length; i < l; i++) node.appendChild(exportSVG(layers[i])); return exportDefinitions(node); } }); };