drawio-offline
Version:
diagrams.net desktop
874 lines (781 loc) • 239 kB
JavaScript
/**
* Copyright (c) 2006-2017, JGraph Ltd
* Copyright (c) 2006-2017, Gaudenz Alder
*/
(function()
{
if (typeof html4 !== 'undefined')
{
/**
* Enables paste from Lucidchart
*/
html4.ATTRIBS['span::data-lucid-content'] = 0;
html4.ATTRIBS['span::data-lucid-type'] = 0;
/**
* Enables custom fonts in labels.
*/
html4.ATTRIBS['font::data-font-src'] = 0;
}
/**
* Specifies the app name. Default is document.title.
*/
Editor.prototype.appName = 'diagrams.net';
/**
* Known file types.
*/
Editor.prototype.diagramFileTypes = [
{description: 'diagramXmlDesc', extension: 'drawio', mimeType: 'text/xml'},
{description: 'diagramPngDesc', extension: 'png', mimeType: 'image/png'},
{description: 'diagramSvgDesc', extension: 'svg', mimeType: 'image/svg'},
{description: 'diagramHtmlDesc', extension: 'html', mimeType: 'text/html'},
{description: 'diagramXmlDesc', extension: 'xml', mimeType: 'text/xml'}];
/**
* Known file types.
*/
Editor.prototype.libraryFileTypes = [{description: 'Library (.drawiolib, .xml)', extensions: ['drawiolib', 'xml']}];
/**
* Additional help text for special file extensions.
*/
Editor.prototype.fileExtensions = [
{ext: 'html', title: 'filetypeHtml'},
{ext: 'png', title: 'filetypePng'},
{ext: 'svg', title: 'filetypeSvg'}];
/**
*
*/
Editor.styles = [{},
{commonStyle: {fontColor: '#5C5C5C', strokeColor: '#006658', fillColor: '#21C0A5'}},
{commonStyle: {fontColor: '#095C86', strokeColor: '#AF45ED', fillColor: '#F694C1'},
edgeStyle: {strokeColor: '#60E696'}},
{commonStyle: {fontColor: '#46495D', strokeColor: '#788AA3', fillColor: '#B2C9AB'}},
{commonStyle: {fontColor: '#5AA9E6', strokeColor: '#FF6392', fillColor: '#FFE45E'}},
{commonStyle: {fontColor: '#1D3557', strokeColor: '#457B9D', fillColor: '#A8DADC'},
graph: {background: '#F1FAEE'}},
{commonStyle: {fontColor: '#393C56', strokeColor: '#E07A5F', fillColor: '#F2CC8F'},
graph: {background: '#F4F1DE', gridColor: '#D4D0C0'}},
{commonStyle: {fontColor: '#143642', strokeColor: '#0F8B8D', fillColor: '#FAE5C7'},
edgeStyle: {strokeColor: '#A8201A'},
graph: {background: '#DAD2D8', gridColor: '#ABA4A9'}},
{commonStyle: {fontColor: '#FEFAE0', strokeColor: '#DDA15E', fillColor: '#BC6C25'},
graph: {background: '#283618', gridColor: '#48632C'}},
{commonStyle: {fontColor: '#E4FDE1', strokeColor: '#028090', fillColor: '#F45B69'},
graph: {background: '#114B5F', gridColor: '#0B3240'}},
{},
{vertexStyle: {strokeColor: '#D0CEE2', fillColor: '#FAD9D5'},
edgeStyle: {strokeColor: '#09555B'},
commonStyle: {fontColor: '#1A1A1A'}},
{vertexStyle: {strokeColor: '#BAC8D3', fillColor: '#09555B', fontColor: '#EEEEEE'},
edgeStyle: {strokeColor: '#0B4D6A'}},
{vertexStyle: {strokeColor: '#D0CEE2', fillColor: '#5D7F99'},
edgeStyle: {strokeColor: '#736CA8'},
commonStyle: {fontColor: '#1A1A1A'}},
{vertexStyle: {strokeColor: '#FFFFFF', fillColor: '#182E3E', fontColor: '#FFFFFF'},
edgeStyle: {strokeColor: '#23445D'},
graph: {background: '#FCE7CD', gridColor: '#CFBDA8'}},
{vertexStyle: {strokeColor: '#FFFFFF', fillColor: '#F08E81'},
edgeStyle: {strokeColor: '#182E3E'},
commonStyle: {fontColor: '#1A1A1A'},
graph: {background: '#B0E3E6', gridColor: '#87AEB0'}},
{vertexStyle: {strokeColor: '#909090', fillColor: '#F5AB50'},
edgeStyle: {strokeColor: '#182E3E'},
commonStyle: {fontColor: '#1A1A1A'},
graph: {background: '#EEEEEE'}},
{vertexStyle: {strokeColor: '#EEEEEE', fillColor: '#56517E', fontColor: '#FFFFFF'},
edgeStyle: {strokeColor: '#182E3E'},
graph: {background: '#FAD9D5', gridColor: '#BFA6A3'}},
{vertexStyle: {strokeColor: '#BAC8D3', fillColor: '#B1DDF0', fontColor: '#182E3E'},
edgeStyle: {strokeColor: '#EEEEEE', fontColor: '#FFFFFF'},
graph: {background: '#09555B', gridColor: '#13B4C2'}},
{vertexStyle: {fillColor: '#EEEEEE', fontColor: '#1A1A1A'},
edgeStyle: {fontColor: '#FFFFFF'},
commonStyle: {strokeColor: '#FFFFFF'},
graph: {background: '#182E3E', gridColor: '#4D94C7'}}
];
/**
*
*/
Editor.saveImage = '';
/**
* Used in the GraphViewer lightbox.
*/
Editor.closeImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/delete.png' : '';
/**
*
*/
Editor.plusImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/plus.png' : '';
/**
*
*/
Editor.spinImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/spin.gif' : '';
/**
*
*/
Editor.globeImage = '';
/**
*
*/
Editor.commentImage = '';
/**
*
*/
Editor.commentImageInverted = '';
/**
*
*/
Editor.userImage = '';
/**
*
*/
Editor.shareImage = '';
/**
*
*/
Editor.syncImage = '';
/**
*
*/
Editor.syncDisabledImage = '';
/**
*
*/
Editor.syncProblemImage = '';
/**
* Used in the GraphViewer lightbox.
*/
Editor.tweetImage = IMAGE_PATH + '/tweet.png';
/**
* Used in the GraphViewer lightbox.
*/
Editor.facebookImage = IMAGE_PATH + '/facebook.png';
/**
* Blank 1x1 pixel transparent PNG image.
*/
Editor.blankImage = '';
/**
* Blank 1x1 pixel transparent PNG image.
*/
Editor.hiResImage = (mxClient.IS_SVG) ? '' : IMAGE_PATH + '/img-hi-res.png';
/**
* Blank 1x1 pixel transparent PNG image.
*/
Editor.loResImage = (mxClient.IS_SVG) ? '' : IMAGE_PATH + '/img-lo-res.png';
/**
* Only needed in browsers with SVG support for export via lightbox toolbar.
*/
Editor.cameraLargeImage = '';
/**
* Broken image symbol for offline SVG.
*/
Editor.svgBrokenImage = Graph.createSvgImage(10, 10, '<rect x="0" y="0" width="10" height="10" stroke="#000" fill="transparent"/><path d="m 0 0 L 10 10 L 0 10 L 10 0" stroke="#000" fill="transparent"/>');
/**
* Error image for not found images
*/
Editor.errorImage = '';
/**
* Error image for not found images
*/
Editor.configurationKey = '.configuration';
/**
* Error image for not found images
*/
Editor.settingsKey = '.drawio-config';
/**
* Default value for custom libraries in mxSettings.
*/
Editor.defaultCustomLibraries = [];
/**
* Default value for custom libraries in mxSettings.
*/
Editor.enableCustomLibraries = true;
/**
* Specifies if custom properties should be enabled.
*/
Editor.enableCustomProperties = true;
/**
* Specifies if custom properties should be enabled.
*/
Editor.enableServiceWorker = urlParams['pwa'] != '0' &&
'serviceWorker' in navigator && (urlParams['offline'] == '1' ||
/.*\.diagrams\.net$/.test(window.location.hostname) ||
/.*\.draw\.io$/.test(window.location.hostname));
/**
* Specifies if XML files should be compressed. Default is true.
*/
Editor.compressXml = true;
/**
* Specifies global variables.
*/
Editor.globalVars = null;
/**
* Disables the shadow option in the format panel.
*/
Editor.shadowOptionEnabled = !mxClient.IS_SF;
/**
* Reference to the config object passed to <configure>.
*/
Editor.config = null;
/**
* Reference to the version of the last config object in
* <configure>. If this is different to the last version in
* mxSettings.parse, then the settings are reset.
*/
Editor.configVersion = null;
/**
* Default border for image export (to allow for sketch style).
*/
Editor.defaultBorder = 5;
/**
* Common properties for all edges.
*/
Editor.commonProperties = [
{name: 'comic', dispName: 'Comic', type: 'bool', defVal: false, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', '0') != '1';
}},
{name: 'jiggle', dispName: 'Jiggle', type: 'float', min: 0, defVal: 1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'comic', '0') == '1' ||
mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'fillWeight', dispName: 'Fill Weight', type: 'int', defVal: -1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'hachureGap', dispName: 'Hachure Gap', type: 'int', defVal: -1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'hachureAngle', dispName: 'Hachure Angle', type: 'int', defVal: -41, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'curveFitting', dispName: 'Curve Fitting', type: 'float', defVal: 0.95, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'simplification', dispName: 'Simplification', type: 'float', defVal: 0, min: 0, max: 1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'disableMultiStroke', dispName: 'Disable Multi Stroke', type: 'bool', defVal: false, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'disableMultiStrokeFill', dispName: 'Disable Multi Stroke Fill', type: 'bool', defVal: false, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'dashOffset', dispName: 'Dash Offset', type: 'int', defVal: -1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'dashGap', dispName: 'Dash Gap', type: 'int', defVal: -1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'zigzagOffset', dispName: 'ZigZag Offset', type: 'int', defVal: -1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'jiggle', dispName: 'Jiggle', type: 'float', min: 0, defVal: 1, isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'comic', '0') == '1' ||
mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}},
{name: 'sketchStyle', dispName: 'Sketch Style', type: 'enum', defVal: 'rough',
enumList: [{val: 'rough', dispName: 'Rough'}, {val: 'comic', dispName: 'Comic'}],
isVisible: function(state, format)
{
return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
}}
];
/**
* Common properties for all edges.
*/
Editor.commonEdgeProperties = [
{type: 'separator'},
{name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
{name: 'sourcePortConstraint', dispName: 'Source Constraint', type: 'enum', defVal: 'none',
enumList: [{val: 'none', dispName: 'None'}, {val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
},
{name: 'targetPortConstraint', dispName: 'Target Constraint', type: 'enum', defVal: 'none',
enumList: [{val: 'none', dispName: 'None'}, {val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
},
{name: 'jettySize', dispName: 'Jetty Size', type: 'int', min: 0, defVal: 'auto', allowAuto: true, isVisible: function(state)
{
return mxUtils.getValue(state.style, mxConstants.STYLE_EDGE, null) == 'orthogonalEdgeStyle';
}},
{name: 'fillOpacity', dispName: 'Fill Opacity', type: 'int', min: 0, max: 100, defVal: 100},
{name: 'strokeOpacity', dispName: 'Stroke Opacity', type: 'int', min: 0, max: 100, defVal: 100},
{name: 'startFill', dispName: 'Start Fill', type: 'bool', defVal: true},
{name: 'endFill', dispName: 'End Fill', type: 'bool', defVal: true},
{name: 'perimeterSpacing', dispName: 'Terminal Spacing', type: 'float', defVal: 0},
{name: 'anchorPointDirection', dispName: 'Anchor Direction', type: 'bool', defVal: true},
{name: 'snapToPoint', dispName: 'Snap to Point', type: 'bool', defVal: false},
{name: 'fixDash', dispName: 'Fixed Dash', type: 'bool', defVal: false},
{name: 'editable', dispName: 'Editable', type: 'bool', defVal: true},
{name: 'metaEdit', dispName: 'Edit Dialog', type: 'bool', defVal: false},
{name: 'backgroundOutline', dispName: 'Background Outline', type: 'bool', defVal: false},
{name: 'bendable', dispName: 'Bendable', type: 'bool', defVal: true},
{name: 'movable', dispName: 'Movable', type: 'bool', defVal: true},
{name: 'cloneable', dispName: 'Cloneable', type: 'bool', defVal: true},
{name: 'deletable', dispName: 'Deletable', type: 'bool', defVal: true},
{name: 'noJump', dispName: 'No Jumps', type: 'bool', defVal: false},
{name: 'flowAnimation', dispName: 'Flow Animation', type: 'bool', defVal: false},
{name: 'ignoreEdge', dispName: 'Ignore Edge', type: 'bool', defVal: false},
{name: 'orthogonalLoop', dispName: 'Loop Routing', type: 'bool', defVal: false},
{name: 'orthogonal', dispName: 'Orthogonal', type: 'bool', defVal: false}
].concat(Editor.commonProperties);
/**
* Common properties for all vertices.
*/
Editor.commonVertexProperties = [
{type: 'separator'},
{name: 'resizeLastRow', dispName: 'Resize Last Row', type: 'bool', getDefaultValue: function(state, format)
{
var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
var graph = format.editorUi.editor.graph;
var style = graph.getCellStyle(cell);
return mxUtils.getValue(style, 'resizeLastRow', '0') == '1';
}, isVisible: function(state, format)
{
var graph = format.editorUi.editor.graph;
return state.vertices.length == 1 && state.edges.length == 0 &&
graph.isTable(state.vertices[0]);
}},
{name: 'resizeLast', dispName: 'Resize Last Column', type: 'bool', getDefaultValue: function(state, format)
{
var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
var graph = format.editorUi.editor.graph;
var style = graph.getCellStyle(cell);
return mxUtils.getValue(style, 'resizeLast', '0') == '1';
}, isVisible: function(state, format)
{
var graph = format.editorUi.editor.graph;
return state.vertices.length == 1 && state.edges.length == 0 &&
graph.isTable(state.vertices[0]);
}},
{name: 'fillOpacity', dispName: 'Fill Opacity', type: 'int', min: 0, max: 100, defVal: 100},
{name: 'strokeOpacity', dispName: 'Stroke Opacity', type: 'int', min: 0, max: 100, defVal: 100},
{name: 'overflow', dispName: 'Text Overflow', defVal: 'visible', type: 'enum',
enumList: [{val: 'visible', dispName: 'Visible'}, {val: 'hidden', dispName: 'Hidden'}, {val: 'block', dispName: 'Block'},
{val: 'fill', dispName: 'Fill'}, {val: 'width', dispName: 'Width'}]
},
{name: 'noLabel', dispName: 'Hide Label', type: 'bool', defVal: false},
{name: 'labelPadding', dispName: 'Label Padding', type: 'float', defVal: 0},
{name: 'direction', dispName: 'Direction', type: 'enum', defVal: 'east',
enumList: [{val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
},
{name: 'portConstraint', dispName: 'Constraint', type: 'enum', defVal: 'none',
enumList: [{val: 'none', dispName: 'None'}, {val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
},
{name: 'portConstraintRotation', dispName: 'Rotate Constraint', type: 'bool', defVal: false},
{name: 'connectable', dispName: 'Connectable', type: 'bool', getDefaultValue: function(state, format)
{
var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
var graph = format.editorUi.editor.graph;
return graph.isCellConnectable(cell);
}, isVisible: function(state, format)
{
return state.vertices.length == 1 && state.edges.length == 0;
}},
{name: 'allowArrows', dispName: 'Allow Arrows', type: 'bool', defVal: true},
{name: 'snapToPoint', dispName: 'Snap to Point', type: 'bool', defVal: false},
{name: 'perimeter', dispName: 'Perimeter', defVal: 'none', type: 'enum',
enumList: [{val: 'none', dispName: 'None'},
{val: 'rectanglePerimeter', dispName: 'Rectangle'}, {val: 'ellipsePerimeter', dispName: 'Ellipse'},
{val: 'rhombusPerimeter', dispName: 'Rhombus'}, {val: 'trianglePerimeter', dispName: 'Triangle'},
{val: 'hexagonPerimeter2', dispName: 'Hexagon'}, {val: 'lifelinePerimeter', dispName: 'Lifeline'},
{val: 'orthogonalPerimeter', dispName: 'Orthogonal'}, {val: 'backbonePerimeter', dispName: 'Backbone'},
{val: 'calloutPerimeter', dispName: 'Callout'}, {val: 'parallelogramPerimeter', dispName: 'Parallelogram'},
{val: 'trapezoidPerimeter', dispName: 'Trapezoid'}, {val: 'stepPerimeter', dispName: 'Step'},
{val: 'centerPerimeter', dispName: 'Center'}]
},
{name: 'fixDash', dispName: 'Fixed Dash', type: 'bool', defVal: false},
{name: 'autosize', dispName: 'Autosize', type: 'bool', defVal: false},
{name: 'container', dispName: 'Container', type: 'bool', defVal: false, isVisible: function(state, format)
{
return state.vertices.length == 1 && state.edges.length == 0;
}},
{name: 'dropTarget', dispName: 'Drop Target', type: 'bool', getDefaultValue: function(state, format)
{
var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
var graph = format.editorUi.editor.graph;
return cell != null && (graph.isSwimlane(cell) || graph.model.getChildCount(cell) > 0);
}, isVisible: function(state, format)
{
return state.vertices.length == 1 && state.edges.length == 0;
}},
{name: 'collapsible', dispName: 'Collapsible', type: 'bool', getDefaultValue: function(state, format)
{
var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
var graph = format.editorUi.editor.graph;
return cell != null && ((graph.isContainer(cell) && state.style['collapsible'] != '0') ||
(!graph.isContainer(cell) && state.style['collapsible'] == '1'));
}, isVisible: function(state, format)
{
return state.vertices.length == 1 && state.edges.length == 0;
}},
{name: 'recursiveResize', dispName: 'Resize Children', type: 'bool', defVal: true, isVisible: function(state, format)
{
return state.vertices.length == 1 && state.edges.length == 0 &&
!format.editorUi.editor.graph.isSwimlane(state.vertices[0]) &&
mxUtils.getValue(state.style, 'childLayout', null) == null;
}},
{name: 'expand', dispName: 'Expand', type: 'bool', defVal: true},
{name: 'part', dispName: 'Part', type: 'bool', defVal: false, isVisible: function(state, format)
{
var model = format.editorUi.editor.graph.model;
return (state.vertices.length > 0) ? model.isVertex(model.getParent(state.vertices[0])) : false;
}},
{name: 'editable', dispName: 'Editable', type: 'bool', defVal: true},
{name: 'metaEdit', dispName: 'Edit Dialog', type: 'bool', defVal: false},
{name: 'backgroundOutline', dispName: 'Background Outline', type: 'bool', defVal: false},
{name: 'movable', dispName: 'Movable', type: 'bool', defVal: true},
{name: 'movableLabel', dispName: 'Movable Label', type: 'bool', defVal: false, isVisible: function(state, format)
{
var geo = (state.vertices.length > 0) ? format.editorUi.editor.graph.getCellGeometry(state.vertices[0]) : null;
return geo != null && !geo.relative;
}},
{name: 'resizable', dispName: 'Resizable', type: 'bool', defVal: true},
{name: 'resizeWidth', dispName: 'Resize Width', type: 'bool', defVal: false},
{name: 'resizeHeight', dispName: 'Resize Height', type: 'bool', defVal: false},
{name: 'rotatable', dispName: 'Rotatable', type: 'bool', defVal: true},
{name: 'cloneable', dispName: 'Cloneable', type: 'bool', defVal: true},
{name: 'deletable', dispName: 'Deletable', type: 'bool', defVal: true},
{name: 'treeFolding', dispName: 'Tree Folding', type: 'bool', defVal: false},
{name: 'treeMoving', dispName: 'Tree Moving', type: 'bool', defVal: false},
{name: 'pointerEvents', dispName: 'Pointer Events', type: 'bool', defVal: true, isVisible: function(state, format)
{
var fillColor = mxUtils.getValue(state.style, mxConstants.STYLE_FILLCOLOR, null);
return format.editorUi.editor.graph.isSwimlane(state.vertices[0]) ||
fillColor == null || fillColor == mxConstants.NONE;
}},
{name: 'moveCells', dispName: 'Move Cells on Fold', type: 'bool', defVal: false, isVisible: function(state, format)
{
return state.vertices.length > 0 && format.editorUi.editor.graph.isContainer(state.vertices[0]);
}}
].concat(Editor.commonProperties);
/**
* Default value for the CSV import dialog.
*/
Editor.defaultCsvValue = '##\n' +
'## Example CSV import. Use ## for comments and # for configuration. Paste CSV below.\n' +
'## The following names are reserved and should not be used (or ignored):\n' +
'## id, tooltip, placeholder(s), link and label (see below)\n' +
'##\n' +
'#\n' +
'## Node label with placeholders and HTML.\n' +
'## Default is \'%name_of_first_column%\'.\n' +
'#\n' +
'# label: %name%<br><i style="color:gray;">%position%</i><br><a href="mailto:%email%">Email</a>\n' +
'#\n' +
'## Node style (placeholders are replaced once).\n' +
'## Default is the current style for nodes.\n' +
'#\n' +
'# style: label;image=%image%;whiteSpace=wrap;html=1;rounded=1;fillColor=%fill%;strokeColor=%stroke%;\n' +
'#\n' +
'## Parent style for nodes with child nodes (placeholders are replaced once).\n' +
'#\n' +
'# parentstyle: swimlane;whiteSpace=wrap;html=1;childLayout=stackLayout;horizontal=1;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;\n' +
'#\n' +
'## Optional column name that contains a reference to a named style in styles.\n' +
'## Default is the current style for nodes.\n' +
'#\n' +
'# stylename: -\n' +
'#\n' +
'## JSON for named styles of the form {"name": "style", "name": "style"} where style is a cell style with\n' +
'## placeholders that are replaced once.\n' +
'#\n' +
'# styles: -\n' +
'#\n' +
'## JSON for variables in styles of the form {"name": "value", "name": "value"} where name is a string\n' +
'## that will replace a placeholder in a style.\n' +
'#\n' +
'# vars: -\n' +
'#\n' +
'## Optional column name that contains a reference to a named label in labels.\n' +
'## Default is the current label.\n' +
'#\n' +
'# labelname: -\n' +
'#\n' +
'## JSON for named labels of the form {"name": "label", "name": "label"} where label is a cell label with\n' +
'## placeholders.\n' +
'#\n' +
'# labels: -\n' +
'#\n' +
'## Uses the given column name as the identity for cells (updates existing cells).\n' +
'## Default is no identity (empty value or -).\n' +
'#\n' +
'# identity: -\n' +
'#\n' +
'## Uses the given column name as the parent reference for cells. Default is no parent (empty or -).\n' +
'## The identity above is used for resolving the reference so it must be specified.\n' +
'#\n' +
'# parent: -\n' +
'#\n' +
'## Adds a prefix to the identity of cells to make sure they do not collide with existing cells (whose\n' +
'## IDs are numbers from 0..n, sometimes with a GUID prefix in the context of realtime collaboration).\n' +
'## Default is csvimport-.\n' +
'#\n' +
'# namespace: csvimport-\n' +
'#\n' +
'## Connections between rows ("from": source colum, "to": target column).\n' +
'## Label, style and invert are optional. Defaults are \'\', current style and false.\n' +
'## If placeholders are used in the style, they are replaced with data from the source.\n' +
'## An optional placeholders can be set to target to use data from the target instead.\n' +
'## In addition to label, an optional fromlabel and tolabel can be used to name the column\n' +
'## that contains the text for the label in the edges source or target (invert ignored).\n' +
'## In addition to those, an optional source and targetlabel can be used to specify a label\n' +
'## that contains placeholders referencing the respective columns in the source or target row.\n' +
'## The label is created in the form fromlabel + sourcelabel + label + tolabel + targetlabel.\n' +
'## Additional labels can be added by using an optional labels array with entries of the\n' +
'## form {"label": string, "x": number, "y": number, "dx": number, "dy": number} where\n' +
'## x is from -1 to 1 along the edge, y is orthogonal, and dx/dy are offsets in pixels.\n' +
'## An optional placeholders with the string value "source" or "target" can be specified\n' +
'## to replace placeholders in the additional label with data from the source or target.\n' +
'## The target column may contain a comma-separated list of values.\n' +
'## Multiple connect entries are allowed.\n' +
'#\n' +
'# connect: {"from": "manager", "to": "name", "invert": true, "label": "manages", \\\n' +
'# "style": "curved=1;endArrow=blockThin;endFill=1;fontSize=11;"}\n' +
'# connect: {"from": "refs", "to": "id", "style": "curved=1;fontSize=11;"}\n' +
'#\n' +
'## Node x-coordinate. Possible value is a column name. Default is empty. Layouts will\n' +
'## override this value.\n' +
'#\n' +
'# left: \n' +
'#\n' +
'## Node y-coordinate. Possible value is a column name. Default is empty. Layouts will\n' +
'## override this value.\n' +
'#\n' +
'# top: \n' +
'#\n' +
'## Node width. Possible value is a number (in px), auto or an @ sign followed by a column\n' +
'## name that contains the value for the width. Default is auto.\n' +
'#\n' +
'# width: auto\n' +
'#\n' +
'## Node height. Possible value is a number (in px), auto or an @ sign followed by a column\n' +
'## name that contains the value for the height. Default is auto.\n' +
'#\n' +
'# height: auto\n' +
'#\n' +
'## Padding for autosize. Default is 0.\n' +
'#\n' +
'# padding: -12\n' +
'#\n' +
'## Comma-separated list of ignored columns for metadata. (These can be\n' +
'## used for connections and styles but will not be added as metadata.)\n' +
'#\n' +
'# ignore: id,image,fill,stroke,refs,manager\n' +
'#\n' +
'## Column to be renamed to link attribute (used as link).\n' +
'#\n' +
'# link: url\n' +
'#\n' +
'## Spacing between nodes. Default is 40.\n' +
'#\n' +
'# nodespacing: 40\n' +
'#\n' +
'## Spacing between levels of hierarchical layouts. Default is 100.\n' +
'#\n' +
'# levelspacing: 100\n' +
'#\n' +
'## Spacing between parallel edges. Default is 40. Use 0 to disable.\n' +
'#\n' +
'# edgespacing: 40\n' +
'#\n' +
'## Name or JSON of layout. Possible values are auto, none, verticaltree, horizontaltree,\n' +
'## verticalflow, horizontalflow, organic, circle or a JSON string as used in Layout, Apply.\n' +
'## Default is auto.\n' +
'#\n' +
'# layout: auto\n' +
'#\n' +
'## ---- CSV below this line. First line are column names. ----\n' +
'name,position,id,location,manager,email,fill,stroke,refs,url,image\n' +
'Tessa Miller,CFO,emi,Office 1,,me@example.com,#dae8fc,#6c8ebf,,https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-3-128.png\n' +
'Edward Morrison,Brand Manager,emo,Office 2,Tessa Miller,me@example.com,#d5e8d4,#82b366,,https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-10-3-128.png\n' +
'Alison Donovan,System Admin,rdo,Office 3,Tessa Miller,me@example.com,#d5e8d4,#82b366,"emo,tva",https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-2-128.png\n' +
'Evan Valet,HR Director,tva,Office 4,Tessa Miller,me@example.com,#d5e8d4,#82b366,,https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-9-2-128.png\n';
/**
* Compresses the given string.
*/
Editor.createRoughCanvas = function(c)
{
var rc = rough.canvas(
{
// Provides expected function but return value is not used
getContext: function()
{
return c;
}
});
rc.draw = function(drawable)
{
var sets = drawable.sets || [];
var o = drawable.options || this.getDefaultOptions();
for (var i = 0; i < sets.length; i++)
{
var drawing = sets[i];
switch (drawing.type)
{
case 'path':
if (o.stroke != null)
{
this._drawToContext(c, drawing, o);
}
break;
case 'fillPath':
this._drawToContext(c, drawing, o);
break;
case 'fillSketch':
this.fillSketch(c, drawing, o);
break;
}
}
};
rc.fillSketch = function(ctx, drawing, o)
{
var strokeColor = c.state.strokeColor;
var strokeWidth = c.state.strokeWidth;
var strokeAlpha = c.state.strokeAlpha;
var dashed = c.state.dashed;
var fweight = o.fillWeight;
if (fweight < 0)
{
fweight = o.strokeWidth / 2;
}
c.setStrokeAlpha(c.state.fillAlpha);
c.setStrokeColor(o.fill || '');
c.setStrokeWidth(fweight);
c.setDashed(false);
this._drawToContext(ctx, drawing, o);
c.setDashed(dashed);
c.setStrokeWidth(strokeWidth);
c.setStrokeColor(strokeColor);
c.setStrokeAlpha(strokeAlpha);
};
rc._drawToContext = function(ctx, drawing, o)
{
ctx.begin();
for (var i = 0; i < drawing.ops.length; i++)
{
var item = drawing.ops[i];
var data = item.data;
switch (item.op)
{
case 'move':
ctx.moveTo(data[0], data[1]);
break;
case 'bcurveTo':
ctx.curveTo(data[0], data[1], data[2], data[3], data[4], data[5]);
break;
case 'lineTo':
ctx.lineTo(data[0], data[1]);
break;
}
};
ctx.end();
if (drawing.type === 'fillPath' && o.filled)
{
ctx.fill();
}
else
{
ctx.stroke();
}
};
return rc;
};
/**
* Uses RoughJs for drawing comic shapes.
*/
(function()
{
/**
* Adds handJiggle style (jiggle=n sets jiggle)
*/
function RoughCanvas(canvas, rc, shape)
{
this.canvas = canvas;
this.rc = rc;
this.shape = shape;
// Avoids "spikes" in the output
this.canvas.setLineJoin('round');
this.canvas.setLineCap('round');
this.originalBegin = this.canvas.begin;
this.canvas.begin = mxUtils.bind(this, RoughCanvas.prototype.begin);
this.originalEnd = this.canvas.end;
this.canvas.end = mxUtils.bind(this, RoughCanvas.prototype.end);
this.originalRect = this.canvas.rect;
this.canvas.rect = mxUtils.bind(this, RoughCanvas.prototype.rect);
this.originalRoundrect = this.canvas.roundrect;
this.canvas.roundrect = mxUtils.bind(this, RoughCanvas.prototype.roundrect);
this.originalEllipse = this.canvas.ellipse;
this.canvas.ellipse = mxUtils.bind(this, RoughCanvas.prototype.ellipse);
this.originalLineTo = this.canvas.lineTo;
this.canvas.lineTo = mxUtils.bind(this, RoughCanvas.prototype.lineTo);
this.originalMoveTo = this.canvas.moveTo;
this.canvas.moveTo = mxUtils.bind(this, RoughCanvas.prototype.moveTo);
this.originalQuadTo = this.canvas.quadTo;
this.canvas.quadTo = mxUtils.bind(this, RoughCanvas.prototype.quadTo);
this.originalCurveTo = this.canvas.curveTo;
this.canvas.curveTo = mxUtils.bind(this, RoughCanvas.prototype.curveTo);
this.originalArcTo = this.canvas.arcTo;
this.canvas.arcTo = mxUtils.bind(this, RoughCanvas.prototype.arcTo);
this.originalClose = this.canvas.close;
this.canvas.close = mxUtils.bind(this, RoughCanvas.prototype.close);
this.originalFill = this.canvas.fill;
this.canvas.fill = mxUtils.bind(this, RoughCanvas.prototype.fill);
this.originalStroke = this.canvas.stroke;
this.canvas.stroke = mxUtils.bind(this, RoughCanvas.prototype.stroke);
this.originalFillAndStroke = this.canvas.fillAndStroke;
this.canvas.fillAndStroke = mxUtils.bind(this, RoughCanvas.prototype.fillAndStroke);
this.path = [];
this.passThrough = false;
};
RoughCanvas.prototype.moveOp = 'M';
RoughCanvas.prototype.lineOp = 'L';
RoughCanvas.prototype.quadOp = 'Q';
RoughCanvas.prototype.curveOp = 'C';
RoughCanvas.prototype.closeOp = 'Z';
RoughCanvas.prototype.getStyle = function(stroke, fill)
{
// Random seed created from cell ID
var seed = 1;
if (this.shape.state != null)
{
var str = this.shape.state.cell.id;
if (str != null)
{
for (var i = 0; i < str.length; i++)
{
seed = ((seed << 5) - seed + str.charCodeAt(i)) << 0;
}
}
}
var style = {strokeWidth: this.canvas.state.strokeWidth, seed: seed, preserveVertices: true};
var defs = this.rc.getDefaultOptions();
if (stroke)
{
style.stroke = this.canvas.state.strokeColor === 'none' ? 'transparent' : this.canvas.state.strokeColor;
}
else
{
delete style.stroke;
}
var gradient = null;
style.filled = fill;
if (fill)
{
style.fill = this.canvas.state.fillColor === 'none' ? '' : this.canvas.state.fillColor;
gradient = this.canvas.state.gradientColor === 'none' ? null : this.canvas.state.gradientColor;
}
else
{
style.fill = '';
}
// Applies cell style
style['bowing'] = mxUtils.getValue(this.shape.style, 'bowing', defs['bowing']);
st