mirror of
https://github.com/svg/svgo.git
synced 2025-08-01 18:46:52 +03:00
add removeOffCanvasPaths plugin (#979)
This commit is contained in:
@ -61,6 +61,7 @@ plugins:
|
||||
- removeStyleElement
|
||||
- removeScriptElement
|
||||
- addAttributesToSVGElement
|
||||
- removeOffCanvasPaths
|
||||
|
||||
# configure the indent (default 4 spaces) used by `--pretty` here:
|
||||
#
|
||||
|
133
plugins/removeOffCanvasPaths.js
Normal file
133
plugins/removeOffCanvasPaths.js
Normal file
@ -0,0 +1,133 @@
|
||||
'use strict';
|
||||
|
||||
exports.type = 'perItem';
|
||||
|
||||
exports.active = false;
|
||||
|
||||
exports.description = 'removes elements that are drawn outside of the viewbox (disabled by default)';
|
||||
|
||||
var SVGO = require('../lib/svgo.js'),
|
||||
_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.isElem('path') && item.hasAttr('d') && 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.attr('d').value))
|
||||
{
|
||||
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.isElem('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.hasAttr('transform') || (item.parentNode && 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.hasAttr('viewBox'))
|
||||
{
|
||||
// Remove commas and plus signs, normalize and trim whitespace
|
||||
viewBoxData = svg.attr('viewBox').value;
|
||||
}
|
||||
else if (svg.hasAttr('height') && svg.hasAttr('width'))
|
||||
{
|
||||
viewBoxData = '0 0 ' + svg.attr('width').value + ' ' + svg.attr('height').value;
|
||||
}
|
||||
|
||||
// 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 SVGO().createContentItem({
|
||||
elem: 'path',
|
||||
prefix: '',
|
||||
local: 'path'
|
||||
});
|
||||
path.addAttr({
|
||||
name: 'd',
|
||||
prefix: '',
|
||||
local: 'd',
|
||||
value: 'M' + m[1] + ' ' + m[2] + 'h' + m[3] + 'v' + m[4] + 'H' + m[1] + 'z'
|
||||
});
|
||||
|
||||
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;
|
||||
}
|
13
test/plugins/removeOffCanvasPaths.01.svg
Normal file
13
test/plugins/removeOffCanvasPaths.01.svg
Normal file
@ -0,0 +1,13 @@
|
||||
<svg viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M10 10 h 80 v 80 h -80 z"/>
|
||||
<path d="M10 -90 h 80 v 80 h -80 z"/>
|
||||
<path d="M110 10 h 80 v 80 h -80 z"/>
|
||||
<path d="M10 110 h 80 v 80 h -80 z"/>
|
||||
<path d="M-90 10 h 80 v 80 h -80 z"/>
|
||||
</svg>
|
||||
|
||||
@@@
|
||||
|
||||
<svg viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M10 10 h 80 v 80 h -80 z"/>
|
||||
</svg>
|
After Width: | Height: | Size: 396 B |
15
test/plugins/removeOffCanvasPaths.02.svg
Normal file
15
test/plugins/removeOffCanvasPaths.02.svg
Normal file
@ -0,0 +1,15 @@
|
||||
<svg height="1000" width="1000" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M10 10 h 80 v 80 h -80 z"/>
|
||||
<path d="M10 -90 h 80 v 80 h -80 z"/>
|
||||
<path d="M110 10 h 80 v 80 h -80 z"/>
|
||||
<path d="M10 110 h 80 v 80 h -80 z"/>
|
||||
<path d="M-90 10 h 80 v 80 h -80 z"/>
|
||||
</svg>
|
||||
|
||||
@@@
|
||||
|
||||
<svg height="1000" width="1000" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M10 10 h 80 v 80 h -80 z"/>
|
||||
<path d="M110 10 h 80 v 80 h -80 z"/>
|
||||
<path d="M10 110 h 80 v 80 h -80 z"/>
|
||||
</svg>
|
After Width: | Height: | Size: 490 B |
12
test/plugins/removeOffCanvasPaths.03.svg
Normal file
12
test/plugins/removeOffCanvasPaths.03.svg
Normal file
@ -0,0 +1,12 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 128 128">
|
||||
<path d="M0 0h128v128H0z" fill="none" stroke="red"/>
|
||||
<path d="M10.14 51.5c4.07 1.56 7.52 4.47 7.37 11.16" fill="none" stroke="#00f"/>
|
||||
<path d="M100 200c4.07 1.56 7.52 4.47 7.37 11.16" fill="none" stroke="#00f"/>
|
||||
</svg>
|
||||
|
||||
@@@
|
||||
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 128 128">
|
||||
<path d="M0 0h128v128H0z" fill="none" stroke="red"/>
|
||||
<path d="M10.14 51.5c4.07 1.56 7.52 4.47 7.37 11.16" fill="none" stroke="#00f"/>
|
||||
</svg>
|
After Width: | Height: | Size: 512 B |
9
test/plugins/removeOffCanvasPaths.04.svg
Normal file
9
test/plugins/removeOffCanvasPaths.04.svg
Normal file
@ -0,0 +1,9 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 128 128">
|
||||
<path d="M20.16 107.3l13.18-12.18m-1.6-5.41l-16.32 6.51M13 84.5h18m77 22.8L94.83 95.12m1.6-5.41l16.32 6.51M115 84.5H98" fill="none" stroke="#444" stroke-width="3"/>
|
||||
</svg>
|
||||
|
||||
@@@
|
||||
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 128 128">
|
||||
<path d="M20.16 107.3l13.18-12.18m-1.6-5.41l-16.32 6.51M13 84.5h18m77 22.8L94.83 95.12m1.6-5.41l16.32 6.51M115 84.5H98" fill="none" stroke="#444" stroke-width="3"/>
|
||||
</svg>
|
After Width: | Height: | Size: 484 B |
15
test/plugins/removeOffCanvasPaths.05.svg
Normal file
15
test/plugins/removeOffCanvasPaths.05.svg
Normal file
@ -0,0 +1,15 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100">
|
||||
<path d="M-100-100h50v50h-50z" fill="red" transform="translate(100 100)"/>
|
||||
<g transform="translate(150 150)">
|
||||
<path d="M-100-100h50v50h-50z" fill="blue"/>
|
||||
</g>
|
||||
</svg>
|
||||
|
||||
@@@
|
||||
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100">
|
||||
<path d="M-100-100h50v50h-50z" fill="red" transform="translate(100 100)"/>
|
||||
<g transform="translate(150 150)">
|
||||
<path d="M-100-100h50v50h-50z" fill="blue"/>
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 506 B |
Reference in New Issue
Block a user