SVGImport: Improve consistency of style handling.

This commit is contained in:
Jürg Lehni 2016-02-11 11:06:09 +01:00
parent 8542eb62b4
commit df57c4adb9
4 changed files with 96 additions and 87 deletions

View file

@ -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, {

View file

@ -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()

View file

@ -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(

View file

@ -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