mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-03 19:45:44 -05:00
SVGImport: Improve consistency of style handling.
This commit is contained in:
parent
8542eb62b4
commit
df57c4adb9
4 changed files with 96 additions and 87 deletions
|
@ -72,9 +72,9 @@ var Style = Base.extend(new function() {
|
||||||
// not supported.
|
// not supported.
|
||||||
var defaults = {
|
var defaults = {
|
||||||
// Paths
|
// Paths
|
||||||
fillColor: undefined,
|
fillColor: null,
|
||||||
fillRule: 'nonzero',
|
fillRule: 'nonzero',
|
||||||
strokeColor: undefined,
|
strokeColor: null,
|
||||||
strokeWidth: 1,
|
strokeWidth: 1,
|
||||||
strokeCap: 'butt',
|
strokeCap: 'butt',
|
||||||
strokeJoin: 'miter',
|
strokeJoin: 'miter',
|
||||||
|
@ -83,11 +83,11 @@ var Style = Base.extend(new function() {
|
||||||
dashOffset: 0,
|
dashOffset: 0,
|
||||||
dashArray: [],
|
dashArray: [],
|
||||||
// Shadows
|
// Shadows
|
||||||
shadowColor: undefined,
|
shadowColor: null,
|
||||||
shadowBlur: 0,
|
shadowBlur: 0,
|
||||||
shadowOffset: new Point(),
|
shadowOffset: new Point(),
|
||||||
// Selection
|
// Selection
|
||||||
selectedColor: undefined
|
selectedColor: null
|
||||||
},
|
},
|
||||||
// For TextItem, override default fillColor and add text-specific properties
|
// For TextItem, override default fillColor and add text-specific properties
|
||||||
textDefaults = new Base(defaults, {
|
textDefaults = new Base(defaults, {
|
||||||
|
|
|
@ -72,11 +72,12 @@ new function() {
|
||||||
function importGroup(node, type, options, isRoot) {
|
function importGroup(node, type, options, isRoot) {
|
||||||
var nodes = node.childNodes,
|
var nodes = node.childNodes,
|
||||||
isClip = type === 'clippath',
|
isClip = type === 'clippath',
|
||||||
|
isDefs = type === 'defs',
|
||||||
item = new Group(),
|
item = new Group(),
|
||||||
project = item._project,
|
project = item._project,
|
||||||
currentStyle = project._currentStyle,
|
currentStyle = project._currentStyle,
|
||||||
children = [];
|
children = [];
|
||||||
if (!isClip) {
|
if (!isClip && !isDefs) {
|
||||||
item = applyAttributes(item, node, isRoot);
|
item = applyAttributes(item, node, isRoot);
|
||||||
// Style on items needs to be handled differently than all other
|
// Style on items needs to be handled differently than all other
|
||||||
// items: We first apply the style to the item, then use it as the
|
// items: We first apply the style to the item, then use it as the
|
||||||
|
@ -90,7 +91,7 @@ new function() {
|
||||||
// e.g. Affinity Designer exports defs as last.
|
// e.g. Affinity Designer exports defs as last.
|
||||||
var defs = node.querySelectorAll('defs');
|
var defs = node.querySelectorAll('defs');
|
||||||
for (var i = 0, l = defs.length; i < l; i++) {
|
for (var i = 0, l = defs.length; i < l; i++) {
|
||||||
importSVG(defs[i], options, false);
|
importNode(defs[i], options, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Collect the children in an array and apply them all at once.
|
// Collect the children in an array and apply them all at once.
|
||||||
|
@ -99,7 +100,7 @@ new function() {
|
||||||
child;
|
child;
|
||||||
if (childNode.nodeType === 1
|
if (childNode.nodeType === 1
|
||||||
&& childNode.nodeName.toLowerCase() !== 'defs'
|
&& childNode.nodeName.toLowerCase() !== 'defs'
|
||||||
&& (child = importSVG(childNode, options, false))
|
&& (child = importNode(childNode, options, false))
|
||||||
&& !(child instanceof SymbolDefinition))
|
&& !(child instanceof SymbolDefinition))
|
||||||
children.push(child);
|
children.push(child);
|
||||||
}
|
}
|
||||||
|
@ -110,7 +111,7 @@ new function() {
|
||||||
item = applyAttributes(item.reduce(), node, isRoot);
|
item = applyAttributes(item.reduce(), node, isRoot);
|
||||||
// Restore currentStyle
|
// Restore currentStyle
|
||||||
project._currentStyle = currentStyle;
|
project._currentStyle = currentStyle;
|
||||||
if (isClip || type === 'defs') {
|
if (isClip || isDefs) {
|
||||||
// We don't want the defs in the DOM. But we might want to use
|
// We don't want the defs in the DOM. But we might want to use
|
||||||
// Symbols for them to save memory?
|
// Symbols for them to save memory?
|
||||||
item.remove();
|
item.remove();
|
||||||
|
@ -197,37 +198,9 @@ new function() {
|
||||||
'#document': function (node, type, options, isRoot) {
|
'#document': function (node, type, options, isRoot) {
|
||||||
var nodes = node.childNodes;
|
var nodes = node.childNodes;
|
||||||
for (var i = 0, l = nodes.length; i < l; i++) {
|
for (var i = 0, l = nodes.length; i < l; i++) {
|
||||||
var child = nodes[i],
|
var child = nodes[i];
|
||||||
next;
|
if (child.nodeType === 1)
|
||||||
if (child.nodeType === 1) {
|
return importNode(child, options, isRoot);
|
||||||
// NOTE: We need to move the SVG node to the current
|
|
||||||
// document, so default styles apply! For this we create and
|
|
||||||
// insert a temporary SVG parent node which is removed again
|
|
||||||
// at the end. This parent node also helps fix a bug on IE.
|
|
||||||
var body = document.body,
|
|
||||||
// No need to inherit styles on Node.js
|
|
||||||
parent = !paper.agent.node && SvgElement.create('svg');
|
|
||||||
if (parent) {
|
|
||||||
body.appendChild(parent);
|
|
||||||
// If no stroke-width is set, IE/Edge appears to have a
|
|
||||||
// default of 0.01px. We can set a default style on the
|
|
||||||
// parent container as a more sensible fall-back.
|
|
||||||
parent.style.strokeWidth = '1px';
|
|
||||||
next = child.nextSibling;
|
|
||||||
parent.appendChild(child);
|
|
||||||
}
|
|
||||||
var item = importSVG(child, options, isRoot);
|
|
||||||
if (parent) {
|
|
||||||
// After import, move things back to how they were:
|
|
||||||
body.removeChild(parent);
|
|
||||||
if (next) {
|
|
||||||
node.insertBefore(child, next);
|
|
||||||
} else {
|
|
||||||
node.appendChild(child);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return item;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
// http://www.w3.org/TR/SVG/struct.html#Groups
|
// http://www.w3.org/TR/SVG/struct.html#Groups
|
||||||
|
@ -581,6 +554,85 @@ new function() {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function importNode(node, options, isRoot) {
|
||||||
|
// jsdom in Node.js uses uppercase values for nodeName...
|
||||||
|
var type = node.nodeName.toLowerCase(),
|
||||||
|
isElement = type !== '#document',
|
||||||
|
body = document.body,
|
||||||
|
container,
|
||||||
|
parent,
|
||||||
|
next;
|
||||||
|
if (isRoot && isElement) {
|
||||||
|
// Set rootSize root element size, fall-back to view size.
|
||||||
|
rootSize = getSize(node, null, null, true)
|
||||||
|
|| paper.getView().getSize();
|
||||||
|
// We need to move the SVG node to the current document, so default
|
||||||
|
// styles apply! For this we create and insert a temporary SVG
|
||||||
|
// container which is removed again at the end. This container also
|
||||||
|
// helps fix a bug on IE. No need to inherit styles on Node.js
|
||||||
|
container = !paper.agent.node && SvgElement.create('svg', {
|
||||||
|
// If no stroke-width is set, IE/Edge appears to have a
|
||||||
|
// default of 0.01px. We can set a default style on the
|
||||||
|
// parent container as a more sensible fall-back. Also, browsers
|
||||||
|
// have a default miter-limit of 4, while Paper.js has 10
|
||||||
|
style: 'stroke-width: 1px; stroke-miterlimit: 10'
|
||||||
|
});
|
||||||
|
if (container) {
|
||||||
|
body.appendChild(container);
|
||||||
|
parent = node.parentNode;
|
||||||
|
next = node.nextSibling;
|
||||||
|
container.appendChild(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Have items imported from SVG not bake in all transformations to their
|
||||||
|
// content and children, as this is how SVG works too, but preserve the
|
||||||
|
// current setting so we can restore it after.
|
||||||
|
var settings = paper.settings,
|
||||||
|
applyMatrix = settings.applyMatrix;
|
||||||
|
settings.applyMatrix = false;
|
||||||
|
var importer = importers[type],
|
||||||
|
item = importer && importer(node, type, options, isRoot) || null;
|
||||||
|
settings.applyMatrix = applyMatrix;
|
||||||
|
if (item) {
|
||||||
|
// Do not apply attributes if this is a #document node.
|
||||||
|
// See importGroup() for an explanation of filtering for Group:
|
||||||
|
if (isElement && !(item instanceof Group))
|
||||||
|
item = applyAttributes(item, node, isRoot);
|
||||||
|
// Support onImportItem callback, to provide mechanism to handle
|
||||||
|
// special attributes (e.g. inkscape:transform-center)
|
||||||
|
var onImport = options.onImport,
|
||||||
|
data = isElement && node.getAttribute('data-paper-data');
|
||||||
|
if (onImport)
|
||||||
|
item = onImport(node, item, options) || item;
|
||||||
|
if (options.expandShapes && item instanceof Shape) {
|
||||||
|
item.remove();
|
||||||
|
item = item.toPath();
|
||||||
|
}
|
||||||
|
if (data)
|
||||||
|
item._data = JSON.parse(data);
|
||||||
|
}
|
||||||
|
if (container) {
|
||||||
|
// After import, move things back to how they were:
|
||||||
|
body.removeChild(container);
|
||||||
|
if (parent) {
|
||||||
|
if (next) {
|
||||||
|
parent.insertBefore(node, next);
|
||||||
|
} else {
|
||||||
|
parent.appendChild(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Clear definitions at the end of import?
|
||||||
|
if (isRoot) {
|
||||||
|
definitions = {};
|
||||||
|
// Now if settings.applyMatrix was set, apply recursively and set
|
||||||
|
// #applyMatrix = true on the item and all children.
|
||||||
|
if (item && Base.pick(options.applyMatrix, applyMatrix))
|
||||||
|
item.matrix.apply(true, true);
|
||||||
|
}
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
|
||||||
function importSVG(source, options, isRoot) {
|
function importSVG(source, options, isRoot) {
|
||||||
if (!source)
|
if (!source)
|
||||||
return null;
|
return null;
|
||||||
|
@ -622,58 +674,13 @@ new function() {
|
||||||
return reader.readAsText(source);
|
return reader.readAsText(source);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof source === 'string') {
|
if (typeof source === 'string') {
|
||||||
node = new window.DOMParser().parseFromString(source,
|
node = new window.DOMParser().parseFromString(source,
|
||||||
'image/svg+xml');
|
'image/svg+xml');
|
||||||
}
|
}
|
||||||
if (!node.nodeName)
|
if (!node.nodeName)
|
||||||
throw new Error('Unsupported SVG source: ' + source);
|
throw new Error('Unsupported SVG source: ' + source);
|
||||||
// jsdom in Node.js uses uppercase values for nodeName...
|
return importNode(node, options, isRoot);
|
||||||
var type = node.nodeName.toLowerCase(),
|
|
||||||
isElement = type !== '#document',
|
|
||||||
importer = importers[type],
|
|
||||||
item,
|
|
||||||
data = isElement && node.getAttribute('data-paper-data'),
|
|
||||||
settings = scope.settings,
|
|
||||||
applyMatrix = settings.applyMatrix;
|
|
||||||
if (isRoot && isElement) {
|
|
||||||
// Set rootSize root element size, fall-back to view size.
|
|
||||||
rootSize = getSize(node, null, null, true)
|
|
||||||
|| scope.getView().getSize();
|
|
||||||
}
|
|
||||||
// Have items imported from SVG not bake in all transformations to their
|
|
||||||
// content and children, as this is how SVG works too, but preserve the
|
|
||||||
// current setting so we can restore it after.
|
|
||||||
settings.applyMatrix = false;
|
|
||||||
item = importer && importer(node, type, options, isRoot) || null;
|
|
||||||
settings.applyMatrix = applyMatrix;
|
|
||||||
if (item) {
|
|
||||||
// Do not apply attributes if this is a #document node.
|
|
||||||
// See importGroup() for an explanation of filtering for Group:
|
|
||||||
if (isElement && !(item instanceof Group))
|
|
||||||
item = applyAttributes(item, node, isRoot);
|
|
||||||
// Support onImportItem callback, to provide mechanism to handle
|
|
||||||
// special attributes (e.g. inkscape:transform-center)
|
|
||||||
var onImport = options.onImport;
|
|
||||||
if (onImport)
|
|
||||||
item = onImport(node, item, options) || item;
|
|
||||||
if (options.expandShapes && item instanceof Shape) {
|
|
||||||
item.remove();
|
|
||||||
item = item.toPath();
|
|
||||||
}
|
|
||||||
if (data)
|
|
||||||
item._data = JSON.parse(data);
|
|
||||||
}
|
|
||||||
// Clear definitions at the end of import?
|
|
||||||
if (isRoot) {
|
|
||||||
definitions = {};
|
|
||||||
// Now if settings.applyMatrix was set, apply recursively and set
|
|
||||||
// #applyMatrix = true on the item and all children.
|
|
||||||
if (item && Base.pick(options.applyMatrix, applyMatrix))
|
|
||||||
item.matrix.apply(true, true);
|
|
||||||
}
|
|
||||||
return item;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NOTE: Documentation is in Item#importSVG()
|
// NOTE: Documentation is in Item#importSVG()
|
||||||
|
|
|
@ -464,6 +464,8 @@ var createSVG = function(str, attrs) {
|
||||||
var node = document.createElementNS('http://www.w3.org/2000/svg', str);
|
var node = document.createElementNS('http://www.w3.org/2000/svg', str);
|
||||||
for (var key in attrs)
|
for (var key in attrs)
|
||||||
node.setAttribute(key, attrs[key]);
|
node.setAttribute(key, attrs[key]);
|
||||||
|
// Paper.js paths do not have a fill by default, SVG does.
|
||||||
|
node.setAttribute('fill', 'none');
|
||||||
return node;
|
return node;
|
||||||
} else {
|
} else {
|
||||||
return new window.DOMParser().parseFromString(
|
return new window.DOMParser().parseFromString(
|
||||||
|
|
|
@ -54,7 +54,7 @@ test('Export SVG ellipse', function() {
|
||||||
cy: 80,
|
cy: 80,
|
||||||
rx: 100,
|
rx: 100,
|
||||||
ry: 50
|
ry: 50
|
||||||
}
|
};
|
||||||
var path = new Path.Ellipse({
|
var path = new Path.Ellipse({
|
||||||
center: new Point(attrs.cx, attrs.cy),
|
center: new Point(attrs.cx, attrs.cy),
|
||||||
radius: new Point(attrs.rx, attrs.ry)
|
radius: new Point(attrs.rx, attrs.ry)
|
||||||
|
@ -67,7 +67,7 @@ test('Export SVG circle', function() {
|
||||||
cx: 100,
|
cx: 100,
|
||||||
cy: 80,
|
cy: 80,
|
||||||
r: 50
|
r: 50
|
||||||
}
|
};
|
||||||
var path = new Path.Circle({
|
var path = new Path.Circle({
|
||||||
center: new Point(attrs.cx, attrs.cy),
|
center: new Point(attrs.cx, attrs.cy),
|
||||||
radius: attrs.r
|
radius: attrs.r
|
||||||
|
|
Loading…
Reference in a new issue