From 1f1ba5ce8987e8b1a642c82da5a0fe6c9a7ea743 Mon Sep 17 00:00:00 2001
From: esikkala <esko.ikkala@aalto.fi>
Date: Mon, 10 Jan 2022 15:38:04 +0200
Subject: [PATCH] Update network config handling

---
 .../facet_results/ResultClassRoute.js         |  12 +-
 .../Cytoscape.js/NetworkConfig.js             | 137 ++----------------
 .../Cytoscape.js/NetworkTools.js              |  59 ++++++++
 3 files changed, 79 insertions(+), 129 deletions(-)
 create mode 100644 src/client/library_configs/Cytoscape.js/NetworkTools.js

diff --git a/src/client/components/facet_results/ResultClassRoute.js b/src/client/components/facet_results/ResultClassRoute.js
index 8b15782a..3a8dd378 100644
--- a/src/client/components/facet_results/ResultClassRoute.js
+++ b/src/client/components/facet_results/ResultClassRoute.js
@@ -296,10 +296,10 @@ const ResultClassRoute = props => {
         pageType = 'facetResults',
         limit,
         optimize,
-        style,
         fitLayout = false,
+        style = null,
         layout = null,
-        preprocess
+        preprocess = null
       } = resultClassConfig
       let networkProps = {
         portalConfig,
@@ -314,12 +314,10 @@ const ResultClassRoute = props => {
         layoutConfig: props.layoutConfig,
         limit,
         optimize,
-        style: networkConfig[style],
         fitLayout,
-        preprocess: networkConfig[preprocess]
-      }
-      if (layout) {
-        networkProps.layout = networkConfig[layout]
+        ...(style && { style }),
+        ...(layout && { layout }),
+        ...(preprocess && { preprocess: networkConfig[preprocess] })
       }
       if (pageType === 'facetResults') {
         networkProps = {
diff --git a/src/client/library_configs/Cytoscape.js/NetworkConfig.js b/src/client/library_configs/Cytoscape.js/NetworkConfig.js
index d3044ad9..a5a3e823 100644
--- a/src/client/library_configs/Cytoscape.js/NetworkConfig.js
+++ b/src/client/library_configs/Cytoscape.js/NetworkConfig.js
@@ -1,65 +1,9 @@
-const maxEdgeWidth = 8
-
-const constrainWidth = width => {
-  return (width ? (width < maxEdgeWidth ? width : maxEdgeWidth) : 1)
-}
-
-// https://js.cytoscape.org/#style
-export const cytoscapeStyle = [
-  {
-    selector: 'node',
-    style: {
-      shape: 'ellipse',
-      'font-size': '12',
-      'background-color': ele => ele.data('color') || '#666',
-      label: ' data(prefLabel)',
-      height: ele => (ele.data('size') || 16 / (ele.data('distance') + 1) || '16px'),
-      width: ele => (ele.data('size') || 16 / (ele.data('distance') + 1) || '16px')
-    }
-  },
-  {
-    selector: 'edge',
-    style: {
-      width: ele => constrainWidth(ele.data('weight')),
-      'line-color': ele => ele.data('color') || '#BBB',
-      'curve-style': 'bezier',
-      // content: ' data(prefLabel) ',
-      'target-arrow-shape': 'triangle',
-      'target-arrow-color': '#999',
-      color: '#555',
-      'font-size': '6',
-      'text-valign': 'top',
-      'text-halign': 'center',
-      'edge-text-rotation': 'autorotate',
-      'text-background-opacity': 1,
-      'text-background-color': 'white',
-      'text-background-shape': 'roundrectangle'
-    }
-  }
-]
-
-// https://js.cytoscape.org/#layouts
-export const coseLayout = {
-  name: 'cose',
-  idealEdgeLength: 100,
-  nodeOverlap: 20,
-  refresh: 20,
-  fit: true,
-  padding: 30,
-  randomize: false,
-  componentSpacing: 100,
-  nodeRepulsion: 400000,
-  edgeElasticity: 100,
-  nestingFactor: 5,
-  gravity: 80,
-  numIter: 1347,
-  initialTemp: 200,
-  coolingFactor: 0.95,
-  minTemp: 1.0
-}
+import { constrainValue, ValueScaler, ColorScaler } from './NetworkTools'
 
 //  preprocess by pagerank
-export const preprocess = elements => {
+export const preprocessPagerank = elements => {
+  const maxEdgeWidth = 8
+
   //  edges
   let arr = elements.edges.map(ele => ele.data.weight)
 
@@ -67,7 +11,6 @@ export const preprocess = elements => {
   let res = (new ValueScaler(1.0, maxEdgeWidth)).fitTransform(arr)
   elements.edges.forEach((ele, i) => { ele.data.weight = res[i] })
 
-  // console.log(elements.nodes)
   // nodes
   arr = elements.nodes.map(ele => ele.data.pagerank)
 
@@ -75,7 +18,7 @@ export const preprocess = elements => {
   res = (new ColorScaler('6px', '24px')).fitTransform(arr)
   elements.nodes.forEach((ele, i) => { ele.data.size = res[i] })
 
-  //  label size
+  // node label font size
   res = (new ValueScaler(8, 12)).fitTransform(arr)
   elements.nodes.forEach((ele, i) => { ele.data.font_size = res[i] })
 
@@ -91,7 +34,7 @@ export const preprocess = elements => {
 
 //  preprocessRelationNetwork
 export const preprocessRelationNetwork = elements => {
-  preprocess(elements)
+  preprocessPagerank(elements)
 
   // nodes
   const arr = elements.nodes.map(ele => ele.data.distance)
@@ -100,7 +43,7 @@ export const preprocessRelationNetwork = elements => {
   let res = (new ColorScaler('24px', '6px')).fitTransform(arr)
   elements.nodes.forEach((ele, i) => { ele.data.size = res[i] })
 
-  //  label size
+  // node label font size
   res = (new ValueScaler(12, 8)).fitTransform(arr)
   elements.nodes.forEach((ele, i) => { ele.data.font_size = res[i] })
 }
@@ -126,7 +69,7 @@ export const preprocessConnections = elements => {
   res = (new ColorScaler('24px', '6px')).fitTransform(arr)
   elements.nodes.forEach((ele, i) => { ele.data.size = res[i] })
 
-  // label size
+  // node label font size
   res = (new ValueScaler(12, 8)).fitTransform(arr)
   elements.nodes.forEach((ele, i) => { ele.data.font_size = res[i] })
 
@@ -144,6 +87,8 @@ export const preprocessConnections = elements => {
 
 //  preprocess by ego node distance
 export const preprocessDistance = elements => {
+  const maxEdgeWidth = 8
+
   //  edges
   let arr = elements.edges.map(ele => ele.data.weight)
 
@@ -179,7 +124,8 @@ export const preprocessPointCloud = elements => {
 }
 
 export const preprocessFamilytree = elements => {
-  // console.log(elements.nodes)
+  const maxEdgeWidth = 8
+
   const nodes = elements.nodes.map(ob => {
     if (ob.data.distance === 0) {
       ob.data.size = '24px'
@@ -195,60 +141,7 @@ export const preprocessFamilytree = elements => {
     }
   })
   elements.nodes = nodes
-}
-
-class ValueScaler {
-  a;
-  b;
-  constructor (low, high) {
-    this.low = low
-    this.high = high
-  }
-
-  fit (vals) {
-    const valmin = Math.min(...vals)
-    const valmax = Math.max(...vals)
-    if (valmax === valmin) {
-      this.a = 0.0
-    } else {
-      this.a = (this.high - this.low) / (valmax - valmin)
-    }
-    this.b = this.low - valmin * this.a
-  }
-
-  transform (vals) {
-    return vals.map(x => { return x * this.a + this.b })
-  }
-
-  fitTransform (vals) {
-    this.fit(vals)
-    return this.transform(vals)
-  }
-}
-
-class ColorScaler extends ValueScaler {
-  col1;
-  col2;
-  constructor (low, high) {
-    super(0.0, 1.0)
-    this.col1 = low
-    this.col2 = high
-  }
-
-  // super.fit(vals)
-
-  _process (s0, s1, r) {
-    const x0 = parseInt(s0)
-    const x1 = parseInt(s1)
-    if (isNaN(x0) || isNaN(x1)) return s0
-    return Math.floor(x0 + (x1 - x0) * r)
-  }
-
-  transform (vals) {
-    const s1 = this.col1.split(/(\d+)/)
-    const s2 = this.col2.split(/(\d+)/)
-    const _vals01 = vals.map(x => { return x * this.a + this.b })
-
-    return _vals01.map(v => s1.map((s, i) => this._process(s, s2[i], v)).join(''))
-  }
+  elements.edges.forEach(edge => {
+    edge.data.weight = constrainValue({ value: edge.data.weight, maxValue: maxEdgeWidth })
+  })
 }
diff --git a/src/client/library_configs/Cytoscape.js/NetworkTools.js b/src/client/library_configs/Cytoscape.js/NetworkTools.js
new file mode 100644
index 00000000..ceba0d42
--- /dev/null
+++ b/src/client/library_configs/Cytoscape.js/NetworkTools.js
@@ -0,0 +1,59 @@
+export const constrainValue = ({ value, maxValue, defaultValue = 1 }) => {
+  return (value ? (value < maxValue ? value : maxValue) : defaultValue)
+}
+
+export class ValueScaler {
+    a;
+    b;
+    constructor (low, high) {
+      this.low = low
+      this.high = high
+    }
+
+    fit (vals) {
+      const valmin = Math.min(...vals)
+      const valmax = Math.max(...vals)
+      if (valmax === valmin) {
+        this.a = 0.0
+      } else {
+        this.a = (this.high - this.low) / (valmax - valmin)
+      }
+      this.b = this.low - valmin * this.a
+    }
+
+    transform (vals) {
+      return vals.map(x => { return x * this.a + this.b })
+    }
+
+    fitTransform (vals) {
+      this.fit(vals)
+      return this.transform(vals)
+    }
+}
+
+export class ColorScaler extends ValueScaler {
+    col1;
+    col2;
+    constructor (low, high) {
+      super(0.0, 1.0)
+      this.col1 = low
+      this.col2 = high
+    }
+
+    // super.fit(vals)
+
+    _process (s0, s1, r) {
+      const x0 = parseInt(s0)
+      const x1 = parseInt(s1)
+      if (isNaN(x0) || isNaN(x1)) return s0
+      return Math.floor(x0 + (x1 - x0) * r)
+    }
+
+    transform (vals) {
+      const s1 = this.col1.split(/(\d+)/)
+      const s2 = this.col2.split(/(\d+)/)
+      const _vals01 = vals.map(x => { return x * this.a + this.b })
+
+      return _vals01.map(v => s1.map((s, i) => this._process(s, s2[i], v)).join(''))
+    }
+}
-- 
GitLab