mermaid/lib/phantomscript.js

263 lines
7.2 KiB
JavaScript
Raw Normal View History

2014-12-20 17:18:38 -08:00
/**
* 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')
2015-04-20 21:21:17 +02:00
2014-12-20 17:18:38 -08:00
var page = webpage.create()
2015-04-20 21:21:17 +02:00
, files = phantom.args.slice(7, phantom.args.length)
2014-12-20 17:18:38 -08:00
, options = {
outputDir: phantom.args[0]
, png: phantom.args[1] === 'true' ? true : false
, svg: phantom.args[2] === 'true' ? true : false
2015-02-02 19:02:45 -08:00
, css: phantom.args[3] !== '' ? phantom.args[3] : '* { margin: 0; padding: 0; }'
, sequenceConfig: phantom.args[4]
2015-04-20 21:21:17 +02:00
, ganttConfig: phantom.args[5]
, verbose: phantom.args[6] === 'true' ? true : false
2014-12-20 17:18:38 -08:00
}
, log = logger(options.verbose)
2015-06-21 17:26:19 +02:00
// If no css is suuplied make sure a fixed witdth is given to the gant renderer
if(phantom.args[3] !== ''){
options.ganttConfig.useWidth = 1200;
}
2015-04-20 21:21:17 +02:00
2015-06-21 17:26:19 +02:00
//console.log('options');
//console.log(options.css);
2015-04-20 21:21:17 +02:00
2014-12-20 17:18:38 -08:00
page.content = [
'<html>'
, '<head>'
, '<style type="text/css">'
2015-02-02 19:02:45 -08:00
, options.css
2014-12-20 17:18:38 -08:00
, '</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 + '")');
};
2014-12-20 17:18:38 -08:00
files.forEach(function(file) {
var contents = fs.read(file)
, filename = file.split(fs.separator).slice(-1)
, oParser = new DOMParser()
, oDOM
, svgContent
2015-06-21 17:26:19 +02:00
, allElements;
2015-05-30 13:00:04 +02:00
2014-12-20 17:18:38 -08:00
// 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, {
2015-04-20 21:21:17 +02:00
contents : contents,
ganttConfig : options.ganttConfig,
sequenceConfig : options.sequenceConfig
})
2014-12-20 17:18:38 -08:00
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) {
2015-06-21 17:26:19 +02:00
var serialize = new XMLSerializer();
2014-12-20 17:18:38 -08:00
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) {
2014-12-20 17:18:38 -08:00
var xmlSerializer = new XMLSerializer()
, contents = data.contents
, sequenceConfig = data.sequenceConfig
2015-04-20 21:21:17 +02:00
, ganttConfig = data.ganttConfig
2014-12-20 17:18:38 -08:00
, 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)
}
2015-04-20 21:21:17 +02:00
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{
2015-05-30 13:00:04 +02:00
console.log('NO gantt config');
2015-06-21 17:26:19 +02:00
sc = document.createElement("script")
scContent = document.createTextNode('mermaid.ganttConfig = {useWidth:1200};')
sc.appendChild(scContent)
2015-04-20 21:21:17 +02:00
2015-06-21 17:26:19 +02:00
document.body.appendChild(sc)
}
2015-05-30 13:00:04 +02:00
2015-06-21 17:26:19 +02:00
mermaid.initialize({
sequenceDiagram:{useMaxWidth:false}
});
mermaid.init();
2014-12-20 17:18:38 -08:00
svg = document.querySelector('svg')
svgValue = xmlSerializer.serializeToString(svg)
2015-06-21 17:26:19 +02:00
//console.log(document.body.outerHTML);
2014-12-20 17:18:38 -08:00
return svgValue
}