mirror of
https://github.com/svg/svgo.git
synced 2025-04-19 10:22:15 +03:00
I saw complaints about `extendDefaultPlugins` api - it cannot be used when svgo is installed globally - it requires svgo to be installed when using svgo-loader or svgo-jsx - it prevents using serializable config formats like json In this diff I introduced the new plugin which is a bundle of all default plugins. ```js module.exports = { plugins: [ 'preset_default', // or { name: 'preset_default', floatPrecision: 4, overrides: { convertPathData: { applyTransforms: false } } } ] } ```
142 lines
3.4 KiB
JavaScript
142 lines
3.4 KiB
JavaScript
'use strict';
|
|
|
|
exports.name = 'removeOffCanvasPaths';
|
|
|
|
exports.type = 'perItem';
|
|
|
|
exports.active = false;
|
|
|
|
exports.description =
|
|
'removes elements that are drawn outside of the viewbox (disabled by default)';
|
|
|
|
const JSAPI = require('../lib/svgo/jsAPI.js');
|
|
|
|
var _path = require('./_path.js'),
|
|
intersects = _path.intersects,
|
|
path2js = _path.path2js,
|
|
viewBox,
|
|
viewBoxJS;
|
|
|
|
/**
|
|
* Remove elements that are drawn outside of the viewbox.
|
|
*
|
|
* @param {Object} item current iteration item
|
|
* @return {Boolean} if false, item will be filtered out
|
|
*
|
|
* @author JoshyPHP
|
|
*/
|
|
exports.fn = function (item) {
|
|
if (
|
|
item.type === 'element' &&
|
|
item.name === 'path' &&
|
|
item.attributes.d != null &&
|
|
typeof viewBox !== 'undefined'
|
|
) {
|
|
// Consider that any item with a transform attribute or a M instruction
|
|
// within the viewBox is visible
|
|
if (hasTransform(item) || pathMovesWithinViewBox(item.attributes.d)) {
|
|
return true;
|
|
}
|
|
|
|
var pathJS = path2js(item);
|
|
if (pathJS.length === 2) {
|
|
// Use a closed clone of the path if it's too short for intersects()
|
|
pathJS = JSON.parse(JSON.stringify(pathJS));
|
|
pathJS.push({ instruction: 'z' });
|
|
}
|
|
|
|
return intersects(viewBoxJS, pathJS);
|
|
}
|
|
if (item.type === 'element' && item.name === 'svg') {
|
|
parseViewBox(item);
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Test whether given item or any of its ancestors has a transform attribute.
|
|
*
|
|
* @param {String} path
|
|
* @return {Boolean}
|
|
*/
|
|
function hasTransform(item) {
|
|
return (
|
|
item.attributes.transform != null ||
|
|
(item.parentNode &&
|
|
item.parentNode.type === 'element' &&
|
|
hasTransform(item.parentNode))
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Parse the viewBox coordinates and compute the JS representation of its path.
|
|
*
|
|
* @param {Object} svg svg element item
|
|
*/
|
|
function parseViewBox(svg) {
|
|
var viewBoxData = '';
|
|
if (svg.attributes.viewBox != null) {
|
|
// Remove commas and plus signs, normalize and trim whitespace
|
|
viewBoxData = svg.attributes.viewBox;
|
|
} else if (svg.attributes.height != null && svg.attributes.width != null) {
|
|
viewBoxData = `0 0 ${svg.attributes.width} ${svg.attributes.height}`;
|
|
}
|
|
|
|
// Remove commas and plus signs, normalize and trim whitespace
|
|
viewBoxData = viewBoxData
|
|
.replace(/[,+]|px/g, ' ')
|
|
.replace(/\s+/g, ' ')
|
|
.replace(/^\s*|\s*$/g, '');
|
|
|
|
// Ensure that the dimensions are 4 values separated by space
|
|
var m = /^(-?\d*\.?\d+) (-?\d*\.?\d+) (\d*\.?\d+) (\d*\.?\d+)$/.exec(
|
|
viewBoxData
|
|
);
|
|
if (!m) {
|
|
return;
|
|
}
|
|
|
|
// Store the viewBox boundaries
|
|
viewBox = {
|
|
left: parseFloat(m[1]),
|
|
top: parseFloat(m[2]),
|
|
right: parseFloat(m[1]) + parseFloat(m[3]),
|
|
bottom: parseFloat(m[2]) + parseFloat(m[4]),
|
|
};
|
|
|
|
var path = new JSAPI({
|
|
type: 'element',
|
|
name: 'path',
|
|
attributes: {
|
|
d: 'M' + m[1] + ' ' + m[2] + 'h' + m[3] + 'v' + m[4] + 'H' + m[1] + 'z',
|
|
},
|
|
content: [],
|
|
});
|
|
|
|
viewBoxJS = path2js(path);
|
|
}
|
|
|
|
/**
|
|
* Test whether given path has a M instruction with coordinates within the viewBox.
|
|
*
|
|
* @param {String} path
|
|
* @return {Boolean}
|
|
*/
|
|
function pathMovesWithinViewBox(path) {
|
|
var regexp = /M\s*(-?\d*\.?\d+)(?!\d)\s*(-?\d*\.?\d+)/g,
|
|
m;
|
|
while (null !== (m = regexp.exec(path))) {
|
|
if (
|
|
m[1] >= viewBox.left &&
|
|
m[1] <= viewBox.right &&
|
|
m[2] >= viewBox.top &&
|
|
m[2] <= viewBox.bottom
|
|
) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|