New Case Study:See how Anthropic automated 95% of dependency reviews with Socket.Learn More
Socket
Sign inDemoInstall
Socket

@datagrok-libraries/ml

Package Overview
Dependencies
Maintainers
0
Versions
156
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@datagrok-libraries/ml - npm Package Compare versions

Comparing version 6.7.6 to 6.8.0

src/MCL/mcl-viewer.d.ts

4

CHANGELOG.md
# ml changelog
## 6.8.0 (2025-01-06)
Refork MCL Viewer
## 6.7.4 (2024-10-08)

@@ -4,0 +8,0 @@

4

package.json

@@ -11,3 +11,3 @@ {

"friendlyName": "Datagrok ML library",
"version": "6.7.6",
"version": "6.8.0",
"description": "Machine learning supporting utilities",

@@ -21,3 +21,3 @@ "dependencies": {

"cash-dom": "^8.1.5",
"datagrok-api": "^1.21.1",
"datagrok-api": "^1.22.0",
"dayjs": "^1.11.13",

@@ -24,0 +24,0 @@ "fastest-levenshtein": "^1.0.12",

import * as DG from 'datagrok-api/dg';
import { KnownMetrics } from '../typed-metrics';
import { DistanceAggregationMethod } from '../distance-matrix/types';
import * as rxjs from 'rxjs';
export type MCLClusterViewerResult = {

@@ -11,4 +12,19 @@ sc: DG.ScatterPlotViewer;

connectivityCol: DG.Column;
i: ArrayLike<number>;
j: ArrayLike<number>;
};
export declare function markovCluster(df: DG.DataFrame, cols: DG.Column[], metrics: KnownMetrics[], weights: number[], aggregationMethod: DistanceAggregationMethod, preprocessingFuncs: (DG.Func | null | undefined)[], preprocessingFuncArgs: any[], threshold?: number, maxIterations?: number, useWebGPU?: boolean, inflate?: number, minClusterSize?: number, scp?: DG.ScatterPlotViewer): Promise<undefined | MCLClusterViewerResult>;
export declare class SCLinesRenderer {
sc: DG.ScatterPlotViewer;
from: ArrayLike<number>;
to: ArrayLike<number>;
shortLineThreshold: number;
width: number;
color: string;
private renderFlag;
renderSub: rxjs.Subscription;
constructor(sc: DG.ScatterPlotViewer, from: ArrayLike<number>, to: ArrayLike<number>, shortLineThreshold: number, width: number, color: string);
render(): void;
destroy(): void;
}
//# sourceMappingURL=clustering-view.d.ts.map

@@ -5,3 +5,2 @@ import * as grok from 'datagrok-api/grok';

import { createMCLWorker } from './index';
import { ScatterPlotCurrentLineStyle, ScatterPlotLinesRenderer } from '@datagrok-libraries/utils/src/render-lines-on-sp';
export async function markovCluster(df, cols, metrics, weights, aggregationMethod, preprocessingFuncs, preprocessingFuncArgs, threshold = 80, maxIterations = 10, useWebGPU = false, inflate = 2, minClusterSize = 5, scp) {

@@ -14,4 +13,8 @@ const scatterPlotProps = {

};
const tv = grok.shell.tableView(df.name) ?? grok.shell.addTableView(df);
const sc = scp ?? tv.scatterPlot({ ...scatterPlotProps, title: 'MCL' });
let tv = null;
let sc = scp;
if (!sc) {
tv = grok.shell.tableView(df.name) ?? grok.shell.addTableView(df);
sc = tv.scatterPlot({ ...scatterPlotProps, title: 'MCL' });
}
ui.setUpdateIndicator(sc.root, true);

@@ -120,15 +123,8 @@ const distanceFnArgs = [];

}
const _scLines = new ScatterPlotLinesRenderer(sc, emberdXColName, emberdYColName, { from: new Uint32Array(filteredIs), to: new Uint32Array(filteredJs),
drawArrows: false, opacity: 0.3, skipMultiLineCalculation: true,
skipShortLines: true, skipMouseOverDetection: true, shortLineThreshold: 6, width: 0.75, color: '128,128,128' }, ScatterPlotCurrentLineStyle.none);
// _scLines.lineClicked.subscribe((args) => {
// const id = args.id;
// args.event.preventDefault();
// args.event.stopImmediatePropagation();
// if ((id ?? -1) === -1)
// return;
// const i = _scLines.lines.from[id];
// const j = _scLines.lines.to[id];
// df.selection.init((index) => index === i || index === j);
// });
// const _scLines = new ScatterPlotLinesRenderer(sc, emberdXColName, emberdYColName,
// {from: new Uint32Array(filteredIs) as any, to: new Uint32Array(filteredJs) as any,
// drawArrows: false, opacity: 0.3, skipMultiLineCalculation: true,
// skipShortLines: true, skipMouseOverDetection: true, shortLineThreshold: 6, width: 0.75, color: '128,128,128'},
// ScatterPlotCurrentLineStyle.none);
//const _scLines = new SCLinesRenderer(sc, filteredIs, filteredJs, 6, 0.75, '128,128,128');
ui.setUpdateIndicator(sc.root, false);

@@ -138,4 +134,67 @@ // sc.close();

// tv.addViewer(scLinesViewer);
return { sc, embedXCol, embedYCol, clusterCol, clusterCounterCol, connectivityCol };
return { sc, embedXCol, embedYCol, clusterCol, clusterCounterCol, connectivityCol, i: filteredIs, j: filteredJs };
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"clustering-view.js","sourceRoot":"","sources":["clustering-view.ts"],"names":[],"mappings":"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","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\n\nimport {createMCLWorker} from './index';\nimport {ScatterPlotCurrentLineStyle, ScatterPlotLinesRenderer} from '@datagrok-libraries/utils/src/render-lines-on-sp';\nimport {KnownMetrics} from '../typed-metrics';\nimport {DistanceAggregationMethod} from '../distance-matrix/types';\nimport {PreprocessFunctionReturnType} from '../functionEditors/dimensionality-reduction-editor';\nimport {Options} from '@datagrok-libraries/utils/src/type-declarations';\n\n\nexport type MCLClusterViewerResult = {\n  sc: DG.ScatterPlotViewer;\n  embedXCol: DG.Column;\n  embedYCol: DG.Column;\n  clusterCol: DG.Column;\n  clusterCounterCol: DG.Column;\n  connectivityCol: DG.Column;\n}\n\nexport async function markovCluster(\n  df: DG.DataFrame, cols: DG.Column[], metrics: KnownMetrics[],\n  weights: number[], aggregationMethod: DistanceAggregationMethod, preprocessingFuncs: (DG.Func | null | undefined)[],\n  preprocessingFuncArgs: any[], threshold: number = 80, maxIterations: number = 10,\n  useWebGPU: boolean = false, inflate: number = 2, minClusterSize: number = 5, scp?: DG.ScatterPlotViewer\n): Promise<undefined | MCLClusterViewerResult> {\n  const scatterPlotProps = {\n    showXAxis: false,\n    showYAxis: false,\n    showXSelector: false,\n    showYSelector: false,\n  };\n  const tv = grok.shell.tableView(df.name) ?? grok.shell.addTableView(df);\n\n  const sc = scp ?? tv.scatterPlot({...scatterPlotProps, title: 'MCL'});\n\n  ui.setUpdateIndicator(sc.root, true);\n  const distanceFnArgs: Options[] = [];\n  const encodedColEntries: PreprocessFunctionReturnType[] = [];\n  for (let i = 0; i < preprocessingFuncs.length; ++i) {\n    const pf = preprocessingFuncs[i];\n    if (pf) {\n      const colInputName = pf.inputs[0].name;\n      const metricInputName = pf.inputs[1].name;\n      const {entries, options}: PreprocessFunctionReturnType =\n                await pf.apply({[colInputName]: cols[i], [metricInputName]: metrics[i],\n                  ...(preprocessingFuncArgs[i] ?? {})});\n      encodedColEntries.push({entries, options});\n      distanceFnArgs.push(options ?? {});\n    } else {\n      const entries = cols[i].toList();\n      const options = {};\n      encodedColEntries.push({entries, options});\n      distanceFnArgs.push(options);\n    }\n  }\n\n  const mclWorker = createMCLWorker(encodedColEntries.map((it) => it.entries),\n    threshold, weights, aggregationMethod, metrics, distanceFnArgs, maxIterations, useWebGPU, inflate);\n\n  const terminateSub = grok.events.onViewerClosed.subscribe((args) => {\n    if (args.args.viewer?.props?.title === sc.props.title && sc.type === args.args?.viewer?.type) {\n      terminateSub.unsubscribe();\n      mclWorker.terminate();\n    }\n  });\n  const res = await mclWorker.promise;\n  if (!res)\n    return;\n  const clusterColName = df.columns.getUnusedName('Cluster (MCL)');\n  const emberdXColName = df.columns.getUnusedName('EmbedX (MCL)');\n  const emberdYColName = df.columns.getUnusedName('EmbedY (MCL)');\n  const clustersCounter: {[_: number]: number} = {};\n  res.clusters.forEach((c) => {\n    if (!clustersCounter[c]) clustersCounter[c] = 0;\n    clustersCounter[c]++;\n  });\n  const connectivity = new Uint32Array(res.embedX.length);\n  for (let i = 0; i < res.is.length; i++) {\n    connectivity[res.is[i]]++;\n    connectivity[res.js[i]]++;\n  }\n  const clusterCounterColName = df.columns.getUnusedName('Cluster size (MCL)');\n  const connectivityColName = df.columns.getUnusedName('Connectivity (MCL)');\n\n  const embedXCol = df.columns.addNewFloat(emberdXColName);\n  embedXCol.init((i) => res.embedX[i]);\n  const embedYCol = df.columns.addNewFloat(emberdYColName);\n  embedYCol.init((i) => res.embedY[i]);\n  const clusterCol = df.columns.addNewString(clusterColName);\n  clusterCol.init((i) => clustersCounter[res.clusters[i]] >= minClusterSize ? res.clusters[i].toString() : '-1');\n  const catColorObj = {'-1': DG.Color.setAlpha(DG.Color.lightBlue, 100)};\n  clusterCol.setTag(DG.TAGS.COLOR_CODING_CATEGORICAL,\n    JSON.stringify(catColorObj));\n  clusterCol.temp[DG.TAGS.COLOR_CODING_CATEGORICAL] = catColorObj;\n  // clusterCol.setCategoryOrder(Array.from(new Set(res.clusters)).sort((a, b) => a - b).map((it) => it.toString()));\n\n  const clusterCounterCol = df.columns.addNewInt(clusterCounterColName);\n  clusterCounterCol.init((i) => clustersCounter[res.clusters[i]]);\n  const connectivityCol = df.columns.addNewInt(connectivityColName);\n  connectivityCol.init((i) => connectivity[i]);\n\n  // df.columns.addNewFloat(emberdXColName).init((i) => res.embedX[i]);\n  // df.columns.addNewFloat(emberdYColName).init((i) => res.embedY[i]);\n  // df.columns.addNewInt(clusterCounterColName).init((i) => clustersCounter[res.clusters[i]]);\n  // df.columns.addNewString(clusterColName).init((i) => res.clusters[i].toString());\n  sc.props.xColumnName = emberdXColName;\n  sc.props.yColumnName = emberdYColName;\n  sc.props.colorColumnName = clusterColName;\n  sc.props.markerDefaultSize = 6;\n  terminateSub.unsubscribe();\n\n  // limit cluster interconnectivity to 20 lines\n\n  const maxInterClusterLinks = 20;\n  // I know pushing to array is slow, but it's not a big deal here as its max 100k pushes in most cases\n  const filteredIs: number[] = [];\n  const filteredJs: number[] = [];\n  const clusterInterconnectivity = new Map<number, Map<number, number>>();\n  for (let i = 0; i < res.is.length; i++) {\n    let from = res.clusters[res.is[i]];\n    let to = res.clusters[res.js[i]];\n    if (from === to) {\n      filteredIs.push(res.is[i]);\n      filteredJs.push(res.js[i]);\n      continue;\n    }\n    if (from > to) {\n      const tmp = from;\n      from = to;\n      to = tmp;\n    }\n\n    let fromMap = clusterInterconnectivity.get(from);\n    if (!fromMap) {\n      fromMap = new Map<number, number>();\n      clusterInterconnectivity.set(from, fromMap);\n    }\n    let count = fromMap.get(to);\n    if (!count)\n      count = 0;\n\n    if (count >= maxInterClusterLinks)\n      continue;\n\n    count++;\n    fromMap.set(to, count);\n    filteredIs.push(res.is[i]);\n    filteredJs.push(res.js[i]);\n  }\n\n\n  const _scLines = new ScatterPlotLinesRenderer(sc, emberdXColName, emberdYColName,\n    {from: new Uint32Array(filteredIs) as any, to: new Uint32Array(filteredJs) as any,\n      drawArrows: false, opacity: 0.3, skipMultiLineCalculation: true,\n      skipShortLines: true, skipMouseOverDetection: true, shortLineThreshold: 6, width: 0.75, color: '128,128,128'},\n    ScatterPlotCurrentLineStyle.none);\n\n  // _scLines.lineClicked.subscribe((args) => {\n  //   const id = args.id;\n  //   args.event.preventDefault();\n  //   args.event.stopImmediatePropagation();\n  //   if ((id ?? -1) === -1)\n  //     return;\n  //   const i = _scLines.lines.from[id];\n  //   const j = _scLines.lines.to[id];\n  //   df.selection.init((index) => index === i || index === j);\n  // });\n\n  ui.setUpdateIndicator(sc.root, false);\n  // sc.close();\n  // const scLinesViewer = new ScatterPlotWithLines(sc, res.is, res.js, emberdXColName, emberdYColName);\n  // tv.addViewer(scLinesViewer);\n  return {sc, embedXCol, embedYCol, clusterCol, clusterCounterCol, connectivityCol};\n}\n\n"]}
export class SCLinesRenderer {
constructor(sc, from, to, shortLineThreshold, width, color) {
this.sc = sc;
this.from = from;
this.to = to;
this.shortLineThreshold = shortLineThreshold;
this.width = width;
this.color = color;
this.renderFlag = false;
this.renderSub = DG.debounce(sc.onAfterDrawScene, 200).subscribe(() => {
if (this.renderFlag) {
this.renderFlag = false;
return;
}
this.renderFlag = true;
const tempSub = sc.onBeforeDrawScene.subscribe((_) => {
this.render();
tempSub.unsubscribe();
});
setTimeout(() => {
this.sc.invalidateCanvas();
// this.renderFlag = false;
});
});
sc.subs.push(this.renderSub);
}
render() {
const xCol = this.sc.dataFrame.getCol(this.sc.props.xColumnName);
const yCol = this.sc.dataFrame.getCol(this.sc.props.yColumnName);
const filter = this.sc.filter;
const positions = new Array(this.sc.dataFrame.rowCount).fill(null)
.map((_, i) => !xCol.isNone(i) && !yCol.isNone(i) && filter.get(i) ? this.sc.pointToScreen(i) : null);
const canvas = this.sc.canvas;
const ctx = canvas.getContext('2d');
if (!ctx)
return;
ctx.strokeStyle = `rgba(${this.color}, 0.3)`;
ctx.lineWidth = this.width;
const shortLineSquared = this.shortLineThreshold * this.shortLineThreshold;
for (let i = 0; i < this.from.length; i++) {
ctx.beginPath();
const from = this.from[i];
const to = this.to[i];
if (positions[from] && positions[to]) {
const fromPos = positions[from];
const toPos = positions[to];
const dx = toPos.x - fromPos.x;
const dy = toPos.y - fromPos.y;
const dist = dx * dx + dy * dy;
if (dist < shortLineSquared)
continue;
ctx.moveTo(fromPos.x, fromPos.y);
ctx.lineTo(toPos.x, toPos.y);
ctx.stroke();
ctx.closePath();
// interestingly, doing stroke for each line turns to be 20 times faster than doing stroke for all lines at once
}
}
}
destroy() {
this.renderSub.unsubscribe();
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"clustering-view.js","sourceRoot":"","sources":["clustering-view.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,IAAI,MAAM,mBAAmB,CAAC;AAC1C,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AACtC,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAEtC,OAAO,EAAC,eAAe,EAAC,MAAM,SAAS,CAAC;AAoBxC,MAAM,CAAC,KAAK,UAAU,aAAa,CACjC,EAAgB,EAAE,IAAiB,EAAE,OAAuB,EAC5D,OAAiB,EAAE,iBAA4C,EAAE,kBAAkD,EACnH,qBAA4B,EAAE,YAAoB,EAAE,EAAE,gBAAwB,EAAE,EAChF,YAAqB,KAAK,EAAE,UAAkB,CAAC,EAAE,iBAAyB,CAAC,EAAE,GAA0B;IAEvG,MAAM,gBAAgB,GAAG;QACvB,SAAS,EAAE,KAAK;QAChB,SAAS,EAAE,KAAK;QAChB,aAAa,EAAE,KAAK;QACpB,aAAa,EAAE,KAAK;KACrB,CAAC;IAEF,IAAI,EAAE,GAAwB,IAAI,CAAC;IACnC,IAAI,EAAE,GAAqC,GAAG,CAAC;IAC/C,IAAI,CAAC,EAAE,EAAE,CAAC;QACR,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;QAClE,EAAE,GAAG,EAAE,CAAC,WAAW,CAAC,EAAC,GAAG,gBAAgB,EAAE,KAAK,EAAE,KAAK,EAAC,CAAC,CAAC;IAC3D,CAAC;IAED,EAAE,CAAC,kBAAkB,CAAC,EAAE,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IACrC,MAAM,cAAc,GAAc,EAAE,CAAC;IACrC,MAAM,iBAAiB,GAAmC,EAAE,CAAC;IAC7D,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,kBAAkB,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE,CAAC;QACnD,MAAM,EAAE,GAAG,kBAAkB,CAAC,CAAC,CAAC,CAAC;QACjC,IAAI,EAAE,EAAE,CAAC;YACP,MAAM,YAAY,GAAG,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YACvC,MAAM,eAAe,GAAG,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YAC1C,MAAM,EAAC,OAAO,EAAE,OAAO,EAAC,GACd,MAAM,EAAE,CAAC,KAAK,CAAC,EAAC,CAAC,YAAY,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,eAAe,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;gBACpE,GAAG,CAAC,qBAAqB,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,EAAC,CAAC,CAAC;YAClD,iBAAiB,CAAC,IAAI,CAAC,EAAC,OAAO,EAAE,OAAO,EAAC,CAAC,CAAC;YAC3C,cAAc,CAAC,IAAI,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC;QACrC,CAAC;aAAM,CAAC;YACN,MAAM,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;YACjC,MAAM,OAAO,GAAG,EAAE,CAAC;YACnB,iBAAiB,CAAC,IAAI,CAAC,EAAC,OAAO,EAAE,OAAO,EAAC,CAAC,CAAC;YAC3C,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC/B,CAAC;IACH,CAAC;IAED,MAAM,SAAS,GAAG,eAAe,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,OAAO,CAAC,EACzE,SAAS,EAAE,OAAO,EAAE,iBAAiB,EAAE,OAAO,EAAE,cAAc,EAAE,aAAa,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;IAErG,MAAM,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;QACjE,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK,EAAE,KAAK,KAAK,EAAE,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;YAC7F,YAAY,CAAC,WAAW,EAAE,CAAC;YAC3B,SAAS,CAAC,SAAS,EAAE,CAAC;QACxB,CAAC;IACH,CAAC,CAAC,CAAC;IACH,MAAM,GAAG,GAAG,MAAM,SAAS,CAAC,OAAO,CAAC;IACpC,IAAI,CAAC,GAAG;QACN,OAAO;IACT,MAAM,cAAc,GAAG,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,eAAe,CAAC,CAAC;IACjE,MAAM,cAAc,GAAG,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC;IAChE,MAAM,cAAc,GAAG,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC;IAChE,MAAM,eAAe,GAA0B,EAAE,CAAC;IAClD,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;QACzB,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC;YAAE,eAAe,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QAChD,eAAe,CAAC,CAAC,CAAC,EAAE,CAAC;IACvB,CAAC,CAAC,CAAC;IACH,MAAM,YAAY,GAAG,IAAI,WAAW,CAAC,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IACxD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACvC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAC1B,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IAC5B,CAAC;IACD,MAAM,qBAAqB,GAAG,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,oBAAoB,CAAC,CAAC;IAC7E,MAAM,mBAAmB,GAAG,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,oBAAoB,CAAC,CAAC;IAE3E,MAAM,SAAS,GAAG,EAAE,CAAC,OAAO,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC;IACzD,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;IACrC,MAAM,SAAS,GAAG,EAAE,CAAC,OAAO,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC;IACzD,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;IACrC,MAAM,UAAU,GAAG,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC;IAC3D,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,cAAc,CAAC,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;IAC/G,MAAM,WAAW,GAAG,EAAC,IAAI,EAAE,EAAE,CAAC,KAAK,CAAC,QAAQ,CAAC,EAAE,CAAC,KAAK,CAAC,SAAS,EAAE,GAAG,CAAC,EAAC,CAAC;IACvE,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,wBAAwB,EAChD,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC;IAC/B,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,wBAAwB,CAAC,GAAG,WAAW,CAAC;IAChE,mHAAmH;IAEnH,MAAM,iBAAiB,GAAG,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;IACtE,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAChE,MAAM,eAAe,GAAG,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;IAClE,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;IAE7C,qEAAqE;IACrE,qEAAqE;IACrE,6FAA6F;IAC7F,mFAAmF;IACnF,EAAE,CAAC,KAAK,CAAC,WAAW,GAAG,cAAc,CAAC;IACtC,EAAE,CAAC,KAAK,CAAC,WAAW,GAAG,cAAc,CAAC;IACtC,EAAE,CAAC,KAAK,CAAC,eAAe,GAAG,cAAc,CAAC;IAC1C,EAAE,CAAC,KAAK,CAAC,iBAAiB,GAAG,CAAC,CAAC;IAC/B,YAAY,CAAC,WAAW,EAAE,CAAC;IAE3B,8CAA8C;IAE9C,MAAM,oBAAoB,GAAG,EAAE,CAAC;IAChC,qGAAqG;IACrG,MAAM,UAAU,GAAa,EAAE,CAAC;IAChC,MAAM,UAAU,GAAa,EAAE,CAAC;IAChC,MAAM,wBAAwB,GAAG,IAAI,GAAG,EAA+B,CAAC;IACxE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACvC,IAAI,IAAI,GAAG,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACnC,IAAI,EAAE,GAAG,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACjC,IAAI,IAAI,KAAK,EAAE,EAAE,CAAC;YAChB,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YAC3B,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YAC3B,SAAS;QACX,CAAC;QACD,IAAI,IAAI,GAAG,EAAE,EAAE,CAAC;YACd,MAAM,GAAG,GAAG,IAAI,CAAC;YACjB,IAAI,GAAG,EAAE,CAAC;YACV,EAAE,GAAG,GAAG,CAAC;QACX,CAAC;QAED,IAAI,OAAO,GAAG,wBAAwB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACjD,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,GAAG,IAAI,GAAG,EAAkB,CAAC;YACpC,wBAAwB,CAAC,GAAG,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;QAC9C,CAAC;QACD,IAAI,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC5B,IAAI,CAAC,KAAK;YACR,KAAK,GAAG,CAAC,CAAC;QAEZ,IAAI,KAAK,IAAI,oBAAoB;YAC/B,SAAS;QAEX,KAAK,EAAE,CAAC;QACR,OAAO,CAAC,GAAG,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QACvB,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAC3B,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7B,CAAC;IAGD,oFAAoF;IACpF,uFAAuF;IACvF,uEAAuE;IACvE,qHAAqH;IACrH,uCAAuC;IAEvC,2FAA2F;IAE3F,EAAE,CAAC,kBAAkB,CAAC,EAAE,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IACtC,cAAc;IACd,sGAAsG;IACtG,+BAA+B;IAE/B,OAAO,EAAC,EAAE,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,EAAE,iBAAiB,EAAE,eAAe,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,UAAU,EAAC,CAAC;AAClH,CAAC;AAGD,MAAM,OAAO,eAAe;IAG1B,YAAmB,EAAwB,EAAS,IAAuB,EAClE,EAAqB,EAAS,kBAA0B,EAAS,KAAa,EAAS,KAAa;QAD1F,OAAE,GAAF,EAAE,CAAsB;QAAS,SAAI,GAAJ,IAAI,CAAmB;QAClE,OAAE,GAAF,EAAE,CAAmB;QAAS,uBAAkB,GAAlB,kBAAkB,CAAQ;QAAS,UAAK,GAAL,KAAK,CAAQ;QAAS,UAAK,GAAL,KAAK,CAAQ;QAHrG,eAAU,GAAG,KAAK,CAAC;QAIzB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,gBAAgB,EAAE,GAAG,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACpE,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACpB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;gBACxB,OAAO;YACT,CAAC;YACD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;YACvB,MAAM,OAAO,GAAG,EAAE,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;gBACnD,IAAI,CAAC,MAAM,EAAE,CAAC;gBACd,OAAO,CAAC,WAAW,EAAE,CAAC;YACxB,CAAC,CAAC,CAAC;YACH,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;gBAC3B,2BAA2B;YAC7B,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC/B,CAAC;IAED,MAAM;QACJ,MAAM,IAAI,GAAG,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QACjE,MAAM,IAAI,GAAG,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QACjE,MAAM,MAAM,GAAG,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC;QAC9B,MAAM,SAAS,GAAG,IAAI,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;aAC/D,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACxG,MAAM,MAAM,GAAG,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC;QAC9B,MAAM,GAAG,GAAG,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACpC,IAAI,CAAC,GAAG;YACN,OAAO;QACT,GAAG,CAAC,WAAW,GAAG,QAAQ,IAAI,CAAC,KAAK,QAAQ,CAAC;QAC7C,GAAG,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC;QAC3B,MAAM,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,kBAAkB,CAAC;QAC3E,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAC1C,GAAG,CAAC,SAAS,EAAE,CAAC;YAChB,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC1B,MAAM,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACtB,IAAI,SAAS,CAAC,IAAI,CAAC,IAAI,SAAS,CAAC,EAAE,CAAC,EAAE,CAAC;gBACrC,MAAM,OAAO,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;gBAChC,MAAM,KAAK,GAAG,SAAS,CAAC,EAAE,CAAC,CAAC;gBAC5B,MAAM,EAAE,GAAG,KAAK,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;gBAC/B,MAAM,EAAE,GAAG,KAAK,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;gBAC/B,MAAM,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;gBAC/B,IAAI,IAAI,GAAG,gBAAgB;oBACzB,SAAS;gBACX,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC;gBACjC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC7B,GAAG,CAAC,MAAM,EAAE,CAAC;gBACb,GAAG,CAAC,SAAS,EAAE,CAAC;gBAChB,gHAAgH;YAClH,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO;QACL,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;IAC/B,CAAC;CACF","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\n\nimport {createMCLWorker} from './index';\nimport {ScatterPlotCurrentLineStyle, ScatterPlotLinesRenderer} from '@datagrok-libraries/utils/src/render-lines-on-sp';\nimport {KnownMetrics} from '../typed-metrics';\nimport {DistanceAggregationMethod} from '../distance-matrix/types';\nimport {PreprocessFunctionReturnType} from '../functionEditors/dimensionality-reduction-editor';\nimport {Options} from '@datagrok-libraries/utils/src/type-declarations';\nimport * as rxjs from 'rxjs';\n\n\nexport type MCLClusterViewerResult = {\n  sc: DG.ScatterPlotViewer;\n  embedXCol: DG.Column;\n  embedYCol: DG.Column;\n  clusterCol: DG.Column;\n  clusterCounterCol: DG.Column;\n  connectivityCol: DG.Column;\n  i: ArrayLike<number>;\n  j: ArrayLike<number>;\n}\n\nexport async function markovCluster(\n  df: DG.DataFrame, cols: DG.Column[], metrics: KnownMetrics[],\n  weights: number[], aggregationMethod: DistanceAggregationMethod, preprocessingFuncs: (DG.Func | null | undefined)[],\n  preprocessingFuncArgs: any[], threshold: number = 80, maxIterations: number = 10,\n  useWebGPU: boolean = false, inflate: number = 2, minClusterSize: number = 5, scp?: DG.ScatterPlotViewer\n): Promise<undefined | MCLClusterViewerResult> {\n  const scatterPlotProps = {\n    showXAxis: false,\n    showYAxis: false,\n    showXSelector: false,\n    showYSelector: false,\n  };\n\n  let tv: DG.TableView | null = null;\n  let sc: DG.ScatterPlotViewer | undefined = scp;\n  if (!sc) {\n    tv = grok.shell.tableView(df.name) ?? grok.shell.addTableView(df);\n    sc = tv.scatterPlot({...scatterPlotProps, title: 'MCL'});\n  }\n\n  ui.setUpdateIndicator(sc.root, true);\n  const distanceFnArgs: Options[] = [];\n  const encodedColEntries: PreprocessFunctionReturnType[] = [];\n  for (let i = 0; i < preprocessingFuncs.length; ++i) {\n    const pf = preprocessingFuncs[i];\n    if (pf) {\n      const colInputName = pf.inputs[0].name;\n      const metricInputName = pf.inputs[1].name;\n      const {entries, options}: PreprocessFunctionReturnType =\n                await pf.apply({[colInputName]: cols[i], [metricInputName]: metrics[i],\n                  ...(preprocessingFuncArgs[i] ?? {})});\n      encodedColEntries.push({entries, options});\n      distanceFnArgs.push(options ?? {});\n    } else {\n      const entries = cols[i].toList();\n      const options = {};\n      encodedColEntries.push({entries, options});\n      distanceFnArgs.push(options);\n    }\n  }\n\n  const mclWorker = createMCLWorker(encodedColEntries.map((it) => it.entries),\n    threshold, weights, aggregationMethod, metrics, distanceFnArgs, maxIterations, useWebGPU, inflate);\n\n  const terminateSub = grok.events.onViewerClosed.subscribe((args) => {\n    if (args.args.viewer?.props?.title === sc.props.title && sc.type === args.args?.viewer?.type) {\n      terminateSub.unsubscribe();\n      mclWorker.terminate();\n    }\n  });\n  const res = await mclWorker.promise;\n  if (!res)\n    return;\n  const clusterColName = df.columns.getUnusedName('Cluster (MCL)');\n  const emberdXColName = df.columns.getUnusedName('EmbedX (MCL)');\n  const emberdYColName = df.columns.getUnusedName('EmbedY (MCL)');\n  const clustersCounter: {[_: number]: number} = {};\n  res.clusters.forEach((c) => {\n    if (!clustersCounter[c]) clustersCounter[c] = 0;\n    clustersCounter[c]++;\n  });\n  const connectivity = new Uint32Array(res.embedX.length);\n  for (let i = 0; i < res.is.length; i++) {\n    connectivity[res.is[i]]++;\n    connectivity[res.js[i]]++;\n  }\n  const clusterCounterColName = df.columns.getUnusedName('Cluster size (MCL)');\n  const connectivityColName = df.columns.getUnusedName('Connectivity (MCL)');\n\n  const embedXCol = df.columns.addNewFloat(emberdXColName);\n  embedXCol.init((i) => res.embedX[i]);\n  const embedYCol = df.columns.addNewFloat(emberdYColName);\n  embedYCol.init((i) => res.embedY[i]);\n  const clusterCol = df.columns.addNewString(clusterColName);\n  clusterCol.init((i) => clustersCounter[res.clusters[i]] >= minClusterSize ? res.clusters[i].toString() : '-1');\n  const catColorObj = {'-1': DG.Color.setAlpha(DG.Color.lightBlue, 100)};\n  clusterCol.setTag(DG.TAGS.COLOR_CODING_CATEGORICAL,\n    JSON.stringify(catColorObj));\n  clusterCol.temp[DG.TAGS.COLOR_CODING_CATEGORICAL] = catColorObj;\n  // clusterCol.setCategoryOrder(Array.from(new Set(res.clusters)).sort((a, b) => a - b).map((it) => it.toString()));\n\n  const clusterCounterCol = df.columns.addNewInt(clusterCounterColName);\n  clusterCounterCol.init((i) => clustersCounter[res.clusters[i]]);\n  const connectivityCol = df.columns.addNewInt(connectivityColName);\n  connectivityCol.init((i) => connectivity[i]);\n\n  // df.columns.addNewFloat(emberdXColName).init((i) => res.embedX[i]);\n  // df.columns.addNewFloat(emberdYColName).init((i) => res.embedY[i]);\n  // df.columns.addNewInt(clusterCounterColName).init((i) => clustersCounter[res.clusters[i]]);\n  // df.columns.addNewString(clusterColName).init((i) => res.clusters[i].toString());\n  sc.props.xColumnName = emberdXColName;\n  sc.props.yColumnName = emberdYColName;\n  sc.props.colorColumnName = clusterColName;\n  sc.props.markerDefaultSize = 6;\n  terminateSub.unsubscribe();\n\n  // limit cluster interconnectivity to 20 lines\n\n  const maxInterClusterLinks = 20;\n  // I know pushing to array is slow, but it's not a big deal here as its max 100k pushes in most cases\n  const filteredIs: number[] = [];\n  const filteredJs: number[] = [];\n  const clusterInterconnectivity = new Map<number, Map<number, number>>();\n  for (let i = 0; i < res.is.length; i++) {\n    let from = res.clusters[res.is[i]];\n    let to = res.clusters[res.js[i]];\n    if (from === to) {\n      filteredIs.push(res.is[i]);\n      filteredJs.push(res.js[i]);\n      continue;\n    }\n    if (from > to) {\n      const tmp = from;\n      from = to;\n      to = tmp;\n    }\n\n    let fromMap = clusterInterconnectivity.get(from);\n    if (!fromMap) {\n      fromMap = new Map<number, number>();\n      clusterInterconnectivity.set(from, fromMap);\n    }\n    let count = fromMap.get(to);\n    if (!count)\n      count = 0;\n\n    if (count >= maxInterClusterLinks)\n      continue;\n\n    count++;\n    fromMap.set(to, count);\n    filteredIs.push(res.is[i]);\n    filteredJs.push(res.js[i]);\n  }\n\n\n  // const _scLines = new ScatterPlotLinesRenderer(sc, emberdXColName, emberdYColName,\n  //   {from: new Uint32Array(filteredIs) as any, to: new Uint32Array(filteredJs) as any,\n  //     drawArrows: false, opacity: 0.3, skipMultiLineCalculation: true,\n  //     skipShortLines: true, skipMouseOverDetection: true, shortLineThreshold: 6, width: 0.75, color: '128,128,128'},\n  //   ScatterPlotCurrentLineStyle.none);\n\n  //const _scLines = new SCLinesRenderer(sc, filteredIs, filteredJs, 6, 0.75, '128,128,128');\n\n  ui.setUpdateIndicator(sc.root, false);\n  // sc.close();\n  // const scLinesViewer = new ScatterPlotWithLines(sc, res.is, res.js, emberdXColName, emberdYColName);\n  // tv.addViewer(scLinesViewer);\n\n  return {sc, embedXCol, embedYCol, clusterCol, clusterCounterCol, connectivityCol, i: filteredIs, j: filteredJs};\n}\n\n\nexport class SCLinesRenderer {\n  private renderFlag = false;\n  renderSub: rxjs.Subscription;\n  constructor(public sc: DG.ScatterPlotViewer, public from: ArrayLike<number>,\n    public to: ArrayLike<number>, public shortLineThreshold: number, public width: number, public color: string) {\n    this.renderSub = DG.debounce(sc.onAfterDrawScene, 200).subscribe(() => {\n      if (this.renderFlag) {\n        this.renderFlag = false;\n        return;\n      }\n      this.renderFlag = true;\n      const tempSub = sc.onBeforeDrawScene.subscribe((_) => {\n        this.render();\n        tempSub.unsubscribe();\n      });\n      setTimeout(() => {\n        this.sc.invalidateCanvas();\n        // this.renderFlag = false;\n      });\n    });\n    sc.subs.push(this.renderSub);\n  }\n\n  render() {\n    const xCol = this.sc.dataFrame.getCol(this.sc.props.xColumnName);\n    const yCol = this.sc.dataFrame.getCol(this.sc.props.yColumnName);\n    const filter = this.sc.filter;\n    const positions = new Array(this.sc.dataFrame.rowCount).fill(null)\n      .map((_, i) => !xCol.isNone(i) && !yCol.isNone(i) && filter.get(i) ? this.sc.pointToScreen(i) : null);\n    const canvas = this.sc.canvas;\n    const ctx = canvas.getContext('2d');\n    if (!ctx)\n      return;\n    ctx.strokeStyle = `rgba(${this.color}, 0.3)`;\n    ctx.lineWidth = this.width;\n    const shortLineSquared = this.shortLineThreshold * this.shortLineThreshold;\n    for (let i = 0; i < this.from.length; i++) {\n      ctx.beginPath();\n      const from = this.from[i];\n      const to = this.to[i];\n      if (positions[from] && positions[to]) {\n        const fromPos = positions[from];\n        const toPos = positions[to];\n        const dx = toPos.x - fromPos.x;\n        const dy = toPos.y - fromPos.y;\n        const dist = dx * dx + dy * dy;\n        if (dist < shortLineSquared)\n          continue;\n        ctx.moveTo(fromPos.x, fromPos.y);\n        ctx.lineTo(toPos.x, toPos.y);\n        ctx.stroke();\n        ctx.closePath();\n        // interestingly, doing stroke for each line turns to be 20 times faster than doing stroke for all lines at once\n      }\n    }\n  }\n\n  destroy() {\n    this.renderSub.unsubscribe();\n  }\n}\n"]}

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc