mirror of
https://github.com/philippemerle/KubeDiagrams.git
synced 2026-05-12 09:26:50 +00:00
431 lines
12 KiB
JavaScript
431 lines
12 KiB
JavaScript
cytoscape.use(cytoscapeDagre);
|
|
cytoscape.use(cytoscapeKlay);
|
|
|
|
let cy;
|
|
let currentLayout;
|
|
|
|
|
|
// Construit la liste d'éléments Cytoscape à partir d'un objet DOT_JSON
|
|
function buildElementsFromDotJson(json) {
|
|
const elements = [];
|
|
const nodes = json.objects || json.nodes || [];
|
|
const edges = json.edges || [];
|
|
addNodesElementsParsedFromNodesJson(elements, nodes);
|
|
addNodesEdgesParsedFromEdgesJson(elements, edges);
|
|
return elements;
|
|
}
|
|
|
|
// Rend le graphe directement depuis un objet DOT_JSON
|
|
window.renderFromDotJson = function (json) {
|
|
try {
|
|
const elements = buildElementsFromDotJson(json);
|
|
load_cytoscape(elements);
|
|
} catch (e) {
|
|
console.error('renderFromDotJson failed:', e);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* initialise a cytoscape instance, create layout selector buttons, add event listeners on save buttons and add
|
|
* an event listener on the file input button to load the cytoscape graph.
|
|
*/
|
|
function setUp() {
|
|
cy = getCyGraph();
|
|
currentLayout = layoutList[0];
|
|
createLayoutSelectorButton();
|
|
document.getElementById("savePNG").addEventListener("click", () => { saveFile("png")});
|
|
document.getElementById("saveJPG").addEventListener("click", () => { saveFile("jpg")});
|
|
//document.getElementById("saveSVG").addEventListener("click", () => { saveFile("svg")});//
|
|
|
|
document.getElementById('fileInput').addEventListener('change', readFileAndloadCytoscapeGraph);
|
|
window.addEventListener('message', (e) => {
|
|
const data = e && e.data;
|
|
if (!data || data.type !== 'KD_LOAD_DOT_JSON' || !data.payload) return;
|
|
try {
|
|
window.renderFromDotJson(data.payload);
|
|
} catch (err) {
|
|
console.error('Failed to handle KD_LOAD_DOT_JSON:', err);
|
|
}
|
|
}, false);
|
|
|
|
// === (Optionnel) Autoload via ?dataUrl=... ===
|
|
try {
|
|
const params = new URLSearchParams(location.search);
|
|
const dataUrl = params.get('dataUrl');
|
|
if (dataUrl) {
|
|
fetch(dataUrl)
|
|
.then(r => r.json())
|
|
.then(json => window.renderFromDotJson(json))
|
|
.catch(err => console.error('Failed to load DOT_JSON via dataUrl:', err));
|
|
}
|
|
} catch (_) { /* ignore */ }
|
|
}
|
|
|
|
|
|
/**
|
|
* return a cytoscape graph.
|
|
* @returns
|
|
*/
|
|
function getCyGraph() {
|
|
return cytoscape({
|
|
container: document.getElementById('paper'),
|
|
elements: undefined,
|
|
hideEdgesOnViewport: true,
|
|
pixelRatio: window.devicePixelRatio,
|
|
style: [
|
|
{
|
|
selector: 'node',
|
|
style: defaultGlobalNodeStyle
|
|
},
|
|
{
|
|
selector: 'node[group = "cluster"]',
|
|
style: clusterOpenStyle
|
|
},
|
|
{
|
|
selector: 'node[group = "node"]',
|
|
style: defaultGroupNodeStyle
|
|
},
|
|
{
|
|
selector: 'edge',
|
|
style: defaultEdgeStyle
|
|
},
|
|
{
|
|
selector: 'edge[dir = "forward"]',
|
|
style: defaultEdgeDirForward
|
|
},
|
|
{
|
|
selector: 'edge[dir = "back"]',
|
|
style: defaultEdgeDirBack
|
|
}
|
|
]
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Read a dot_json file to add nodes and edges in a elements list used to load a cytoscape instance.
|
|
* @param {*} event
|
|
*/
|
|
function readFileAndloadCytoscapeGraph(event) {
|
|
const file = event.target.files[0];
|
|
const reader = new FileReader();
|
|
const elements = [];
|
|
|
|
reader.onload = function(e) {
|
|
let content = e.target.result;
|
|
const json = JSON.parse(content);
|
|
const nodes = json.objects;
|
|
const edges = json.edges;
|
|
addNodesElementsParsedFromNodesJson(elements, nodes);
|
|
addNodesEdgesParsedFromEdgesJson(elements, edges);
|
|
load_cytoscape(elements);
|
|
};
|
|
reader.onload = function(e) {
|
|
try {
|
|
const content = e.target.result;
|
|
const json = JSON.parse(content);
|
|
const elements = buildElementsFromDotJson(json);
|
|
load_cytoscape(elements);
|
|
} catch (err) {
|
|
console.error('Invalid DOT_JSON file:', err);
|
|
}
|
|
};
|
|
reader.readAsText(file);
|
|
}
|
|
|
|
/**
|
|
* Remove precedent elements before to add the new elements then create tool tip and context menus for the
|
|
* cytoscape instance.
|
|
* @param {*} elements
|
|
*/
|
|
function load_cytoscape(elements) {
|
|
cy.nodes().remove();
|
|
cy.add(elements);
|
|
createTooltip("node");
|
|
createTooltip("edge");
|
|
createAndGetContextMenu(cy);
|
|
cy.layout(currentLayout).run();
|
|
}
|
|
|
|
function load_layout(layout) {
|
|
currentLayout = layout;
|
|
cy.layout(currentLayout).run();
|
|
}
|
|
|
|
/**
|
|
* Add nodes elements created from nodesJson data list in the elements list.
|
|
* @param {*} elements
|
|
* @param {*} nodesJson - nodes data list
|
|
* @returns
|
|
*/
|
|
function addNodesElementsParsedFromNodesJson(elements, nodesJson) {
|
|
let parent = {};
|
|
|
|
for (let i in nodesJson) {
|
|
let node = {
|
|
data: {
|
|
id: nodesJson[i]._gvid,
|
|
group: (nodesJson[i].nodes) ? 'cluster' : 'node',
|
|
isClose : false,
|
|
label: (nodesJson[i].label.includes('<') && nodesJson[i].label.includes('>')) ? nodesJson[i].tooltip : nodesJson[i].label,
|
|
bs: getCorrespondingBorderStyle(nodesJson[i].style),
|
|
bgcolor: getCorrespondingColor(nodesJson[i].bgcolor ?? 'blue'),
|
|
bc: nodesJson[i].pencolor ?? 'gray',
|
|
parent: parent[nodesJson[i]._gvid] ?? '',
|
|
fontsize: nodesJson[i].fontsize ?? '',
|
|
fontfamily: nodesJson[i].fontname ?? '',
|
|
fontcolor: nodesJson[i].fontcolor ?? '',
|
|
image: (nodesJson[i].image) ? nodesJson[i].image : '',
|
|
tooltip: nodesJson[i].tooltip ?? ''
|
|
}
|
|
}
|
|
|
|
if (nodesJson[i].nodes) {
|
|
for (let child of nodesJson[i].nodes) {
|
|
parent[child] = nodesJson[i]._gvid;
|
|
}
|
|
}
|
|
|
|
if (nodesJson[i].subgraphs) {
|
|
for (let sub of nodesJson[i].subgraphs) {
|
|
parent[sub] = nodesJson[i]._gvid;
|
|
}
|
|
}
|
|
addClassToElement(node);
|
|
elements.push(node);
|
|
}
|
|
|
|
return elements;
|
|
}
|
|
|
|
/**
|
|
* Add nodes elements created from edgesJson data list in the elements list.
|
|
* @param {*} elements
|
|
* @param {*} edgesJson
|
|
*/
|
|
function addNodesEdgesParsedFromEdgesJson(elements, edgesJson) {
|
|
for (let e in edgesJson) {
|
|
let edge = {
|
|
data: {
|
|
id: 'e' + edgesJson[e]._gvid,
|
|
group: 'edge',
|
|
dir: edgesJson[e].dir,
|
|
source: edgesJson[e].tail,
|
|
target: edgesJson[e].head,
|
|
color: edgesJson[e].color,
|
|
line_style: edgesJson[e].style ?? 'solid',
|
|
xlabel: edgesJson[e].xlabel ?? '',
|
|
fontsize: edgesJson[e].fontsize ?? '',
|
|
fontfamily: edgesJson[e].fontname ?? '',
|
|
fontcolor: edgesJson[e].fontcolor ?? '',
|
|
tooltip: edgesJson[e].tooltip ?? '',
|
|
}
|
|
}
|
|
addClassToElement(edge);
|
|
elements.push(edge);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get border style based on the what's in the style parameter.
|
|
* @param {*} style
|
|
* @returns
|
|
*/
|
|
function getCorrespondingBorderStyle(style) {
|
|
if (style.includes('dashed')) {
|
|
return 'dashed';
|
|
}
|
|
else {
|
|
return 'solid';
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the corresponding color.
|
|
* @param {*} color
|
|
* @returns
|
|
*/
|
|
function getCorrespondingColor(color) {
|
|
return (color == 'transparent') ? '#ffffff00' : color;
|
|
}
|
|
|
|
/**
|
|
* Create a HTML tooltip with the events of the cytoscape instance
|
|
* @param {*} elementType
|
|
*/
|
|
function createTooltip(elementType) {
|
|
const tooltip = document.getElementById('tooltip');
|
|
let to;
|
|
|
|
cy.on('mouseover', elementType, function(event) {
|
|
const element = event.target;
|
|
to = setTimeout(() => {
|
|
tooltip.style.display = 'block';
|
|
tooltip.innerText = element.data('tooltip');
|
|
}, 1000);
|
|
});
|
|
|
|
cy.on('mouseout', elementType, function(event) {
|
|
clearTimeout(to);
|
|
tooltip.style.display = 'none';
|
|
});
|
|
|
|
cy.on('mousemove', function(event) {
|
|
tooltip.style.left = (event.originalEvent.pageX + 10) + 'px';
|
|
tooltip.style.top = (event.originalEvent.pageY + 10) + 'px';
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Create layout selector buttons after the fileInput
|
|
*/
|
|
function createLayoutSelectorButton() {
|
|
const layoutButtons = document.getElementById("layoutButtons");
|
|
for (let layout of layoutList) {
|
|
let button = document.createElement("button")
|
|
button.id = layout.name;
|
|
button.textContent = layout.name;
|
|
button.addEventListener("click", () => {load_layout(layout)} );
|
|
layoutButtons.appendChild(button);
|
|
}
|
|
}
|
|
|
|
|
|
// Modifier ici pour récupérer les données via un blob
|
|
// function saveFile(format) {
|
|
// const data = getFileAs(format);
|
|
|
|
// if (format === 'svg') {
|
|
// console.log(data);
|
|
// const blob = new Blob([data], { type: 'image/svg+xml' });
|
|
// const url = URL.createObjectURL(blob);
|
|
// const a = document.createElement('a');
|
|
// a.href = url;
|
|
// a.download = 'graph.svg';
|
|
// a.click();
|
|
// setTimeout(() => URL.revokeObjectURL(url), 0);
|
|
// return;
|
|
// }
|
|
// const a = document.createElement('a');
|
|
// a.href = data;
|
|
// a.download = 'graph.' + format;
|
|
// a.click();
|
|
// }
|
|
|
|
/**
|
|
* Save file in png or jpg format.
|
|
* @param {*} format
|
|
*/
|
|
function saveFile(format) {
|
|
const img = getFileAs(format);
|
|
|
|
const ele = document.createElement('a');
|
|
ele.href = img;
|
|
ele.download = 'graph.' + format;
|
|
ele.click();
|
|
}
|
|
|
|
/**
|
|
* Get a representation of the graph as an image of the requested format
|
|
* @param {*} format
|
|
* @returns
|
|
*/
|
|
function getFileAs(format) {
|
|
switch (format) {
|
|
case 'png': return cy.png({ full: true });
|
|
case 'jpg': return cy.jpg({ full: true });
|
|
//case 'svg': return cy.svg({ full: true, scale: 5});//
|
|
default: throw new Error('Format inconnu: ' + format);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Create and return a context menus for the cytoscape instance
|
|
* @param {*} cy
|
|
* @returns
|
|
*/
|
|
function createAndGetContextMenu(cy) {
|
|
return cy.contextMenus({
|
|
evtType: 'cxttap',
|
|
menuItems: [itemOpenClose]
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Check the element's group and then add a style class based on the group and his values.
|
|
* @param {*} element
|
|
*/
|
|
function addClassToElement(element) {
|
|
let classesName = [];
|
|
let style;
|
|
if (element.data.group == "edge") {
|
|
style = getDefaultEdgeStyleFromEdgeValues(element);
|
|
classesName.push(findAndGetClassStyle(style, edgeStyleList).selector.replace(/^\./, ''));
|
|
}
|
|
else {
|
|
style = getDefaultGlobalNodeStyleFromNodeValues(element);
|
|
classesName.push(findAndGetClassStyle(style, nodeStyleList).selector.replace(/^\./, ''));
|
|
if (element.data.group == "cluster") {
|
|
style = getDefaultClusterStyleFromClusterValues(element);
|
|
classesName.push(findAndGetClassStyle(style, clusterStyleList).selector.replace(/^\./, ''))
|
|
}
|
|
}
|
|
element.classes = classesName.join(' ');
|
|
}
|
|
|
|
/**
|
|
* Look for a style class similar to the styleTarget in the styleList. if a style class is founded,
|
|
* so the method return the style class founded otherwise add the styleTarget in the cytoscape instance and
|
|
* in the styleList then return the styleTarget
|
|
* @param {*} styleTarget
|
|
* @param {*} styleList
|
|
* @returns
|
|
*/
|
|
function findAndGetClassStyle(styleTarget, styleList) {
|
|
for (let style of styleList) {
|
|
if (equals(styleTarget.style, style.style)) {
|
|
return style;
|
|
}
|
|
}
|
|
addStyleToCytoscapeGraph(styleTarget);
|
|
styleList.push(styleTarget);
|
|
return styleTarget;
|
|
}
|
|
|
|
/**
|
|
* Add a new style in the style attribute of the cytoscape instance.
|
|
* @param {*} style
|
|
*/
|
|
function addStyleToCytoscapeGraph(style) {
|
|
const existingStyle = cy.style().json();
|
|
cy.style([...existingStyle, style]);
|
|
}
|
|
|
|
/**
|
|
* Check if two object are similar.
|
|
* @param {*} o1
|
|
* @param {*} o2
|
|
* @returns
|
|
*/
|
|
function equals(o1, o2) {
|
|
if (o1 === o2) return true;
|
|
if (typeof o1 !== "object" || typeof o2 !== "object" || o1 == null || o2 == null) {
|
|
return false;
|
|
}
|
|
|
|
const k1 = Object.keys(o1);
|
|
const k2 = Object.keys(o2);
|
|
|
|
if (k1.length !== k2.length) return false;
|
|
|
|
for (let k of k1) {
|
|
if (!k2.includes(k) || !equals(o1[k], o2[k])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
window.addEventListener("load", setUp); |