mirror of
https://github.com/mermaid-js/mermaid.git
synced 2025-01-14 06:43:25 +08:00
253 lines
6.8 KiB
JavaScript
253 lines
6.8 KiB
JavaScript
/**
|
|
* Credits:
|
|
* - SVG Processing from the NYTimes svg-crowbar, under an MIT license
|
|
* https://github.com/NYTimes/svg-crowbar
|
|
* - Thanks to the grunticon project for some guidance
|
|
* https://github.com/filamentgroup/grunticon
|
|
*/
|
|
|
|
phantom.onError = function(msg, trace) {
|
|
var msgStack = ['PHANTOM ERROR: ' + msg]
|
|
if (trace && trace.length) {
|
|
msgStack.push('TRACE:')
|
|
trace.forEach(function(t) {
|
|
msgStack.push(
|
|
' -> '
|
|
+ (t.file || t.sourceURL)
|
|
+ ': '
|
|
+ t.line
|
|
+ (t.function ? ' (in function ' + t.function +')' : '')
|
|
)
|
|
})
|
|
}
|
|
system.stderr.write(msgStack.join('\n'))
|
|
phantom.exit(1)
|
|
}
|
|
|
|
var system = require('system')
|
|
, fs = require('fs')
|
|
, webpage = require('webpage')
|
|
|
|
|
|
var page = webpage.create()
|
|
, files = phantom.args.slice(7, phantom.args.length)
|
|
, options = {
|
|
outputDir: phantom.args[0]
|
|
, png: phantom.args[1] === 'true' ? true : false
|
|
, svg: phantom.args[2] === 'true' ? true : false
|
|
, css: phantom.args[3] !== '' ? phantom.args[3] : '* { margin: 0; padding: 0; }'
|
|
, sequenceConfig: phantom.args[4]
|
|
, ganttConfig: phantom.args[5]
|
|
, verbose: phantom.args[6] === 'true' ? true : false
|
|
}
|
|
, log = logger(options.verbose)
|
|
|
|
|
|
console.log('options');
|
|
console.log(phantom.args.length);
|
|
|
|
page.content = [
|
|
'<html>'
|
|
, '<head>'
|
|
, '<style type="text/css">'
|
|
, options.css
|
|
, '</style>'
|
|
, '</head>'
|
|
, '<body>'
|
|
, '</body>'
|
|
, '</html>'
|
|
].join('\n')
|
|
|
|
page.injectJs('../dist/mermaid.js')
|
|
page.onConsoleMessage = function(msg, lineNum, sourceId) {
|
|
console.log('CONSOLE: ' + msg + ' (from line #' + lineNum + ' in "' + sourceId + '")');
|
|
};
|
|
files.forEach(function(file) {
|
|
var contents = fs.read(file)
|
|
, filename = file.split(fs.separator).slice(-1)
|
|
, oParser = new DOMParser()
|
|
, oDOM
|
|
, svgContent
|
|
, allElements
|
|
|
|
console.log('Gantconfig: '+options.ganttConfig);
|
|
|
|
// this JS is executed in this statement is sandboxed, even though it doesn't
|
|
// look like it. we need to serialize then unserialize the svgContent that's
|
|
// taken from the DOM
|
|
svgContent = page.evaluate(executeInPage, {
|
|
contents : contents,
|
|
ganttConfig : options.ganttConfig,
|
|
sequenceConfig : options.sequenceConfig
|
|
})
|
|
oDOM = oParser.parseFromString(svgContent, "text/xml")
|
|
|
|
resolveSVGElement(oDOM.firstChild)
|
|
|
|
// traverse the SVG, and replace all foreignObject elements
|
|
// can be removed when https://github.com/knsv/mermaid/issues/58 is resolved
|
|
allElements = traverse(oDOM)
|
|
for (var i = 0, len = allElements.length; i < len; i++) {
|
|
resolveForeignObjects(allElements[i])
|
|
}
|
|
|
|
if (options.png) {
|
|
page.viewportSize = {
|
|
width: ~~oDOM.documentElement.attributes.getNamedItem('width').value
|
|
, height: ~~oDOM.documentElement.attributes.getNamedItem('height').value
|
|
}
|
|
|
|
page.render(options.outputDir + fs.separator + filename + '.png')
|
|
log('saved png: ' + filename + '.png')
|
|
}
|
|
|
|
if (options.svg) {
|
|
var serialize = new XMLSerializer()
|
|
fs.write(
|
|
options.outputDir + fs.separator + filename + '.svg'
|
|
, serialize.serializeToString(oDOM)
|
|
, 'w'
|
|
)
|
|
log('saved svg: ' + filename + '.svg')
|
|
}
|
|
})
|
|
|
|
phantom.exit()
|
|
|
|
function logger(_verbose) {
|
|
var verbose = _verbose
|
|
|
|
return function(_message, _level) {
|
|
var level = level
|
|
, message = _message
|
|
, log
|
|
|
|
log = level === 'error' ? system.stderr : system.stdout
|
|
|
|
if (verbose) {
|
|
log.write(message + '\n')
|
|
}
|
|
}
|
|
}
|
|
|
|
function traverse(obj){
|
|
var tree = []
|
|
|
|
tree.push(obj)
|
|
visit(obj)
|
|
|
|
function visit(node) {
|
|
if (node && node.hasChildNodes()) {
|
|
var child = node.firstChild
|
|
while (child) {
|
|
if (child.nodeType === 1 && child.nodeName != 'SCRIPT'){
|
|
tree.push(child)
|
|
visit(child)
|
|
}
|
|
child = child.nextSibling
|
|
}
|
|
}
|
|
}
|
|
|
|
return tree
|
|
}
|
|
|
|
function resolveSVGElement(element) {
|
|
var prefix = {
|
|
xmlns: "http://www.w3.org/2000/xmlns/"
|
|
, xlink: "http://www.w3.org/1999/xlink"
|
|
, svg: "http://www.w3.org/2000/svg"
|
|
}
|
|
, doctype = '<!DOCTYPE svg:svg PUBLIC'
|
|
+ ' "-//W3C//DTD XHTML 1.1 plus MathML 2.0 plus SVG 1.1//EN"'
|
|
+ ' "http://www.w3.org/2002/04/xhtml-math-svg/xhtml-math-svg.dtd">'
|
|
|
|
element.setAttribute("version", "1.1")
|
|
// removing attributes so they aren't doubled up
|
|
element.removeAttribute("xmlns")
|
|
element.removeAttribute("xlink")
|
|
// These are needed for the svg
|
|
if (!element.hasAttributeNS(prefix.xmlns, "xmlns")) {
|
|
element.setAttributeNS(prefix.xmlns, "xmlns", prefix.svg)
|
|
}
|
|
if (!element.hasAttributeNS(prefix.xmlns, "xmlns:xlink")) {
|
|
element.setAttributeNS(prefix.xmlns, "xmlns:xlink", prefix.xlink)
|
|
}
|
|
}
|
|
|
|
function resolveForeignObjects(element) {
|
|
var children
|
|
, textElement
|
|
, textSpan
|
|
|
|
if (element.tagName === 'foreignObject') {
|
|
textElement = document.createElement('text')
|
|
textSpan = document.createElement('tspan')
|
|
textSpan.setAttribute(
|
|
'style'
|
|
, 'font-size: 11.5pt; font-family: "sans-serif";'
|
|
)
|
|
textSpan.setAttribute('x', 0)
|
|
textSpan.setAttribute('y', 14.5)
|
|
textSpan.textContent = element.textContent
|
|
|
|
textElement.appendChild(textSpan)
|
|
element.parentElement.appendChild(textElement)
|
|
element.parentElement.removeChild(element)
|
|
}
|
|
}
|
|
|
|
// The sandboxed function that's executed in-page by phantom
|
|
function executeInPage(data) {
|
|
var xmlSerializer = new XMLSerializer()
|
|
, contents = data.contents
|
|
, sequenceConfig = data.sequenceConfig
|
|
, ganttConfig = data.ganttConfig
|
|
, toRemove
|
|
, el
|
|
, elContent
|
|
, svg
|
|
, svgValue
|
|
|
|
toRemove = document.getElementsByClassName('mermaid')
|
|
if (toRemove && toRemove.length) {
|
|
for (var i = 0, len = toRemove.length; i < len; i++) {
|
|
toRemove[i].parentNode.removeChild(toRemove[i])
|
|
}
|
|
}
|
|
|
|
el = document.createElement("div")
|
|
el.className = 'mermaid'
|
|
elContent = document.createTextNode(contents)
|
|
el.appendChild(elContent)
|
|
|
|
document.body.appendChild(el)
|
|
|
|
if(typeof sequenceConfig !== undefined && sequenceConfig !== 'undefined'){
|
|
sc = document.createElement("script")
|
|
scContent = document.createTextNode('mermaid.sequenceConfig = JSON.parse(' + JSON.stringify(sequenceConfig) + ');')
|
|
sc.appendChild(scContent)
|
|
|
|
document.body.appendChild(sc)
|
|
}
|
|
|
|
if(typeof ganttConfig !== undefined && ganttConfig !== 'undefined'){
|
|
sc = document.createElement("script")
|
|
scContent = document.createTextNode('mermaid.ganttConfig = JSON.parse(' + JSON.stringify(ganttConfig) + ');')
|
|
sc.appendChild(scContent)
|
|
|
|
document.body.appendChild(sc)
|
|
}else{
|
|
console.log('NO gantt config');
|
|
}
|
|
|
|
|
|
mermaid.init();
|
|
console.log(document.body.outerHTML);
|
|
|
|
svg = document.querySelector('svg')
|
|
svgValue = xmlSerializer.serializeToString(svg)
|
|
|
|
return svgValue
|
|
}
|