mirror of
https://github.com/weaveworks/scope.git
synced 2026-03-02 17:50:39 +00:00
145 lines
3.9 KiB
JavaScript
145 lines
3.9 KiB
JavaScript
// adapted from https://github.com/NYTimes/svg-crowbar
|
|
import _ from 'lodash';
|
|
|
|
const doctype = '<?xml version="1.0" standalone="no"?><!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">';
|
|
const prefix = {
|
|
xmlns: 'http://www.w3.org/2000/xmlns/',
|
|
xlink: 'http://www.w3.org/1999/xlink',
|
|
svg: 'http://www.w3.org/2000/svg'
|
|
};
|
|
const cssSkipValues = {
|
|
'auto': true,
|
|
'0px 0px': true,
|
|
'visible': true,
|
|
'pointer': true
|
|
};
|
|
|
|
function setInlineStyles(svg, target, emptySvgDeclarationComputed) {
|
|
function explicitlySetStyle(element, targetEl) {
|
|
const cSSStyleDeclarationComputed = getComputedStyle(element);
|
|
let value;
|
|
let computedStyleStr = '';
|
|
_.each(cSSStyleDeclarationComputed, key => {
|
|
value = cSSStyleDeclarationComputed.getPropertyValue(key);
|
|
if (value !== emptySvgDeclarationComputed.getPropertyValue(key) && !cssSkipValues[value]) {
|
|
computedStyleStr += key + ':' + value + ';';
|
|
}
|
|
});
|
|
targetEl.setAttribute('style', computedStyleStr);
|
|
}
|
|
|
|
function traverse(obj) {
|
|
const tree = [];
|
|
|
|
function visit(node) {
|
|
if (node && node.hasChildNodes()) {
|
|
let child = node.firstChild;
|
|
while (child) {
|
|
if (child.nodeType === 1 && child.nodeName !== 'SCRIPT') {
|
|
tree.push(child);
|
|
visit(child);
|
|
}
|
|
child = child.nextSibling;
|
|
}
|
|
}
|
|
}
|
|
|
|
tree.push(obj);
|
|
visit(obj);
|
|
return tree;
|
|
}
|
|
|
|
// make sure logo shows up
|
|
svg.setAttribute('class', 'exported');
|
|
|
|
// hardcode computed css styles inside svg
|
|
const allElements = traverse(svg);
|
|
const allTargetElements = traverse(target);
|
|
let i = allElements.length;
|
|
while (i--) {
|
|
explicitlySetStyle(allElements[i], allTargetElements[i]);
|
|
}
|
|
|
|
// set font
|
|
target.setAttribute('style', 'font-family: "Roboto", sans-serif;');
|
|
}
|
|
|
|
function download(source, name) {
|
|
let filename = 'untitled';
|
|
|
|
if (name) {
|
|
filename = name;
|
|
} else if (window.document.title) {
|
|
filename = window.document.title.replace(/[^a-z0-9]/gi, '-').toLowerCase()
|
|
+ '-' + (+new Date);
|
|
}
|
|
|
|
const url = window.URL.createObjectURL(new Blob(source,
|
|
{'type': 'text\/xml'}
|
|
));
|
|
|
|
const a = document.createElement('a');
|
|
document.body.appendChild(a);
|
|
a.setAttribute('class', 'svg-crowbar');
|
|
a.setAttribute('download', filename + '.svg');
|
|
a.setAttribute('href', url);
|
|
a.style.display = 'none';
|
|
a.click();
|
|
|
|
setTimeout(function() {
|
|
window.URL.revokeObjectURL(url);
|
|
}, 10);
|
|
}
|
|
|
|
function getSVG(doc, emptySvgDeclarationComputed) {
|
|
const svg = document.getElementById('nodes-chart-canvas');
|
|
const target = svg.cloneNode(true);
|
|
|
|
target.setAttribute('version', '1.1');
|
|
|
|
// removing attributes so they aren't doubled up
|
|
target.removeAttribute('xmlns');
|
|
target.removeAttribute('xlink');
|
|
|
|
// These are needed for the svg
|
|
if (!target.hasAttributeNS(prefix.xmlns, 'xmlns')) {
|
|
target.setAttributeNS(prefix.xmlns, 'xmlns', prefix.svg);
|
|
}
|
|
|
|
if (!target.hasAttributeNS(prefix.xmlns, 'xmlns:xlink')) {
|
|
target.setAttributeNS(prefix.xmlns, 'xmlns:xlink', prefix.xlink);
|
|
}
|
|
|
|
setInlineStyles(svg, target, emptySvgDeclarationComputed);
|
|
|
|
const source = (new XMLSerializer()).serializeToString(target);
|
|
|
|
return [doctype + source];
|
|
}
|
|
|
|
function cleanup() {
|
|
const crowbarElements = document.querySelectorAll('.svg-crowbar');
|
|
|
|
[].forEach.call(crowbarElements, function(el) {
|
|
el.parentNode.removeChild(el);
|
|
});
|
|
|
|
// hide embedded logo
|
|
const svg = document.getElementById('nodes-chart-canvas');
|
|
svg.setAttribute('class', '');
|
|
}
|
|
|
|
export function saveGraph(filename) {
|
|
window.URL = (window.URL || window.webkitURL);
|
|
|
|
// add empty svg element
|
|
const emptySvg = window.document.createElementNS(prefix.svg, 'svg');
|
|
window.document.body.appendChild(emptySvg);
|
|
const emptySvgDeclarationComputed = getComputedStyle(emptySvg);
|
|
|
|
const svgSource = getSVG(document, emptySvgDeclarationComputed);
|
|
download(svgSource, filename);
|
|
|
|
cleanup();
|
|
}
|