mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-03 19:45:44 -05:00
Implement Matrix support in SvgExporter.
This commit is contained in:
parent
a6b8201cf4
commit
d9bfc2fa35
2 changed files with 50 additions and 29 deletions
|
@ -471,6 +471,7 @@ var Matrix = this.Matrix = Base.extend(/** @lends Matrix# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
getScaling: function() {
|
getScaling: function() {
|
||||||
|
// http://math.stackexchange.com/questions/13150/
|
||||||
var hor = Math.sqrt(this._a * this._a + this._c * this._c),
|
var hor = Math.sqrt(this._a * this._a + this._c * this._c),
|
||||||
ver = Math.sqrt(this._b * this._b + this._d * this._d);
|
ver = Math.sqrt(this._b * this._b + this._d * this._d);
|
||||||
return Point.create(this._a < 0 ? -hor : hor, this._b < 0 ? -ver : ver);
|
return Point.create(this._a < 0 ? -hor : hor, this._b < 0 ? -ver : ver);
|
||||||
|
|
|
@ -51,38 +51,60 @@ var SvgExporter = this.SvgExporter = new function() {
|
||||||
return segments[index1]._point.getDistance(segments[index2]._point);
|
return segments[index1]._point.getDistance(segments[index2]._point);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getTransform(item) {
|
||||||
|
var matrix = item._matrix.createShiftless(),
|
||||||
|
trans = matrix._inverseTransform(item._matrix.getTranslation()),
|
||||||
|
attrs = {
|
||||||
|
x: trans.x,
|
||||||
|
y: trans.y
|
||||||
|
};
|
||||||
|
if (matrix.isIdentity())
|
||||||
|
return attrs;
|
||||||
|
// See if we can formulate this matrix as simple scale / rotate commands
|
||||||
|
// Note: getScaling() returns values also when it's not a simple scale,
|
||||||
|
// but angle is only != null if it is, so check for that.
|
||||||
|
var transform = [],
|
||||||
|
angle = matrix.getRotation(),
|
||||||
|
scale = matrix.getScaling();
|
||||||
|
if (angle != null) {
|
||||||
|
transform.push(angle
|
||||||
|
? 'rotate(' + angle + ',' + formatPoint(center) +')'
|
||||||
|
: 'scale(' + formatPoint(scale) +')');
|
||||||
|
} else {
|
||||||
|
transform.push('matrix(' + matrix.getValues().join(',') + ')');
|
||||||
|
}
|
||||||
|
attrs.transform = transform.join(' ');
|
||||||
|
return attrs;
|
||||||
|
}
|
||||||
|
|
||||||
function exportGroup(group) {
|
function exportGroup(group) {
|
||||||
var svg = createElement('g'),
|
var attrs = getTransform(group),
|
||||||
children = group._children;
|
children = group._children;
|
||||||
|
// Override default SVG style on groups, then apply style.
|
||||||
|
attrs.fill = 'none';
|
||||||
|
var svg = createElement('g', attrs);
|
||||||
for (var i = 0, l = children.length; i < l; i++)
|
for (var i = 0, l = children.length; i < l; i++)
|
||||||
svg.appendChild(SvgExporter.exportItem(children[i]));
|
svg.appendChild(SvgExporter.exportItem(children[i]));
|
||||||
// Override default SVG style on groups, then apply style.
|
return svg;
|
||||||
return setAttributes(svg, {
|
|
||||||
fill: 'none'
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function exportText(item) {
|
function exportText(item) {
|
||||||
var point = item.getPoint(),
|
var attrs = getTransform(item),
|
||||||
style = item._style,
|
style = item._style;
|
||||||
attrs = {
|
|
||||||
x: point._x,
|
|
||||||
y: point._y
|
|
||||||
};
|
|
||||||
if (style._font != null)
|
if (style._font != null)
|
||||||
attrs['font-family'] = style._font;
|
attrs['font-family'] = style._font;
|
||||||
if (style._fontSize != null)
|
if (style._fontSize != null)
|
||||||
attrs['font-size'] = style._fontSize;
|
attrs['font-size'] = style._fontSize;
|
||||||
var svg = createElement('text', attrs);
|
var svg = createElement('text', attrs);
|
||||||
svg.textContent = item.getContent();
|
svg.textContent = item._content;
|
||||||
svg.setAttribute('transform','rotate(' + item.matrix.getRotation() + ',' + item.getPoint()._x + ',' + item.getPoint()._y +')');
|
|
||||||
return svg;
|
return svg;
|
||||||
}
|
}
|
||||||
|
|
||||||
function exportPath(path) {
|
function exportPath(path) {
|
||||||
var segments = path._segments,
|
var segments = path._segments,
|
||||||
|
center = path.getPosition(true),
|
||||||
type = determineType(path, segments),
|
type = determineType(path, segments),
|
||||||
angle = determineAngle(path, segments, type),
|
angle = determineAngle(path, segments, type, center),
|
||||||
attrs;
|
attrs;
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'path':
|
case 'path':
|
||||||
|
@ -105,7 +127,7 @@ var SvgExporter = this.SvgExporter = new function() {
|
||||||
var width = getDistance(segments, 0, 3),
|
var width = getDistance(segments, 0, 3),
|
||||||
height = getDistance(segments, 0, 1),
|
height = getDistance(segments, 0, 1),
|
||||||
// Counter-compensate the determined rotation angle
|
// Counter-compensate the determined rotation angle
|
||||||
point = segments[1]._point.rotate(-angle, path.getPosition());
|
point = segments[1]._point.rotate(-angle, center);
|
||||||
attrs = {
|
attrs = {
|
||||||
x: point.x,
|
x: point.x,
|
||||||
y: point.y,
|
y: point.y,
|
||||||
|
@ -129,7 +151,7 @@ var SvgExporter = this.SvgExporter = new function() {
|
||||||
right = segments[4]._point, // top-right side point
|
right = segments[4]._point, // top-right side point
|
||||||
point = left.subtract(right.subtract(left).normalize(rx))
|
point = left.subtract(right.subtract(left).normalize(rx))
|
||||||
// Counter-compensate the determined rotation angle
|
// Counter-compensate the determined rotation angle
|
||||||
.rotate(-angle, path.getPosition());
|
.rotate(-angle, center);
|
||||||
attrs = {
|
attrs = {
|
||||||
x: point.x,
|
x: point.x,
|
||||||
y: point.y,
|
y: point.y,
|
||||||
|
@ -150,31 +172,29 @@ var SvgExporter = this.SvgExporter = new function() {
|
||||||
};
|
};
|
||||||
break;
|
break;
|
||||||
case 'circle':
|
case 'circle':
|
||||||
var radius = getDistance(segments, 0, 2) / 2,
|
var radius = getDistance(segments, 0, 2) / 2;
|
||||||
center = path.getPosition();
|
|
||||||
attrs = {
|
attrs = {
|
||||||
cx: center._x,
|
cx: center.x,
|
||||||
cy: center._y,
|
cy: center.y,
|
||||||
r: radius
|
r: radius
|
||||||
};
|
};
|
||||||
break;
|
break;
|
||||||
case 'ellipse':
|
case 'ellipse':
|
||||||
var rx = getDistance(segments, 2, 0) / 2,
|
var rx = getDistance(segments, 2, 0) / 2,
|
||||||
ry = getDistance(segments, 3, 1) / 2,
|
ry = getDistance(segments, 3, 1) / 2;
|
||||||
center = path.getPosition();
|
|
||||||
attrs = {
|
attrs = {
|
||||||
cx: center._x,
|
cx: center.x,
|
||||||
cy: center._y,
|
cy: center.y,
|
||||||
rx: rx,
|
rx: rx,
|
||||||
ry: ry
|
ry: ry
|
||||||
};
|
};
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
var svg = createElement(type, attrs);
|
|
||||||
if (angle) {
|
if (angle) {
|
||||||
svg.setAttribute('transform', 'rotate(' + formatNumber(angle) + ','
|
attrs.transform = 'rotate(' + formatNumber(angle) + ','
|
||||||
+ formatPoint(path.getPosition()) + ')');
|
+ formatPoint(center) + ')';
|
||||||
}
|
}
|
||||||
|
var svg = createElement(type, attrs);
|
||||||
return svg;
|
return svg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -213,7 +233,7 @@ var SvgExporter = this.SvgExporter = new function() {
|
||||||
return parts.join(' ');
|
return parts.join(' ');
|
||||||
}
|
}
|
||||||
|
|
||||||
function determineAngle(path, segments, type) {
|
function determineAngle(path, segments, type, center) {
|
||||||
// If the object is a circle, ellipse, rectangle, or rounded rectangle,
|
// If the object is a circle, ellipse, rectangle, or rounded rectangle,
|
||||||
// see if they are placed at an angle.
|
// see if they are placed at an angle.
|
||||||
var topCenter = type === 'rect'
|
var topCenter = type === 'rect'
|
||||||
|
@ -224,7 +244,7 @@ var SvgExporter = this.SvgExporter = new function() {
|
||||||
? segments[1]._point
|
? segments[1]._point
|
||||||
: null;
|
: null;
|
||||||
if (topCenter) {
|
if (topCenter) {
|
||||||
var angle = topCenter.subtract(path.getPosition()).getAngle() + 90;
|
var angle = topCenter.subtract(center).getAngle() + 90;
|
||||||
return Numerical.isZero(angle) ? 0 : angle;
|
return Numerical.isZero(angle) ? 0 : angle;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
|
Loading…
Reference in a new issue