2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
const CONFIG = {
|
2021-03-31 14:24:46 +00:00
|
|
|
'nodeSize': 8,
|
2021-03-30 14:30:50 +00:00
|
|
|
'baseUrl': 'https://www.securityvision.io/wiki/index.php/',
|
|
|
|
'dataUrl': '../semantic_data.json',
|
2021-03-31 14:24:46 +00:00
|
|
|
'preSimulate': false, // run simulation before starting, so we don't start with lines jumping around
|
2021-03-30 14:30:50 +00:00
|
|
|
'labels': {
|
|
|
|
'rotate': true,
|
|
|
|
},
|
2021-04-16 09:15:57 +00:00
|
|
|
'countries': ['Austria', 'Italy', 'Belgium', 'Latvia', 'Bulgaria', 'Lithuania', 'Croatia', 'Luxembourg', 'Cyprus', 'Malta', 'Czechia', 'Netherlands', 'Denmark', 'Poland', 'Estonia', 'Portugal', 'Finland ', 'Romania', 'France', 'Slovakia', 'Germany', 'Slovenia', 'Greece', 'Spain', 'Hungary', 'Sweden', 'Ireland'],
|
2021-03-31 14:24:46 +00:00
|
|
|
'eu': {
|
|
|
|
'lonMin': -10,
|
2021-03-31 15:02:00 +00:00
|
|
|
'lonMax': 35,
|
|
|
|
'center': [11, 47],
|
2021-04-16 09:15:57 +00:00
|
|
|
},
|
|
|
|
"filters": ["Institution", "Deployments", "Technology", "Dataset"],
|
2021-03-30 14:30:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
// thanks to https://bl.ocks.org/denisemauldin/cdd667cbaf7b45d600a634c8ae32fae5
|
|
|
|
var graph, store;
|
|
|
|
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
var filteredCategories = [];//= ["Institution"];
|
|
|
|
|
|
|
|
CONFIG.filters.forEach(f => {
|
|
|
|
let labelEl = document.createElement('label')
|
|
|
|
let inputEl = document.createElement('input')
|
|
|
|
let textEl = document.createElement('span');
|
|
|
|
inputEl.type = "checkbox";
|
|
|
|
textEl.innerText = f;
|
|
|
|
labelEl.appendChild(inputEl);
|
|
|
|
labelEl.appendChild(textEl);
|
|
|
|
|
|
|
|
if(!filteredCategories.includes(f)) {
|
|
|
|
inputEl.checked = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inputEl.addEventListener('change', function(e){
|
|
|
|
if(e.target.checked) {
|
|
|
|
filteredCategories.forEach((d, i) => {
|
|
|
|
if(d == f) {
|
|
|
|
filteredCategories.splice(i, 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
if(!filteredCategories.includes(f)) {
|
|
|
|
filteredCategories.push(f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
filter();
|
|
|
|
update();
|
|
|
|
})
|
|
|
|
|
|
|
|
document.getElementById('filters').appendChild(labelEl);
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
function getSizeForNode(node) {
|
|
|
|
// if (node.hasOwnProperty('https://schema.org/thumbnailUrl')) return nodeSize;
|
|
|
|
// if (weights[node['@id']]) return nodeSize * weights[node['@id']];
|
|
|
|
// if (node['@id'] == firstNodeId) return nodeSize * 1.2;
|
|
|
|
// // everynode has at least one link. these should equal 1
|
|
|
|
// return nodeSize * (.7 + Math.min(20, linkMap[node['@id']].length) / 40);
|
|
|
|
if (node.parent) {
|
|
|
|
return 2;
|
|
|
|
}
|
2021-03-31 14:24:46 +00:00
|
|
|
// if(getCategories(node).indexOf('City') !== -1) {
|
|
|
|
// return 2;
|
|
|
|
// }
|
2021-03-30 14:30:50 +00:00
|
|
|
return CONFIG.nodeSize;
|
2021-03-29 18:49:50 +00:00
|
|
|
}
|
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
function splitText(text) {
|
|
|
|
var characters = [" ", "-", "_", '\xAD'];
|
|
|
|
var charSplitPos = {};
|
|
|
|
var mid = Math.floor(text.length / 2);
|
|
|
|
var splitPos = false;
|
|
|
|
var splitPosChar = false;
|
|
|
|
// split sentences
|
|
|
|
var _iteratorNormalCompletion6 = true;
|
|
|
|
var _didIteratorError6 = false;
|
|
|
|
var _iteratorError6 = undefined;
|
|
|
|
|
|
|
|
try {
|
|
|
|
for (var _iterator6 = characters[Symbol.iterator](), _step6; !(_iteratorNormalCompletion6 = (_step6 = _iterator6.next()).done); _iteratorNormalCompletion6 = true) {
|
|
|
|
var char = _step6.value;
|
|
|
|
|
|
|
|
if (text.indexOf(char) < 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
var tmid = text.substr(0, mid).lastIndexOf(char);
|
|
|
|
if (tmid === -1) {
|
|
|
|
tmid = text.indexOf(char);
|
|
|
|
}
|
|
|
|
tmid += 1; // we want to cut _after_ the character
|
|
|
|
// console.log("Char", char, tmid);
|
|
|
|
if (splitPos === false || Math.abs(tmid - mid) < Math.abs(splitPos - mid)) {
|
|
|
|
// console.log("least!");
|
|
|
|
splitPos = tmid;
|
|
|
|
splitPosChar = char;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// console.log("pos",splitPos)
|
|
|
|
|
|
|
|
} catch (err) {
|
|
|
|
_didIteratorError6 = true;
|
|
|
|
_iteratorError6 = err;
|
|
|
|
} finally {
|
|
|
|
try {
|
|
|
|
if (!_iteratorNormalCompletion6 && _iterator6.return) {
|
|
|
|
_iterator6.return();
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (_didIteratorError6) {
|
|
|
|
throw _iteratorError6;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
if (splitPos === false) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
var text1 = text.substr(0, splitPos).trim();
|
|
|
|
var text2 = text.substr(splitPos).trim();
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
if (splitPosChar == '\xAD') {
|
|
|
|
text1 += "-";
|
|
|
|
}
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
// find most equal split
|
|
|
|
return [text1, text2];
|
|
|
|
};
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
function getTitle(obj) {
|
|
|
|
if (obj.parent) {
|
|
|
|
return "sub of " + obj.parent.split('#', 1)[0].replace(/_/g, " ")
|
|
|
|
}
|
|
|
|
return obj['@id'].split('#', 1)[0].replace(/_/g, " ")
|
|
|
|
}
|
2021-03-31 14:24:46 +00:00
|
|
|
function getCategories(obj) {
|
|
|
|
if (!obj._INST) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
return obj['_INST'].map(classId => classId.split('#', 1)[0]);
|
|
|
|
}
|
2021-03-30 14:30:50 +00:00
|
|
|
function getClasses(obj) {
|
|
|
|
if (!obj._INST)
|
|
|
|
return 'node';
|
2021-03-31 14:24:46 +00:00
|
|
|
const classes = getCategories(obj);
|
2021-03-30 14:30:50 +00:00
|
|
|
return 'node ' + classes.join(' ');
|
|
|
|
}
|
|
|
|
function getUrl(obj) {
|
|
|
|
return CONFIG.baseUrl + obj['@id'].split('#', 1)[0];
|
|
|
|
}
|
2021-03-29 18:49:50 +00:00
|
|
|
|
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
// see also: http://bl.ocks.org/dwtkns/4973620
|
|
|
|
let width = window.innerWidth;
|
|
|
|
let height = window.innerHeight;
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
const svg = d3.select("svg")
|
|
|
|
|
|
|
|
|
|
|
|
// SET UP MAP:
|
|
|
|
const projection = d3.geoHill()
|
|
|
|
.rotate([-12, 0, 0])
|
2021-04-16 09:15:57 +00:00
|
|
|
.translate([width / 2, height * 1.5])
|
|
|
|
.scale(height * 1.5);
|
2021-03-31 15:02:00 +00:00
|
|
|
|
|
|
|
const proj = d3.geoPath().projection(projection);
|
|
|
|
const graticule = d3.geoGraticule10();
|
2021-04-16 09:15:57 +00:00
|
|
|
const euCenter = projection(CONFIG.eu.center);
|
2021-03-31 15:02:00 +00:00
|
|
|
|
|
|
|
const container = svg.append("g").attr("id", "container");
|
|
|
|
// container.append("circle").attr("cx", euCenter[0]).attr("cy", euCenter[1]).attr("r", 500).attr("fill","red")
|
|
|
|
|
|
|
|
const g_countries = container.append("g").attr("id", "countries");
|
|
|
|
const g_borders = container.append("g").attr("id", "borders");
|
|
|
|
const g_graticule = container.append("g")
|
|
|
|
.append('path')
|
|
|
|
.attr("class", "graticule")
|
|
|
|
.attr("fill", "none")
|
2021-04-16 09:15:57 +00:00
|
|
|
.attr('d', proj(graticule))
|
2021-03-31 15:02:00 +00:00
|
|
|
.attr("stroke-width", "!px")
|
|
|
|
.attr("stroke", (n) => {
|
|
|
|
return "lightgray";
|
|
|
|
});
|
|
|
|
;
|
2021-03-29 18:49:50 +00:00
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
function sizeWindow() {
|
|
|
|
width = window.innerWidth;
|
|
|
|
height = window.innerHeight;
|
|
|
|
|
|
|
|
svg
|
|
|
|
.attr("viewBox", [0, 0, width, height])
|
|
|
|
.attr("width", width)
|
|
|
|
.attr("height", height);
|
2021-04-16 09:15:57 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
//
|
2021-03-31 15:02:00 +00:00
|
|
|
// update();
|
|
|
|
}
|
|
|
|
sizeWindow()
|
|
|
|
window.addEventListener('resize', sizeWindow);
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
svg.call(d3.zoom().scaleExtent([0.3, 8]).on("start", function () {
|
|
|
|
svg.node().classList.add("dragging");
|
|
|
|
}).on("end", function () {
|
|
|
|
svg.node().classList.remove("dragging");
|
|
|
|
}).on("zoom", function ({ transform }) {
|
|
|
|
container.attr("transform", transform);
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var node = container.append("g")
|
|
|
|
.attr('class', 'nodes')
|
|
|
|
.selectAll(".node");
|
|
|
|
var link = container.append("g")
|
|
|
|
.attr('class', 'links')
|
|
|
|
.selectAll(".link");
|
|
|
|
// let linkLines = link.selectAll('line');
|
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
|
|
|
|
// REQUEST ATLAS & GRAPH
|
2021-03-30 14:30:50 +00:00
|
|
|
const req_data = new Request(CONFIG.dataUrl, { method: 'GET' });
|
|
|
|
const req_world = new Request('https://cdn.jsdelivr.net/npm/world-atlas@2/countries-110m.json', { method: 'GET' });
|
|
|
|
Promise.all([fetch(req_data), fetch(req_world)])
|
|
|
|
.then(([res_data, res_world]) => {
|
|
|
|
return Promise.all([res_data.json(), res_world.json()]);
|
|
|
|
})
|
|
|
|
.then(([data, world]) => {
|
|
|
|
buildGraph(data, world);
|
|
|
|
}).catch(error => {
|
|
|
|
console.error(error);
|
|
|
|
});
|
|
|
|
;
|
2021-03-31 15:02:00 +00:00
|
|
|
|
|
|
|
let linkCounts = [];
|
|
|
|
const simulation = d3.forceSimulation()
|
|
|
|
.force("link", d3.forceLink()
|
|
|
|
.id(d => d['@id'])
|
|
|
|
.iterations(2) // increase to make more rigid
|
|
|
|
.distance((l) => {
|
|
|
|
// if (getCategories(l.source).indexOf('City') || getCategories(l.target).indexOf('City')) {
|
|
|
|
// // if(l.source.lat || l.target.lat) {
|
|
|
|
// return 2;
|
|
|
|
// }
|
|
|
|
return 10;
|
|
|
|
})
|
|
|
|
.strength((l) => {
|
|
|
|
if (linkCounts.length < 1) {
|
|
|
|
// replicate from d3-force/src/link.js so we have access to this in our own strength function
|
|
|
|
linkCounts = new Array(store.nodes.length)
|
|
|
|
for (let i = 0; i < store.links.length; ++i) {
|
|
|
|
let link = store.links[i];
|
|
|
|
linkCounts[link.source.index] = (linkCounts[link.source.index] || 0) + 1;
|
|
|
|
linkCounts[link.target.index] = (linkCounts[link.target.index] || 0) + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (store.nodesBorderingEu.indexOf(l.source) !== -1 || store.nodesBorderingEu.indexOf(l.target) !== -1) {
|
|
|
|
// console.log('outside', l.target)
|
|
|
|
return 0.0001;
|
|
|
|
}
|
|
|
|
// original:
|
|
|
|
return 1 / Math.min(linkCounts[l.source.index], linkCounts[l.target.index]);
|
|
|
|
})
|
|
|
|
)
|
|
|
|
// .force("charge", d3.forceManyBody()
|
|
|
|
// .strength(-10)
|
|
|
|
// )
|
|
|
|
// .force("center", d3.forceCenter(width / 2, height / 2))
|
|
|
|
.force("collision", d3.forceCollide(function (d) {
|
|
|
|
return getSizeForNode(d) * 1.5; // avoid overlapping nodes
|
|
|
|
}))
|
2021-04-16 09:15:57 +00:00
|
|
|
.force("outsideEu", d3.forceRadial(height/2.7, euCenter[0], euCenter[1])
|
2021-03-31 15:02:00 +00:00
|
|
|
.strength(function (node, idx) {
|
|
|
|
// return 1;
|
|
|
|
if (store.nodesBorderingEu.indexOf(node) !== -1) {
|
|
|
|
// console.log(node, store.nodesBorderingEu.indexOf(node) !== -1);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
})
|
|
|
|
);
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
function buildGraph(data, world) {
|
2021-03-29 18:49:50 +00:00
|
|
|
// land = topojson.feature(world, world.objects.land)
|
|
|
|
const borders = topojson.mesh(world, world.objects.countries, (a, b) => a !== b)
|
|
|
|
const countries = topojson.feature(world, world.objects.countries).features;
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
|
2021-03-30 14:30:50 +00:00
|
|
|
const nodes = data.nodes.filter(n => n._INST || n.parent).map(d => Object.create(d));
|
2021-03-31 14:24:46 +00:00
|
|
|
console.log('all nodes', nodes);
|
2021-03-30 14:30:50 +00:00
|
|
|
const nodeMap = Object.fromEntries(nodes.map(d => [d['@id'], d]));
|
|
|
|
const links = data.links.filter(l => nodeMap[l.source] && nodeMap[l.target]).map(d => Object.create(d));
|
2021-04-16 09:15:57 +00:00
|
|
|
console.log(nodeMap, links);
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
graph = {
|
2021-04-16 09:15:57 +00:00
|
|
|
"nodes": [],
|
|
|
|
"links": [],
|
2021-03-31 15:02:00 +00:00
|
|
|
}
|
|
|
|
store = {
|
|
|
|
"nodes": [...nodes],
|
|
|
|
"links": [...links],
|
|
|
|
"nodesBorderingEu": links.filter(l => l.name == 'City' && (nodeMap[l.target].lon < CONFIG.eu.lonMin || nodeMap[l.target].lon > CONFIG.eu.lonMax)).map(l => nodeMap[l.source]),
|
|
|
|
}
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
filter();
|
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
// the .source and .target attributes are still ID's. Only after initialisation of the force are they replaced with their representative objects
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
const nodesEastOfEu = links.filter(l => l.name == 'City' && (nodeMap[l.target].lon < CONFIG.eu.lonMin)).map(l => nodeMap[l.source]);
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
var c = g_countries.selectAll("path")
|
2021-03-29 18:49:50 +00:00
|
|
|
.data(countries)
|
|
|
|
.enter()
|
|
|
|
.append("path")
|
|
|
|
.attr("class", "countries")
|
|
|
|
.attr("d", proj)
|
|
|
|
.attr("fill", (n) => {
|
2021-03-30 14:30:50 +00:00
|
|
|
if (CONFIG.countries.indexOf(n.properties.name) !== -1) {
|
2021-03-29 18:49:50 +00:00
|
|
|
return '';
|
|
|
|
}
|
2021-04-16 09:15:57 +00:00
|
|
|
return "rgba(200,200,200,.7)";
|
2021-03-29 18:49:50 +00:00
|
|
|
});
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
g_borders
|
2021-03-29 18:49:50 +00:00
|
|
|
.append("path")
|
|
|
|
.attr("class", "borders")
|
|
|
|
.attr("d", proj(borders))
|
|
|
|
.attr("fill", "none")
|
|
|
|
.attr("stroke-width", "2px")
|
|
|
|
.attr("stroke", (n) => {
|
|
|
|
return "white";
|
|
|
|
});
|
|
|
|
|
2021-03-31 14:24:46 +00:00
|
|
|
nodes.forEach(function (d) {
|
|
|
|
d.x = euCenter[0];
|
|
|
|
d.y = euCenter[1];
|
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
if (store.nodesBorderingEu.indexOf(d) !== -1) {
|
|
|
|
if (nodesEastOfEu.indexOf(d) !== -1) {
|
2021-03-31 14:24:46 +00:00
|
|
|
d.x = 466.5836692678423;
|
|
|
|
d.y = 466.3493609705728;
|
|
|
|
} else {
|
|
|
|
d.x = 1406.608195836305;
|
|
|
|
d.y = 807.9332721328062;
|
|
|
|
}
|
2021-03-31 15:02:00 +00:00
|
|
|
|
2021-03-31 14:24:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (d.lon && d.lat) {
|
|
|
|
// console.log("fix node", d);
|
|
|
|
var p = projection([d.lon, d.lat]);
|
|
|
|
d.x = p[0];
|
|
|
|
d.y = p[1];
|
|
|
|
d.fx = p[0];
|
|
|
|
d.fy = p[1];
|
|
|
|
}
|
|
|
|
})
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
update();
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
return svg.node();
|
|
|
|
}
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
function update() {
|
|
|
|
|
|
|
|
|
|
|
|
// see also: https://www.createwithdata.com/enter-exit-with-d3-join/
|
|
|
|
node = node.data(graph.nodes, d => d.id)
|
|
|
|
.join(function (enter) {
|
|
|
|
let group = enter.append("g").attr("class", getClasses);
|
|
|
|
group.call(drag(simulation));
|
|
|
|
group.on("click", (evt, n) => selectNode(evt, n, node));
|
|
|
|
group.append('circle').attr("r", getSizeForNode);
|
|
|
|
var nodeTitle = group.append('text').attr("class", "nodeTitle").attr("y", "5");
|
|
|
|
nodeTitle
|
|
|
|
.each(function (node, nodes) {
|
|
|
|
var textLength = void 0;
|
|
|
|
const self = d3.select(this);
|
|
|
|
const titleText = getTitle(node);
|
|
|
|
var titleTexts = false;
|
|
|
|
if (titleText.length > 20) {
|
|
|
|
titleTexts = splitText(titleText);
|
|
|
|
}
|
|
|
|
if (titleTexts !== false) {
|
|
|
|
const tspan1 = self.append("tspan").text(titleTexts[0]).attr("y", "-10").attr("x", "0");
|
|
|
|
const tspan = self.append("tspan").text(titleTexts[1]).attr("y", "10").attr("x", "0");
|
|
|
|
const textLength1 = tspan.node().getComputedTextLength();
|
|
|
|
const textLength2 = tspan.node().getComputedTextLength();
|
|
|
|
textLength = Math.max(textLength1, textLength2);
|
|
|
|
} else {
|
|
|
|
self.text(titleText);
|
|
|
|
textLength = self.node().getComputedTextLength();
|
|
|
|
}
|
|
|
|
// scale according to text length:
|
|
|
|
if (textLength > getSizeForNode(node) * 2) {
|
|
|
|
self.attr('transform', 'scale(' + getSizeForNode(node) * 2 / textLength / 1.05 + ')');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return group;
|
2021-03-30 14:30:50 +00:00
|
|
|
});
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
// let linkText;
|
|
|
|
// let linkLine;
|
|
|
|
link = link
|
|
|
|
.data(graph.links)
|
|
|
|
.join(
|
|
|
|
enter => {
|
|
|
|
console.log(enter);
|
|
|
|
let group = enter.append("g").attr("class", "link");
|
|
|
|
group.append("line").attr("marker-end", "url(#arrowHead)").attr('id', (d,i) => 'linkid_' + i);
|
|
|
|
group.filter((l) => l.name != "City").append("text").attr("class", "labelText").text(function (l) {
|
|
|
|
return l.name;
|
|
|
|
});
|
|
|
|
// group.append("text")
|
|
|
|
// .attr("class", "labelText")
|
|
|
|
// .attr("dx", 20)
|
|
|
|
// .attr("dy", 0)
|
|
|
|
// .style("fill", "red")
|
|
|
|
// .append("textPath")
|
|
|
|
// .attr("xlink:href", function (d, i) { return "#linkid_" + i; })
|
|
|
|
// .attr("startOffset","50%")
|
|
|
|
// .text((d,i) => d.name );
|
|
|
|
return group;
|
|
|
|
}
|
|
|
|
)
|
|
|
|
;
|
|
|
|
// let linkEnter = link.enter()
|
|
|
|
// let linkLine = link.selectAll('line');
|
|
|
|
// let newLinks = link.enter().append("g").attr("class","link");
|
|
|
|
// newLinks.append("line").attr("marker-end", "url(#arrowHead)");
|
|
|
|
// newLinks.filter((l) => l.name != "City").append("text").text(function (l) {
|
|
|
|
// return l.name;
|
|
|
|
// });
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
// link.exit().remove();
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
// let linkLine = link.selectAll('line');
|
|
|
|
// let linkText = link.selectAll('text');
|
|
|
|
// // }) function(update){ return update }, function(exit) {
|
|
|
|
// // exit.remove();
|
|
|
|
// // })
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
// ;
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
// console.log(link, linkText, linkLine)
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
simulation.nodes(graph.nodes);
|
|
|
|
simulation.on("tick", () => {
|
|
|
|
|
|
|
|
// console.log('t', link._groups[0].length);
|
2021-03-30 14:30:50 +00:00
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
link.each(function (d) {
|
|
|
|
let sourceX, targetX, midX, dx, dy, angle;
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
// This mess makes the arrows exactly perfect.
|
|
|
|
// thanks to http://bl.ocks.org/curran/9b73eb564c1c8a3d8f3ab207de364bf4
|
|
|
|
if (d.source.x < d.target.x) {
|
|
|
|
sourceX = d.source.x;
|
|
|
|
targetX = d.target.x;
|
|
|
|
} else if (d.target.x < d.source.x) {
|
|
|
|
targetX = d.target.x;
|
|
|
|
sourceX = d.source.x;
|
|
|
|
} else if (d.target.isCircle) {
|
|
|
|
targetX = sourceX = d.target.x;
|
|
|
|
} else if (d.source.isCircle) {
|
|
|
|
targetX = sourceX = d.source.x;
|
|
|
|
} else {
|
|
|
|
midX = (d.source.x + d.target.x) / 2;
|
|
|
|
if (midX > d.target.x) {
|
|
|
|
midX = d.target.x;
|
|
|
|
} else if (midX > d.source.x) {
|
|
|
|
midX = d.source.x;
|
|
|
|
} else if (midX < d.target.x) {
|
|
|
|
midX = d.target.x;
|
|
|
|
} else if (midX < d.source.x) {
|
|
|
|
midX = d.source.x;
|
|
|
|
}
|
|
|
|
targetX = sourceX = midX;
|
|
|
|
}
|
|
|
|
|
|
|
|
dx = targetX - sourceX;
|
|
|
|
dy = d.target.y - d.source.y;
|
|
|
|
angle = Math.atan2(dx, dy);
|
|
|
|
|
|
|
|
/* DISABLED
|
|
|
|
srcSize = (typeof nodePositions[d.source.index] != 'undefined') ? selectedNodeSize : nodeSize;
|
|
|
|
tgtSize = (typeof nodePositions[d.target.index] != 'undefined') ? selectedNodeSize : nodeSize;
|
|
|
|
*/
|
|
|
|
var srcSize = getSizeForNode(d.source);
|
|
|
|
var tgtSize = getSizeForNode(d.target);
|
|
|
|
|
|
|
|
// Compute the line endpoint such that the arrow
|
|
|
|
// is touching the edge of the node rectangle perfectly.
|
|
|
|
d.sourceX = sourceX + Math.sin(angle) * srcSize;
|
|
|
|
d.targetX = targetX - Math.sin(angle) * tgtSize;
|
|
|
|
d.sourceY = d.source.y + Math.cos(angle) * srcSize;
|
|
|
|
d.targetY = d.target.y - Math.cos(angle) * tgtSize;
|
2021-04-16 09:15:57 +00:00
|
|
|
|
|
|
|
let rel = d3.select(this);
|
|
|
|
rel.select('line')
|
|
|
|
.attr("x1", d.sourceX)
|
|
|
|
.attr("y1", d.sourceY)
|
|
|
|
.attr("x2", d.targetX)
|
|
|
|
.attr("y2", d.targetY)
|
|
|
|
|
|
|
|
rel.select('text')
|
|
|
|
.attr("transform", function (d) {
|
|
|
|
const dx = (d.target.x - d.source.x) / 2;
|
|
|
|
const dy = (d.target.y - d.source.y) / 2;
|
|
|
|
const x = d.source.x + dx;
|
|
|
|
const y = d.source.y + dy;
|
|
|
|
const deg = Math.atan(dy / dx) * 180 / Math.PI;
|
|
|
|
// if dx/dy == 0/0 -> deg == NaN
|
|
|
|
if (isNaN(deg)) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
// return "";
|
|
|
|
return "translate(" + x + " " + y + ") rotate(" + (CONFIG.labels.rotate ? deg : 0) + ")";
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
})/*.attr("x1", function (d) {
|
2021-03-30 14:30:50 +00:00
|
|
|
return d.sourceX;
|
|
|
|
}).attr("y1", function (d) {
|
|
|
|
return d.sourceY;
|
|
|
|
}).attr("x2", function (d) {
|
|
|
|
return d.targetX;
|
|
|
|
}).attr("y2", function (d) {
|
|
|
|
return d.targetY;
|
2021-04-16 09:15:57 +00:00
|
|
|
});*/
|
|
|
|
// linkText
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
node
|
|
|
|
.attr("transform", d => `translate(${d.x}, ${d.y})`);
|
|
|
|
});
|
|
|
|
|
2021-03-31 15:02:00 +00:00
|
|
|
simulation.force("link")
|
|
|
|
.links(graph.links);
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
// simulate the first bit without drawing, so we don't have the 'jumping' graph in the beginning
|
|
|
|
if (CONFIG.preSimulate) {
|
|
|
|
for (var i = 0, n = Math.ceil(Math.log(simulation.alphaMin()) / Math.log(1 - simulation.alphaDecay())); i < n; ++i) {
|
|
|
|
simulation.tick();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// filter function
|
|
|
|
function filter() {
|
|
|
|
// add and remove nodes from data based on type filters
|
|
|
|
store.nodes.forEach(function (n) {
|
|
|
|
const cats = getCategories(n);
|
|
|
|
if (cats.every(c => filteredCategories.includes(c))) {
|
|
|
|
// hide
|
|
|
|
graph.nodes.forEach(function (d, i) {
|
|
|
|
if (n['@id'] === d['@id']) {
|
|
|
|
graph.nodes.splice(i, 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// add back
|
|
|
|
if (!graph.nodes.includes(n)) {
|
|
|
|
graph.nodes.push(n);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
// add and remove links from data based on availability of nodes
|
|
|
|
let checkOn;
|
|
|
|
if(typeof store.links[0].source !== 'object') {
|
|
|
|
checkOn = graph.nodes.map(n => n['@id'])
|
|
|
|
} else {
|
|
|
|
checkOn = graph.nodes
|
|
|
|
}
|
|
|
|
|
|
|
|
store.links.forEach(function (l) {
|
|
|
|
|
|
|
|
if (!checkOn.includes(l.source) || !checkOn.includes(l.target)) {
|
|
|
|
// hide
|
|
|
|
console.log('hide!')
|
|
|
|
graph.links.forEach(function (d, i) {
|
|
|
|
if (l === d) {
|
|
|
|
graph.links.splice(i, 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
} else if (!graph.links.includes(l)) {
|
|
|
|
graph.links[graph.links.length] = l;
|
|
|
|
// console.log('add', l, graph.links)
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
// console.log(graph.links)
|
2021-03-30 14:30:50 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const drag = simulation => {
|
|
|
|
|
2021-03-31 14:24:46 +00:00
|
|
|
let ignoreDrag = false;
|
2021-03-30 14:30:50 +00:00
|
|
|
function dragstarted(event) {
|
2021-03-31 14:24:46 +00:00
|
|
|
if (event.subject.fx) ignoreDrag = true;
|
2021-03-30 14:30:50 +00:00
|
|
|
if (!event.active) simulation.alphaTarget(0.3).restart();
|
|
|
|
event.subject.fx = event.subject.x;
|
|
|
|
event.subject.fy = event.subject.y;
|
|
|
|
}
|
|
|
|
|
|
|
|
function dragged(event) {
|
2021-03-31 14:24:46 +00:00
|
|
|
if (ignoreDrag) return;
|
2021-03-30 14:30:50 +00:00
|
|
|
event.subject.fx = event.x;
|
|
|
|
event.subject.fy = event.y;
|
|
|
|
}
|
|
|
|
|
|
|
|
function dragended(event) {
|
2021-03-31 14:24:46 +00:00
|
|
|
if (ignoreDrag) {
|
|
|
|
ignoreDrag = false;
|
|
|
|
return;
|
|
|
|
}
|
2021-03-30 14:30:50 +00:00
|
|
|
if (!event.active) simulation.alphaTarget(0);
|
|
|
|
event.subject.fx = null;
|
|
|
|
event.subject.fy = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return d3.drag()
|
|
|
|
.on("start", dragstarted)
|
|
|
|
.on("drag", dragged)
|
|
|
|
.on("end", dragended);
|
|
|
|
};
|
|
|
|
|
|
|
|
function selectNode(evt, node, d3Node) {
|
|
|
|
console.log(evt, node, d3Node);
|
|
|
|
document.querySelectorAll('svg .node').forEach(n => n.classList.remove('selected'));
|
|
|
|
d3Node._groups[0][node.index].classList.add('selected');
|
|
|
|
|
|
|
|
infoEl = document.getElementById('nodeInfo');
|
|
|
|
infoEl.classList.remove('hidden');
|
|
|
|
|
|
|
|
const url = getUrl(node);
|
|
|
|
const hrefEl = infoEl.querySelector('.nodeHref');
|
|
|
|
hrefEl.textContent = getTitle(node);
|
|
|
|
hrefEl.setAttribute('href', url);
|
|
|
|
infoEl.querySelector('.nodeContents').src = url;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-04-16 09:15:57 +00:00
|
|
|
const closeEl = document.getElementById('closeInfo');
|
|
|
|
if(closeEl) {
|
|
|
|
|
|
|
|
document.getElementById('closeInfo').addEventListener('click', (evt) => {
|
|
|
|
document.querySelectorAll('svg .node').forEach(n => n.classList.remove('selected'));
|
|
|
|
document.getElementById('nodeInfo').classList.add('hidden');
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|